forked from roadrunner2/macbook12-spi-driver
-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathapple-touchbar.c
1500 lines (1226 loc) · 40 KB
/
apple-touchbar.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
// SPDX-License-Identifier: GPL-2.0
/*
* Apple Touch Bar Driver
*
* Copyright (c) 2017-2018 Ronald Tschalär
*/
/*
* Recent MacBookPro models (MacBookPro 13,[23] and later) have a touch bar,
* which is exposed via several USB interfaces. MacOS supports a fancy mode
* where arbitrary buttons can be defined; this driver currently only
* supports the simple mode that consists of 3 predefined layouts
* (escape-only, esc + special keys, and esc + function keys).
*
* The first USB HID interface supports two reports, an input report that
* is used to report the key presses, and an output report which can be
* used to set the touch bar "mode": touch bar off (in which case no touches
* are reported at all), escape key only, escape + 12 function keys, and
* escape + several special keys (including brightness, audio volume,
* etc). The second interface supports several, complex reports, most of
* which are unknown at this time, but one of which has been determined to
* allow for controlling of the touch bar's brightness: off (though touches
* are still reported), dimmed, and full brightness. This driver makes
* use of these two reports.
*/
#define dev_fmt(fmt) "tb: " fmt
#include <linux/device.h>
#include <linux/hid.h>
#include <linux/input.h>
#include <linux/jiffies.h>
#include <linux/ktime.h>
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/spinlock.h>
#include <linux/string.h>
#include <linux/sysfs.h>
#include <linux/usb/ch9.h>
#include <linux/usb.h>
#include <linux/workqueue.h>
#include "hid-ids.h"
#include "apple-ibridge.h"
#define HID_UP_APPLE 0xff120000
#define HID_USAGE_MODE (HID_UP_CUSTOM | 0x0004)
#define HID_USAGE_APPLE_APP (HID_UP_APPLE | 0x0001)
#define HID_USAGE_DISP (HID_UP_APPLE | 0x0021)
#define HID_USAGE_DISP_AUX1 (HID_UP_APPLE | 0x0020)
#define APPLETB_MAX_TB_KEYS 13 /* ESC, F1-F12 */
#define APPLETB_CMD_MODE_ESC 0
#define APPLETB_CMD_MODE_FN 1
#define APPLETB_CMD_MODE_SPCL 2
#define APPLETB_CMD_MODE_OFF 3
#define APPLETB_CMD_MODE_UPD 254
#define APPLETB_CMD_MODE_NONE 255
#define APPLETB_CMD_DISP_ON 1
#define APPLETB_CMD_DISP_DIM 2
#define APPLETB_CMD_DISP_OFF 4
#define APPLETB_CMD_DISP_UPD 254
#define APPLETB_CMD_DISP_NONE 255
#define APPLETB_FN_MODE_FKEYS 0
#define APPLETB_FN_MODE_NORM 1
#define APPLETB_FN_MODE_INV 2
#define APPLETB_FN_MODE_SPCL 3
#define APPLETB_FN_MODE_ESC 4
#define APPLETB_FN_MODE_MAX APPLETB_FN_MODE_ESC
#define APPLETB_DEVID_KEYBOARD 1
#define APPLETB_DEVID_TOUCHPAD 2
#define APPLETB_MAX_DIM_TIME 30
#define APPLETB_FEATURE_IS_T1 BIT(0)
static int appletb_tb_def_idle_timeout = 5 * 60;
module_param_named(idle_timeout, appletb_tb_def_idle_timeout, int, 0444);
MODULE_PARM_DESC(idle_timeout, "Default touch bar idle timeout:\n"
" [>0] - turn touch bar display off after no keyboard, trackpad, or touch bar input has been received for this many seconds;\n"
" the display will be turned back on as soon as new input is received\n"
" 0 - turn touch bar display off (input does not turn it on again)\n"
" -1 - turn touch bar display on (does not turn off automatically)\n"
" -2 - disable touch bar completely");
static int appletb_tb_def_dim_timeout = -2;
module_param_named(dim_timeout, appletb_tb_def_dim_timeout, int, 0444);
MODULE_PARM_DESC(dim_timeout, "Default touch bar dim timeout:\n"
" >0 - dim touch bar display after no keyboard, trackpad, or touch bar input has been received for this many seconds\n"
" the display will be returned to full brightness as soon as new input is received\n"
" 0 - dim touch bar display (input does not return it to full brightness)\n"
" -1 - disable timeout (touch bar never dimmed)\n"
" [-2] - calculate timeout based on idle-timeout");
static int appletb_tb_def_fn_mode = APPLETB_FN_MODE_NORM;
module_param_named(fnmode, appletb_tb_def_fn_mode, int, 0444);
MODULE_PARM_DESC(fnmode, "Default Fn key mode:\n"
" 0 - function-keys only\n"
" [1] - fn key switches from special to function-keys\n"
" 2 - inverse of 1\n"
" 3 - special keys only\n"
" 4 - escape key only");
static ssize_t idle_timeout_show(struct device *dev,
struct device_attribute *attr, char *buf);
static ssize_t idle_timeout_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t size);
static DEVICE_ATTR_RW(idle_timeout);
static ssize_t dim_timeout_show(struct device *dev,
struct device_attribute *attr, char *buf);
static ssize_t dim_timeout_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t size);
static DEVICE_ATTR_RW(dim_timeout);
static ssize_t fnmode_show(struct device *dev, struct device_attribute *attr,
char *buf);
static ssize_t fnmode_store(struct device *dev, struct device_attribute *attr,
const char *buf, size_t size);
static DEVICE_ATTR_RW(fnmode);
static struct attribute *appletb_attrs[] = {
&dev_attr_idle_timeout.attr,
&dev_attr_dim_timeout.attr,
&dev_attr_fnmode.attr,
NULL,
};
static const struct attribute_group appletb_attr_group = {
.attrs = appletb_attrs,
};
struct appletb_device {
bool active;
struct device *log_dev;
struct hid_field *mode_field;
struct hid_field *disp_field;
struct hid_field *disp_field_aux1;
struct appletb_iface_info {
struct hid_device *hdev;
struct usb_interface *usb_iface;
bool suspended;
} mode_iface, disp_iface;
struct input_handler inp_handler;
struct input_handle kbd_handle;
struct input_handle tpd_handle;
bool last_tb_keys_pressed[APPLETB_MAX_TB_KEYS];
bool last_tb_keys_translated[APPLETB_MAX_TB_KEYS];
bool last_fn_pressed;
ktime_t last_event_time;
unsigned char cur_tb_mode;
unsigned char pnd_tb_mode;
unsigned char cur_tb_disp;
unsigned char pnd_tb_disp;
bool tb_autopm_off;
bool restore_autopm;
struct delayed_work tb_work;
/* protects most of the above */
spinlock_t tb_lock;
int dim_timeout;
int idle_timeout;
bool dim_to_is_calc;
int fn_mode;
bool is_t1;
};
struct appletb_key_translation {
u16 from;
u16 to;
};
static const struct appletb_key_translation appletb_fn_codes[] = {
{ KEY_F1, KEY_BRIGHTNESSDOWN },
{ KEY_F2, KEY_BRIGHTNESSUP },
{ KEY_F3, KEY_SCALE }, /* not used */
{ KEY_F4, KEY_DASHBOARD }, /* not used */
{ KEY_F5, KEY_KBDILLUMDOWN },
{ KEY_F6, KEY_KBDILLUMUP },
{ KEY_F7, KEY_PREVIOUSSONG },
{ KEY_F8, KEY_PLAYPAUSE },
{ KEY_F9, KEY_NEXTSONG },
{ KEY_F10, KEY_MUTE },
{ KEY_F11, KEY_VOLUMEDOWN },
{ KEY_F12, KEY_VOLUMEUP },
};
static struct appletb_device *appletb_dev;
static bool appletb_disable_autopm(struct hid_device *hdev)
{
int rc;
rc = hid_hw_power(hdev, PM_HINT_FULLON);
if (rc == 0)
return true;
hid_err(hdev,
"Failed to disable auto-pm on touch bar device (%d)\n", rc);
return false;
}
/*
* While the mode functionality is listed as a valid hid report in the usb
* interface descriptor, on a T1 it's not sent that way. Instead it's sent with
* different request-type and without a leading report-id in the data. Hence
* we need to send it as a custom usb control message rather via any of the
* standard hid_hw_*request() functions. The device might return EPIPE for a
* while after setting the display mode on T1 models, so retrying should be
* done on those models.
*/
static int appletb_set_tb_mode(struct appletb_device *tb_dev,
unsigned char mode)
{
struct hid_report *report;
void *buf;
bool autopm_off = false;
int rc;
if (!tb_dev->mode_iface.hdev)
return -ENOTCONN;
report = tb_dev->mode_field->report;
if (tb_dev->is_t1) {
buf = kmemdup(&mode, 1, GFP_KERNEL);
} else {
char data[] = { report->id, mode };
buf = kmemdup(data, sizeof(data), GFP_KERNEL);
}
if (!buf)
return -ENOMEM;
autopm_off = appletb_disable_autopm(tb_dev->mode_iface.hdev);
if (tb_dev->is_t1) {
int tries = 0;
struct usb_device *dev = interface_to_usbdev(tb_dev->mode_iface.usb_iface);
__u8 ifnum = tb_dev->mode_iface.usb_iface->cur_altsetting->desc.bInterfaceNumber;
do {
rc = usb_control_msg(dev, usb_sndctrlpipe(dev, 0), HID_REQ_SET_REPORT,
USB_DIR_OUT | USB_RECIP_INTERFACE | USB_TYPE_VENDOR,
(report->type + 1) << 8 | report->id,
ifnum, buf, 1, 2000);
if (rc != -EPIPE)
break;
usleep_range(1000 << tries, 3000 << tries);
} while (++tries < 5);
} else {
rc = hid_hw_raw_request(tb_dev->mode_iface.hdev, report->id,
(__u8 *) buf, 2, report->type,
HID_REQ_SET_REPORT);
}
if (rc < 0)
dev_err(tb_dev->log_dev,
"Failed to set touch bar mode to %u (%d)\n", mode, rc);
if (autopm_off)
hid_hw_power(tb_dev->mode_iface.hdev, PM_HINT_NORMAL);
kfree(buf);
return rc;
}
static int appletb_set_tb_disp(struct appletb_device *tb_dev,
unsigned char disp)
{
struct hid_report *report;
int rc;
if (!tb_dev->disp_iface.hdev)
return -ENOTCONN;
report = tb_dev->disp_field->report;
rc = hid_set_field(tb_dev->disp_field_aux1, 0, 1);
if (rc) {
dev_err(tb_dev->log_dev,
"Failed to set display report field (%d)\n", rc);
return rc;
}
rc = hid_set_field(tb_dev->disp_field, 0, disp);
if (rc) {
dev_err(tb_dev->log_dev,
"Failed to set display report field (%d)\n", rc);
return rc;
}
/*
* Keep the USB interface powered on while the touch bar display is on
* for better responsiveness.
*/
if (disp != APPLETB_CMD_DISP_OFF && !tb_dev->tb_autopm_off)
tb_dev->tb_autopm_off =
appletb_disable_autopm(report->device);
hid_hw_request(tb_dev->disp_iface.hdev, report, HID_REQ_SET_REPORT);
if (disp == APPLETB_CMD_DISP_OFF && tb_dev->tb_autopm_off) {
hid_hw_power(tb_dev->disp_iface.hdev, PM_HINT_NORMAL);
tb_dev->tb_autopm_off = false;
}
return rc;
}
static bool appletb_any_tb_key_pressed(struct appletb_device *tb_dev)
{
return !!memchr_inv(tb_dev->last_tb_keys_pressed, 0,
sizeof(tb_dev->last_tb_keys_pressed));
}
static void appletb_schedule_tb_update(struct appletb_device *tb_dev, s64 secs)
{
schedule_delayed_work(&tb_dev->tb_work, msecs_to_jiffies(secs * 1000));
}
static void appletb_set_tb_worker(struct work_struct *work)
{
struct appletb_device *tb_dev =
container_of(work, struct appletb_device, tb_work.work);
s64 time_left = 0, min_timeout, time_to_off;
unsigned char pending_mode;
unsigned char pending_disp;
unsigned char current_disp;
bool restore_autopm;
bool any_tb_key_pressed, need_reschedule;
int rc1 = 1, rc2 = 1;
unsigned long flags;
spin_lock_irqsave(&tb_dev->tb_lock, flags);
/* handle explicit mode-change request */
pending_mode = tb_dev->pnd_tb_mode;
pending_disp = tb_dev->pnd_tb_disp;
restore_autopm = tb_dev->restore_autopm;
spin_unlock_irqrestore(&tb_dev->tb_lock, flags);
if (pending_mode != APPLETB_CMD_MODE_NONE)
rc1 = appletb_set_tb_mode(tb_dev, pending_mode);
if (pending_mode != APPLETB_CMD_MODE_NONE &&
pending_disp != APPLETB_CMD_DISP_NONE)
msleep(25);
if (pending_disp != APPLETB_CMD_DISP_NONE)
rc2 = appletb_set_tb_disp(tb_dev, pending_disp);
if (restore_autopm && tb_dev->tb_autopm_off)
appletb_disable_autopm(tb_dev->disp_field->report->device);
spin_lock_irqsave(&tb_dev->tb_lock, flags);
need_reschedule = false;
if (rc1 == 0) {
tb_dev->cur_tb_mode = pending_mode;
if (tb_dev->pnd_tb_mode == pending_mode)
tb_dev->pnd_tb_mode = APPLETB_CMD_MODE_NONE;
else
need_reschedule = true;
}
if (rc2 == 0) {
tb_dev->cur_tb_disp = pending_disp;
if (tb_dev->pnd_tb_disp == pending_disp)
tb_dev->pnd_tb_disp = APPLETB_CMD_DISP_NONE;
else
need_reschedule = true;
}
current_disp = tb_dev->cur_tb_disp;
tb_dev->restore_autopm = false;
/* calculate time left to next timeout */
if (tb_dev->idle_timeout == -2 || tb_dev->idle_timeout == 0)
min_timeout = -1;
else if (tb_dev->idle_timeout == -1)
min_timeout = tb_dev->dim_timeout;
else if (tb_dev->dim_timeout <= 0)
min_timeout = tb_dev->idle_timeout;
else
min_timeout = min(tb_dev->dim_timeout, tb_dev->idle_timeout);
if (min_timeout > 0) {
s64 idle_time =
(ktime_ms_delta(ktime_get(), tb_dev->last_event_time) +
500) / 1000;
time_left = max(min_timeout - idle_time, 0LL);
if (tb_dev->idle_timeout <= 0)
time_to_off = -1;
else if (idle_time >= tb_dev->idle_timeout)
time_to_off = 0;
else
time_to_off = tb_dev->idle_timeout - idle_time;
} else {
/* not used - just to appease the compiler */
time_to_off = 0;
}
any_tb_key_pressed = appletb_any_tb_key_pressed(tb_dev);
spin_unlock_irqrestore(&tb_dev->tb_lock, flags);
dev_dbg(tb_dev->log_dev, "timeout calc: idle_timeout=%d dim_timeout=%d min_timeout=%lld time_left=%lld need_reschedule=%d any_tb_key_pressed=%d\n",
tb_dev->idle_timeout, tb_dev->dim_timeout, min_timeout,
time_left, need_reschedule, any_tb_key_pressed);
/* a new command arrived while we were busy - handle it */
if (need_reschedule) {
appletb_schedule_tb_update(tb_dev, 0);
return;
}
/* if no idle/dim timeout, we're done */
if (min_timeout <= 0)
return;
/* manage idle/dim timeout */
if (time_left > 0) {
/* we fired too soon or had a mode-change - re-schedule */
appletb_schedule_tb_update(tb_dev, time_left);
} else if (any_tb_key_pressed) {
/* keys are still pressed - re-schedule */
appletb_schedule_tb_update(tb_dev, min_timeout);
} else {
/* dim or idle timeout reached */
int next_disp = (time_to_off == 0) ? APPLETB_CMD_DISP_OFF :
APPLETB_CMD_DISP_DIM;
if (next_disp != current_disp &&
appletb_set_tb_disp(tb_dev, next_disp) == 0) {
spin_lock_irqsave(&tb_dev->tb_lock, flags);
tb_dev->cur_tb_disp = next_disp;
spin_unlock_irqrestore(&tb_dev->tb_lock, flags);
}
if (time_to_off > 0)
appletb_schedule_tb_update(tb_dev, time_to_off);
}
}
static u16 appletb_fn_to_special(u16 code)
{
int idx;
for (idx = 0; idx < ARRAY_SIZE(appletb_fn_codes); idx++) {
if (appletb_fn_codes[idx].from == code)
return appletb_fn_codes[idx].to;
}
return 0;
}
static unsigned char appletb_get_cur_tb_mode(struct appletb_device *tb_dev)
{
return tb_dev->pnd_tb_mode != APPLETB_CMD_MODE_NONE ?
tb_dev->pnd_tb_mode : tb_dev->cur_tb_mode;
}
static unsigned char appletb_get_cur_tb_disp(struct appletb_device *tb_dev)
{
return tb_dev->pnd_tb_disp != APPLETB_CMD_DISP_NONE ?
tb_dev->pnd_tb_disp : tb_dev->cur_tb_disp;
}
static unsigned char appletb_get_fn_tb_mode(struct appletb_device *tb_dev)
{
switch (tb_dev->fn_mode) {
case APPLETB_FN_MODE_ESC:
return APPLETB_CMD_MODE_ESC;
case APPLETB_FN_MODE_FKEYS:
return APPLETB_CMD_MODE_FN;
case APPLETB_FN_MODE_SPCL:
return APPLETB_CMD_MODE_SPCL;
case APPLETB_FN_MODE_INV:
return (tb_dev->last_fn_pressed) ? APPLETB_CMD_MODE_SPCL :
APPLETB_CMD_MODE_FN;
case APPLETB_FN_MODE_NORM:
default:
return (tb_dev->last_fn_pressed) ? APPLETB_CMD_MODE_FN :
APPLETB_CMD_MODE_SPCL;
}
}
/*
* Switch touch bar mode and display when mode or display not the desired ones.
*/
static void appletb_update_touchbar_no_lock(struct appletb_device *tb_dev,
bool force)
{
unsigned char want_mode;
unsigned char want_disp;
bool need_update = false;
/*
* Calculate the new modes:
* idle_timeout:
* -2 mode/disp off
* -1 mode on, disp on/dim
* 0 mode on, disp off
* >0 mode on, disp off after idle_timeout seconds
* dim_timeout (only valid if idle_timeout > 0 || idle_timeout == -1):
* -1 disp never dimmed
* 0 disp always dimmed
* >0 disp dim after dim_timeout seconds
*/
if (tb_dev->idle_timeout == -2) {
want_mode = APPLETB_CMD_MODE_OFF;
want_disp = APPLETB_CMD_DISP_OFF;
} else {
want_mode = appletb_get_fn_tb_mode(tb_dev);
want_disp = tb_dev->idle_timeout == 0 ? APPLETB_CMD_DISP_OFF :
tb_dev->dim_timeout == 0 ? APPLETB_CMD_DISP_DIM :
APPLETB_CMD_DISP_ON;
}
/*
* See if we need to update the touch bar, taking into account that we
* generally don't want to switch modes while a touch bar key is
* pressed.
*/
if (appletb_get_cur_tb_mode(tb_dev) != want_mode &&
!appletb_any_tb_key_pressed(tb_dev)) {
tb_dev->pnd_tb_mode = want_mode;
need_update = true;
}
if (appletb_get_cur_tb_disp(tb_dev) != want_disp &&
(!appletb_any_tb_key_pressed(tb_dev) ||
want_disp != APPLETB_CMD_DISP_OFF)) {
tb_dev->pnd_tb_disp = want_disp;
need_update = true;
}
if (force)
need_update = true;
/* schedule the update if desired */
dev_dbg_ratelimited(tb_dev->log_dev,
"update: need_update=%d, want_mode=%d, cur-mode=%d, want_disp=%d, cur-disp=%d\n",
need_update, want_mode, tb_dev->cur_tb_mode,
want_disp, tb_dev->cur_tb_disp);
if (need_update) {
cancel_delayed_work(&tb_dev->tb_work);
appletb_schedule_tb_update(tb_dev, 0);
}
}
static void appletb_update_touchbar(struct appletb_device *tb_dev, bool force)
{
unsigned long flags;
spin_lock_irqsave(&tb_dev->tb_lock, flags);
if (tb_dev->active)
appletb_update_touchbar_no_lock(tb_dev, force);
spin_unlock_irqrestore(&tb_dev->tb_lock, flags);
}
static void appletb_set_idle_timeout(struct appletb_device *tb_dev, int new)
{
tb_dev->idle_timeout = new;
if (tb_dev->dim_to_is_calc && tb_dev->idle_timeout > 0)
tb_dev->dim_timeout = new - min(APPLETB_MAX_DIM_TIME, new / 3);
else if (tb_dev->dim_to_is_calc)
tb_dev->dim_timeout = -1;
}
static ssize_t idle_timeout_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct appletb_device *tb_dev = dev_get_drvdata(dev);
return snprintf(buf, PAGE_SIZE, "%d\n", tb_dev->idle_timeout);
}
static ssize_t idle_timeout_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t size)
{
struct appletb_device *tb_dev = dev_get_drvdata(dev);
long new;
int rc;
rc = kstrtol(buf, 0, &new);
if (rc || new > INT_MAX || new < -2)
return -EINVAL;
appletb_set_idle_timeout(tb_dev, new);
appletb_update_touchbar(tb_dev, true);
return size;
}
static void appletb_set_dim_timeout(struct appletb_device *tb_dev, int new)
{
if (new == -2) {
tb_dev->dim_to_is_calc = true;
appletb_set_idle_timeout(tb_dev, tb_dev->idle_timeout);
} else {
tb_dev->dim_to_is_calc = false;
tb_dev->dim_timeout = new;
}
}
static ssize_t dim_timeout_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct appletb_device *tb_dev = dev_get_drvdata(dev);
return snprintf(buf, PAGE_SIZE, "%d\n",
tb_dev->dim_to_is_calc ? -2 : tb_dev->dim_timeout);
}
static ssize_t dim_timeout_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t size)
{
struct appletb_device *tb_dev = dev_get_drvdata(dev);
long new;
int rc;
rc = kstrtol(buf, 0, &new);
if (rc || new > INT_MAX || new < -2)
return -EINVAL;
appletb_set_dim_timeout(tb_dev, new);
appletb_update_touchbar(tb_dev, true);
return size;
}
static ssize_t fnmode_show(struct device *dev, struct device_attribute *attr,
char *buf)
{
struct appletb_device *tb_dev = dev_get_drvdata(dev);
return snprintf(buf, PAGE_SIZE, "%d\n", tb_dev->fn_mode);
}
static ssize_t fnmode_store(struct device *dev, struct device_attribute *attr,
const char *buf, size_t size)
{
struct appletb_device *tb_dev = dev_get_drvdata(dev);
long new;
int rc;
rc = kstrtol(buf, 0, &new);
if (rc || new > APPLETB_FN_MODE_MAX || new < 0)
return -EINVAL;
tb_dev->fn_mode = new;
appletb_update_touchbar(tb_dev, false);
return size;
}
static int appletb_tb_key_to_slot(unsigned int code)
{
switch (code) {
case KEY_ESC:
return 0;
case KEY_F1:
case KEY_F2:
case KEY_F3:
case KEY_F4:
case KEY_F5:
case KEY_F6:
case KEY_F7:
case KEY_F8:
case KEY_F9:
case KEY_F10:
return code - KEY_F1 + 1;
case KEY_F11:
case KEY_F12:
return code - KEY_F11 + 11;
default:
return -1;
}
}
static int appletb_hid_event(struct hid_device *hdev, struct hid_field *field,
struct hid_usage *usage, __s32 value)
{
struct appletb_device *tb_dev = hid_get_drvdata(hdev);
unsigned int new_code = 0;
unsigned long flags;
bool send_dummy = false;
bool send_trnsl = false;
int slot;
int rc = 0;
if ((usage->hid & HID_USAGE_PAGE) != HID_UP_KEYBOARD ||
usage->type != EV_KEY)
return 0;
/*
* Skip non-touch-bar keys.
*
* Either the touch bar itself or usbhid generate a slew of key-down
* events for all the meta keys. None of which we're at all interested
* in.
*/
slot = appletb_tb_key_to_slot(usage->code);
if (slot < 0)
return 0;
spin_lock_irqsave(&tb_dev->tb_lock, flags);
if (!tb_dev->active) {
spin_unlock_irqrestore(&tb_dev->tb_lock, flags);
return 0;
}
new_code = appletb_fn_to_special(usage->code);
if (value != 2)
tb_dev->last_tb_keys_pressed[slot] = value;
tb_dev->last_event_time = ktime_get();
appletb_update_touchbar_no_lock(tb_dev, false);
/*
* We want to suppress touch bar keys while the touch bar is off, but
* we do want to wake up the screen if it's asleep, so generate a dummy
* event in that case.
*/
if (tb_dev->cur_tb_mode == APPLETB_CMD_MODE_OFF ||
tb_dev->cur_tb_disp == APPLETB_CMD_DISP_OFF) {
send_dummy = true;
rc = 1;
/* translate special keys */
} else if (new_code &&
((value > 0 &&
appletb_get_cur_tb_mode(tb_dev) == APPLETB_CMD_MODE_SPCL)
||
(value == 0 && tb_dev->last_tb_keys_translated[slot]))) {
tb_dev->last_tb_keys_translated[slot] = true;
send_trnsl = true;
rc = 1;
/* everything else handled normally */
} else {
tb_dev->last_tb_keys_translated[slot] = false;
}
spin_unlock_irqrestore(&tb_dev->tb_lock, flags);
/*
* Need to send these input events outside of the lock, as otherwise
* we can run into the following deadlock:
* Task 1 Task 2
* appletb_hid_event() input_event()
* acquire tb_lock acquire dev->event_lock
* input_event() appletb_inp_event()
* acquire dev->event_lock acquire tb_lock
*/
if (send_dummy) {
input_event(field->hidinput->input, EV_KEY, KEY_UNKNOWN, 1);
input_event(field->hidinput->input, EV_KEY, KEY_UNKNOWN, 0);
} else if (send_trnsl) {
input_event(field->hidinput->input, usage->type, new_code,
value);
}
return rc;
}
static void appletb_inp_event(struct input_handle *handle, unsigned int type,
unsigned int code, int value)
{
struct appletb_device *tb_dev = handle->private;
unsigned long flags;
spin_lock_irqsave(&tb_dev->tb_lock, flags);
if (!tb_dev->active) {
spin_unlock_irqrestore(&tb_dev->tb_lock, flags);
return;
}
if (type == EV_KEY && code == KEY_FN && value != 2)
tb_dev->last_fn_pressed = value;
tb_dev->last_event_time = ktime_get();
appletb_update_touchbar_no_lock(tb_dev, false);
spin_unlock_irqrestore(&tb_dev->tb_lock, flags);
}
/* Find and save the usb-device associated with the touch bar input device */
static struct usb_interface *appletb_get_usb_iface(struct hid_device *hdev)
{
struct device *dev = &hdev->dev;
while (dev && !(dev->type && dev->type->name &&
!strcmp(dev->type->name, "usb_interface")))
dev = dev->parent;
return dev ? to_usb_interface(dev) : NULL;
}
static int appletb_inp_connect(struct input_handler *handler,
struct input_dev *dev,
const struct input_device_id *id)
{
struct appletb_device *tb_dev = handler->private;
struct input_handle *handle;
int rc;
if (id->driver_info == APPLETB_DEVID_KEYBOARD) {
handle = &tb_dev->kbd_handle;
handle->name = "tbkbd";
} else if (id->driver_info == APPLETB_DEVID_TOUCHPAD) {
handle = &tb_dev->tpd_handle;
handle->name = "tbtpad";
} else {
dev_err(tb_dev->log_dev, "Unknown device id (%lu)\n",
id->driver_info);
return -ENOENT;
}
if (handle->dev) {
dev_err(tb_dev->log_dev,
"Duplicate connect to %s input device\n", handle->name);
return -EEXIST;
}
handle->open = 0;
handle->dev = input_get_device(dev);
handle->handler = handler;
handle->private = tb_dev;
rc = input_register_handle(handle);
if (rc)
goto err_free_dev;
rc = input_open_device(handle);
if (rc)
goto err_unregister_handle;
dev_dbg(tb_dev->log_dev, "Connected to %s input device\n",
handle == &tb_dev->kbd_handle ? "keyboard" : "touchpad");
return 0;
err_unregister_handle:
input_unregister_handle(handle);
err_free_dev:
input_put_device(handle->dev);
handle->dev = NULL;
return rc;
}
static void appletb_inp_disconnect(struct input_handle *handle)
{
struct appletb_device *tb_dev = handle->private;
input_close_device(handle);
input_unregister_handle(handle);
dev_dbg(tb_dev->log_dev, "Disconnected from %s input device\n",
handle == &tb_dev->kbd_handle ? "keyboard" : "touchpad");
input_put_device(handle->dev);
handle->dev = NULL;
}
static int appletb_input_configured(struct hid_device *hdev,
struct hid_input *hidinput)
{
int idx;
struct input_dev *input = hidinput->input;
/*
* Clear various input capabilities that are blindly set by the hid
* driver (usbkbd.c)
*/
memset(input->evbit, 0, sizeof(input->evbit));
memset(input->keybit, 0, sizeof(input->keybit));
memset(input->ledbit, 0, sizeof(input->ledbit));
/* set our actual capabilities */
__set_bit(EV_KEY, input->evbit);
__set_bit(EV_REP, input->evbit);
__set_bit(EV_MSC, input->evbit); /* hid-input generates MSC_SCAN */
for (idx = 0; idx < ARRAY_SIZE(appletb_fn_codes); idx++) {
input_set_capability(input, EV_KEY, appletb_fn_codes[idx].from);
input_set_capability(input, EV_KEY, appletb_fn_codes[idx].to);
}
input_set_capability(input, EV_KEY, KEY_ESC);
input_set_capability(input, EV_KEY, KEY_UNKNOWN);
return 0;
}
static struct appletb_iface_info *
appletb_get_iface_info(struct appletb_device *tb_dev, struct hid_device *hdev)
{
if (hdev == tb_dev->mode_iface.hdev)
return &tb_dev->mode_iface;
if (hdev == tb_dev->disp_iface.hdev)
return &tb_dev->disp_iface;
return NULL;
}
/**
* appletb_find_report_field() - Find the field in the report with the given
* usage.
* @report: the report to search
* @field_usage: the usage of the field to search for
*
* Returns: the hid field if found, or NULL if none found.
*/
static struct hid_field *appletb_find_report_field(struct hid_report *report,
unsigned int field_usage)
{
int f, u;
for (f = 0; f < report->maxfield; f++) {
struct hid_field *field = report->field[f];
if (field->logical == field_usage)
return field;
for (u = 0; u < field->maxusage; u++) {
if (field->usage[u].hid == field_usage)
return field;
}
}
return NULL;
}
/**
* appletb_find_hid_field() - Search all the reports of the device for the
* field with the given usage.
* @hdev: the device whose reports to search
* @application: the usage of application collection that the field must
* belong to
* @field_usage: the usage of the field to search for
*
* Returns: the hid field if found, or NULL if none found.
*/
static struct hid_field *appletb_find_hid_field(struct hid_device *hdev,
unsigned int application,
unsigned int field_usage)
{
static const int report_types[] = { HID_INPUT_REPORT, HID_OUTPUT_REPORT,
HID_FEATURE_REPORT };
struct hid_report *report;
struct hid_field *field;
int t;
for (t = 0; t < ARRAY_SIZE(report_types); t++) {
struct list_head *report_list =
&hdev->report_enum[report_types[t]].report_list;
list_for_each_entry(report, report_list, list) {
if (report->application != application)
continue;
field = appletb_find_report_field(report, field_usage);
if (field)
return field;
}
}
return NULL;