/* Copyright 2019 The Chromium OS Authors. All rights reserved. * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. * * Test USB Type-C Dual Role Port, Audio Accessory, and Try.SRC Device module. */ #include "charge_manager.h" #include "mock/tcpc_mock.h" #include "mock/usb_mux_mock.h" #include "system.h" #include "task.h" #include "test_util.h" #include "timer.h" #include "usb_mux.h" #include "usb_pd_tcpm.h" #include "usb_sm_checks.h" #include "usb_tc_sm.h" #define PORT0 0 /* * Amount of time to wait after a specified timeout. Allows for an extra loop * through statemachine plus 1000 calls to clock */ #define FUDGE (6 * MSEC) /* Unreachable time in future */ #define TIMER_DISABLED 0xffffffffffffffff /* Install Mock TCPC and MUX drivers */ const struct tcpc_config_t tcpc_config[CONFIG_USB_PD_PORT_MAX_COUNT] = { { .drv = &mock_tcpc_driver, }, }; const struct usb_mux usb_muxes[CONFIG_USB_PD_PORT_MAX_COUNT] = { { .driver = &mock_usb_mux_driver, } }; void charge_manager_set_ceil(int port, enum ceil_requestor requestor, int ceil) { /* Do Nothing, but needed for linking */ } void pd_resume_check_pr_swap_needed(int port) { /* Do Nothing, but needed for linking */ } /* Vbus is turned on at the board level, so mock it here for our purposes */ static bool board_vbus_enabled[CONFIG_USB_PD_PORT_MAX_COUNT]; static bool mock_get_vbus_enabled(int port) { return board_vbus_enabled[port]; } static void mock_set_vbus_enabled(int port, bool enabled) { board_vbus_enabled[port] = enabled; } static void mock_reset_vbus_enabled(void) { int i; for (i = 0; i < CONFIG_USB_PD_PORT_MAX_COUNT; i++) mock_set_vbus_enabled(i, false); } int pd_set_power_supply_ready(int port) { mock_set_vbus_enabled(port, true); return EC_SUCCESS; } void pd_power_supply_reset(int port) { mock_set_vbus_enabled(port, false); } __maybe_unused static int test_mux_con_dis_as_src(void) { mock_tcpc.should_print_call = false; mock_usb_mux.num_set_calls = 0; /* Update CC lines send state machine event to process */ mock_tcpc.cc1 = TYPEC_CC_VOLT_RD; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); pd_set_dual_role(0, PD_DRP_TOGGLE_ON); /* This wait trainsitions through AttachWait.SRC then Attached.SRC */ task_wait_event(SECOND); /* We are in Attached.SRC now */ TEST_EQ(mock_usb_mux.state, USB_PD_MUX_USB_ENABLED, "%d"); TEST_EQ(mock_usb_mux.num_set_calls, 1, "%d"); mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* This wait will go through TryWait.SNK then to Unattached.SNK */ task_wait_event(10 * SECOND); /* We are in Unattached.SNK. The mux should have detached */ TEST_EQ(mock_usb_mux.state, USB_PD_MUX_NONE, "%d"); TEST_EQ(mock_usb_mux.num_set_calls, 2, "%d"); return EC_SUCCESS; } __maybe_unused static int test_mux_con_dis_as_snk(void) { mock_tcpc.should_print_call = false; mock_usb_mux.num_set_calls = 0; /* * we expect a PD-capable partner to be able to check below * whether it is data capable. */ tc_pd_connection(0, 1); /* Update CC lines send state machine event to process */ mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* This wait will go through AttachWait.SNK to Attached.SNK */ task_wait_event(5 * SECOND); /* * We are in Attached.SNK now, but the port partner isn't data capable * so we should not connect the USB data mux. */ TEST_EQ(mock_usb_mux.state, USB_PD_MUX_NONE, "%d"); mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; mock_tcpc.vbus_level = 0; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* This wait will go through TryWait.SNK then to Unattached.SNK */ task_wait_event(10 * SECOND); /* We are in Unattached.SNK. The mux should have detached */ TEST_EQ(mock_usb_mux.state, USB_PD_MUX_NONE, "%d"); TEST_LE(mock_usb_mux.num_set_calls, 2, "%d"); return EC_SUCCESS; } __maybe_unused static int test_power_role_set(void) { mock_tcpc.num_calls_to_set_header = 0; /* * We need to allow auto toggling to see the port partner attach * as a sink */ pd_set_dual_role(PORT0, PD_DRP_TOGGLE_ON); /* Update CC lines send state machine event to process */ mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RD; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); task_wait_event(10 * SECOND); /* We are in Attached.SRC now */ TEST_EQ(mock_tcpc.last.power_role, PD_ROLE_SOURCE, "%d"); TEST_EQ(mock_tcpc.last.data_role, PD_ROLE_DFP, "%d"); /* * We allow 2 separate calls to update the header since power and data * role updates can be separate calls depending on the state is came * from. */ TEST_LE(mock_tcpc.num_calls_to_set_header, 2, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_cc1_default(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC USB-DEF on CC1"); mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_DEF; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; mock_tcpc.vbus_level = 1; /* * In this test we are expecting value of polarity, which is set by * default for tcpc mock. Initialize it with something else, in order * to catch possible errors. */ mock_tcpc.last.polarity = POLARITY_COUNT; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC1, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_cc1_1A5(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC USB-1A5 on CC1"); mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_1_5; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC1, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_cc1_3A0(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC USB-3A0 on CC1"); mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC1, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_cc2_default(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC USB-DEF on CC2"); mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_DEF; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_cc2_1A5(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC USB-1A5 on CC2"); mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_1_5; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_cc2_3A0(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC USB-3A0 on CC2"); mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_dts_cc1_default(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC DTS-Default on CC1"); mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_1_5; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC1_DTS, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_dts_cc1_1A5(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC DTS-1A5 on CC1"); mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_1_5; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_DEF; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC1_DTS, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_dts_cc1_3A0(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC DTS-1A5 on CC1"); mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_DEF; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC1_DTS, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_dts_cc2_default(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC DTS-Default on CC2"); mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_1_5; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2_DTS, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_dts_cc2_1A5(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC DTS-1A5 on CC2"); mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_DEF; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_1_5; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2_DTS, "%d"); return EC_SUCCESS; } __maybe_unused static int test_polarity_dts_cc2_3A0(void) { /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC DTS-1A5 on CC2"); mock_tcpc.cc1 = TYPEC_CC_VOLT_RP_DEF; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE + FUDGE); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2_DTS, "%d"); return EC_SUCCESS; } /* Record any calls that would change our CCs to Rp */ static int changes_to_rp; static int record_changes_to_rp(int port, int pull) { if (pull == TYPEC_CC_RP) ++changes_to_rp; return EC_SUCCESS; }; __maybe_unused static int test_try_src_disabled(void) { changes_to_rp = 0; mock_tcpc.callbacks.set_cc = &record_changes_to_rp; tc_try_src_override(TRY_SRC_OVERRIDE_OFF); /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC"); mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Wait a long time past many potential transitions */ task_wait_event(10 * SECOND); TEST_EQ(mock_tcpc.last.cc, TYPEC_CC_RD, "%d"); TEST_EQ(changes_to_rp, 0, "%d"); TEST_EQ(mock_tcpc.last.power_role, PD_ROLE_SINK, "%d"); TEST_EQ(mock_tcpc.last.data_role, PD_ROLE_UFP, "%d"); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2, "%d"); TEST_EQ(tc_is_attached_snk(PORT0), true, "%d"); return EC_SUCCESS; } /* Act like a PD device that switches to opposite role */ static int switch_to_opposite_role(int port, int pull) { static enum tcpc_cc_pull last_pull = -1; if (pull == last_pull) return EC_SUCCESS; last_pull = pull; if (pull == TYPEC_CC_RP) { /* If host is setting Rp, then CCs will negotiate as SNK */ mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RD; mock_tcpc.vbus_level = 0; ccprints("[Test] Partner presents SNK"); } else if (pull == TYPEC_CC_RD) { /* If host is setting Rd, then CCs will negotiate as SRC */ mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; ccprints("[Test] Partner presents SRC with Vbus ON"); } task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); return EC_SUCCESS; }; __maybe_unused static int test_try_src_partner_switches(void) { mock_tcpc.callbacks.set_cc = &switch_to_opposite_role; tc_try_src_override(TRY_SRC_OVERRIDE_ON); /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC"); mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* We are in AttachWait.SNK now */ /* Before tCCDebounce elapses, we should still be a SNK */ task_wait_event(PD_T_CC_DEBOUNCE / 2); TEST_EQ(mock_tcpc.last.cc, TYPEC_CC_RD, "%d"); task_wait_event(PD_T_CC_DEBOUNCE / 2); /* We are in Try.SRC now */ /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE / 2); TEST_EQ(mock_tcpc.last.cc, TYPEC_CC_RP, "%d"); /* Wait for tCCDebounce to elapse, then should be SRC */ task_wait_event(PD_T_CC_DEBOUNCE); TEST_EQ(mock_tcpc.last.power_role, PD_ROLE_SOURCE, "%d"); TEST_EQ(mock_tcpc.last.data_role, PD_ROLE_DFP, "%d"); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2, "%d"); TEST_EQ(tc_is_attached_src(PORT0), true, "%d"); return EC_SUCCESS; } /* Act like a non-PD charger that always presents Vbus and Rp lines */ static int dumb_src_charger_cc_response(int port, int pull) { static enum tcpc_cc_pull last_pull = -1; if (pull == last_pull) return EC_SUCCESS; last_pull = pull; if (pull == TYPEC_CC_RP) { /* If host is setting Rp, then CCs will open */ mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; } else if (pull == TYPEC_CC_RD) { /* If host is setting Rd, then CCs will negotiate */ mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; } mock_tcpc.vbus_level = 1; ccprints("[Test] Partner presents SRC with Vbus ON"); task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); return EC_SUCCESS; }; __maybe_unused static int test_try_src_partner_does_not_switch_vbus(void) { tc_try_src_override(TRY_SRC_OVERRIDE_ON); mock_tcpc.callbacks.set_cc = &dumb_src_charger_cc_response; /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC"); mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* We are in AttachWait.SNK now */ /* Before tCCDebounce elapses, we should still be a SNK */ task_wait_event(PD_T_CC_DEBOUNCE / 2); TEST_EQ(mock_tcpc.last.cc, TYPEC_CC_RD, "%d"); task_wait_event(PD_T_CC_DEBOUNCE / 2); /* We are in Try.SRC now */ /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE / 2); TEST_EQ(mock_tcpc.last.cc, TYPEC_CC_RP, "%d"); /* * Wait for tTryTimeout to elapse, then should be * presenting SNK resistors again but not connected yet, until we * debounce Vbus. */ task_wait_event(PD_T_TRY_TIMEOUT); TEST_EQ(mock_tcpc.last.power_role, PD_ROLE_SINK, "%d"); TEST_EQ(tc_is_attached_snk(PORT0), false, "%d"); /* Once we debouce Vbus, then we should be connected */ task_wait_event(PD_T_CC_DEBOUNCE); TEST_EQ(mock_tcpc.last.power_role, PD_ROLE_SINK, "%d"); TEST_EQ(mock_tcpc.last.data_role, PD_ROLE_UFP, "%d"); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2, "%d"); TEST_EQ(tc_is_attached_snk(PORT0), true, "%d"); return EC_SUCCESS; } /* Act like a PD charger that will drop Vbus when CC lines are open */ static int src_charger_drops_vbus_cc_response(int port, int pull) { static enum tcpc_cc_pull last_pull = -1; if (pull == last_pull) return EC_SUCCESS; last_pull = pull; if (pull == TYPEC_CC_RP) { /* If host is setting Rp, then CCs will open */ mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; mock_tcpc.vbus_level = 0; ccprints("[Test] Partner presents SRC with Vbus OFF"); } else if (pull == TYPEC_CC_RD) { /* If host is setting Rd, then CCs will negotiate */ mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; ccprints("[Test] Partner presents SRC with Vbus ON"); } task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); return EC_SUCCESS; }; __maybe_unused static int test_try_src_partner_does_not_switch_no_vbus(void) { tc_try_src_override(TRY_SRC_OVERRIDE_ON); mock_tcpc.callbacks.set_cc = &src_charger_drops_vbus_cc_response; /* Update CC lines send state machine event to process */ ccprints("[Test] Partner connects as SRC"); mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* We are in AttachWait.SNK now */ /* Before tCCDebounce elapses, we should still be a SNK */ task_wait_event(PD_T_CC_DEBOUNCE / 2); TEST_EQ(mock_tcpc.last.cc, TYPEC_CC_RD, "%d"); task_wait_event(PD_T_CC_DEBOUNCE / 2); /* We are in Try.SRC now */ /* Before tCCDebounce elapses, we should SRC */ task_wait_event(PD_T_CC_DEBOUNCE / 2); TEST_EQ(mock_tcpc.last.cc, TYPEC_CC_RP, "%d"); /* * Wait for tTryTimeout to elapse, then should be * presenting SNK resistors again but not connected yet, until we * debounce Vbus. */ task_wait_event(PD_T_DRP_TRY); TEST_EQ(mock_tcpc.last.power_role, PD_ROLE_SINK, "%d"); TEST_EQ(tc_is_attached_snk(PORT0), false, "%d"); /* Once we debouce Vbus, then we should be connected */ task_wait_event(PD_T_CC_DEBOUNCE); TEST_EQ(mock_tcpc.last.power_role, PD_ROLE_SINK, "%d"); TEST_EQ(mock_tcpc.last.data_role, PD_ROLE_UFP, "%d"); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC2, "%d"); TEST_EQ(tc_is_attached_snk(PORT0), true, "%d"); return EC_SUCCESS; } /* Record the cc voltages */ static enum tcpc_cc_pull cc_pull[16]; static int cc_pull_count; static int record_cc_pull(int port, int pull) { if (cc_pull_count < ARRAY_SIZE(cc_pull)) cc_pull[cc_pull_count++] = pull; return EC_SUCCESS; }; __maybe_unused static int test_cc_open_on_normal_reset(void) { uint32_t flags = system_get_reset_flags(); cc_pull_count = 0; mock_tcpc.callbacks.set_cc = &record_cc_pull; system_clear_reset_flags(EC_RESET_FLAG_POWER_ON); task_set_event(TASK_ID_PD_C0, TASK_EVENT_RESET_DONE); task_wait_event(SECOND * 10); /* Ensure that the first CC set call was to open (error recovery). */ TEST_GT(cc_pull_count, 0, "%d"); TEST_EQ(cc_pull[0], TYPEC_CC_OPEN, "%d"); /* Ensure that the second CC set call was to Rd (sink) */ TEST_GT(cc_pull_count, 1, "%d"); TEST_EQ(cc_pull[1], TYPEC_CC_RD, "%d"); /* Reset system flags after test */ system_set_reset_flags(flags); return EC_SUCCESS; } __maybe_unused static int test_cc_rd_on_por_reset(void) { uint32_t flags = system_get_reset_flags(); cc_pull_count = 0; mock_tcpc.callbacks.set_cc = &record_cc_pull; system_set_reset_flags(EC_RESET_FLAG_POWER_ON); task_set_event(TASK_ID_PD_C0, TASK_EVENT_RESET_DONE); task_wait_event(SECOND * 10); /* Ensure that the first CC set call was to Rd (sink) */ TEST_GT(cc_pull_count, 0, "%d"); TEST_EQ(cc_pull[0], TYPEC_CC_RD, "%d"); /* Reset system flags after test */ system_clear_reset_flags(~flags); return EC_SUCCESS; } __maybe_unused static int test_auto_toggle_delay(void) { uint64_t time; /* Start with auto toggle disabled so we can time the transition */ pd_set_dual_role(PORT0, PD_DRP_TOGGLE_OFF); task_wait_event(SECOND); /* Enabled auto toggle and start the timer for the transition */ pd_set_dual_role(PORT0, PD_DRP_TOGGLE_ON); time = get_time().val; /* * Ensure we do not transition to auto toggle from Rd or Rp in less time * than tDRP minimum (50 ms) * dcSRC.DRP minimum (30%) = 15 ms. * Otherwise we can confuse external partners with the first transition * to auto toggle. */ task_wait_event(SECOND); TEST_GT(mock_tcpc.first_call_to_enable_auto_toggle - time, (uint64_t)15 * MSEC, "%" PRIu64); return EC_SUCCESS; } __maybe_unused static int test_auto_toggle_delay_early_connect(void) { cc_pull_count = 0; mock_tcpc.callbacks.set_cc = &record_cc_pull; mock_tcpc.first_call_to_enable_auto_toggle = TIMER_DISABLED; /* Start with auto toggle disabled */ pd_set_dual_role(PORT0, PD_DRP_TOGGLE_OFF); task_wait_event(SECOND); /* Enabled auto toggle */ pd_set_dual_role(PORT0, PD_DRP_TOGGLE_ON); /* Wait less than tDRP_SNK(40ms) and tDRP_SRC(30ms) */ task_wait_event(MIN(PD_T_DRP_SNK, PD_T_DRP_SRC) - (10 * MSEC)); /* Have partner connect as SRC */ mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_RP_3_0; mock_tcpc.vbus_level = 1; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* Ensure the auto toggle enable was never called */ task_wait_event(SECOND); TEST_EQ(mock_tcpc.first_call_to_enable_auto_toggle, TIMER_DISABLED, "%" PRIu64); /* Ensure that the first CC set call was to Rd. */ TEST_GT(cc_pull_count, 0, "%d"); TEST_EQ(cc_pull[0], TYPEC_CC_RD, "%d"); return EC_SUCCESS; } /* TODO(b/153071799): test as SNK monitor for Vbus disconnect (not CC line) */ __maybe_unused static int test_typec_dis_as_src(void) { mock_tcpc.should_print_call = false; /* Update CC lines send state machine event to process */ mock_tcpc.cc1 = TYPEC_CC_VOLT_RD; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); pd_set_dual_role(0, PD_DRP_TOGGLE_ON); /* This wait trainsitions through AttachWait.SRC then Attached.SRC */ task_wait_event(SECOND); /* * We are in Attached.SRC now, verify: * - Vbus was turned on * - Rp is set * - polarity is detected as CC1 * - Rp was set to default configured level */ TEST_EQ(mock_tcpc.last.cc, TYPEC_CC_RP, "%d"); TEST_EQ(mock_tcpc.last.polarity, POLARITY_CC1, "%d"); TEST_EQ(mock_tcpc.last.rp, CONFIG_USB_PD_PULLUP, "%d"); TEST_EQ(mock_get_vbus_enabled(0), true, "%d"); /* Force a detach through CC open */ mock_tcpc.cc1 = TYPEC_CC_VOLT_OPEN; mock_tcpc.cc2 = TYPEC_CC_VOLT_OPEN; task_set_event(TASK_ID_PD_C0, PD_EVENT_CC); /* This wait will go through TryWait.SNK then to Unattached.SNK */ task_wait_event(10 * SECOND); /* We are in Unattached.SNK. Verify Vbus has been removed */ TEST_EQ(mock_get_vbus_enabled(0), false, "%d"); return EC_SUCCESS; } /* Reset the mocks before each test */ void before_test(void) { mock_usb_mux_reset(); mock_tcpc_reset(); mock_reset_vbus_enabled(); /* Restart the PD task and let it settle */ task_set_event(TASK_ID_PD_C0, TASK_EVENT_RESET_DONE); task_wait_event(SECOND); /* Print out TCPC calls for easier debugging */ mock_tcpc.should_print_call = true; } void run_test(int argc, char **argv) { test_reset(); RUN_TEST(test_polarity_cc1_default); RUN_TEST(test_polarity_cc1_1A5); RUN_TEST(test_polarity_cc1_3A0); RUN_TEST(test_polarity_cc2_default); RUN_TEST(test_polarity_cc2_1A5); RUN_TEST(test_polarity_cc2_3A0); RUN_TEST(test_polarity_dts_cc1_default); RUN_TEST(test_polarity_dts_cc1_1A5); RUN_TEST(test_polarity_dts_cc1_3A0); RUN_TEST(test_polarity_dts_cc2_default); RUN_TEST(test_polarity_dts_cc2_1A5); RUN_TEST(test_polarity_dts_cc2_3A0); RUN_TEST(test_mux_con_dis_as_src); RUN_TEST(test_mux_con_dis_as_snk); RUN_TEST(test_power_role_set); RUN_TEST(test_typec_dis_as_src); RUN_TEST(test_try_src_disabled); RUN_TEST(test_try_src_partner_switches); RUN_TEST(test_try_src_partner_does_not_switch_vbus); RUN_TEST(test_try_src_partner_does_not_switch_no_vbus); RUN_TEST(test_cc_open_on_normal_reset); RUN_TEST(test_cc_rd_on_por_reset); RUN_TEST(test_auto_toggle_delay); RUN_TEST(test_auto_toggle_delay_early_connect); /* Do basic state machine validity checks last. */ RUN_TEST(test_tc_no_parent_cycles); RUN_TEST(test_tc_all_states_named); test_print_result(); }