usb3503.c 10.0 KB
Newer Older
1
// SPDX-License-Identifier: GPL-2.0+
2 3 4 5 6 7
/*
 * Driver for SMSC USB3503 USB 2.0 hub controller driver
 *
 * Copyright (c) 2012-2013 Dongjin Kim (tobetter@gmail.com)
 */

8
#include <linux/clk.h>
9 10 11 12 13
#include <linux/i2c.h>
#include <linux/gpio.h>
#include <linux/delay.h>
#include <linux/slab.h>
#include <linux/module.h>
D
Dongjin Kim 已提交
14
#include <linux/of_gpio.h>
15 16
#include <linux/platform_device.h>
#include <linux/platform_data/usb3503.h>
17
#include <linux/regmap.h>
18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41

#define USB3503_VIDL		0x00
#define USB3503_VIDM		0x01
#define USB3503_PIDL		0x02
#define USB3503_PIDM		0x03
#define USB3503_DIDL		0x04
#define USB3503_DIDM		0x05

#define USB3503_CFG1		0x06
#define USB3503_SELF_BUS_PWR	(1 << 7)

#define USB3503_CFG2		0x07
#define USB3503_CFG3		0x08
#define USB3503_NRD		0x09

#define USB3503_PDS		0x0a

#define USB3503_SP_ILOCK	0xe7
#define USB3503_SPILOCK_CONNECT	(1 << 1)
#define USB3503_SPILOCK_CONFIG	(1 << 0)

#define USB3503_CFGP		0xee
#define USB3503_CLKSUSP		(1 << 7)

42 43
#define USB3503_RESET		0xff

44 45
struct usb3503 {
	enum usb3503_mode	mode;
46 47
	struct regmap		*regmap;
	struct device		*dev;
48
	struct clk		*clk;
49
	u8	port_off_mask;
50 51 52
	int	gpio_intn;
	int	gpio_reset;
	int	gpio_connect;
53
	bool	secondary_ref_clk;
54 55
};

56
static int usb3503_reset(struct usb3503 *hub, int state)
57
{
58 59 60 61 62
	if (!state && gpio_is_valid(hub->gpio_connect))
		gpio_set_value_cansleep(hub->gpio_connect, 0);

	if (gpio_is_valid(hub->gpio_reset))
		gpio_set_value_cansleep(hub->gpio_reset, state);
63

64
	/* Wait T_HUBINIT == 4ms for hub logic to stabilize */
65
	if (state)
66
		usleep_range(4000, 10000);
67 68 69 70

	return 0;
}

71
static int usb3503_connect(struct usb3503 *hub)
72
{
73
	struct device *dev = hub->dev;
74
	int err;
75

76
	usb3503_reset(hub, 1);
77

78
	if (hub->regmap) {
79
		/* SP_ILOCK: set connect_n, config_n for config */
80
		err = regmap_write(hub->regmap, USB3503_SP_ILOCK,
81
			   (USB3503_SPILOCK_CONNECT
82 83
				 | USB3503_SPILOCK_CONFIG));
		if (err < 0) {
84
			dev_err(dev, "SP_ILOCK failed (%d)\n", err);
85
			return err;
86 87
		}

88
		/* PDS : Set the ports which are disabled in self-powered mode. */
89
		if (hub->port_off_mask) {
90 91
			err = regmap_update_bits(hub->regmap, USB3503_PDS,
					hub->port_off_mask,
92 93
					hub->port_off_mask);
			if (err < 0) {
94
				dev_err(dev, "PDS failed (%d)\n", err);
95
				return err;
96
			}
97 98
		}

99
		/* CFG1 : Set SELF_BUS_PWR, this enables self-powered operation. */
100 101 102
		err = regmap_update_bits(hub->regmap, USB3503_CFG1,
					 USB3503_SELF_BUS_PWR,
					 USB3503_SELF_BUS_PWR);
103
		if (err < 0) {
104
			dev_err(dev, "CFG1 failed (%d)\n", err);
105
			return err;
106 107 108
		}

		/* SP_LOCK: clear connect_n, config_n for hub connect */
109 110 111
		err = regmap_update_bits(hub->regmap, USB3503_SP_ILOCK,
					 (USB3503_SPILOCK_CONNECT
					  | USB3503_SPILOCK_CONFIG), 0);
112
		if (err < 0) {
113
			dev_err(dev, "SP_ILOCK failed (%d)\n", err);
114
			return err;
115
		}
116
	}
117

118 119
	if (gpio_is_valid(hub->gpio_connect))
		gpio_set_value_cansleep(hub->gpio_connect, 1);
120

121 122 123 124 125 126 127 128 129 130 131 132 133 134
	hub->mode = USB3503_MODE_HUB;
	dev_info(dev, "switched to HUB mode\n");

	return 0;
}

static int usb3503_switch_mode(struct usb3503 *hub, enum usb3503_mode mode)
{
	struct device *dev = hub->dev;
	int err = 0;

	switch (mode) {
	case USB3503_MODE_HUB:
		err = usb3503_connect(hub);
135 136 137
		break;

	case USB3503_MODE_STANDBY:
138
		usb3503_reset(hub, 0);
139
		dev_info(dev, "switched to STANDBY mode\n");
140 141 142
		break;

	default:
143
		dev_err(dev, "unknown mode is requested\n");
144 145 146 147 148 149 150
		err = -EINVAL;
		break;
	}

	return err;
}

151 152 153 154 155 156 157
static const struct regmap_config usb3503_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,

	.max_register = USB3503_RESET,
};

158
static int usb3503_probe(struct usb3503 *hub)
159
{
160 161 162 163
	struct device *dev = hub->dev;
	struct usb3503_platform_data *pdata = dev_get_platdata(dev);
	struct device_node *np = dev->of_node;
	int err;
164
	u32 mode = USB3503_MODE_HUB;
165 166
	const u32 *property;
	int len;
167

D
Dongjin Kim 已提交
168
	if (pdata) {
169
		hub->port_off_mask	= pdata->port_off_mask;
170 171 172 173
		hub->gpio_intn		= pdata->gpio_intn;
		hub->gpio_connect	= pdata->gpio_connect;
		hub->gpio_reset		= pdata->gpio_reset;
		hub->mode		= pdata->initial_mode;
D
Dongjin Kim 已提交
174
	} else if (np) {
175
		struct clk *clk;
176
		u32 rate = 0;
177 178
		hub->port_off_mask = 0;

179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200
		if (!of_property_read_u32(np, "refclk-frequency", &rate)) {
			switch (rate) {
			case 38400000:
			case 26000000:
			case 19200000:
			case 12000000:
				hub->secondary_ref_clk = 0;
				break;
			case 24000000:
			case 27000000:
			case 25000000:
			case 50000000:
				hub->secondary_ref_clk = 1;
				break;
			default:
				dev_err(dev,
					"unsupported reference clock rate (%d)\n",
					(int) rate);
				return -EINVAL;
			}
		}

201 202
		clk = devm_clk_get(dev, "refclk");
		if (IS_ERR(clk) && PTR_ERR(clk) != -ENOENT) {
203 204
			dev_err(dev, "unable to request refclk (%ld)\n",
					PTR_ERR(clk));
205 206 207 208 209 210
			return PTR_ERR(clk);
		}

		if (!IS_ERR(clk)) {
			hub->clk = clk;

211
			if (rate != 0) {
212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228
				err = clk_set_rate(hub->clk, rate);
				if (err) {
					dev_err(dev,
						"unable to set reference clock rate to %d\n",
						(int) rate);
					return err;
				}
			}

			err = clk_prepare_enable(hub->clk);
			if (err) {
				dev_err(dev,
					"unable to enable reference clock\n");
				return err;
			}
		}

229 230 231 232 233 234 235 236 237 238
		property = of_get_property(np, "disabled-ports", &len);
		if (property && (len / sizeof(u32)) > 0) {
			int i;
			for (i = 0; i < len / sizeof(u32); i++) {
				u32 port = be32_to_cpu(property[i]);
				if ((1 <= port) && (port <= 3))
					hub->port_off_mask |= (1 << port);
			}
		}

239
		hub->gpio_intn	= of_get_named_gpio(np, "intn-gpios", 0);
D
Dongjin Kim 已提交
240 241
		if (hub->gpio_intn == -EPROBE_DEFER)
			return -EPROBE_DEFER;
242
		hub->gpio_connect = of_get_named_gpio(np, "connect-gpios", 0);
D
Dongjin Kim 已提交
243 244
		if (hub->gpio_connect == -EPROBE_DEFER)
			return -EPROBE_DEFER;
245
		hub->gpio_reset = of_get_named_gpio(np, "reset-gpios", 0);
D
Dongjin Kim 已提交
246 247 248 249
		if (hub->gpio_reset == -EPROBE_DEFER)
			return -EPROBE_DEFER;
		of_property_read_u32(np, "initial-mode", &mode);
		hub->mode = mode;
250 251
	}

252 253 254
	if (hub->port_off_mask && !hub->regmap)
		dev_err(dev, "Ports disabled with no control interface\n");

255
	if (gpio_is_valid(hub->gpio_intn)) {
256 257 258 259
		int val = hub->secondary_ref_clk ? GPIOF_OUT_INIT_LOW :
						   GPIOF_OUT_INIT_HIGH;
		err = devm_gpio_request_one(dev, hub->gpio_intn, val,
					    "usb3503 intn");
260
		if (err) {
261
			dev_err(dev,
262
				"unable to request GPIO %d as interrupt pin (%d)\n",
263
				hub->gpio_intn, err);
264
			return err;
265 266 267 268
		}
	}

	if (gpio_is_valid(hub->gpio_connect)) {
269
		err = devm_gpio_request_one(dev, hub->gpio_connect,
270
				GPIOF_OUT_INIT_LOW, "usb3503 connect");
271
		if (err) {
272 273 274
			dev_err(dev,
				"unable to request GPIO %d as connect pin (%d)\n",
				hub->gpio_connect, err);
275
			return err;
276 277 278 279
		}
	}

	if (gpio_is_valid(hub->gpio_reset)) {
280
		err = devm_gpio_request_one(dev, hub->gpio_reset,
281 282
				GPIOF_OUT_INIT_LOW, "usb3503 reset");
		if (err) {
283 284 285
			dev_err(dev,
				"unable to request GPIO %d as reset pin (%d)\n",
				hub->gpio_reset, err);
286
			return err;
287 288 289
		}
	}

D
Dongjin Kim 已提交
290
	usb3503_switch_mode(hub, hub->mode);
291

292
	dev_info(dev, "%s: probed in %s mode\n", __func__,
293 294 295 296 297
			(hub->mode == USB3503_MODE_HUB) ? "hub" : "standby");

	return 0;
}

298 299 300 301 302 303 304
static int usb3503_i2c_probe(struct i2c_client *i2c,
			     const struct i2c_device_id *id)
{
	struct usb3503 *hub;
	int err;

	hub = devm_kzalloc(&i2c->dev, sizeof(struct usb3503), GFP_KERNEL);
305
	if (!hub)
306 307 308 309 310 311 312 313 314 315 316 317 318 319
		return -ENOMEM;

	i2c_set_clientdata(i2c, hub);
	hub->regmap = devm_regmap_init_i2c(i2c, &usb3503_regmap_config);
	if (IS_ERR(hub->regmap)) {
		err = PTR_ERR(hub->regmap);
		dev_err(&i2c->dev, "Failed to initialise regmap: %d\n", err);
		return err;
	}
	hub->dev = &i2c->dev;

	return usb3503_probe(hub);
}

320 321 322 323 324 325 326 327 328 329 330
static int usb3503_i2c_remove(struct i2c_client *i2c)
{
	struct usb3503 *hub;

	hub = i2c_get_clientdata(i2c);
	if (hub->clk)
		clk_disable_unprepare(hub->clk);

	return 0;
}

331 332 333 334 335
static int usb3503_platform_probe(struct platform_device *pdev)
{
	struct usb3503 *hub;

	hub = devm_kzalloc(&pdev->dev, sizeof(struct usb3503), GFP_KERNEL);
336
	if (!hub)
337 338
		return -ENOMEM;
	hub->dev = &pdev->dev;
339
	platform_set_drvdata(pdev, hub);
340 341 342 343

	return usb3503_probe(hub);
}

344 345 346 347 348 349 350 351 352 353 354
static int usb3503_platform_remove(struct platform_device *pdev)
{
	struct usb3503 *hub;

	hub = platform_get_drvdata(pdev);
	if (hub->clk)
		clk_disable_unprepare(hub->clk);

	return 0;
}

J
Joonyoung Shim 已提交
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
#ifdef CONFIG_PM_SLEEP
static int usb3503_i2c_suspend(struct device *dev)
{
	struct i2c_client *client = to_i2c_client(dev);
	struct usb3503 *hub = i2c_get_clientdata(client);

	usb3503_switch_mode(hub, USB3503_MODE_STANDBY);

	if (hub->clk)
		clk_disable_unprepare(hub->clk);

	return 0;
}

static int usb3503_i2c_resume(struct device *dev)
{
	struct i2c_client *client = to_i2c_client(dev);
	struct usb3503 *hub = i2c_get_clientdata(client);

	if (hub->clk)
		clk_prepare_enable(hub->clk);

	usb3503_switch_mode(hub, hub->mode);

	return 0;
}
#endif

static SIMPLE_DEV_PM_OPS(usb3503_i2c_pm_ops, usb3503_i2c_suspend,
		usb3503_i2c_resume);

386 387 388 389 390 391
static const struct i2c_device_id usb3503_id[] = {
	{ USB3503_I2C_NAME, 0 },
	{ }
};
MODULE_DEVICE_TABLE(i2c, usb3503_id);

D
Dongjin Kim 已提交
392 393 394
#ifdef CONFIG_OF
static const struct of_device_id usb3503_of_match[] = {
	{ .compatible = "smsc,usb3503", },
395
	{ .compatible = "smsc,usb3503a", },
D
Dongjin Kim 已提交
396 397 398 399 400
	{},
};
MODULE_DEVICE_TABLE(of, usb3503_of_match);
#endif

401
static struct i2c_driver usb3503_i2c_driver = {
402 403
	.driver = {
		.name = USB3503_I2C_NAME,
J
Joonyoung Shim 已提交
404
		.pm = &usb3503_i2c_pm_ops,
D
Dongjin Kim 已提交
405
		.of_match_table = of_match_ptr(usb3503_of_match),
406
	},
407
	.probe		= usb3503_i2c_probe,
408
	.remove		= usb3503_i2c_remove,
409 410 411
	.id_table	= usb3503_id,
};

412 413 414 415 416 417
static struct platform_driver usb3503_platform_driver = {
	.driver = {
		.name = USB3503_I2C_NAME,
		.of_match_table = of_match_ptr(usb3503_of_match),
	},
	.probe		= usb3503_platform_probe,
418
	.remove		= usb3503_platform_remove,
419 420 421 422 423 424
};

static int __init usb3503_init(void)
{
	int err;

425
	err = i2c_add_driver(&usb3503_i2c_driver);
426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443
	if (err != 0)
		pr_err("usb3503: Failed to register I2C driver: %d\n", err);

	err = platform_driver_register(&usb3503_platform_driver);
	if (err != 0)
		pr_err("usb3503: Failed to register platform driver: %d\n",
		       err);

	return 0;
}
module_init(usb3503_init);

static void __exit usb3503_exit(void)
{
	platform_driver_unregister(&usb3503_platform_driver);
	i2c_del_driver(&usb3503_i2c_driver);
}
module_exit(usb3503_exit);
444 445 446 447

MODULE_AUTHOR("Dongjin Kim <tobetter@gmail.com>");
MODULE_DESCRIPTION("USB3503 USB HUB driver");
MODULE_LICENSE("GPL");