blob: 974aeb625ac92474ef064d5aec2e2ed046e41707 [file] [log] [blame]
Brandon Breitenstein99b38a92019-12-19 23:12:58 -08001/* SPDX-License-Identifier: GPL-2.0-or-later */
2
John Zhao3c463712022-01-10 15:49:37 -08003#define __SIMPLE_DEVICE__
4
Brandon Breitenstein29144552020-11-17 11:29:45 -08005#include <bootmode.h>
Brandon Breitenstein99b38a92019-12-19 23:12:58 -08006#include <console/console.h>
7#include <device/pci.h>
John Zhao3c463712022-01-10 15:49:37 -08008#include <intelblocks/p2sb.h>
9#include <intelblocks/pcr.h>
Brandon Breitenstein99b38a92019-12-19 23:12:58 -080010#include <intelblocks/pmc_ipc.h>
Tim Wawrzynczak59a621a2021-03-22 10:43:42 -060011#include <intelblocks/systemagent.h>
Tim Wawrzynczakeb6ebc02021-03-22 16:39:57 -060012#include <intelblocks/tcss.h>
13#include <inttypes.h>
Sridhar Siricillaf9ee35e2022-12-08 16:15:16 +053014#include <lib.h>
Brandon Breitenstein29144552020-11-17 11:29:45 -080015#include <security/vboot/vboot_common.h>
Brandon Breitenstein99b38a92019-12-19 23:12:58 -080016#include <soc/pci_devs.h>
Tim Wawrzynczak59a621a2021-03-22 10:43:42 -060017#include <soc/pcr_ids.h>
18#include <soc/tcss.h>
Deepti Deshattyc146daf2021-05-20 21:01:52 +053019#include <drivers/intel/pmc_mux/conn/chip.h>
Brandon Breitenstein99b38a92019-12-19 23:12:58 -080020
Tim Wawrzynczak59a621a2021-03-22 10:43:42 -060021#define BIAS_CTRL_VW_INDEX_SHIFT 16
22#define BIAS_CTRL_BIT_POS_SHIFT 8
Kapil Porwal400f1aa2023-07-10 11:40:16 +000023#define WAIT_FOR_DISPLAYPORT_TIMEOUT_MS 1000
24#define WAIT_FOR_DP_MODE_ENTRY_TIMEOUT_MS 500
25#define WAIT_FOR_HPD_TIMEOUT_MS 3000
Tim Wawrzynczak59a621a2021-03-22 10:43:42 -060026
Brandon Breitenstein99b38a92019-12-19 23:12:58 -080027static uint32_t tcss_make_conn_cmd(int u, int u3, int u2, int ufp, int hsl,
28 int sbu, int acc)
29{
30 return TCSS_CD_FIELD(USAGE, u) |
31 TCSS_CD_FIELD(USB3, u3) |
32 TCSS_CD_FIELD(USB2, u2) |
33 TCSS_CD_FIELD(UFP, ufp) |
34 TCSS_CD_FIELD(HSL, hsl) |
35 TCSS_CD_FIELD(SBU, sbu) |
36 TCSS_CD_FIELD(ACC, acc);
37}
38
39static uint32_t tcss_make_alt_mode_cmd_buf_0(int u, int u3, int m)
40{
41 return TCSS_ALT_FIELD(USAGE, u) |
42 TCSS_ALT_FIELD(USB3, u3) |
43 TCSS_ALT_FIELD(MODE, m);
44
45}
46
47static uint32_t tcss_make_alt_mode_cmd_buf_1(int p, int c, int ufp, int dp)
48{
49 return TCSS_ALT_FIELD(POLARITY, p) |
50 TCSS_ALT_FIELD(CABLE, c) |
51 TCSS_ALT_FIELD(UFP, ufp) |
52 TCSS_ALT_FIELD(DP_MODE, dp);
53}
54
55static uint32_t tcss_make_safe_mode_cmd(int u, int u3)
56{
57 return TCSS_CD_FIELD(USAGE, u) |
58 TCSS_CD_FIELD(USB3, u3);
59}
60
61
62static uint32_t tcss_make_hpd_mode_cmd(int u, int u3, int hpd_lvl, int hpd_irq)
63{
64 return TCSS_HPD_FIELD(USAGE, u) |
65 TCSS_HPD_FIELD(USB3, u3) |
66 TCSS_HPD_FIELD(LVL, hpd_lvl) |
67 TCSS_HPD_FIELD(IRQ, hpd_irq);
68
69}
70
71static int send_pmc_req(int cmd_type, const struct pmc_ipc_buffer *req,
72 struct pmc_ipc_buffer *res, uint32_t size)
73{
Brandon Breitenstein99b38a92019-12-19 23:12:58 -080074 uint32_t cmd_reg;
75 uint32_t res_reg;
76 int tries = 2;
77 int r;
78
79 cmd_reg = pmc_make_ipc_cmd(PMC_IPC_USBC_CMD_ID, PMC_IPC_USBC_SUBCMD_ID,
80 size);
81
82 printk(BIOS_DEBUG, "Raw Buffer output 0 %08" PRIx32 "\n", req->buf[0]);
83 printk(BIOS_DEBUG, "Raw Buffer output 1 %08" PRIx32 "\n", req->buf[1]);
84
85 do {
86 r = pmc_send_ipc_cmd(cmd_reg, req, res);
87 if (r < 0) {
88 printk(BIOS_ERR, "pmc_send_ipc_cmd failed\n");
89 return -1;
90 }
91
92 res_reg = res->buf[0];
93 if (cmd_type == CONNECT_REQ) {
94 if (!TCSS_CONN_STATUS_HAS_FAILED(res_reg)) {
95 printk(BIOS_DEBUG, "pmc_send_ipc_cmd succeeded\n");
96 return 0;
97 }
98
99 if (TCSS_CONN_STATUS_IS_FATAL(res_reg)) {
100 printk(BIOS_ERR, "pmc_send_ipc_cmd status: fatal\n");
101 return -1;
102 }
103 } else {
104 if (!TCSS_STATUS_HAS_FAILED(res_reg)) {
105 printk(BIOS_DEBUG, "pmc_send_ipc_cmd succeeded\n");
106 return 0;
107 }
108
109 if (TCSS_STATUS_IS_FATAL(res_reg)) {
110 printk(BIOS_ERR, "pmc_send_ipc_cmd status: fatal\n");
111 return -1;
112 }
113 }
114 } while (--tries >= 0);
115
116 printk(BIOS_ERR, "pmc_send_ipc_cmd failed after retries\n");
117 return -1;
118}
119
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800120static int send_pmc_disconnect_request(int port, const struct tcss_port_map *port_map)
121{
122 uint32_t cmd;
123 struct pmc_ipc_buffer req = { 0 };
124 struct pmc_ipc_buffer rsp;
125
126 cmd = tcss_make_conn_cmd(PMC_IPC_TCSS_DISC_REQ_RES, port_map->usb3_port,
127 port_map->usb2_port, 0, 0, 0, 0);
128
129 req.buf[0] = cmd;
130
131 printk(BIOS_DEBUG, "port C%d DISC req: usage %d usb3 %d usb2 %d\n",
132 port,
133 GET_TCSS_CD_FIELD(USAGE, cmd),
134 GET_TCSS_CD_FIELD(USB3, cmd),
135 GET_TCSS_CD_FIELD(USB2, cmd));
136
137 return send_pmc_req(CONNECT_REQ, &req, &rsp, PMC_IPC_DISC_REQ_SIZE);
138}
139
Derek Huang7f22bc42021-08-25 17:47:25 +0800140static int send_pmc_connect_request(int port, const struct usbc_mux_info *mux_data,
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800141 const struct tcss_port_map *port_map)
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800142{
143 uint32_t cmd;
144 struct pmc_ipc_buffer req = { 0 };
Furquan Shaikh338d6682021-03-03 09:45:50 -0800145 struct pmc_ipc_buffer rsp;
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800146
147 cmd = tcss_make_conn_cmd(
148 PMC_IPC_TCSS_CONN_REQ_RES,
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800149 port_map->usb3_port,
150 port_map->usb2_port,
151 mux_data->ufp,
152 mux_data->polarity,
153 mux_data->polarity,
Derek Huang7f22bc42021-08-25 17:47:25 +0800154 mux_data->dbg_acc);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800155
156 req.buf[0] = cmd;
157
158 printk(BIOS_DEBUG, "port C%d CONN req: usage %d usb3 %d usb2 %d "
159 "ufp %d ori_hsl %d ori_sbu %d dbg_acc %d\n",
160 port,
161 GET_TCSS_CD_FIELD(USAGE, cmd),
162 GET_TCSS_CD_FIELD(USB3, cmd),
163 GET_TCSS_CD_FIELD(USB2, cmd),
164 GET_TCSS_CD_FIELD(UFP, cmd),
165 GET_TCSS_CD_FIELD(HSL, cmd),
166 GET_TCSS_CD_FIELD(SBU, cmd),
167 GET_TCSS_CD_FIELD(ACC, cmd));
168
Furquan Shaikh338d6682021-03-03 09:45:50 -0800169 return send_pmc_req(CONNECT_REQ, &req, &rsp, PMC_IPC_CONN_REQ_SIZE);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800170}
171
Derek Huang7f22bc42021-08-25 17:47:25 +0800172static int send_pmc_safe_mode_request(int port, const struct usbc_mux_info *mux_data,
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800173 const struct tcss_port_map *port_map)
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800174{
175 uint32_t cmd;
176 struct pmc_ipc_buffer req = { 0 };
Furquan Shaikh338d6682021-03-03 09:45:50 -0800177 struct pmc_ipc_buffer rsp;
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800178
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800179 cmd = tcss_make_safe_mode_cmd(PMC_IPC_TCSS_SAFE_MODE_REQ_RES, port_map->usb3_port);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800180
181 req.buf[0] = cmd;
182
183 printk(BIOS_DEBUG, "port C%d SAFE req: usage %d usb3 %d\n",
184 port,
185 GET_TCSS_CD_FIELD(USAGE, cmd),
186 GET_TCSS_CD_FIELD(USB3, cmd));
187
Furquan Shaikh338d6682021-03-03 09:45:50 -0800188 return send_pmc_req(SAFE_REQ, &req, &rsp, PMC_IPC_SAFE_REQ_SIZE);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800189}
190
Derek Huang7f22bc42021-08-25 17:47:25 +0800191static int send_pmc_dp_hpd_request(int port, const struct usbc_mux_info *mux_data,
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800192 const struct tcss_port_map *port_map)
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800193{
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800194 struct pmc_ipc_buffer req = { 0 };
Furquan Shaikh338d6682021-03-03 09:45:50 -0800195 struct pmc_ipc_buffer rsp;
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800196 uint32_t cmd;
197
198 cmd = tcss_make_hpd_mode_cmd(
199 PMC_IPC_TCSS_HPD_REQ_RES,
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800200 port_map->usb3_port,
201 mux_data->hpd_lvl,
202 mux_data->hpd_irq);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800203
204 req.buf[0] = cmd;
205
Furquan Shaikh338d6682021-03-03 09:45:50 -0800206 return send_pmc_req(HPD_REQ, &req, &rsp, PMC_IPC_HPD_REQ_SIZE);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800207}
208
Sridhar Siricillaf9ee35e2022-12-08 16:15:16 +0530209static uint8_t get_dp_mode(uint8_t dp_pin_mode)
210{
211 switch (dp_pin_mode) {
212 case MODE_DP_PIN_A:
213 case MODE_DP_PIN_B:
214 case MODE_DP_PIN_C:
215 case MODE_DP_PIN_D:
216 case MODE_DP_PIN_E:
217 case MODE_DP_PIN_F:
218 return log2(dp_pin_mode) + 1;
219 default:
220 return 0;
221 }
222}
223
Derek Huang7f22bc42021-08-25 17:47:25 +0800224static int send_pmc_dp_mode_request(int port, const struct usbc_mux_info *mux_data,
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800225 const struct tcss_port_map *port_map)
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800226{
227 uint32_t cmd;
228 uint8_t dp_mode;
229 int ret;
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800230
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800231 struct pmc_ipc_buffer req = { 0 };
Furquan Shaikh338d6682021-03-03 09:45:50 -0800232 struct pmc_ipc_buffer rsp;
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800233
234 cmd = tcss_make_alt_mode_cmd_buf_0(
235 PMC_IPC_TCSS_ALTMODE_REQ_RES,
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800236 port_map->usb3_port,
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800237 PMC_IPC_DP_MODE);
238
239 req.buf[0] = cmd;
240
241 printk(BIOS_DEBUG, "port C%d ALT_1 req: usage %d usb3 %d dp_mode %d\n",
242 port,
243 GET_TCSS_ALT_FIELD(USAGE, cmd),
244 GET_TCSS_ALT_FIELD(USB3, cmd),
245 GET_TCSS_ALT_FIELD(MODE, cmd));
246
Sridhar Siricillaf9ee35e2022-12-08 16:15:16 +0530247 dp_mode = get_dp_mode(mux_data->dp_pin_mode);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800248 cmd = tcss_make_alt_mode_cmd_buf_1(
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800249 mux_data->polarity,
250 mux_data->cable,
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800251 0, /* ufp is not supported in DP ALT Mode request */
252 dp_mode);
253
254 printk(BIOS_DEBUG, "port C%d ALT_2 req: polarity %d cable %d ufp %d "
255 "dp_mode %d\n",
256 port,
257 GET_TCSS_ALT_FIELD(POLARITY, cmd),
258 GET_TCSS_ALT_FIELD(CABLE, cmd),
259 GET_TCSS_ALT_FIELD(UFP, cmd),
260 GET_TCSS_ALT_FIELD(DP_MODE, cmd));
261
262 req.buf[1] = cmd;
263
Furquan Shaikh338d6682021-03-03 09:45:50 -0800264 ret = send_pmc_req(DP_REQ, &req, &rsp, PMC_IPC_ALT_REQ_SIZE);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800265 if (ret)
266 return ret;
267
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800268 send_pmc_dp_hpd_request(port, mux_data, port_map);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800269 return 0;
270}
271
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800272static void tcss_init_mux(int port, const struct tcss_port_map *port_map)
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800273{
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800274 int ret;
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800275
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800276 ret = send_pmc_disconnect_request(port, port_map);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800277 if (ret)
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800278 printk(BIOS_ERR, "Failed to setup port:%d to initial state\n", port);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800279}
280
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800281static void tcss_configure_dp_mode(const struct tcss_port_map *port_map, size_t num_ports)
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800282{
Kapil Porwal400f1aa2023-07-10 11:40:16 +0000283 int ret, port_bitmask;
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800284 size_t i;
Derek Huang7f22bc42021-08-25 17:47:25 +0800285 const struct usbc_ops *ops;
286 struct usbc_mux_info mux_info;
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800287 const struct tcss_port_map *port_info;
Brandon Breitenstein29144552020-11-17 11:29:45 -0800288
289 if (!display_init_required())
290 return;
291
Derek Huang7f22bc42021-08-25 17:47:25 +0800292 ops = usbc_get_ops();
293 if (ops == NULL)
294 return;
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800295
Kapil Porwal400f1aa2023-07-10 11:40:16 +0000296 port_bitmask = ops->dp_ops.wait_for_connection(WAIT_FOR_DISPLAYPORT_TIMEOUT_MS);
297 if (!port_bitmask) /* No DP device is connected */
298 return;
299
Derek Huang7f22bc42021-08-25 17:47:25 +0800300 for (i = 0; i < num_ports; i++) {
Kapil Porwal400f1aa2023-07-10 11:40:16 +0000301 if (!(port_bitmask & BIT(i)))
302 continue;
303
304 ret = ops->dp_ops.enter_dp_mode(i);
305 if (ret < 0)
306 continue;
307
308 ret = ops->dp_ops.wait_for_dp_mode_entry(i, WAIT_FOR_DP_MODE_ENTRY_TIMEOUT_MS);
309 if (ret < 0)
310 continue;
311
312 ret = ops->dp_ops.wait_for_hpd(i, WAIT_FOR_HPD_TIMEOUT_MS);
313 if (ret < 0)
314 continue;
315
Derek Huang7f22bc42021-08-25 17:47:25 +0800316 ret = ops->mux_ops.get_mux_info(i, &mux_info);
Kapil Porwal400f1aa2023-07-10 11:40:16 +0000317 if (ret < 0)
Derek Huang7f22bc42021-08-25 17:47:25 +0800318 continue;
319
320 port_info = &port_map[i];
321
322 ret = send_pmc_connect_request(i, &mux_info, port_info);
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800323 if (ret) {
Matt DeVillierb33778e2023-08-04 09:35:32 -0500324 printk(BIOS_ERR, "Port %zu connect request failed\n", i);
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800325 continue;
326 }
Derek Huang7f22bc42021-08-25 17:47:25 +0800327 ret = send_pmc_safe_mode_request(i, &mux_info, port_info);
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800328 if (ret) {
Matt DeVillierb33778e2023-08-04 09:35:32 -0500329 printk(BIOS_ERR, "Port %zu safe mode request failed\n", i);
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800330 continue;
331 }
332
Derek Huang7f22bc42021-08-25 17:47:25 +0800333 ret = send_pmc_dp_mode_request(i, &mux_info, port_info);
334 if (ret) {
Matt DeVillierb33778e2023-08-04 09:35:32 -0500335 printk(BIOS_ERR, "Port C%zu mux set failed with error %d\n", i, ret);
Derek Huang7f22bc42021-08-25 17:47:25 +0800336 } else {
Matt DeVillierb33778e2023-08-04 09:35:32 -0500337 printk(BIOS_INFO, "Port C%zu is configured to DP mode!\n", i);
Derek Huang7f22bc42021-08-25 17:47:25 +0800338 return;
339 }
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800340 }
341}
342
CoolStar9c7433e2023-02-07 14:44:54 -0800343static void tcss_configure_usb_mode(const struct tcss_port_map *port_map, size_t num_ports)
344{
345 int ret;
346 size_t i;
347 const struct usbc_ops *ops;
348 struct usbc_mux_info mux_info;
349 const struct tcss_port_map *port_info;
350
351 ops = usbc_get_ops();
352 if (ops == NULL)
353 return;
354
355 for (i = 0; i < num_ports; i++) {
356 ret = ops->mux_ops.get_mux_info(i, &mux_info);
357 if ((ret < 0) || !mux_info.usb || (mux_info.dp && mux_info.hpd_lvl))
358 continue;
359
360 port_info = &port_map[i];
361 ret = send_pmc_connect_request(i, &mux_info, port_info);
362 if (ret) {
363 printk(BIOS_ERR, "Port %zu connect request failed\n", i);
364 continue;
365 }
366 }
367}
368
Tim Wawrzynczak59a621a2021-03-22 10:43:42 -0600369static uint32_t calc_bias_ctrl_reg_value(gpio_t pad)
370{
371 unsigned int vw_index, vw_bit;
372 const unsigned int cpu_pid = gpio_get_pad_cpu_portid(pad);
373 if (!gpio_get_vw_info(pad, &vw_index, &vw_bit) || !cpu_pid)
374 return 0;
375
376 return vw_index << BIAS_CTRL_VW_INDEX_SHIFT |
377 vw_bit << BIAS_CTRL_BIT_POS_SHIFT |
378 cpu_pid;
379}
380
John848b4252022-03-09 17:51:56 -0800381void tcss_configure_aux_bias_pads_regbar(
John Zhao3c463712022-01-10 15:49:37 -0800382 const struct typec_aux_bias_pads *pads)
Tim Wawrzynczak59a621a2021-03-22 10:43:42 -0600383{
384 for (size_t i = 0; i < MAX_TYPE_C_PORTS; i++) {
385 if (pads[i].pad_auxn_dc && pads[i].pad_auxp_dc) {
386 REGBAR32(PID_IOM, IOM_AUX_BIAS_CTRL_PULLUP_OFFSET(i)) =
387 calc_bias_ctrl_reg_value(pads[i].pad_auxp_dc);
388 REGBAR32(PID_IOM, IOM_AUX_BIAS_CTRL_PULLDOWN_OFFSET(i)) =
389 calc_bias_ctrl_reg_value(pads[i].pad_auxn_dc);
390 }
391 }
392}
393
John2bcc5f32022-03-27 09:06:25 -0700394void ioe_tcss_configure_aux_bias_pads_sbi(
395 const struct typec_aux_bias_pads *pads)
396{
397 for (size_t i = 0; i < MAX_TYPE_C_PORTS; i++) {
398 if (pads[i].pad_auxn_dc && pads[i].pad_auxp_dc) {
399 ioe_p2sb_sbi_write(PID_IOM, IOM_AUX_BIAS_CTRL_PULLUP_OFFSET(i),
400 calc_bias_ctrl_reg_value(pads[i].pad_auxp_dc));
401 ioe_p2sb_sbi_write(PID_IOM, IOM_AUX_BIAS_CTRL_PULLDOWN_OFFSET(i),
402 calc_bias_ctrl_reg_value(pads[i].pad_auxn_dc));
403 }
404 }
405}
406
Deepti Deshattyc146daf2021-05-20 21:01:52 +0530407const struct tcss_port_map *tcss_get_port_info(size_t *num_ports)
408{
409 static struct tcss_port_map port_map[MAX_TYPE_C_PORTS];
410 size_t active_ports = 0;
411 size_t port;
412
413 for (port = 0; port < MAX_TYPE_C_PORTS; port++) {
414 const struct device_path conn_path[] = {
415 {.type = DEVICE_PATH_PCI, .pci.devfn = PCH_DEVFN_PMC},
416 {.type = DEVICE_PATH_GENERIC, .generic.id = 0, .generic.subid = 0},
417 {.type = DEVICE_PATH_GENERIC, .generic.id = port},
418 };
419 const struct device *conn = find_dev_nested_path(pci_root_bus(), conn_path,
420 ARRAY_SIZE(conn_path));
421 unsigned int usb2_port, usb3_port;
422
Reka Normane78669e2022-03-30 19:49:31 +1100423 if (!is_dev_enabled(conn))
Deepti Deshattyc146daf2021-05-20 21:01:52 +0530424 continue;
425
426 if (CONFIG(DRIVERS_INTEL_PMC) &&
427 intel_pmc_mux_conn_get_ports(conn, &usb2_port, &usb3_port)) {
428 port_map[active_ports].usb2_port = usb2_port;
429 port_map[active_ports].usb3_port = usb3_port;
430 ++active_ports;
431 }
432 }
433
434 *num_ports = active_ports;
435 return port_map;
436}
437
Tim Wawrzynczak59a621a2021-03-22 10:43:42 -0600438void tcss_configure(const struct typec_aux_bias_pads aux_bias_pads[MAX_TYPE_C_PORTS])
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800439{
440 const struct tcss_port_map *port_map;
441 size_t num_ports;
442 size_t i;
443
Deepti Deshattyc146daf2021-05-20 21:01:52 +0530444 port_map = tcss_get_port_info(&num_ports);
Sean Rhodes72016022024-01-10 20:38:05 +0000445 if ((port_map == NULL) || platform_is_resuming())
Brandon Breitensteind8774f62021-03-03 14:09:22 -0800446 return;
Brandon Breitenstein29144552020-11-17 11:29:45 -0800447
Sean Rhodes72016022024-01-10 20:38:05 +0000448 if (CONFIG(TCSS_HAS_USBC_OPS))
Sridhar Siricilladb8a97a2022-04-29 15:13:38 +0530449 for (i = 0; i < num_ports; i++)
450 tcss_init_mux(i, &port_map[i]);
Subrata Banik97a45e62022-12-01 16:58:13 +0530451
Sean Rhodes72016022024-01-10 20:38:05 +0000452 /* This should be performed before alternate modes are entered */
453 if (tcss_ops.configure_aux_bias_pads)
454 tcss_ops.configure_aux_bias_pads(aux_bias_pads);
Subrata Banik97a45e62022-12-01 16:58:13 +0530455
Sean Rhodes72016022024-01-10 20:38:05 +0000456 if (CONFIG(ENABLE_TCSS_DISPLAY_DETECTION))
457 tcss_configure_dp_mode(port_map, num_ports);
CoolStar9c7433e2023-02-07 14:44:54 -0800458
Sean Rhodes72016022024-01-10 20:38:05 +0000459 if (CONFIG(ENABLE_TCSS_USB_DETECTION))
460 tcss_configure_usb_mode(port_map, num_ports);
Brandon Breitenstein99b38a92019-12-19 23:12:58 -0800461}
John Zhao3c463712022-01-10 15:49:37 -0800462
John848b4252022-03-09 17:51:56 -0800463bool tcss_valid_tbt_auth(void)
John Zhao3c463712022-01-10 15:49:37 -0800464{
John848b4252022-03-09 17:51:56 -0800465 return REGBAR32(PID_IOM, IOM_CSME_IMR_TBT_STATUS) & TBT_VALID_AUTHENTICATION;
John Zhao3c463712022-01-10 15:49:37 -0800466}