Linux 6.2-rc1
[linux-block.git] / drivers / usb / typec / altmodes / displayport.c
CommitLineData
0e3bb7d6 1// SPDX-License-Identifier: GPL-2.0
50c9bd05 2/*
0e3bb7d6
HK
3 * USB Typec-C DisplayPort Alternate Mode driver
4 *
5 * Copyright (C) 2018 Intel Corporation
6 * Author: Heikki Krogerus <heikki.krogerus@linux.intel.com>
7 *
8 * DisplayPort is trademark of VESA (www.vesa.org)
9 */
10
11#include <linux/delay.h>
12#include <linux/mutex.h>
13#include <linux/module.h>
7f811394 14#include <linux/property.h>
0e3bb7d6
HK
15#include <linux/usb/pd_vdo.h>
16#include <linux/usb/typec_dp.h>
7f811394 17#include <drm/drm_connector.h>
318ebed2 18#include "displayport.h"
0e3bb7d6 19
4c93cad8
KT
20#define DP_HEADER(_dp, ver, cmd) (VDO((_dp)->alt->svid, 1, ver, cmd) \
21 | VDO_OPOS(USB_TYPEC_DP_MODE))
0e3bb7d6
HK
22
23enum {
24 DP_CONF_USB,
25 DP_CONF_DFP_D,
26 DP_CONF_UFP_D,
27 DP_CONF_DUAL_D,
28};
29
0e3bb7d6
HK
30/* Pin assignments that use USB3.1 Gen2 signaling to carry DP protocol */
31#define DP_PIN_ASSIGN_GEN2_BR_MASK (BIT(DP_PIN_ASSIGN_A) | \
32 BIT(DP_PIN_ASSIGN_B))
33
34/* Pin assignments that use DP v1.3 signaling to carry DP protocol */
35#define DP_PIN_ASSIGN_DP_BR_MASK (BIT(DP_PIN_ASSIGN_C) | \
36 BIT(DP_PIN_ASSIGN_D) | \
37 BIT(DP_PIN_ASSIGN_E) | \
38 BIT(DP_PIN_ASSIGN_F))
39
40/* DP only pin assignments */
41#define DP_PIN_ASSIGN_DP_ONLY_MASK (BIT(DP_PIN_ASSIGN_A) | \
42 BIT(DP_PIN_ASSIGN_C) | \
43 BIT(DP_PIN_ASSIGN_E))
44
45/* Pin assignments where one channel is for USB */
46#define DP_PIN_ASSIGN_MULTI_FUNC_MASK (BIT(DP_PIN_ASSIGN_B) | \
47 BIT(DP_PIN_ASSIGN_D) | \
48 BIT(DP_PIN_ASSIGN_F))
49
50enum dp_state {
51 DP_STATE_IDLE,
52 DP_STATE_ENTER,
53 DP_STATE_UPDATE,
54 DP_STATE_CONFIGURE,
55 DP_STATE_EXIT,
56};
57
58struct dp_altmode {
59 struct typec_displayport_data data;
60
61 enum dp_state state;
7f811394 62 bool hpd;
0e3bb7d6
HK
63
64 struct mutex lock; /* device lock */
65 struct work_struct work;
66 struct typec_altmode *alt;
67 const struct typec_altmode *port;
7f811394 68 struct fwnode_handle *connector_fwnode;
0e3bb7d6
HK
69};
70
71static int dp_altmode_notify(struct dp_altmode *dp)
72{
fc27e046
HG
73 unsigned long conf;
74 u8 state;
75
76 if (dp->data.conf) {
77 state = get_count_order(DP_CONF_GET_PIN_ASSIGN(dp->data.conf));
78 conf = TYPEC_MODAL_STATE(state);
79 } else {
80 conf = TYPEC_STATE_USB;
81 }
0e3bb7d6 82
fc27e046 83 return typec_altmode_notify(dp->alt, conf, &dp->data);
0e3bb7d6
HK
84}
85
86static int dp_altmode_configure(struct dp_altmode *dp, u8 con)
87{
88 u32 conf = DP_CONF_SIGNALING_DP; /* Only DP signaling supported */
89 u8 pin_assign = 0;
90
91 switch (con) {
92 case DP_STATUS_CON_DISABLED:
93 return 0;
94 case DP_STATUS_CON_DFP_D:
95 conf |= DP_CONF_UFP_U_AS_DFP_D;
96 pin_assign = DP_CAP_UFP_D_PIN_ASSIGN(dp->alt->vdo) &
97 DP_CAP_DFP_D_PIN_ASSIGN(dp->port->vdo);
98 break;
99 case DP_STATUS_CON_UFP_D:
100 case DP_STATUS_CON_BOTH: /* NOTE: First acting as DP source */
101 conf |= DP_CONF_UFP_U_AS_UFP_D;
c1e5c2f0
PS
102 pin_assign = DP_CAP_PIN_ASSIGN_UFP_D(dp->alt->vdo) &
103 DP_CAP_PIN_ASSIGN_DFP_D(dp->port->vdo);
0e3bb7d6
HK
104 break;
105 default:
106 break;
107 }
108
109 /* Determining the initial pin assignment. */
110 if (!DP_CONF_GET_PIN_ASSIGN(dp->data.conf)) {
111 /* Is USB together with DP preferred */
112 if (dp->data.status & DP_STATUS_PREFER_MULTI_FUNC &&
113 pin_assign & DP_PIN_ASSIGN_MULTI_FUNC_MASK)
114 pin_assign &= DP_PIN_ASSIGN_MULTI_FUNC_MASK;
09fed4d6 115 else if (pin_assign & DP_PIN_ASSIGN_DP_ONLY_MASK)
0e3bb7d6
HK
116 pin_assign &= DP_PIN_ASSIGN_DP_ONLY_MASK;
117
118 if (!pin_assign)
119 return -EINVAL;
120
121 conf |= DP_CONF_SET_PIN_ASSIGN(pin_assign);
122 }
123
124 dp->data.conf = conf;
125
126 return 0;
127}
128
129static int dp_altmode_status_update(struct dp_altmode *dp)
130{
131 bool configured = !!DP_CONF_GET_PIN_ASSIGN(dp->data.conf);
7f811394 132 bool hpd = !!(dp->data.status & DP_STATUS_HPD_STATE);
0e3bb7d6
HK
133 u8 con = DP_STATUS_CONNECTION(dp->data.status);
134 int ret = 0;
135
136 if (configured && (dp->data.status & DP_STATUS_SWITCH_TO_USB)) {
137 dp->data.conf = 0;
138 dp->state = DP_STATE_CONFIGURE;
139 } else if (dp->data.status & DP_STATUS_EXIT_DP_MODE) {
140 dp->state = DP_STATE_EXIT;
141 } else if (!(con & DP_CONF_CURRENTLY(dp->data.conf))) {
142 ret = dp_altmode_configure(dp, con);
143 if (!ret)
144 dp->state = DP_STATE_CONFIGURE;
7f811394
HG
145 } else {
146 if (dp->hpd != hpd) {
147 drm_connector_oob_hotplug_event(dp->connector_fwnode);
148 dp->hpd = hpd;
149 }
0e3bb7d6
HK
150 }
151
152 return ret;
153}
154
155static int dp_altmode_configured(struct dp_altmode *dp)
156{
0e3bb7d6 157 sysfs_notify(&dp->alt->dev.kobj, "displayport", "configuration");
0e3bb7d6
HK
158 sysfs_notify(&dp->alt->dev.kobj, "displayport", "pin_assignment");
159
fc27e046 160 return dp_altmode_notify(dp);
0e3bb7d6
HK
161}
162
163static int dp_altmode_configure_vdm(struct dp_altmode *dp, u32 conf)
164{
4c93cad8
KT
165 int svdm_version = typec_altmode_get_svdm_version(dp->alt);
166 u32 header;
0e3bb7d6
HK
167 int ret;
168
4c93cad8
KT
169 if (svdm_version < 0)
170 return svdm_version;
171
172 header = DP_HEADER(dp, svdm_version, DP_CMD_CONFIGURE);
0e3bb7d6
HK
173 ret = typec_altmode_notify(dp->alt, TYPEC_STATE_SAFE, &dp->data);
174 if (ret) {
175 dev_err(&dp->alt->dev,
176 "unable to put to connector to safe mode\n");
177 return ret;
178 }
179
180 ret = typec_altmode_vdm(dp->alt, header, &conf, 2);
fc27e046
HG
181 if (ret)
182 dp_altmode_notify(dp);
0e3bb7d6
HK
183
184 return ret;
185}
186
187static void dp_altmode_work(struct work_struct *work)
188{
189 struct dp_altmode *dp = container_of(work, struct dp_altmode, work);
4c93cad8 190 int svdm_version;
0e3bb7d6
HK
191 u32 header;
192 u32 vdo;
193 int ret;
194
195 mutex_lock(&dp->lock);
196
197 switch (dp->state) {
198 case DP_STATE_ENTER:
8face9aa 199 ret = typec_altmode_enter(dp->alt, NULL);
5789051f 200 if (ret && ret != -EBUSY)
0e3bb7d6
HK
201 dev_err(&dp->alt->dev, "failed to enter mode\n");
202 break;
203 case DP_STATE_UPDATE:
4c93cad8
KT
204 svdm_version = typec_altmode_get_svdm_version(dp->alt);
205 if (svdm_version < 0)
206 break;
207 header = DP_HEADER(dp, svdm_version, DP_CMD_STATUS_UPDATE);
0e3bb7d6
HK
208 vdo = 1;
209 ret = typec_altmode_vdm(dp->alt, header, &vdo, 2);
210 if (ret)
211 dev_err(&dp->alt->dev,
212 "unable to send Status Update command (%d)\n",
213 ret);
214 break;
215 case DP_STATE_CONFIGURE:
216 ret = dp_altmode_configure_vdm(dp, dp->data.conf);
217 if (ret)
218 dev_err(&dp->alt->dev,
219 "unable to send Configure command (%d)\n", ret);
220 break;
221 case DP_STATE_EXIT:
222 if (typec_altmode_exit(dp->alt))
223 dev_err(&dp->alt->dev, "Exit Mode Failed!\n");
224 break;
225 default:
226 break;
227 }
228
229 dp->state = DP_STATE_IDLE;
230
231 mutex_unlock(&dp->lock);
232}
233
234static void dp_altmode_attention(struct typec_altmode *alt, const u32 vdo)
235{
236 struct dp_altmode *dp = typec_altmode_get_drvdata(alt);
237 u8 old_state;
238
239 mutex_lock(&dp->lock);
240
241 old_state = dp->state;
242 dp->data.status = vdo;
243
244 if (old_state != DP_STATE_IDLE)
245 dev_warn(&alt->dev, "ATTENTION while processing state %d\n",
246 old_state);
247
248 if (dp_altmode_status_update(dp))
249 dev_warn(&alt->dev, "%s: status update failed\n", __func__);
250
251 if (dp_altmode_notify(dp))
252 dev_err(&alt->dev, "%s: notification failed\n", __func__);
253
254 if (old_state == DP_STATE_IDLE && dp->state != DP_STATE_IDLE)
255 schedule_work(&dp->work);
256
257 mutex_unlock(&dp->lock);
258}
259
260static int dp_altmode_vdm(struct typec_altmode *alt,
261 const u32 hdr, const u32 *vdo, int count)
262{
263 struct dp_altmode *dp = typec_altmode_get_drvdata(alt);
264 int cmd_type = PD_VDO_CMDT(hdr);
265 int cmd = PD_VDO_CMD(hdr);
266 int ret = 0;
267
268 mutex_lock(&dp->lock);
269
270 if (dp->state != DP_STATE_IDLE) {
271 ret = -EBUSY;
272 goto err_unlock;
273 }
274
275 switch (cmd_type) {
276 case CMDT_RSP_ACK:
277 switch (cmd) {
278 case CMD_ENTER_MODE:
279 dp->state = DP_STATE_UPDATE;
280 break;
281 case CMD_EXIT_MODE:
282 dp->data.status = 0;
283 dp->data.conf = 0;
284 break;
285 case DP_CMD_STATUS_UPDATE:
286 dp->data.status = *vdo;
287 ret = dp_altmode_status_update(dp);
288 break;
289 case DP_CMD_CONFIGURE:
290 ret = dp_altmode_configured(dp);
291 break;
292 default:
293 break;
294 }
295 break;
296 case CMDT_RSP_NAK:
297 switch (cmd) {
298 case DP_CMD_CONFIGURE:
299 dp->data.conf = 0;
300 ret = dp_altmode_configured(dp);
301 break;
302 default:
303 break;
304 }
305 break;
306 default:
307 break;
308 }
309
310 if (dp->state != DP_STATE_IDLE)
311 schedule_work(&dp->work);
312
313err_unlock:
314 mutex_unlock(&dp->lock);
315 return ret;
316}
317
318static int dp_altmode_activate(struct typec_altmode *alt, int activate)
319{
8face9aa
HK
320 return activate ? typec_altmode_enter(alt, NULL) :
321 typec_altmode_exit(alt);
0e3bb7d6
HK
322}
323
324static const struct typec_altmode_ops dp_altmode_ops = {
325 .attention = dp_altmode_attention,
326 .vdm = dp_altmode_vdm,
327 .activate = dp_altmode_activate,
328};
329
330static const char * const configurations[] = {
331 [DP_CONF_USB] = "USB",
332 [DP_CONF_DFP_D] = "source",
333 [DP_CONF_UFP_D] = "sink",
334};
335
336static ssize_t
337configuration_store(struct device *dev, struct device_attribute *attr,
338 const char *buf, size_t size)
339{
340 struct dp_altmode *dp = dev_get_drvdata(dev);
341 u32 conf;
342 u32 cap;
343 int con;
98a1a0c7 344 int ret = 0;
0e3bb7d6
HK
345
346 con = sysfs_match_string(configurations, buf);
347 if (con < 0)
348 return con;
349
350 mutex_lock(&dp->lock);
351
352 if (dp->state != DP_STATE_IDLE) {
353 ret = -EBUSY;
354 goto err_unlock;
355 }
356
357 cap = DP_CAP_CAPABILITY(dp->alt->vdo);
358
359 if ((con == DP_CONF_DFP_D && !(cap & DP_CAP_DFP_D)) ||
98a1a0c7
CIK
360 (con == DP_CONF_UFP_D && !(cap & DP_CAP_UFP_D))) {
361 ret = -EINVAL;
362 goto err_unlock;
363 }
0e3bb7d6
HK
364
365 conf = dp->data.conf & ~DP_CONF_DUAL_D;
366 conf |= con;
367
368 if (dp->alt->active) {
369 ret = dp_altmode_configure_vdm(dp, conf);
370 if (ret)
371 goto err_unlock;
372 }
373
374 dp->data.conf = conf;
375
376err_unlock:
377 mutex_unlock(&dp->lock);
378
379 return ret ? ret : size;
380}
381
382static ssize_t configuration_show(struct device *dev,
383 struct device_attribute *attr, char *buf)
384{
385 struct dp_altmode *dp = dev_get_drvdata(dev);
386 int len;
387 u8 cap;
388 u8 cur;
389 int i;
390
391 mutex_lock(&dp->lock);
392
393 cap = DP_CAP_CAPABILITY(dp->alt->vdo);
394 cur = DP_CONF_CURRENTLY(dp->data.conf);
395
396 len = sprintf(buf, "%s ", cur ? "USB" : "[USB]");
397
398 for (i = 1; i < ARRAY_SIZE(configurations); i++) {
399 if (i == cur)
400 len += sprintf(buf + len, "[%s] ", configurations[i]);
401 else if ((i == DP_CONF_DFP_D && cap & DP_CAP_DFP_D) ||
402 (i == DP_CONF_UFP_D && cap & DP_CAP_UFP_D))
403 len += sprintf(buf + len, "%s ", configurations[i]);
404 }
405
406 mutex_unlock(&dp->lock);
407
408 buf[len - 1] = '\n';
409 return len;
410}
411static DEVICE_ATTR_RW(configuration);
412
413static const char * const pin_assignments[] = {
414 [DP_PIN_ASSIGN_A] = "A",
415 [DP_PIN_ASSIGN_B] = "B",
416 [DP_PIN_ASSIGN_C] = "C",
417 [DP_PIN_ASSIGN_D] = "D",
418 [DP_PIN_ASSIGN_E] = "E",
419 [DP_PIN_ASSIGN_F] = "F",
420};
421
422static ssize_t
423pin_assignment_store(struct device *dev, struct device_attribute *attr,
424 const char *buf, size_t size)
425{
426 struct dp_altmode *dp = dev_get_drvdata(dev);
427 u8 assignments;
428 u32 conf;
429 int ret;
430
431 ret = sysfs_match_string(pin_assignments, buf);
432 if (ret < 0)
433 return ret;
434
435 conf = DP_CONF_SET_PIN_ASSIGN(BIT(ret));
436 ret = 0;
437
438 mutex_lock(&dp->lock);
439
440 if (conf & dp->data.conf)
441 goto out_unlock;
442
443 if (dp->state != DP_STATE_IDLE) {
444 ret = -EBUSY;
445 goto out_unlock;
446 }
447
448 if (DP_CONF_CURRENTLY(dp->data.conf) == DP_CONF_DFP_D)
449 assignments = DP_CAP_UFP_D_PIN_ASSIGN(dp->alt->vdo);
450 else
451 assignments = DP_CAP_DFP_D_PIN_ASSIGN(dp->alt->vdo);
452
453 if (!(DP_CONF_GET_PIN_ASSIGN(conf) & assignments)) {
454 ret = -EINVAL;
455 goto out_unlock;
456 }
457
458 conf |= dp->data.conf & ~DP_CONF_PIN_ASSIGNEMENT_MASK;
459
460 /* Only send Configure command if a configuration has been set */
461 if (dp->alt->active && DP_CONF_CURRENTLY(dp->data.conf)) {
462 ret = dp_altmode_configure_vdm(dp, conf);
463 if (ret)
464 goto out_unlock;
465 }
466
467 dp->data.conf = conf;
468
469out_unlock:
470 mutex_unlock(&dp->lock);
471
472 return ret ? ret : size;
473}
474
475static ssize_t pin_assignment_show(struct device *dev,
476 struct device_attribute *attr, char *buf)
477{
478 struct dp_altmode *dp = dev_get_drvdata(dev);
479 u8 assignments;
480 int len = 0;
481 u8 cur;
482 int i;
483
484 mutex_lock(&dp->lock);
485
486 cur = get_count_order(DP_CONF_GET_PIN_ASSIGN(dp->data.conf));
487
488 if (DP_CONF_CURRENTLY(dp->data.conf) == DP_CONF_DFP_D)
489 assignments = DP_CAP_UFP_D_PIN_ASSIGN(dp->alt->vdo);
490 else
491 assignments = DP_CAP_DFP_D_PIN_ASSIGN(dp->alt->vdo);
492
493 for (i = 0; assignments; assignments >>= 1, i++) {
494 if (assignments & 1) {
495 if (i == cur)
496 len += sprintf(buf + len, "[%s] ",
497 pin_assignments[i]);
498 else
499 len += sprintf(buf + len, "%s ",
500 pin_assignments[i]);
501 }
502 }
503
504 mutex_unlock(&dp->lock);
505
506 buf[len - 1] = '\n';
507 return len;
508}
509static DEVICE_ATTR_RW(pin_assignment);
510
511static struct attribute *dp_altmode_attrs[] = {
512 &dev_attr_configuration.attr,
513 &dev_attr_pin_assignment.attr,
514 NULL
515};
516
517static const struct attribute_group dp_altmode_group = {
518 .name = "displayport",
519 .attrs = dp_altmode_attrs,
520};
521
d266e968 522int dp_altmode_probe(struct typec_altmode *alt)
0e3bb7d6
HK
523{
524 const struct typec_altmode *port = typec_altmode_get_partner(alt);
7f811394 525 struct fwnode_handle *fwnode;
0e3bb7d6
HK
526 struct dp_altmode *dp;
527 int ret;
528
529 /* FIXME: Port can only be DFP_U. */
530
531 /* Make sure we have compatiple pin configurations */
532 if (!(DP_CAP_DFP_D_PIN_ASSIGN(port->vdo) &
533 DP_CAP_UFP_D_PIN_ASSIGN(alt->vdo)) &&
534 !(DP_CAP_UFP_D_PIN_ASSIGN(port->vdo) &
535 DP_CAP_DFP_D_PIN_ASSIGN(alt->vdo)))
536 return -ENODEV;
537
538 ret = sysfs_create_group(&alt->dev.kobj, &dp_altmode_group);
539 if (ret)
540 return ret;
541
542 dp = devm_kzalloc(&alt->dev, sizeof(*dp), GFP_KERNEL);
543 if (!dp)
544 return -ENOMEM;
545
546 INIT_WORK(&dp->work, dp_altmode_work);
547 mutex_init(&dp->lock);
548 dp->port = port;
549 dp->alt = alt;
550
551 alt->desc = "DisplayPort";
552 alt->ops = &dp_altmode_ops;
553
7f811394
HG
554 fwnode = dev_fwnode(alt->dev.parent->parent); /* typec_port fwnode */
555 dp->connector_fwnode = fwnode_find_reference(fwnode, "displayport", 0);
556 if (IS_ERR(dp->connector_fwnode))
557 dp->connector_fwnode = NULL;
558
0e3bb7d6
HK
559 typec_altmode_set_drvdata(alt, dp);
560
561 dp->state = DP_STATE_ENTER;
562 schedule_work(&dp->work);
563
564 return 0;
565}
d266e968 566EXPORT_SYMBOL_GPL(dp_altmode_probe);
0e3bb7d6 567
d266e968 568void dp_altmode_remove(struct typec_altmode *alt)
0e3bb7d6
HK
569{
570 struct dp_altmode *dp = typec_altmode_get_drvdata(alt);
571
572 sysfs_remove_group(&alt->dev.kobj, &dp_altmode_group);
573 cancel_work_sync(&dp->work);
7f811394
HG
574
575 if (dp->connector_fwnode) {
576 if (dp->hpd)
577 drm_connector_oob_hotplug_event(dp->connector_fwnode);
578
579 fwnode_handle_put(dp->connector_fwnode);
580 }
0e3bb7d6 581}
d266e968 582EXPORT_SYMBOL_GPL(dp_altmode_remove);
0e3bb7d6
HK
583
584static const struct typec_device_id dp_typec_id[] = {
585 { USB_TYPEC_DP_SID, USB_TYPEC_DP_MODE },
586 { },
587};
588MODULE_DEVICE_TABLE(typec, dp_typec_id);
589
590static struct typec_altmode_driver dp_altmode_driver = {
591 .id_table = dp_typec_id,
592 .probe = dp_altmode_probe,
593 .remove = dp_altmode_remove,
594 .driver = {
595 .name = "typec_displayport",
596 .owner = THIS_MODULE,
597 },
598};
599module_typec_altmode_driver(dp_altmode_driver);
600
601MODULE_AUTHOR("Heikki Krogerus <heikki.krogerus@linux.intel.com>");
602MODULE_LICENSE("GPL v2");
603MODULE_DESCRIPTION("DisplayPort Alternate Mode");