xref: /linux/drivers/platform/surface/surface_aggregator_registry.c (revision 69050f8d6d075dc01af7a5f2f550a8067510366f)
1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3  * Surface System Aggregator Module (SSAM) client device registry.
4  *
5  * Registry for non-platform/non-ACPI SSAM client devices, i.e. devices that
6  * cannot be auto-detected. Provides device-hubs and performs instantiation
7  * for these devices.
8  *
9  * Copyright (C) 2020-2022 Maximilian Luz <luzmaximilian@gmail.com>
10  */
11 
12 #include <linux/acpi.h>
13 #include <linux/kernel.h>
14 #include <linux/module.h>
15 #include <linux/of.h>
16 #include <linux/platform_device.h>
17 #include <linux/property.h>
18 #include <linux/types.h>
19 
20 #include <linux/surface_aggregator/device.h>
21 
22 
23 /* -- Device registry. ------------------------------------------------------ */
24 
25 /*
26  * SSAM device names follow the SSAM module alias, meaning they are prefixed
27  * with 'ssam:', followed by domain, category, target ID, instance ID, and
28  * function, each encoded as two-digit hexadecimal, separated by ':'. In other
29  * words, it follows the scheme
30  *
31  *      ssam:dd:cc:tt:ii:ff
32  *
33  * Where, 'dd', 'cc', 'tt', 'ii', and 'ff' are the two-digit hexadecimal
34  * values mentioned above, respectively.
35  */
36 
37 /* Root node. */
38 static const struct software_node ssam_node_root = {
39 	.name = "ssam_platform_hub",
40 };
41 
42 /* KIP device hub (connects keyboard cover devices on Surface Pro 8). */
43 static const struct software_node ssam_node_hub_kip = {
44 	.name = "ssam:00:00:01:0e:00",
45 	.parent = &ssam_node_root,
46 };
47 
48 /* Base device hub (devices attached to Surface Book 3 base). */
49 static const struct software_node ssam_node_hub_base = {
50 	.name = "ssam:00:00:01:11:00",
51 	.parent = &ssam_node_root,
52 };
53 
54 /* AC adapter. */
55 static const struct software_node ssam_node_bat_ac = {
56 	.name = "ssam:01:02:01:01:01",
57 	.parent = &ssam_node_root,
58 };
59 
60 /* Primary battery. */
61 static const struct software_node ssam_node_bat_main = {
62 	.name = "ssam:01:02:01:01:00",
63 	.parent = &ssam_node_root,
64 };
65 
66 /* Secondary battery (Surface Book 3). */
67 static const struct software_node ssam_node_bat_sb3base = {
68 	.name = "ssam:01:02:02:01:00",
69 	.parent = &ssam_node_hub_base,
70 };
71 
72 /* Platform profile / performance-mode device without a fan. */
73 static const struct software_node ssam_node_tmp_perf_profile = {
74 	.name = "ssam:01:03:01:00:01",
75 	.parent = &ssam_node_root,
76 };
77 
78 /* Platform profile / performance-mode device with a fan, such that
79  * the fan controller profile can also be switched.
80  */
81 static const struct property_entry ssam_node_tmp_perf_profile_has_fan[] = {
82 	PROPERTY_ENTRY_BOOL("has_fan"),
83 	{ }
84 };
85 
86 static const struct software_node ssam_node_tmp_perf_profile_with_fan = {
87 	.name = "ssam:01:03:01:00:01",
88 	.parent = &ssam_node_root,
89 	.properties = ssam_node_tmp_perf_profile_has_fan,
90 };
91 
92 /* Thermal sensors. */
93 static const struct software_node ssam_node_tmp_sensors = {
94 	.name = "ssam:01:03:01:00:02",
95 	.parent = &ssam_node_root,
96 };
97 
98 /* Fan speed function. */
99 static const struct software_node ssam_node_fan_speed = {
100 	.name = "ssam:01:05:01:01:01",
101 	.parent = &ssam_node_root,
102 };
103 
104 /* Tablet-mode switch via KIP subsystem. */
105 static const struct software_node ssam_node_kip_tablet_switch = {
106 	.name = "ssam:01:0e:01:00:01",
107 	.parent = &ssam_node_root,
108 };
109 
110 /* DTX / detachment-system device (Surface Book 3). */
111 static const struct software_node ssam_node_bas_dtx = {
112 	.name = "ssam:01:11:01:00:00",
113 	.parent = &ssam_node_root,
114 };
115 
116 /* HID keyboard (SAM, TID=1). */
117 static const struct software_node ssam_node_hid_sam_keyboard = {
118 	.name = "ssam:01:15:01:01:00",
119 	.parent = &ssam_node_root,
120 };
121 
122 /* HID pen stash (SAM, TID=1; pen taken / stashed away evens). */
123 static const struct software_node ssam_node_hid_sam_penstash = {
124 	.name = "ssam:01:15:01:02:00",
125 	.parent = &ssam_node_root,
126 };
127 
128 /* HID touchpad (SAM, TID=1). */
129 static const struct software_node ssam_node_hid_sam_touchpad = {
130 	.name = "ssam:01:15:01:03:00",
131 	.parent = &ssam_node_root,
132 };
133 
134 /* HID device instance 6 (SAM, TID=1, HID sensor collection). */
135 static const struct software_node ssam_node_hid_sam_sensors = {
136 	.name = "ssam:01:15:01:06:00",
137 	.parent = &ssam_node_root,
138 };
139 
140 /* HID device instance 7 (SAM, TID=1, UCM UCSI HID client). */
141 static const struct software_node ssam_node_hid_sam_ucm_ucsi = {
142 	.name = "ssam:01:15:01:07:00",
143 	.parent = &ssam_node_root,
144 };
145 
146 /* HID system controls (SAM, TID=1). */
147 static const struct software_node ssam_node_hid_sam_sysctrl = {
148 	.name = "ssam:01:15:01:08:00",
149 	.parent = &ssam_node_root,
150 };
151 
152 /* HID keyboard. */
153 static const struct software_node ssam_node_hid_main_keyboard = {
154 	.name = "ssam:01:15:02:01:00",
155 	.parent = &ssam_node_root,
156 };
157 
158 /* HID touchpad. */
159 static const struct software_node ssam_node_hid_main_touchpad = {
160 	.name = "ssam:01:15:02:03:00",
161 	.parent = &ssam_node_root,
162 };
163 
164 /* HID device instance 5 (unknown HID device). */
165 static const struct software_node ssam_node_hid_main_iid5 = {
166 	.name = "ssam:01:15:02:05:00",
167 	.parent = &ssam_node_root,
168 };
169 
170 /* HID keyboard (base hub). */
171 static const struct software_node ssam_node_hid_base_keyboard = {
172 	.name = "ssam:01:15:02:01:00",
173 	.parent = &ssam_node_hub_base,
174 };
175 
176 /* HID touchpad (base hub). */
177 static const struct software_node ssam_node_hid_base_touchpad = {
178 	.name = "ssam:01:15:02:03:00",
179 	.parent = &ssam_node_hub_base,
180 };
181 
182 /* HID device instance 5 (unknown HID device, base hub). */
183 static const struct software_node ssam_node_hid_base_iid5 = {
184 	.name = "ssam:01:15:02:05:00",
185 	.parent = &ssam_node_hub_base,
186 };
187 
188 /* HID device instance 6 (unknown HID device, base hub). */
189 static const struct software_node ssam_node_hid_base_iid6 = {
190 	.name = "ssam:01:15:02:06:00",
191 	.parent = &ssam_node_hub_base,
192 };
193 
194 /* HID keyboard (KIP hub). */
195 static const struct software_node ssam_node_hid_kip_keyboard = {
196 	.name = "ssam:01:15:02:01:00",
197 	.parent = &ssam_node_hub_kip,
198 };
199 
200 /* HID pen stash (KIP hub; pen taken / stashed away evens). */
201 static const struct software_node ssam_node_hid_kip_penstash = {
202 	.name = "ssam:01:15:02:02:00",
203 	.parent = &ssam_node_hub_kip,
204 };
205 
206 /* HID touchpad (KIP hub). */
207 static const struct software_node ssam_node_hid_kip_touchpad = {
208 	.name = "ssam:01:15:02:03:00",
209 	.parent = &ssam_node_hub_kip,
210 };
211 
212 /* HID device instance 5 (KIP hub, type-cover firmware update). */
213 static const struct software_node ssam_node_hid_kip_fwupd = {
214 	.name = "ssam:01:15:02:05:00",
215 	.parent = &ssam_node_hub_kip,
216 };
217 
218 /* Tablet-mode switch via POS subsystem. */
219 static const struct software_node ssam_node_pos_tablet_switch = {
220 	.name = "ssam:01:26:01:00:01",
221 	.parent = &ssam_node_root,
222 };
223 
224 /*
225  * Devices for 5th- and 6th-generations models:
226  * - Surface Book 2,
227  * - Surface Laptop 1 and 2,
228  * - Surface Pro 5 and 6.
229  */
230 static const struct software_node *ssam_node_group_gen5[] = {
231 	&ssam_node_root,
232 	&ssam_node_tmp_perf_profile,
233 	NULL,
234 };
235 
236 /* Devices for Surface Book 3. */
237 static const struct software_node *ssam_node_group_sb3[] = {
238 	&ssam_node_root,
239 	&ssam_node_hub_base,
240 	&ssam_node_bat_ac,
241 	&ssam_node_bat_main,
242 	&ssam_node_bat_sb3base,
243 	&ssam_node_tmp_perf_profile,
244 	&ssam_node_bas_dtx,
245 	&ssam_node_hid_base_keyboard,
246 	&ssam_node_hid_base_touchpad,
247 	&ssam_node_hid_base_iid5,
248 	&ssam_node_hid_base_iid6,
249 	NULL,
250 };
251 
252 /* Devices for Surface Laptop 3 and 4. */
253 static const struct software_node *ssam_node_group_sl3[] = {
254 	&ssam_node_root,
255 	&ssam_node_bat_ac,
256 	&ssam_node_bat_main,
257 	&ssam_node_tmp_perf_profile,
258 	&ssam_node_hid_main_keyboard,
259 	&ssam_node_hid_main_touchpad,
260 	&ssam_node_hid_main_iid5,
261 	NULL,
262 };
263 
264 /* Devices for Surface Laptop 5. */
265 static const struct software_node *ssam_node_group_sl5[] = {
266 	&ssam_node_root,
267 	&ssam_node_bat_ac,
268 	&ssam_node_bat_main,
269 	&ssam_node_tmp_perf_profile_with_fan,
270 	&ssam_node_tmp_sensors,
271 	&ssam_node_fan_speed,
272 	&ssam_node_hid_main_keyboard,
273 	&ssam_node_hid_main_touchpad,
274 	&ssam_node_hid_main_iid5,
275 	&ssam_node_hid_sam_ucm_ucsi,
276 	NULL,
277 };
278 
279 /* Devices for Surface Laptop 6. */
280 static const struct software_node *ssam_node_group_sl6[] = {
281 	&ssam_node_root,
282 	&ssam_node_bat_ac,
283 	&ssam_node_bat_main,
284 	&ssam_node_tmp_perf_profile_with_fan,
285 	&ssam_node_tmp_sensors,
286 	&ssam_node_fan_speed,
287 	&ssam_node_hid_main_keyboard,
288 	&ssam_node_hid_main_touchpad,
289 	&ssam_node_hid_main_iid5,
290 	&ssam_node_hid_sam_sensors,
291 	&ssam_node_hid_sam_ucm_ucsi,
292 	NULL,
293 };
294 
295 /* Devices for Surface Laptop 7. */
296 static const struct software_node *ssam_node_group_sl7[] = {
297 	&ssam_node_root,
298 	&ssam_node_bat_ac,
299 	&ssam_node_bat_main,
300 	&ssam_node_tmp_perf_profile_with_fan,
301 	&ssam_node_fan_speed,
302 	&ssam_node_hid_sam_keyboard,
303 	/* TODO: evaluate thermal sensors devices when we get a driver for that */
304 	NULL,
305 };
306 
307 /* Devices for Surface Laptop Studio 1. */
308 static const struct software_node *ssam_node_group_sls1[] = {
309 	&ssam_node_root,
310 	&ssam_node_bat_ac,
311 	&ssam_node_bat_main,
312 	&ssam_node_tmp_perf_profile,
313 	&ssam_node_pos_tablet_switch,
314 	&ssam_node_hid_sam_keyboard,
315 	&ssam_node_hid_sam_penstash,
316 	&ssam_node_hid_sam_touchpad,
317 	&ssam_node_hid_sam_sensors,
318 	&ssam_node_hid_sam_ucm_ucsi,
319 	&ssam_node_hid_sam_sysctrl,
320 	NULL,
321 };
322 
323 /* Devices for Surface Laptop Studio 2. */
324 static const struct software_node *ssam_node_group_sls2[] = {
325 	&ssam_node_root,
326 	&ssam_node_bat_ac,
327 	&ssam_node_bat_main,
328 	&ssam_node_tmp_perf_profile_with_fan,
329 	&ssam_node_tmp_sensors,
330 	&ssam_node_fan_speed,
331 	&ssam_node_pos_tablet_switch,
332 	&ssam_node_hid_sam_keyboard,
333 	&ssam_node_hid_sam_penstash,
334 	&ssam_node_hid_sam_sensors,
335 	&ssam_node_hid_sam_ucm_ucsi,
336 	NULL,
337 };
338 
339 /* Devices for Surface Laptop Go. */
340 static const struct software_node *ssam_node_group_slg1[] = {
341 	&ssam_node_root,
342 	&ssam_node_bat_ac,
343 	&ssam_node_bat_main,
344 	&ssam_node_tmp_perf_profile,
345 	NULL,
346 };
347 
348 /* Devices for Surface Pro 7 and Surface Pro 7+. */
349 static const struct software_node *ssam_node_group_sp7[] = {
350 	&ssam_node_root,
351 	&ssam_node_bat_ac,
352 	&ssam_node_bat_main,
353 	&ssam_node_tmp_perf_profile,
354 	NULL,
355 };
356 
357 /* Devices for Surface Pro 8 */
358 static const struct software_node *ssam_node_group_sp8[] = {
359 	&ssam_node_root,
360 	&ssam_node_hub_kip,
361 	&ssam_node_bat_ac,
362 	&ssam_node_bat_main,
363 	&ssam_node_tmp_perf_profile,
364 	&ssam_node_kip_tablet_switch,
365 	&ssam_node_hid_kip_keyboard,
366 	&ssam_node_hid_kip_penstash,
367 	&ssam_node_hid_kip_touchpad,
368 	&ssam_node_hid_kip_fwupd,
369 	&ssam_node_hid_sam_sensors,
370 	&ssam_node_hid_sam_ucm_ucsi,
371 	NULL,
372 };
373 
374 /* Devices for Surface Pro 9, 10 and 11 (Intel/x86) */
375 static const struct software_node *ssam_node_group_sp9[] = {
376 	&ssam_node_root,
377 	&ssam_node_hub_kip,
378 	&ssam_node_bat_ac,
379 	&ssam_node_bat_main,
380 	&ssam_node_tmp_perf_profile_with_fan,
381 	&ssam_node_tmp_sensors,
382 	&ssam_node_fan_speed,
383 	&ssam_node_pos_tablet_switch,
384 	&ssam_node_hid_kip_keyboard,
385 	&ssam_node_hid_kip_penstash,
386 	&ssam_node_hid_kip_touchpad,
387 	&ssam_node_hid_kip_fwupd,
388 	&ssam_node_hid_sam_sensors,
389 	&ssam_node_hid_sam_ucm_ucsi,
390 	NULL,
391 };
392 
393 /* Devices for Surface Pro 9 5G (ARM/QCOM) */
394 static const struct software_node *ssam_node_group_sp9_5g[] = {
395 	&ssam_node_root,
396 	&ssam_node_hub_kip,
397 	&ssam_node_bat_ac,
398 	&ssam_node_bat_main,
399 	&ssam_node_tmp_sensors,
400 	&ssam_node_hid_kip_keyboard,
401 	&ssam_node_hid_kip_penstash,
402 	&ssam_node_hid_kip_touchpad,
403 	&ssam_node_hid_kip_fwupd,
404 	&ssam_node_hid_sam_sensors,
405 	&ssam_node_kip_tablet_switch,
406 	NULL,
407 };
408 
409 /* Devices for Surface Pro 11 (ARM/QCOM) */
410 static const struct software_node *ssam_node_group_sp11[] = {
411 	&ssam_node_root,
412 	&ssam_node_hub_kip,
413 	&ssam_node_bat_ac,
414 	&ssam_node_bat_main,
415 	&ssam_node_tmp_sensors,
416 	&ssam_node_hid_kip_keyboard,
417 	&ssam_node_hid_kip_penstash,
418 	&ssam_node_hid_kip_touchpad,
419 	&ssam_node_hid_kip_fwupd,
420 	&ssam_node_hid_sam_sensors,
421 	&ssam_node_kip_tablet_switch,
422 	NULL,
423 };
424 
425 /* -- SSAM platform/meta-hub driver. ---------------------------------------- */
426 
427 static const struct acpi_device_id ssam_platform_hub_acpi_match[] = {
428 	/* Surface Pro 4, 5, and 6 (OMBR < 0x10) */
429 	{ "MSHW0081", (unsigned long)ssam_node_group_gen5 },
430 
431 	/* Surface Pro 6 (OMBR >= 0x10) */
432 	{ "MSHW0111", (unsigned long)ssam_node_group_gen5 },
433 
434 	/* Surface Pro 7 */
435 	{ "MSHW0116", (unsigned long)ssam_node_group_sp7 },
436 
437 	/* Surface Pro 7+ */
438 	{ "MSHW0119", (unsigned long)ssam_node_group_sp7 },
439 
440 	/* Surface Pro 8 */
441 	{ "MSHW0263", (unsigned long)ssam_node_group_sp8 },
442 
443 	/* Surface Pro 9 */
444 	{ "MSHW0343", (unsigned long)ssam_node_group_sp9 },
445 
446 	/* Surface Pro 10 */
447 	{ "MSHW0510", (unsigned long)ssam_node_group_sp9 },
448 
449 	/* Surface Pro 11 */
450 	{ "MSHW0583", (unsigned long)ssam_node_group_sp9 },
451 
452 	/* Surface Book 2 */
453 	{ "MSHW0107", (unsigned long)ssam_node_group_gen5 },
454 
455 	/* Surface Book 3 */
456 	{ "MSHW0117", (unsigned long)ssam_node_group_sb3 },
457 
458 	/* Surface Laptop 1 */
459 	{ "MSHW0086", (unsigned long)ssam_node_group_gen5 },
460 
461 	/* Surface Laptop 2 */
462 	{ "MSHW0112", (unsigned long)ssam_node_group_gen5 },
463 
464 	/* Surface Laptop 3 (13", Intel) */
465 	{ "MSHW0114", (unsigned long)ssam_node_group_sl3 },
466 
467 	/* Surface Laptop 3 (15", AMD) and 4 (15", AMD) */
468 	{ "MSHW0110", (unsigned long)ssam_node_group_sl3 },
469 
470 	/* Surface Laptop 4 (13", Intel) */
471 	{ "MSHW0250", (unsigned long)ssam_node_group_sl3 },
472 
473 	/* Surface Laptop 5 */
474 	{ "MSHW0350", (unsigned long)ssam_node_group_sl5 },
475 
476 	/* Surface Laptop 6 */
477 	{ "MSHW0530", (unsigned long)ssam_node_group_sl6 },
478 
479 	/* Surface Laptop Go 1 */
480 	{ "MSHW0118", (unsigned long)ssam_node_group_slg1 },
481 
482 	/* Surface Laptop Go 2 */
483 	{ "MSHW0290", (unsigned long)ssam_node_group_slg1 },
484 
485 	/* Surface Laptop Go 3 */
486 	{ "MSHW0440", (unsigned long)ssam_node_group_slg1 },
487 
488 	/* Surface Laptop Studio 1 */
489 	{ "MSHW0123", (unsigned long)ssam_node_group_sls1 },
490 
491 	/* Surface Laptop Studio 2 */
492 	{ "MSHW0360", (unsigned long)ssam_node_group_sls2 },
493 
494 	{ },
495 };
496 MODULE_DEVICE_TABLE(acpi, ssam_platform_hub_acpi_match);
497 
498 static const struct of_device_id ssam_platform_hub_of_match[] __maybe_unused = {
499 	/* Surface Pro 9 5G (ARM/QCOM) */
500 	{ .compatible = "microsoft,arcata", (void *)ssam_node_group_sp9_5g },
501 	/* Surface Pro 11 (ARM/QCOM) */
502 	{ .compatible = "microsoft,denali", (void *)ssam_node_group_sp11 },
503 	/* Surface Laptop 7 */
504 	{ .compatible = "microsoft,romulus13", (void *)ssam_node_group_sl7 },
505 	{ .compatible = "microsoft,romulus15", (void *)ssam_node_group_sl7 },
506 	{ },
507 };
508 
509 static int ssam_platform_hub_probe(struct platform_device *pdev)
510 {
511 	const struct software_node **nodes;
512 	struct ssam_controller *ctrl;
513 	struct fwnode_handle *root;
514 	int status;
515 
516 	nodes = (const struct software_node **)acpi_device_get_match_data(&pdev->dev);
517 	if (!nodes) {
518 		nodes = (const struct software_node **)of_machine_get_match_data(ssam_platform_hub_of_match);
519 		if (!nodes)
520 			return -ENODEV;
521 	}
522 
523 	/*
524 	 * As we're adding the SSAM client devices as children under this device
525 	 * and not the SSAM controller, we need to add a device link to the
526 	 * controller to ensure that we remove all of our devices before the
527 	 * controller is removed. This also guarantees proper ordering for
528 	 * suspend/resume of the devices on this hub.
529 	 */
530 	ctrl = ssam_client_bind(&pdev->dev);
531 	if (IS_ERR(ctrl))
532 		return PTR_ERR(ctrl) == -ENODEV ? -EPROBE_DEFER : PTR_ERR(ctrl);
533 
534 	status = software_node_register_node_group(nodes);
535 	if (status)
536 		return status;
537 
538 	root = software_node_fwnode(&ssam_node_root);
539 	if (!root) {
540 		software_node_unregister_node_group(nodes);
541 		return -ENOENT;
542 	}
543 
544 	set_secondary_fwnode(&pdev->dev, root);
545 
546 	status = __ssam_register_clients(&pdev->dev, ctrl, root);
547 	if (status) {
548 		set_secondary_fwnode(&pdev->dev, NULL);
549 		software_node_unregister_node_group(nodes);
550 	}
551 
552 	platform_set_drvdata(pdev, nodes);
553 	return status;
554 }
555 
556 static void ssam_platform_hub_remove(struct platform_device *pdev)
557 {
558 	const struct software_node **nodes = platform_get_drvdata(pdev);
559 
560 	ssam_remove_clients(&pdev->dev);
561 	set_secondary_fwnode(&pdev->dev, NULL);
562 	software_node_unregister_node_group(nodes);
563 }
564 
565 static struct platform_driver ssam_platform_hub_driver = {
566 	.probe = ssam_platform_hub_probe,
567 	.remove = ssam_platform_hub_remove,
568 	.driver = {
569 		.name = "surface_aggregator_platform_hub",
570 		.acpi_match_table = ssam_platform_hub_acpi_match,
571 		.probe_type = PROBE_PREFER_ASYNCHRONOUS,
572 	},
573 };
574 module_platform_driver(ssam_platform_hub_driver);
575 
576 MODULE_ALIAS("platform:surface_aggregator_platform_hub");
577 MODULE_AUTHOR("Maximilian Luz <luzmaximilian@gmail.com>");
578 MODULE_DESCRIPTION("Device-registry for Surface System Aggregator Module");
579 MODULE_LICENSE("GPL");
580