pmic_glink_altmode.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2019-2020, The Linux Foundation. All rights reserved.
  4. * Copyright (c) 2022, Linaro Ltd
  5. */
  6. #include <linux/auxiliary_bus.h>
  7. #include <linux/bitfield.h>
  8. #include <linux/cleanup.h>
  9. #include <linux/module.h>
  10. #include <linux/of.h>
  11. #include <linux/of_device.h>
  12. #include <linux/mutex.h>
  13. #include <linux/property.h>
  14. #include <linux/soc/qcom/pdr.h>
  15. #include <drm/bridge/aux-bridge.h>
  16. #include <linux/usb/pd.h>
  17. #include <linux/usb/typec_altmode.h>
  18. #include <linux/usb/typec_dp.h>
  19. #include <linux/usb/typec_mux.h>
  20. #include <linux/usb/typec_retimer.h>
  21. #include <linux/usb/typec_tbt.h>
  22. #include <linux/soc/qcom/pmic_glink.h>
  23. #define PMIC_GLINK_MAX_PORTS 3
  24. #define USBC_SC8180X_NOTIFY_IND 0x13
  25. #define USBC_CMD_WRITE_REQ 0x15
  26. #define USBC_NOTIFY_IND 0x16
  27. #define ALTMODE_PAN_EN 0x10
  28. #define ALTMODE_PAN_ACK 0x11
  29. struct usbc_write_req {
  30. struct pmic_glink_hdr hdr;
  31. __le32 cmd;
  32. __le32 arg;
  33. __le32 reserved;
  34. };
  35. struct usbc_sc8280x_dp_data {
  36. u8 pin_assignment : 6;
  37. u8 hpd_state : 1;
  38. u8 hpd_irq : 1;
  39. u8 res[7];
  40. };
  41. /* Used for both TBT and USB4 notifications */
  42. struct usbc_sc8280x_tbt_data {
  43. u8 usb_speed : 3;
  44. u8 cable_type : 3;
  45. /* This field is NOP on USB4, all cables support rounded rates by spec */
  46. u8 rounded_cable : 1;
  47. u8 power_limited : 1;
  48. u8 res[11];
  49. };
  50. struct usbc_notify {
  51. struct pmic_glink_hdr hdr;
  52. u8 port_idx;
  53. u8 orientation;
  54. u8 mux_ctrl;
  55. #define MUX_CTRL_STATE_NO_CONN 0
  56. #define MUX_CTRL_STATE_USB3_ONLY 1
  57. #define MUX_CTRL_STATE_DP4LN 2
  58. #define MUX_CTRL_STATE_USB3_DP 3
  59. #define MUX_CTRL_STATE_TUNNELING 4
  60. u8 res;
  61. __le16 vid;
  62. __le16 svid;
  63. union usbc_sc8280x_extended_data {
  64. struct usbc_sc8280x_dp_data dp;
  65. struct usbc_sc8280x_tbt_data tbt;
  66. } extended_data;
  67. };
  68. struct usbc_sc8180x_notify {
  69. struct pmic_glink_hdr hdr;
  70. __le32 notification;
  71. __le32 reserved[2];
  72. };
  73. enum pmic_glink_altmode_pin_assignment {
  74. DPAM_HPD_OUT,
  75. DPAM_HPD_A,
  76. DPAM_HPD_B,
  77. DPAM_HPD_C,
  78. DPAM_HPD_D,
  79. DPAM_HPD_E,
  80. DPAM_HPD_F,
  81. };
  82. struct pmic_glink_altmode;
  83. #define work_to_altmode_port(w) container_of((w), struct pmic_glink_altmode_port, work)
  84. struct pmic_glink_altmode_port {
  85. struct pmic_glink_altmode *altmode;
  86. unsigned int index;
  87. struct typec_switch *typec_switch;
  88. struct typec_mux *typec_mux;
  89. struct typec_mux_state state;
  90. struct typec_retimer *typec_retimer;
  91. struct typec_retimer_state retimer_state;
  92. struct typec_altmode dp_alt;
  93. struct typec_altmode tbt_alt;
  94. struct work_struct work;
  95. struct auxiliary_device *bridge;
  96. enum typec_orientation orientation;
  97. u16 svid;
  98. struct usbc_sc8280x_tbt_data tbt_data;
  99. u8 dp_data;
  100. u8 mode;
  101. u8 hpd_state;
  102. u8 hpd_irq;
  103. u8 mux_ctrl;
  104. };
  105. #define work_to_altmode(w) container_of((w), struct pmic_glink_altmode, enable_work)
  106. struct pmic_glink_altmode {
  107. struct device *dev;
  108. unsigned int owner_id;
  109. /* To synchronize WRITE_REQ acks */
  110. struct mutex lock;
  111. struct completion pan_ack;
  112. struct pmic_glink_client *client;
  113. struct work_struct enable_work;
  114. struct pmic_glink_altmode_port ports[PMIC_GLINK_MAX_PORTS];
  115. };
  116. static int pmic_glink_altmode_request(struct pmic_glink_altmode *altmode, u32 cmd, u32 arg)
  117. {
  118. struct usbc_write_req req = {};
  119. unsigned long left;
  120. int ret;
  121. /*
  122. * The USBC_CMD_WRITE_REQ ack doesn't identify the request, so wait for
  123. * one ack at a time.
  124. */
  125. guard(mutex)(&altmode->lock);
  126. req.hdr.owner = cpu_to_le32(altmode->owner_id);
  127. req.hdr.type = cpu_to_le32(PMIC_GLINK_REQ_RESP);
  128. req.hdr.opcode = cpu_to_le32(USBC_CMD_WRITE_REQ);
  129. req.cmd = cpu_to_le32(cmd);
  130. req.arg = cpu_to_le32(arg);
  131. ret = pmic_glink_send(altmode->client, &req, sizeof(req));
  132. if (ret) {
  133. dev_err(altmode->dev, "failed to send altmode request: %#x (%d)\n", cmd, ret);
  134. return ret;
  135. }
  136. left = wait_for_completion_timeout(&altmode->pan_ack, 5 * HZ);
  137. if (!left) {
  138. dev_err(altmode->dev, "timeout waiting for altmode request ack for: %#x\n", cmd);
  139. return -ETIMEDOUT;
  140. }
  141. return 0;
  142. }
  143. static void pmic_glink_altmode_enable_dp(struct pmic_glink_altmode *altmode,
  144. struct pmic_glink_altmode_port *port,
  145. u8 mode, bool hpd_state,
  146. bool hpd_irq)
  147. {
  148. struct typec_displayport_data dp_data = {};
  149. int ret;
  150. dp_data.status = DP_STATUS_ENABLED;
  151. if (hpd_state)
  152. dp_data.status |= DP_STATUS_HPD_STATE;
  153. if (hpd_irq)
  154. dp_data.status |= DP_STATUS_IRQ_HPD;
  155. dp_data.conf = DP_CONF_SET_PIN_ASSIGN(mode);
  156. port->state.alt = &port->dp_alt;
  157. port->state.data = &dp_data;
  158. port->state.mode = TYPEC_MODAL_STATE(mode);
  159. ret = typec_mux_set(port->typec_mux, &port->state);
  160. if (ret)
  161. dev_err(altmode->dev, "failed to switch mux to DP: %d\n", ret);
  162. port->retimer_state.alt = &port->dp_alt;
  163. port->retimer_state.data = &dp_data;
  164. port->retimer_state.mode = TYPEC_MODAL_STATE(mode);
  165. ret = typec_retimer_set(port->typec_retimer, &port->retimer_state);
  166. if (ret)
  167. dev_err(altmode->dev, "failed to setup retimer to DP: %d\n", ret);
  168. }
  169. static void pmic_glink_altmode_enable_tbt(struct pmic_glink_altmode *altmode,
  170. struct pmic_glink_altmode_port *port)
  171. {
  172. struct usbc_sc8280x_tbt_data *tbt = &port->tbt_data;
  173. struct typec_thunderbolt_data tbt_data = {};
  174. u32 cable_speed;
  175. int ret;
  176. /* Device Discover Mode VDO */
  177. tbt_data.device_mode = TBT_MODE;
  178. tbt_data.device_mode |= TBT_SET_ADAPTER(TBT_ADAPTER_TBT3);
  179. /* Cable Discover Mode VDO */
  180. tbt_data.cable_mode = TBT_MODE;
  181. if (tbt->usb_speed == 0) {
  182. cable_speed = TBT_CABLE_USB3_PASSIVE;
  183. } else if (tbt->usb_speed == 1) {
  184. cable_speed = TBT_CABLE_10_AND_20GBPS;
  185. } else {
  186. dev_err(altmode->dev,
  187. "Got illegal TBT3 cable speed value (%u), falling back to passive\n",
  188. tbt->usb_speed);
  189. cable_speed = TBT_CABLE_USB3_PASSIVE;
  190. }
  191. tbt_data.cable_mode |= TBT_SET_CABLE_SPEED(cable_speed);
  192. if (tbt->cable_type) {
  193. tbt_data.cable_mode |= TBT_CABLE_ACTIVE_PASSIVE;
  194. tbt_data.cable_mode |= TBT_SET_CABLE_ROUNDED(tbt->rounded_cable);
  195. }
  196. /* Enter Mode VDO */
  197. tbt_data.enter_vdo |= TBT_MODE;
  198. tbt_data.enter_vdo |= TBT_ENTER_MODE_CABLE_SPEED(cable_speed);
  199. if (tbt->cable_type) {
  200. tbt_data.enter_vdo |= TBT_CABLE_ACTIVE_PASSIVE;
  201. tbt_data.enter_vdo |= TBT_SET_CABLE_ROUNDED(tbt->rounded_cable);
  202. }
  203. port->state.alt = &port->tbt_alt;
  204. port->state.data = &tbt_data;
  205. port->state.mode = TYPEC_MODAL_STATE(port->mode);
  206. ret = typec_mux_set(port->typec_mux, &port->state);
  207. if (ret)
  208. dev_err(altmode->dev, "failed to switch mux to USB: %d\n", ret);
  209. port->retimer_state.alt = &port->tbt_alt;
  210. port->retimer_state.data = &tbt_data;
  211. port->retimer_state.mode = TYPEC_MODAL_STATE(port->mode);
  212. ret = typec_retimer_set(port->typec_retimer, &port->retimer_state);
  213. if (ret)
  214. dev_err(altmode->dev, "failed to setup retimer to USB: %d\n", ret);
  215. }
  216. static void pmic_glink_altmode_enable_usb4(struct pmic_glink_altmode *altmode,
  217. struct pmic_glink_altmode_port *port)
  218. {
  219. struct usbc_sc8280x_tbt_data *tbt = &port->tbt_data;
  220. struct enter_usb_data data = {};
  221. int ret;
  222. data.eudo = FIELD_PREP(EUDO_USB_MODE_MASK, EUDO_USB_MODE_USB4);
  223. if (tbt->usb_speed == 0) {
  224. data.eudo |= FIELD_PREP(EUDO_CABLE_SPEED_MASK, EUDO_CABLE_SPEED_USB4_GEN2);
  225. } else if (tbt->usb_speed == 1) {
  226. data.eudo |= FIELD_PREP(EUDO_CABLE_SPEED_MASK, EUDO_CABLE_SPEED_USB4_GEN3);
  227. } else {
  228. pr_err("Got illegal USB4 cable speed value (%u), falling back to G2\n",
  229. tbt->usb_speed);
  230. data.eudo |= FIELD_PREP(EUDO_CABLE_SPEED_MASK, EUDO_CABLE_SPEED_USB4_GEN2);
  231. }
  232. data.eudo |= FIELD_PREP(EUDO_CABLE_TYPE_MASK, tbt->cable_type);
  233. port->state.alt = NULL;
  234. port->state.data = &data;
  235. port->state.mode = TYPEC_MODE_USB4;
  236. ret = typec_mux_set(port->typec_mux, &port->state);
  237. if (ret)
  238. dev_err(altmode->dev, "failed to switch mux to USB: %d\n", ret);
  239. port->retimer_state.alt = NULL;
  240. port->retimer_state.data = &data;
  241. port->retimer_state.mode = TYPEC_MODE_USB4;
  242. ret = typec_retimer_set(port->typec_retimer, &port->retimer_state);
  243. if (ret)
  244. dev_err(altmode->dev, "failed to setup retimer to USB: %d\n", ret);
  245. }
  246. static void pmic_glink_altmode_enable_usb(struct pmic_glink_altmode *altmode,
  247. struct pmic_glink_altmode_port *port)
  248. {
  249. int ret;
  250. port->state.alt = NULL;
  251. port->state.data = NULL;
  252. port->state.mode = TYPEC_STATE_USB;
  253. ret = typec_mux_set(port->typec_mux, &port->state);
  254. if (ret)
  255. dev_err(altmode->dev, "failed to switch mux to USB: %d\n", ret);
  256. port->retimer_state.alt = NULL;
  257. port->retimer_state.data = NULL;
  258. port->retimer_state.mode = TYPEC_STATE_USB;
  259. ret = typec_retimer_set(port->typec_retimer, &port->retimer_state);
  260. if (ret)
  261. dev_err(altmode->dev, "failed to setup retimer to USB: %d\n", ret);
  262. }
  263. static void pmic_glink_altmode_safe(struct pmic_glink_altmode *altmode,
  264. struct pmic_glink_altmode_port *port)
  265. {
  266. int ret;
  267. port->state.alt = NULL;
  268. port->state.data = NULL;
  269. port->state.mode = TYPEC_STATE_SAFE;
  270. ret = typec_mux_set(port->typec_mux, &port->state);
  271. if (ret)
  272. dev_err(altmode->dev, "failed to switch mux to safe mode: %d\n", ret);
  273. port->retimer_state.alt = NULL;
  274. port->retimer_state.data = NULL;
  275. port->retimer_state.mode = TYPEC_STATE_SAFE;
  276. ret = typec_retimer_set(port->typec_retimer, &port->retimer_state);
  277. if (ret)
  278. dev_err(altmode->dev, "failed to setup retimer to USB: %d\n", ret);
  279. }
  280. static void pmic_glink_altmode_worker(struct work_struct *work)
  281. {
  282. struct pmic_glink_altmode_port *alt_port = work_to_altmode_port(work);
  283. struct pmic_glink_altmode *altmode = alt_port->altmode;
  284. enum drm_connector_status conn_status;
  285. typec_switch_set(alt_port->typec_switch, alt_port->orientation);
  286. /*
  287. * MUX_CTRL_STATE_DP4LN/USB3_DP may only be set if SVID=DP, but we need
  288. * to special-case the SVID=DP && mux_ctrl=NO_CONN case to deliver a
  289. * HPD notification
  290. */
  291. if (alt_port->svid == USB_TYPEC_DP_SID) {
  292. if (alt_port->mux_ctrl == MUX_CTRL_STATE_NO_CONN) {
  293. pmic_glink_altmode_safe(altmode, alt_port);
  294. } else {
  295. pmic_glink_altmode_enable_dp(altmode, alt_port,
  296. alt_port->mode,
  297. alt_port->hpd_state,
  298. alt_port->hpd_irq);
  299. }
  300. if (alt_port->hpd_state)
  301. conn_status = connector_status_connected;
  302. else
  303. conn_status = connector_status_disconnected;
  304. drm_aux_hpd_bridge_notify(&alt_port->bridge->dev, conn_status);
  305. } else if (alt_port->mux_ctrl == MUX_CTRL_STATE_TUNNELING) {
  306. if (alt_port->svid == USB_TYPEC_TBT_SID)
  307. pmic_glink_altmode_enable_tbt(altmode, alt_port);
  308. else
  309. pmic_glink_altmode_enable_usb4(altmode, alt_port);
  310. } else if (alt_port->mux_ctrl == MUX_CTRL_STATE_USB3_ONLY) {
  311. pmic_glink_altmode_enable_usb(altmode, alt_port);
  312. } else if (alt_port->mux_ctrl == MUX_CTRL_STATE_NO_CONN) {
  313. pmic_glink_altmode_safe(altmode, alt_port);
  314. } else {
  315. dev_err(altmode->dev, "Got unknown mux_ctrl: %u on port %u, forcing safe mode\n",
  316. alt_port->mux_ctrl, alt_port->index);
  317. pmic_glink_altmode_safe(altmode, alt_port);
  318. }
  319. pmic_glink_altmode_request(altmode, ALTMODE_PAN_ACK, alt_port->index);
  320. }
  321. static enum typec_orientation pmic_glink_altmode_orientation(unsigned int orientation)
  322. {
  323. if (orientation == 0)
  324. return TYPEC_ORIENTATION_NORMAL;
  325. else if (orientation == 1)
  326. return TYPEC_ORIENTATION_REVERSE;
  327. else
  328. return TYPEC_ORIENTATION_NONE;
  329. }
  330. #define SC8180X_PORT_MASK 0x000000ff
  331. #define SC8180X_ORIENTATION_MASK 0x0000ff00
  332. #define SC8180X_MUX_MASK 0x00ff0000
  333. #define SC8180X_MODE_MASK 0x3f000000
  334. #define SC8180X_HPD_STATE_MASK 0x40000000
  335. #define SC8180X_HPD_IRQ_MASK 0x80000000
  336. static void pmic_glink_altmode_sc8180xp_notify(struct pmic_glink_altmode *altmode,
  337. const void *data, size_t len)
  338. {
  339. struct pmic_glink_altmode_port *alt_port;
  340. const struct usbc_sc8180x_notify *msg;
  341. u32 notification;
  342. u8 orientation;
  343. u8 hpd_state;
  344. u8 hpd_irq;
  345. u16 svid;
  346. u8 port;
  347. u8 mode;
  348. u8 mux;
  349. if (len != sizeof(*msg)) {
  350. dev_warn(altmode->dev, "invalid length of USBC_NOTIFY indication: %zd\n", len);
  351. return;
  352. }
  353. msg = data;
  354. notification = le32_to_cpu(msg->notification);
  355. port = FIELD_GET(SC8180X_PORT_MASK, notification);
  356. orientation = FIELD_GET(SC8180X_ORIENTATION_MASK, notification);
  357. mux = FIELD_GET(SC8180X_MUX_MASK, notification);
  358. mode = FIELD_GET(SC8180X_MODE_MASK, notification);
  359. hpd_state = FIELD_GET(SC8180X_HPD_STATE_MASK, notification);
  360. hpd_irq = FIELD_GET(SC8180X_HPD_IRQ_MASK, notification);
  361. svid = mux == 2 ? USB_TYPEC_DP_SID : 0;
  362. if (port >= ARRAY_SIZE(altmode->ports) || !altmode->ports[port].altmode) {
  363. dev_dbg(altmode->dev, "notification on undefined port %d\n", port);
  364. return;
  365. }
  366. alt_port = &altmode->ports[port];
  367. alt_port->orientation = pmic_glink_altmode_orientation(orientation);
  368. alt_port->svid = svid;
  369. alt_port->mode = mode;
  370. alt_port->hpd_state = hpd_state;
  371. alt_port->hpd_irq = hpd_irq;
  372. schedule_work(&alt_port->work);
  373. }
  374. #define SC8280XP_DPAM_MASK 0x3f
  375. #define SC8280XP_HPD_STATE_MASK BIT(6)
  376. #define SC8280XP_HPD_IRQ_MASK BIT(7)
  377. static void pmic_glink_altmode_sc8280xp_notify(struct pmic_glink_altmode *altmode,
  378. u16 svid, const void *data, size_t len)
  379. {
  380. struct pmic_glink_altmode_port *alt_port;
  381. const struct usbc_sc8280x_tbt_data *tbt;
  382. const struct usbc_sc8280x_dp_data *dp;
  383. const struct usbc_notify *notify;
  384. u8 orientation;
  385. u8 port;
  386. if (len != sizeof(*notify)) {
  387. dev_warn(altmode->dev, "invalid length USBC_NOTIFY_IND: %zd\n",
  388. len);
  389. return;
  390. }
  391. notify = data;
  392. port = notify->port_idx;
  393. orientation = notify->orientation;
  394. if (port >= ARRAY_SIZE(altmode->ports) || !altmode->ports[port].altmode) {
  395. dev_dbg(altmode->dev, "notification on undefined port %d\n", port);
  396. return;
  397. }
  398. alt_port = &altmode->ports[port];
  399. alt_port->orientation = pmic_glink_altmode_orientation(orientation);
  400. alt_port->svid = svid;
  401. alt_port->mux_ctrl = notify->mux_ctrl;
  402. if (svid == USB_TYPEC_DP_SID) {
  403. dp = &notify->extended_data.dp;
  404. alt_port->mode = dp->pin_assignment - DPAM_HPD_A;
  405. alt_port->hpd_state = dp->hpd_state;
  406. alt_port->hpd_irq = dp->hpd_irq;
  407. } else if (alt_port->mux_ctrl == MUX_CTRL_STATE_TUNNELING) {
  408. /* Valid for both USB4 and TBT3 */
  409. tbt = &notify->extended_data.tbt;
  410. alt_port->tbt_data = *tbt;
  411. }
  412. schedule_work(&alt_port->work);
  413. }
  414. static void pmic_glink_altmode_callback(const void *data, size_t len, void *priv)
  415. {
  416. struct pmic_glink_altmode *altmode = priv;
  417. const struct pmic_glink_hdr *hdr = data;
  418. u16 opcode;
  419. u16 svid;
  420. opcode = le32_to_cpu(hdr->opcode) & 0xff;
  421. svid = le32_to_cpu(hdr->opcode) >> 16;
  422. switch (opcode) {
  423. case USBC_CMD_WRITE_REQ:
  424. complete(&altmode->pan_ack);
  425. break;
  426. case USBC_NOTIFY_IND:
  427. pmic_glink_altmode_sc8280xp_notify(altmode, svid, data, len);
  428. break;
  429. case USBC_SC8180X_NOTIFY_IND:
  430. pmic_glink_altmode_sc8180xp_notify(altmode, data, len);
  431. break;
  432. }
  433. }
  434. static void pmic_glink_altmode_put_retimer(void *data)
  435. {
  436. typec_retimer_put(data);
  437. }
  438. static void pmic_glink_altmode_put_mux(void *data)
  439. {
  440. typec_mux_put(data);
  441. }
  442. static void pmic_glink_altmode_put_switch(void *data)
  443. {
  444. typec_switch_put(data);
  445. }
  446. static void pmic_glink_altmode_enable_worker(struct work_struct *work)
  447. {
  448. struct pmic_glink_altmode *altmode = work_to_altmode(work);
  449. int ret;
  450. ret = pmic_glink_altmode_request(altmode, ALTMODE_PAN_EN, 0);
  451. if (ret)
  452. dev_err(altmode->dev, "failed to request altmode notifications: %d\n", ret);
  453. }
  454. static void pmic_glink_altmode_pdr_notify(void *priv, int state)
  455. {
  456. struct pmic_glink_altmode *altmode = priv;
  457. if (state == SERVREG_SERVICE_STATE_UP)
  458. schedule_work(&altmode->enable_work);
  459. }
  460. static const struct of_device_id pmic_glink_altmode_of_quirks[] = {
  461. { .compatible = "qcom,sc8180x-pmic-glink", .data = (void *)PMIC_GLINK_OWNER_USBC },
  462. {}
  463. };
  464. static int pmic_glink_altmode_probe(struct auxiliary_device *adev,
  465. const struct auxiliary_device_id *id)
  466. {
  467. struct pmic_glink_altmode_port *alt_port;
  468. struct pmic_glink_altmode *altmode;
  469. const struct of_device_id *match;
  470. struct fwnode_handle *fwnode;
  471. struct device *dev = &adev->dev;
  472. u32 port;
  473. int ret;
  474. altmode = devm_kzalloc(dev, sizeof(*altmode), GFP_KERNEL);
  475. if (!altmode)
  476. return -ENOMEM;
  477. altmode->dev = dev;
  478. match = of_match_device(pmic_glink_altmode_of_quirks, dev->parent);
  479. if (match)
  480. altmode->owner_id = (unsigned long)match->data;
  481. else
  482. altmode->owner_id = PMIC_GLINK_OWNER_USBC_PAN;
  483. INIT_WORK(&altmode->enable_work, pmic_glink_altmode_enable_worker);
  484. init_completion(&altmode->pan_ack);
  485. mutex_init(&altmode->lock);
  486. device_for_each_child_node(dev, fwnode) {
  487. ret = fwnode_property_read_u32(fwnode, "reg", &port);
  488. if (ret < 0) {
  489. dev_err(dev, "missing reg property of %pOFn\n", fwnode);
  490. fwnode_handle_put(fwnode);
  491. return ret;
  492. }
  493. if (port >= ARRAY_SIZE(altmode->ports)) {
  494. dev_warn(dev, "invalid connector number, ignoring\n");
  495. continue;
  496. }
  497. if (altmode->ports[port].altmode) {
  498. dev_err(dev, "multiple connector definition for port %u\n", port);
  499. fwnode_handle_put(fwnode);
  500. return -EINVAL;
  501. }
  502. alt_port = &altmode->ports[port];
  503. alt_port->altmode = altmode;
  504. alt_port->index = port;
  505. INIT_WORK(&alt_port->work, pmic_glink_altmode_worker);
  506. alt_port->bridge = devm_drm_dp_hpd_bridge_alloc(dev, to_of_node(fwnode));
  507. if (IS_ERR(alt_port->bridge)) {
  508. fwnode_handle_put(fwnode);
  509. return PTR_ERR(alt_port->bridge);
  510. }
  511. alt_port->dp_alt.svid = USB_TYPEC_DP_SID;
  512. alt_port->dp_alt.mode = USB_TYPEC_DP_MODE;
  513. alt_port->dp_alt.active = 1;
  514. alt_port->tbt_alt.svid = USB_TYPEC_TBT_SID;
  515. alt_port->tbt_alt.mode = TYPEC_TBT_MODE;
  516. alt_port->tbt_alt.active = 1;
  517. alt_port->typec_mux = fwnode_typec_mux_get(fwnode);
  518. if (IS_ERR(alt_port->typec_mux)) {
  519. fwnode_handle_put(fwnode);
  520. return dev_err_probe(dev, PTR_ERR(alt_port->typec_mux),
  521. "failed to acquire mode-switch for port: %d\n",
  522. port);
  523. }
  524. ret = devm_add_action_or_reset(dev, pmic_glink_altmode_put_mux,
  525. alt_port->typec_mux);
  526. if (ret) {
  527. fwnode_handle_put(fwnode);
  528. return ret;
  529. }
  530. alt_port->typec_retimer = fwnode_typec_retimer_get(fwnode);
  531. if (IS_ERR(alt_port->typec_retimer)) {
  532. fwnode_handle_put(fwnode);
  533. return dev_err_probe(dev, PTR_ERR(alt_port->typec_retimer),
  534. "failed to acquire retimer-switch for port: %d\n",
  535. port);
  536. }
  537. ret = devm_add_action_or_reset(dev, pmic_glink_altmode_put_retimer,
  538. alt_port->typec_retimer);
  539. if (ret) {
  540. fwnode_handle_put(fwnode);
  541. return ret;
  542. }
  543. alt_port->typec_switch = fwnode_typec_switch_get(fwnode);
  544. if (IS_ERR(alt_port->typec_switch)) {
  545. fwnode_handle_put(fwnode);
  546. return dev_err_probe(dev, PTR_ERR(alt_port->typec_switch),
  547. "failed to acquire orientation-switch for port: %d\n",
  548. port);
  549. }
  550. ret = devm_add_action_or_reset(dev, pmic_glink_altmode_put_switch,
  551. alt_port->typec_switch);
  552. if (ret) {
  553. fwnode_handle_put(fwnode);
  554. return ret;
  555. }
  556. }
  557. for (port = 0; port < ARRAY_SIZE(altmode->ports); port++) {
  558. alt_port = &altmode->ports[port];
  559. if (!alt_port->bridge)
  560. continue;
  561. ret = devm_drm_dp_hpd_bridge_add(dev, alt_port->bridge);
  562. if (ret)
  563. return ret;
  564. }
  565. altmode->client = devm_pmic_glink_client_alloc(dev,
  566. altmode->owner_id,
  567. pmic_glink_altmode_callback,
  568. pmic_glink_altmode_pdr_notify,
  569. altmode);
  570. if (IS_ERR(altmode->client))
  571. return PTR_ERR(altmode->client);
  572. pmic_glink_client_register(altmode->client);
  573. return 0;
  574. }
  575. static const struct auxiliary_device_id pmic_glink_altmode_id_table[] = {
  576. { .name = "pmic_glink.altmode", },
  577. {},
  578. };
  579. MODULE_DEVICE_TABLE(auxiliary, pmic_glink_altmode_id_table);
  580. static struct auxiliary_driver pmic_glink_altmode_driver = {
  581. .name = "pmic_glink_altmode",
  582. .probe = pmic_glink_altmode_probe,
  583. .id_table = pmic_glink_altmode_id_table,
  584. };
  585. module_auxiliary_driver(pmic_glink_altmode_driver);
  586. MODULE_DESCRIPTION("Qualcomm PMIC GLINK Altmode driver");
  587. MODULE_LICENSE("GPL");