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