Linux Audio

Check our new training course

Loading...
v5.9
  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			  &reg_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_update_bits_base(hd3ss3220->regmap,
129				      HD3SS3220_REG_CN_STAT_CTRL,
130				      HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS,
131				      HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS,
132				      NULL, false, true);
133	if (err < 0)
134		return IRQ_NONE;
135
136	return IRQ_HANDLED;
137}
138
139static irqreturn_t hd3ss3220_irq_handler(int irq, void *data)
140{
141	struct i2c_client *client = to_i2c_client(data);
142	struct hd3ss3220 *hd3ss3220 = i2c_get_clientdata(client);
143
144	return hd3ss3220_irq(hd3ss3220);
145}
146
147static const struct regmap_config config = {
148	.reg_bits = 8,
149	.val_bits = 8,
150	.max_register = 0x0A,
151};
152
153static int hd3ss3220_probe(struct i2c_client *client,
154		const struct i2c_device_id *id)
155{
156	struct typec_capability typec_cap = { };
157	struct hd3ss3220 *hd3ss3220;
158	struct fwnode_handle *connector;
159	int ret;
160	unsigned int data;
161
162	hd3ss3220 = devm_kzalloc(&client->dev, sizeof(struct hd3ss3220),
163				 GFP_KERNEL);
164	if (!hd3ss3220)
165		return -ENOMEM;
166
167	i2c_set_clientdata(client, hd3ss3220);
168
169	hd3ss3220->dev = &client->dev;
170	hd3ss3220->regmap = devm_regmap_init_i2c(client, &config);
171	if (IS_ERR(hd3ss3220->regmap))
172		return PTR_ERR(hd3ss3220->regmap);
173
174	hd3ss3220_set_source_pref(hd3ss3220,
175				  HD3SS3220_REG_GEN_CTRL_SRC_PREF_DRP_DEFAULT);
 
176	connector = device_get_named_child_node(hd3ss3220->dev, "connector");
177	if (!connector)
178		return -ENODEV;
 
 
 
 
 
 
 
 
 
 
179
180	hd3ss3220->role_sw = fwnode_usb_role_switch_get(connector);
181	if (IS_ERR(hd3ss3220->role_sw)) {
182		ret = PTR_ERR(hd3ss3220->role_sw);
183		goto err_put_fwnode;
184	}
185
186	typec_cap.prefer_role = TYPEC_NO_PREFERRED_ROLE;
187	typec_cap.driver_data = hd3ss3220;
188	typec_cap.type = TYPEC_PORT_DRP;
189	typec_cap.data = TYPEC_PORT_DRD;
190	typec_cap.ops = &hd3ss3220_ops;
191	typec_cap.fwnode = connector;
192
193	hd3ss3220->port = typec_register_port(&client->dev, &typec_cap);
194	if (IS_ERR(hd3ss3220->port)) {
195		ret = PTR_ERR(hd3ss3220->port);
196		goto err_put_role;
197	}
198
199	hd3ss3220_set_role(hd3ss3220);
200	ret = regmap_read(hd3ss3220->regmap, HD3SS3220_REG_CN_STAT_CTRL, &data);
201	if (ret < 0)
202		goto err_unreg_port;
203
204	if (data & HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS) {
205		ret = regmap_write(hd3ss3220->regmap,
206				HD3SS3220_REG_CN_STAT_CTRL,
207				data | HD3SS3220_REG_CN_STAT_CTRL_INT_STATUS);
208		if (ret < 0)
209			goto err_unreg_port;
210	}
211
212	if (client->irq > 0) {
213		ret = devm_request_threaded_irq(&client->dev, client->irq, NULL,
214					hd3ss3220_irq_handler,
215					IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
216					"hd3ss3220", &client->dev);
217		if (ret)
218			goto err_unreg_port;
219	}
220
221	ret = i2c_smbus_read_byte_data(client, HD3SS3220_REG_DEV_REV);
222	if (ret < 0)
223		goto err_unreg_port;
224
225	fwnode_handle_put(connector);
226
227	dev_info(&client->dev, "probed revision=0x%x\n", ret);
228
229	return 0;
230err_unreg_port:
231	typec_unregister_port(hd3ss3220->port);
232err_put_role:
233	usb_role_switch_put(hd3ss3220->role_sw);
234err_put_fwnode:
235	fwnode_handle_put(connector);
236
237	return ret;
238}
239
240static int hd3ss3220_remove(struct i2c_client *client)
241{
242	struct hd3ss3220 *hd3ss3220 = i2c_get_clientdata(client);
243
244	typec_unregister_port(hd3ss3220->port);
245	usb_role_switch_put(hd3ss3220->role_sw);
246
247	return 0;
248}
249
250static const struct of_device_id dev_ids[] = {
251	{ .compatible = "ti,hd3ss3220"},
252	{}
253};
254MODULE_DEVICE_TABLE(of, dev_ids);
255
256static struct i2c_driver hd3ss3220_driver = {
257	.driver = {
258		.name = "hd3ss3220",
259		.of_match_table = of_match_ptr(dev_ids),
260	},
261	.probe = hd3ss3220_probe,
262	.remove =  hd3ss3220_remove,
263};
264
265module_i2c_driver(hd3ss3220_driver);
266
267MODULE_AUTHOR("Biju Das <biju.das@bp.renesas.com>");
268MODULE_DESCRIPTION("TI HD3SS3220 DRP Port Controller Driver");
269MODULE_LICENSE("GPL");
v6.2
  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			  &reg_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");