Loading...
1// SPDX-License-Identifier: GPL-2.0+
2/*
3 * TI HD3SS3220 Type-C DRP Port Controller Driver
4 *
5 * Copyright (C) 2019 Renesas Electronics Corp.
6 */
7
8#include <linux/module.h>
9#include <linux/i2c.h>
10#include <linux/usb/role.h>
11#include <linux/irqreturn.h>
12#include <linux/interrupt.h>
13#include <linux/regmap.h>
14#include <linux/slab.h>
15#include <linux/usb/typec.h>
16#include <linux/delay.h>
17#include <linux/workqueue.h>
18
19#define HD3SS3220_REG_CN_STAT_CTRL 0x09
20#define HD3SS3220_REG_GEN_CTRL 0x0A
21#define HD3SS3220_REG_DEV_REV 0xA0
22
23/* Register HD3SS3220_REG_CN_STAT_CTRL*/
24#define HD3SS3220_REG_CN_STAT_CTRL_ATTACHED_STATE_MASK (BIT(7) | BIT(6))
25#define HD3SS3220_REG_CN_STAT_CTRL_AS_DFP BIT(6)
26#define HD3SS3220_REG_CN_STAT_CTRL_AS_UFP BIT(7)
27#define HD3SS3220_REG_CN_STAT_CTRL_TO_ACCESSORY (BIT(7) | BIT(6))
28#define HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS BIT(4)
29
30/* Register HD3SS3220_REG_GEN_CTRL*/
31#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_MASK (BIT(2) | BIT(1))
32#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_DEFAULT 0x00
33#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SNK BIT(1)
34#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SRC (BIT(2) | BIT(1))
35
36struct hd3ss3220 {
37 struct device *dev;
38 struct regmap *regmap;
39 struct usb_role_switch *role_sw;
40 struct typec_port *port;
41 struct delayed_work output_poll_work;
42 enum usb_role role_state;
43 bool poll;
44};
45
46static int hd3ss3220_set_source_pref(struct hd3ss3220 *hd3ss3220, int src_pref)
47{
48 return regmap_update_bits(hd3ss3220->regmap, HD3SS3220_REG_GEN_CTRL,
49 HD3SS3220_REG_GEN_CTRL_SRC_PREF_MASK,
50 src_pref);
51}
52
53static enum usb_role hd3ss3220_get_attached_state(struct hd3ss3220 *hd3ss3220)
54{
55 unsigned int reg_val;
56 enum usb_role attached_state;
57 int ret;
58
59 ret = regmap_read(hd3ss3220->regmap, HD3SS3220_REG_CN_STAT_CTRL,
60 ®_val);
61 if (ret < 0)
62 return ret;
63
64 switch (reg_val & HD3SS3220_REG_CN_STAT_CTRL_ATTACHED_STATE_MASK) {
65 case HD3SS3220_REG_CN_STAT_CTRL_AS_DFP:
66 attached_state = USB_ROLE_HOST;
67 break;
68 case HD3SS3220_REG_CN_STAT_CTRL_AS_UFP:
69 attached_state = USB_ROLE_DEVICE;
70 break;
71 default:
72 attached_state = USB_ROLE_NONE;
73 break;
74 }
75
76 return attached_state;
77}
78
79static int hd3ss3220_dr_set(struct typec_port *port, enum typec_data_role role)
80{
81 struct hd3ss3220 *hd3ss3220 = typec_get_drvdata(port);
82 enum usb_role role_val;
83 int pref, ret = 0;
84
85 if (role == TYPEC_HOST) {
86 role_val = USB_ROLE_HOST;
87 pref = HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SRC;
88 } else {
89 role_val = USB_ROLE_DEVICE;
90 pref = HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SNK;
91 }
92
93 ret = hd3ss3220_set_source_pref(hd3ss3220, pref);
94 usleep_range(10, 100);
95
96 usb_role_switch_set_role(hd3ss3220->role_sw, role_val);
97 typec_set_data_role(hd3ss3220->port, role);
98
99 return ret;
100}
101
102static const struct typec_operations hd3ss3220_ops = {
103 .dr_set = hd3ss3220_dr_set
104};
105
106static void hd3ss3220_set_role(struct hd3ss3220 *hd3ss3220)
107{
108 enum usb_role role_state = hd3ss3220_get_attached_state(hd3ss3220);
109
110 usb_role_switch_set_role(hd3ss3220->role_sw, role_state);
111 if (role_state == USB_ROLE_NONE)
112 hd3ss3220_set_source_pref(hd3ss3220,
113 HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_DEFAULT);
114
115 switch (role_state) {
116 case USB_ROLE_HOST:
117 typec_set_data_role(hd3ss3220->port, TYPEC_HOST);
118 break;
119 case USB_ROLE_DEVICE:
120 typec_set_data_role(hd3ss3220->port, TYPEC_DEVICE);
121 break;
122 default:
123 break;
124 }
125
126 hd3ss3220->role_state = role_state;
127}
128
129static void output_poll_execute(struct work_struct *work)
130{
131 struct delayed_work *delayed_work = to_delayed_work(work);
132 struct hd3ss3220 *hd3ss3220 = container_of(delayed_work,
133 struct hd3ss3220,
134 output_poll_work);
135 enum usb_role role_state = hd3ss3220_get_attached_state(hd3ss3220);
136
137 if (hd3ss3220->role_state != role_state)
138 hd3ss3220_set_role(hd3ss3220);
139
140 schedule_delayed_work(&hd3ss3220->output_poll_work, HZ);
141}
142
143static irqreturn_t hd3ss3220_irq(struct hd3ss3220 *hd3ss3220)
144{
145 int err;
146
147 hd3ss3220_set_role(hd3ss3220);
148 err = regmap_write_bits(hd3ss3220->regmap, HD3SS3220_REG_CN_STAT_CTRL,
149 HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS,
150 HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS);
151 if (err < 0)
152 return IRQ_NONE;
153
154 return IRQ_HANDLED;
155}
156
157static irqreturn_t hd3ss3220_irq_handler(int irq, void *data)
158{
159 struct i2c_client *client = to_i2c_client(data);
160 struct hd3ss3220 *hd3ss3220 = i2c_get_clientdata(client);
161
162 return hd3ss3220_irq(hd3ss3220);
163}
164
165static const struct regmap_config config = {
166 .reg_bits = 8,
167 .val_bits = 8,
168 .max_register = 0x0A,
169};
170
171static int hd3ss3220_probe(struct i2c_client *client)
172{
173 struct typec_capability typec_cap = { };
174 struct hd3ss3220 *hd3ss3220;
175 struct fwnode_handle *connector, *ep;
176 int ret;
177 unsigned int data;
178
179 hd3ss3220 = devm_kzalloc(&client->dev, sizeof(struct hd3ss3220),
180 GFP_KERNEL);
181 if (!hd3ss3220)
182 return -ENOMEM;
183
184 i2c_set_clientdata(client, hd3ss3220);
185
186 hd3ss3220->dev = &client->dev;
187 hd3ss3220->regmap = devm_regmap_init_i2c(client, &config);
188 if (IS_ERR(hd3ss3220->regmap))
189 return PTR_ERR(hd3ss3220->regmap);
190
191 hd3ss3220_set_source_pref(hd3ss3220,
192 HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_DEFAULT);
193 /* For backward compatibility check the connector child node first */
194 connector = device_get_named_child_node(hd3ss3220->dev, "connector");
195 if (connector) {
196 hd3ss3220->role_sw = fwnode_usb_role_switch_get(connector);
197 } else {
198 ep = fwnode_graph_get_next_endpoint(dev_fwnode(hd3ss3220->dev), NULL);
199 if (!ep)
200 return -ENODEV;
201 connector = fwnode_graph_get_remote_port_parent(ep);
202 fwnode_handle_put(ep);
203 if (!connector)
204 return -ENODEV;
205 hd3ss3220->role_sw = usb_role_switch_get(hd3ss3220->dev);
206 }
207
208 if (IS_ERR(hd3ss3220->role_sw)) {
209 ret = PTR_ERR(hd3ss3220->role_sw);
210 goto err_put_fwnode;
211 }
212
213 typec_cap.prefer_role = TYPEC_NO_PREFERRED_ROLE;
214 typec_cap.driver_data = hd3ss3220;
215 typec_cap.type = TYPEC_PORT_DRP;
216 typec_cap.data = TYPEC_PORT_DRD;
217 typec_cap.ops = &hd3ss3220_ops;
218 typec_cap.fwnode = connector;
219
220 hd3ss3220->port = typec_register_port(&client->dev, &typec_cap);
221 if (IS_ERR(hd3ss3220->port)) {
222 ret = PTR_ERR(hd3ss3220->port);
223 goto err_put_role;
224 }
225
226 hd3ss3220_set_role(hd3ss3220);
227 ret = regmap_read(hd3ss3220->regmap, HD3SS3220_REG_CN_STAT_CTRL, &data);
228 if (ret < 0)
229 goto err_unreg_port;
230
231 if (data & HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS) {
232 ret = regmap_write(hd3ss3220->regmap,
233 HD3SS3220_REG_CN_STAT_CTRL,
234 data | HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS);
235 if (ret < 0)
236 goto err_unreg_port;
237 }
238
239 if (client->irq > 0) {
240 ret = devm_request_threaded_irq(&client->dev, client->irq, NULL,
241 hd3ss3220_irq_handler,
242 IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
243 "hd3ss3220", &client->dev);
244 if (ret)
245 goto err_unreg_port;
246 } else {
247 INIT_DELAYED_WORK(&hd3ss3220->output_poll_work, output_poll_execute);
248 hd3ss3220->poll = true;
249 }
250
251 ret = i2c_smbus_read_byte_data(client, HD3SS3220_REG_DEV_REV);
252 if (ret < 0)
253 goto err_unreg_port;
254
255 fwnode_handle_put(connector);
256
257 if (hd3ss3220->poll)
258 schedule_delayed_work(&hd3ss3220->output_poll_work, HZ);
259
260 dev_info(&client->dev, "probed revision=0x%x\n", ret);
261
262 return 0;
263err_unreg_port:
264 typec_unregister_port(hd3ss3220->port);
265err_put_role:
266 usb_role_switch_put(hd3ss3220->role_sw);
267err_put_fwnode:
268 fwnode_handle_put(connector);
269
270 return ret;
271}
272
273static void hd3ss3220_remove(struct i2c_client *client)
274{
275 struct hd3ss3220 *hd3ss3220 = i2c_get_clientdata(client);
276
277 if (hd3ss3220->poll)
278 cancel_delayed_work_sync(&hd3ss3220->output_poll_work);
279
280 typec_unregister_port(hd3ss3220->port);
281 usb_role_switch_put(hd3ss3220->role_sw);
282}
283
284static const struct of_device_id dev_ids[] = {
285 { .compatible = "ti,hd3ss3220"},
286 {}
287};
288MODULE_DEVICE_TABLE(of, dev_ids);
289
290static struct i2c_driver hd3ss3220_driver = {
291 .driver = {
292 .name = "hd3ss3220",
293 .of_match_table = dev_ids,
294 },
295 .probe = hd3ss3220_probe,
296 .remove = hd3ss3220_remove,
297};
298
299module_i2c_driver(hd3ss3220_driver);
300
301MODULE_AUTHOR("Biju Das <biju.das@bp.renesas.com>");
302MODULE_DESCRIPTION("TI HD3SS3220 DRP Port Controller Driver");
303MODULE_LICENSE("GPL");
1// SPDX-License-Identifier: GPL-2.0+
2/*
3 * TI HD3SS3220 Type-C DRP Port Controller Driver
4 *
5 * Copyright (C) 2019 Renesas Electronics Corp.
6 */
7
8#include <linux/module.h>
9#include <linux/i2c.h>
10#include <linux/usb/role.h>
11#include <linux/irqreturn.h>
12#include <linux/interrupt.h>
13#include <linux/regmap.h>
14#include <linux/slab.h>
15#include <linux/usb/typec.h>
16#include <linux/delay.h>
17
18#define HD3SS3220_REG_CN_STAT_CTRL 0x09
19#define HD3SS3220_REG_GEN_CTRL 0x0A
20#define HD3SS3220_REG_DEV_REV 0xA0
21
22/* Register HD3SS3220_REG_CN_STAT_CTRL*/
23#define HD3SS3220_REG_CN_STAT_CTRL_ATTACHED_STATE_MASK (BIT(7) | BIT(6))
24#define HD3SS3220_REG_CN_STAT_CTRL_AS_DFP BIT(6)
25#define HD3SS3220_REG_CN_STAT_CTRL_AS_UFP BIT(7)
26#define HD3SS3220_REG_CN_STAT_CTRL_TO_ACCESSORY (BIT(7) | BIT(6))
27#define HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS BIT(4)
28
29/* Register HD3SS3220_REG_GEN_CTRL*/
30#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_MASK (BIT(2) | BIT(1))
31#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_DEFAULT 0x00
32#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SNK BIT(1)
33#define HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SRC (BIT(2) | BIT(1))
34
35struct hd3ss3220 {
36 struct device *dev;
37 struct regmap *regmap;
38 struct usb_role_switch *role_sw;
39 struct typec_port *port;
40};
41
42static int hd3ss3220_set_source_pref(struct hd3ss3220 *hd3ss3220, int src_pref)
43{
44 return regmap_update_bits(hd3ss3220->regmap, HD3SS3220_REG_GEN_CTRL,
45 HD3SS3220_REG_GEN_CTRL_SRC_PREF_MASK,
46 src_pref);
47}
48
49static enum usb_role hd3ss3220_get_attached_state(struct hd3ss3220 *hd3ss3220)
50{
51 unsigned int reg_val;
52 enum usb_role attached_state;
53 int ret;
54
55 ret = regmap_read(hd3ss3220->regmap, HD3SS3220_REG_CN_STAT_CTRL,
56 ®_val);
57 if (ret < 0)
58 return ret;
59
60 switch (reg_val & HD3SS3220_REG_CN_STAT_CTRL_ATTACHED_STATE_MASK) {
61 case HD3SS3220_REG_CN_STAT_CTRL_AS_DFP:
62 attached_state = USB_ROLE_HOST;
63 break;
64 case HD3SS3220_REG_CN_STAT_CTRL_AS_UFP:
65 attached_state = USB_ROLE_DEVICE;
66 break;
67 default:
68 attached_state = USB_ROLE_NONE;
69 break;
70 }
71
72 return attached_state;
73}
74
75static int hd3ss3220_dr_set(struct typec_port *port, enum typec_data_role role)
76{
77 struct hd3ss3220 *hd3ss3220 = typec_get_drvdata(port);
78 enum usb_role role_val;
79 int pref, ret = 0;
80
81 if (role == TYPEC_HOST) {
82 role_val = USB_ROLE_HOST;
83 pref = HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SRC;
84 } else {
85 role_val = USB_ROLE_DEVICE;
86 pref = HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_TRY_SNK;
87 }
88
89 ret = hd3ss3220_set_source_pref(hd3ss3220, pref);
90 usleep_range(10, 100);
91
92 usb_role_switch_set_role(hd3ss3220->role_sw, role_val);
93 typec_set_data_role(hd3ss3220->port, role);
94
95 return ret;
96}
97
98static const struct typec_operations hd3ss3220_ops = {
99 .dr_set = hd3ss3220_dr_set
100};
101
102static void hd3ss3220_set_role(struct hd3ss3220 *hd3ss3220)
103{
104 enum usb_role role_state = hd3ss3220_get_attached_state(hd3ss3220);
105
106 usb_role_switch_set_role(hd3ss3220->role_sw, role_state);
107 if (role_state == USB_ROLE_NONE)
108 hd3ss3220_set_source_pref(hd3ss3220,
109 HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_DEFAULT);
110
111 switch (role_state) {
112 case USB_ROLE_HOST:
113 typec_set_data_role(hd3ss3220->port, TYPEC_HOST);
114 break;
115 case USB_ROLE_DEVICE:
116 typec_set_data_role(hd3ss3220->port, TYPEC_DEVICE);
117 break;
118 default:
119 break;
120 }
121}
122
123static irqreturn_t hd3ss3220_irq(struct hd3ss3220 *hd3ss3220)
124{
125 int err;
126
127 hd3ss3220_set_role(hd3ss3220);
128 err = regmap_write_bits(hd3ss3220->regmap, HD3SS3220_REG_CN_STAT_CTRL,
129 HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS,
130 HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS);
131 if (err < 0)
132 return IRQ_NONE;
133
134 return IRQ_HANDLED;
135}
136
137static irqreturn_t hd3ss3220_irq_handler(int irq, void *data)
138{
139 struct i2c_client *client = to_i2c_client(data);
140 struct hd3ss3220 *hd3ss3220 = i2c_get_clientdata(client);
141
142 return hd3ss3220_irq(hd3ss3220);
143}
144
145static const struct regmap_config config = {
146 .reg_bits = 8,
147 .val_bits = 8,
148 .max_register = 0x0A,
149};
150
151static int hd3ss3220_probe(struct i2c_client *client)
152{
153 struct typec_capability typec_cap = { };
154 struct hd3ss3220 *hd3ss3220;
155 struct fwnode_handle *connector, *ep;
156 int ret;
157 unsigned int data;
158
159 hd3ss3220 = devm_kzalloc(&client->dev, sizeof(struct hd3ss3220),
160 GFP_KERNEL);
161 if (!hd3ss3220)
162 return -ENOMEM;
163
164 i2c_set_clientdata(client, hd3ss3220);
165
166 hd3ss3220->dev = &client->dev;
167 hd3ss3220->regmap = devm_regmap_init_i2c(client, &config);
168 if (IS_ERR(hd3ss3220->regmap))
169 return PTR_ERR(hd3ss3220->regmap);
170
171 hd3ss3220_set_source_pref(hd3ss3220,
172 HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_DEFAULT);
173 /* For backward compatibility check the connector child node first */
174 connector = device_get_named_child_node(hd3ss3220->dev, "connector");
175 if (connector) {
176 hd3ss3220->role_sw = fwnode_usb_role_switch_get(connector);
177 } else {
178 ep = fwnode_graph_get_next_endpoint(dev_fwnode(hd3ss3220->dev), NULL);
179 if (!ep)
180 return -ENODEV;
181 connector = fwnode_graph_get_remote_port_parent(ep);
182 fwnode_handle_put(ep);
183 if (!connector)
184 return -ENODEV;
185 hd3ss3220->role_sw = usb_role_switch_get(hd3ss3220->dev);
186 }
187
188 if (IS_ERR(hd3ss3220->role_sw)) {
189 ret = PTR_ERR(hd3ss3220->role_sw);
190 goto err_put_fwnode;
191 }
192
193 typec_cap.prefer_role = TYPEC_NO_PREFERRED_ROLE;
194 typec_cap.driver_data = hd3ss3220;
195 typec_cap.type = TYPEC_PORT_DRP;
196 typec_cap.data = TYPEC_PORT_DRD;
197 typec_cap.ops = &hd3ss3220_ops;
198 typec_cap.fwnode = connector;
199
200 hd3ss3220->port = typec_register_port(&client->dev, &typec_cap);
201 if (IS_ERR(hd3ss3220->port)) {
202 ret = PTR_ERR(hd3ss3220->port);
203 goto err_put_role;
204 }
205
206 hd3ss3220_set_role(hd3ss3220);
207 ret = regmap_read(hd3ss3220->regmap, HD3SS3220_REG_CN_STAT_CTRL, &data);
208 if (ret < 0)
209 goto err_unreg_port;
210
211 if (data & HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS) {
212 ret = regmap_write(hd3ss3220->regmap,
213 HD3SS3220_REG_CN_STAT_CTRL,
214 data | HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS);
215 if (ret < 0)
216 goto err_unreg_port;
217 }
218
219 if (client->irq > 0) {
220 ret = devm_request_threaded_irq(&client->dev, client->irq, NULL,
221 hd3ss3220_irq_handler,
222 IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
223 "hd3ss3220", &client->dev);
224 if (ret)
225 goto err_unreg_port;
226 }
227
228 ret = i2c_smbus_read_byte_data(client, HD3SS3220_REG_DEV_REV);
229 if (ret < 0)
230 goto err_unreg_port;
231
232 fwnode_handle_put(connector);
233
234 dev_info(&client->dev, "probed revision=0x%x\n", ret);
235
236 return 0;
237err_unreg_port:
238 typec_unregister_port(hd3ss3220->port);
239err_put_role:
240 usb_role_switch_put(hd3ss3220->role_sw);
241err_put_fwnode:
242 fwnode_handle_put(connector);
243
244 return ret;
245}
246
247static void hd3ss3220_remove(struct i2c_client *client)
248{
249 struct hd3ss3220 *hd3ss3220 = i2c_get_clientdata(client);
250
251 typec_unregister_port(hd3ss3220->port);
252 usb_role_switch_put(hd3ss3220->role_sw);
253}
254
255static const struct of_device_id dev_ids[] = {
256 { .compatible = "ti,hd3ss3220"},
257 {}
258};
259MODULE_DEVICE_TABLE(of, dev_ids);
260
261static struct i2c_driver hd3ss3220_driver = {
262 .driver = {
263 .name = "hd3ss3220",
264 .of_match_table = of_match_ptr(dev_ids),
265 },
266 .probe_new = hd3ss3220_probe,
267 .remove = hd3ss3220_remove,
268};
269
270module_i2c_driver(hd3ss3220_driver);
271
272MODULE_AUTHOR("Biju Das <biju.das@bp.renesas.com>");
273MODULE_DESCRIPTION("TI HD3SS3220 DRP Port Controller Driver");
274MODULE_LICENSE("GPL");