-
Notifications
You must be signed in to change notification settings - Fork 30
/
Copy pathfl2000_drv.c
201 lines (156 loc) · 4.92 KB
/
fl2000_drv.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
// SPDX-License-Identifier: GPL-2.0
/*
* (C) Copyright 2017, Fresco Logic, Incorporated.
* (C) Copyright 2018-2020, Artem Mygaiev
*/
#include "fl2000.h"
#define USB_DRIVER_NAME "fl2000_usb"
#define USB_CLASS_AV 0x10
#define USB_SUBCLASS_AV_CONTROL 0x01
#define USB_SUBCLASS_AV_VIDEO 0x02
#define USB_SUBCLASS_AV_AUDIO 0x03
#define USB_VENDOR_FRESCO_LOGIC 0x1D5C
#define USB_PRODUCT_FL2000 0x2000
#define FL2000_ALL_IFS \
(BIT(FL2000_USBIF_AVCONTROL) | BIT(FL2000_USBIF_STREAMING) | BIT(FL2000_USBIF_INTERRUPT))
static struct usb_device_id fl2000_id_table[] = {
{ USB_DEVICE_INTERFACE_CLASS(USB_VENDOR_FRESCO_LOGIC, USB_PRODUCT_FL2000, USB_CLASS_AV) },
{},
};
MODULE_DEVICE_TABLE(usb, fl2000_id_table);
/* Devices that are independent of interfaces, created for the lifetime of USB device instance */
struct fl2000_devs {
struct regmap *regmap;
struct i2c_adapter *adapter;
struct component_match *match;
int active_if;
};
static struct component_master_ops fl2000_master_ops = {
.bind = fl2000_drm_bind,
.unbind = fl2000_drm_unbind,
};
static int fl2000_compare(struct device *dev, void *data)
{
struct i2c_client *client = i2c_verify_client(dev);
static const char *const fl2000_supported_bridges[] = {
"it66121", /* IT66121 driver name*/
};
UNUSED(data);
if (!client)
return 0;
/* Check this is a supported DRM bridge */
for (int i = 0; i < ARRAY_SIZE(fl2000_supported_bridges); i++)
if (!strncmp(fl2000_supported_bridges[i], client->name, sizeof(client->name)))
return 1; /* Must be not 0 for success */
return 0;
}
static struct fl2000_devs *fl2000_get_devices(struct usb_device *usb_dev)
{
struct fl2000_devs *devs;
devs = devm_kzalloc(&usb_dev->dev, sizeof(*devs), GFP_KERNEL);
if (!devs)
return (ERR_PTR(-ENOMEM));
devs->regmap = fl2000_regmap_init(usb_dev);
if (IS_ERR(devs->regmap))
return ERR_CAST(devs->regmap);
devs->adapter = fl2000_i2c_init(usb_dev);
if (IS_ERR(devs->adapter))
return ERR_CAST(devs->adapter);
component_match_add(&devs->adapter->dev, &devs->match, fl2000_compare, NULL);
dev_set_drvdata(&usb_dev->dev, devs);
return devs;
}
/* TODO: Halt driver on initialization failure */
static int fl2000_probe(struct usb_interface *interface, const struct usb_device_id *usb_dev_id)
{
int ret = 0;
u8 iface_num = interface->cur_altsetting->desc.bInterfaceNumber;
struct usb_device *usb_dev = interface_to_usbdev(interface);
struct fl2000_devs *devs = dev_get_drvdata(&usb_dev->dev);
UNUSED(usb_dev_id);
if (usb_dev->speed < USB_SPEED_HIGH) {
dev_err(&usb_dev->dev, "USB 1.1 is not supported!");
return -ENODEV;
}
if (!devs) {
devs = fl2000_get_devices(usb_dev);
if (IS_ERR(devs)) {
dev_err(&usb_dev->dev, "Cannot initialize I2C and regmap!");
return -ENODEV;
}
}
switch (iface_num) {
case FL2000_USBIF_AVCONTROL:
case FL2000_USBIF_STREAMING:
case FL2000_USBIF_INTERRUPT:
devs->active_if |= BIT(iface_num);
break;
default: /* Device does not have any other interfaces */
dev_warn(&interface->dev, "What interface %d?", iface_num);
ret = -ENODEV;
break;
}
/* When all interfaces are up - proceed with registration */
if (devs->active_if == FL2000_ALL_IFS) {
ret = component_master_add_with_match(&devs->adapter->dev, &fl2000_master_ops,
devs->match);
if (ret) {
dev_err(&usb_dev->dev, "Cannot register component master (%d)", ret);
return ret;
}
}
return ret;
}
static void fl2000_disconnect(struct usb_interface *interface)
{
u8 iface_num = interface->cur_altsetting->desc.bInterfaceNumber;
struct usb_device *usb_dev = interface_to_usbdev(interface);
struct fl2000_devs *devs = dev_get_drvdata(&usb_dev->dev);
if (!devs)
return;
if (devs->active_if == FL2000_ALL_IFS)
component_master_del(&devs->adapter->dev, &fl2000_master_ops);
switch (iface_num) {
case FL2000_USBIF_AVCONTROL:
case FL2000_USBIF_STREAMING:
case FL2000_USBIF_INTERRUPT:
devs->active_if &= ~BIT(iface_num);
break;
default: /* Device does not have any other interfaces */
dev_warn(&interface->dev, "What interface %d?", iface_num);
break;
}
}
static int fl2000_suspend(struct usb_interface *interface, pm_message_t message)
{
struct usb_device *usb_dev = interface_to_usbdev(interface);
UNUSED(message);
dev_dbg(&usb_dev->dev, "resume");
/* TODO: suspend
* drm_mode_config_helper_suspend()
*/
return 0;
}
static int fl2000_resume(struct usb_interface *interface)
{
struct usb_device *usb_dev = interface_to_usbdev(interface);
dev_dbg(&usb_dev->dev, "suspend");
/* TODO: resume
* drm_mode_config_helper_resume()
*/
return 0;
}
static struct usb_driver fl2000_driver = {
.name = USB_DRIVER_NAME,
.probe = fl2000_probe,
.disconnect = fl2000_disconnect,
.suspend = fl2000_suspend,
.resume = fl2000_resume,
.id_table = fl2000_id_table,
.supports_autosuspend = false,
.disable_hub_initiated_lpm = true,
};
module_usb_driver(fl2000_driver);
MODULE_AUTHOR("Artem Mygaiev");
MODULE_DESCRIPTION("FL2000 USB display driver");
MODULE_LICENSE("GPL v2");