spi-orion.c 12.8 KB
Newer Older
S
Shadi Ammouri 已提交
1
/*
G
Grant Likely 已提交
2
 * Marvell Orion SPI controller driver
S
Shadi Ammouri 已提交
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
 *
 * Author: Shadi Ammouri <shadi@marvell.com>
 * Copyright (C) 2007-2008 Marvell Ltd.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 */

#include <linux/interrupt.h>
#include <linux/delay.h>
#include <linux/platform_device.h>
#include <linux/err.h>
#include <linux/io.h>
#include <linux/spi/spi.h>
18
#include <linux/module.h>
19
#include <linux/pm_runtime.h>
A
Andrew Lunn 已提交
20
#include <linux/of.h>
21
#include <linux/of_device.h>
22
#include <linux/clk.h>
23
#include <linux/sizes.h>
S
Shadi Ammouri 已提交
24 25 26 27
#include <asm/unaligned.h>

#define DRIVER_NAME			"orion_spi"

28 29 30
/* Runtime PM autosuspend timeout: PM is fairly light on this driver */
#define SPI_AUTOSUSPEND_TIMEOUT		200

S
Shadi Ammouri 已提交
31 32 33 34 35 36 37 38 39
#define ORION_NUM_CHIPSELECTS		1 /* only one slave is supported*/
#define ORION_SPI_WAIT_RDY_MAX_LOOP	2000 /* in usec */

#define ORION_SPI_IF_CTRL_REG		0x00
#define ORION_SPI_IF_CONFIG_REG		0x04
#define ORION_SPI_DATA_OUT_REG		0x08
#define ORION_SPI_DATA_IN_REG		0x0c
#define ORION_SPI_INT_CAUSE_REG		0x10

40 41
#define ORION_SPI_MODE_CPOL		(1 << 11)
#define ORION_SPI_MODE_CPHA		(1 << 12)
S
Shadi Ammouri 已提交
42 43
#define ORION_SPI_IF_8_16_BIT_MODE	(1 << 5)
#define ORION_SPI_CLK_PRESCALE_MASK	0x1F
44
#define ARMADA_SPI_CLK_PRESCALE_MASK	0xDF
45 46
#define ORION_SPI_MODE_MASK		(ORION_SPI_MODE_CPOL | \
					 ORION_SPI_MODE_CPHA)
S
Shadi Ammouri 已提交
47

48 49 50 51 52 53 54 55 56 57 58 59
enum orion_spi_type {
	ORION_SPI,
	ARMADA_SPI,
};

struct orion_spi_dev {
	enum orion_spi_type	typ;
	unsigned int		min_divisor;
	unsigned int		max_divisor;
	u32			prescale_mask;
};

S
Shadi Ammouri 已提交
60 61 62
struct orion_spi {
	struct spi_master	*master;
	void __iomem		*base;
63
	struct clk              *clk;
64
	const struct orion_spi_dev *devdata;
S
Shadi Ammouri 已提交
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
};

static inline void __iomem *spi_reg(struct orion_spi *orion_spi, u32 reg)
{
	return orion_spi->base + reg;
}

static inline void
orion_spi_setbits(struct orion_spi *orion_spi, u32 reg, u32 mask)
{
	void __iomem *reg_addr = spi_reg(orion_spi, reg);
	u32 val;

	val = readl(reg_addr);
	val |= mask;
	writel(val, reg_addr);
}

static inline void
orion_spi_clrbits(struct orion_spi *orion_spi, u32 reg, u32 mask)
{
	void __iomem *reg_addr = spi_reg(orion_spi, reg);
	u32 val;

	val = readl(reg_addr);
	val &= ~mask;
	writel(val, reg_addr);
}

static int orion_spi_baudrate_set(struct spi_device *spi, unsigned int speed)
{
	u32 tclk_hz;
	u32 rate;
	u32 prescale;
	u32 reg;
	struct orion_spi *orion_spi;
101
	const struct orion_spi_dev *devdata;
S
Shadi Ammouri 已提交
102 103

	orion_spi = spi_master_get_devdata(spi->master);
104
	devdata = orion_spi->devdata;
S
Shadi Ammouri 已提交
105

106
	tclk_hz = clk_get_rate(orion_spi->clk);
S
Shadi Ammouri 已提交
107

108 109 110
	if (devdata->typ == ARMADA_SPI) {
		unsigned int clk, spr, sppr, sppr2, err;
		unsigned int best_spr, best_sppr, best_err;
S
Shadi Ammouri 已提交
111

112 113 114
		best_err = speed;
		best_spr = 0;
		best_sppr = 0;
S
Shadi Ammouri 已提交
115

116 117 118 119 120 121 122 123
		/* Iterate over the valid range looking for best fit */
		for (sppr = 0; sppr < 8; sppr++) {
			sppr2 = 0x1 << sppr;

			spr = tclk_hz / sppr2;
			spr = DIV_ROUND_UP(spr, speed);
			if ((spr == 0) || (spr > 15))
				continue;
S
Shadi Ammouri 已提交
124

125 126
			clk = tclk_hz / (spr * sppr2);
			err = speed - clk;
S
Shadi Ammouri 已提交
127

128 129 130 131 132 133
			if (err < best_err) {
				best_spr = spr;
				best_sppr = sppr;
				best_err = err;
			}
		}
S
Shadi Ammouri 已提交
134

135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150
		if ((best_sppr == 0) && (best_spr == 0))
			return -EINVAL;

		prescale = ((best_sppr & 0x6) << 5) |
			((best_sppr & 0x1) << 4) | best_spr;
	} else {
		/*
		 * the supported rates are: 4,6,8...30
		 * round up as we look for equal or less speed
		 */
		rate = DIV_ROUND_UP(tclk_hz, speed);
		rate = roundup(rate, 2);

		/* check if requested speed is too small */
		if (rate > 30)
			return -EINVAL;
S
Shadi Ammouri 已提交
151

152 153 154 155 156 157
		if (rate < 4)
			rate = 4;

		/* Convert the rate to SPI clock divisor value.	*/
		prescale = 0x10 + rate/2;
	}
S
Shadi Ammouri 已提交
158 159

	reg = readl(spi_reg(orion_spi, ORION_SPI_IF_CONFIG_REG));
160
	reg = ((reg & ~devdata->prescale_mask) | prescale);
S
Shadi Ammouri 已提交
161 162 163 164 165
	writel(reg, spi_reg(orion_spi, ORION_SPI_IF_CONFIG_REG));

	return 0;
}

166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182
static void
orion_spi_mode_set(struct spi_device *spi)
{
	u32 reg;
	struct orion_spi *orion_spi;

	orion_spi = spi_master_get_devdata(spi->master);

	reg = readl(spi_reg(orion_spi, ORION_SPI_IF_CONFIG_REG));
	reg &= ~ORION_SPI_MODE_MASK;
	if (spi->mode & SPI_CPOL)
		reg |= ORION_SPI_MODE_CPOL;
	if (spi->mode & SPI_CPHA)
		reg |= ORION_SPI_MODE_CPHA;
	writel(reg, spi_reg(orion_spi, ORION_SPI_IF_CONFIG_REG));
}

S
Shadi Ammouri 已提交
183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201
/*
 * called only when no transfer is active on the bus
 */
static int
orion_spi_setup_transfer(struct spi_device *spi, struct spi_transfer *t)
{
	struct orion_spi *orion_spi;
	unsigned int speed = spi->max_speed_hz;
	unsigned int bits_per_word = spi->bits_per_word;
	int	rc;

	orion_spi = spi_master_get_devdata(spi->master);

	if ((t != NULL) && t->speed_hz)
		speed = t->speed_hz;

	if ((t != NULL) && t->bits_per_word)
		bits_per_word = t->bits_per_word;

202 203
	orion_spi_mode_set(spi);

S
Shadi Ammouri 已提交
204 205 206 207
	rc = orion_spi_baudrate_set(spi, speed);
	if (rc)
		return rc;

208 209 210 211 212 213 214 215
	if (bits_per_word == 16)
		orion_spi_setbits(orion_spi, ORION_SPI_IF_CONFIG_REG,
				  ORION_SPI_IF_8_16_BIT_MODE);
	else
		orion_spi_clrbits(orion_spi, ORION_SPI_IF_CONFIG_REG,
				  ORION_SPI_IF_8_16_BIT_MODE);

	return 0;
S
Shadi Ammouri 已提交
216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232
}

static void orion_spi_set_cs(struct orion_spi *orion_spi, int enable)
{
	if (enable)
		orion_spi_setbits(orion_spi, ORION_SPI_IF_CTRL_REG, 0x1);
	else
		orion_spi_clrbits(orion_spi, ORION_SPI_IF_CTRL_REG, 0x1);
}

static inline int orion_spi_wait_till_ready(struct orion_spi *orion_spi)
{
	int i;

	for (i = 0; i < ORION_SPI_WAIT_RDY_MAX_LOOP; i++) {
		if (readl(spi_reg(orion_spi, ORION_SPI_INT_CAUSE_REG)))
			return 1;
J
Jingoo Han 已提交
233 234

		udelay(1);
S
Shadi Ammouri 已提交
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
	}

	return -1;
}

static inline int
orion_spi_write_read_8bit(struct spi_device *spi,
			  const u8 **tx_buf, u8 **rx_buf)
{
	void __iomem *tx_reg, *rx_reg, *int_reg;
	struct orion_spi *orion_spi;

	orion_spi = spi_master_get_devdata(spi->master);
	tx_reg = spi_reg(orion_spi, ORION_SPI_DATA_OUT_REG);
	rx_reg = spi_reg(orion_spi, ORION_SPI_DATA_IN_REG);
	int_reg = spi_reg(orion_spi, ORION_SPI_INT_CAUSE_REG);

	/* clear the interrupt cause register */
	writel(0x0, int_reg);

	if (tx_buf && *tx_buf)
		writel(*(*tx_buf)++, tx_reg);
	else
		writel(0, tx_reg);

	if (orion_spi_wait_till_ready(orion_spi) < 0) {
		dev_err(&spi->dev, "TXS timed out\n");
		return -1;
	}

	if (rx_buf && *rx_buf)
		*(*rx_buf)++ = readl(rx_reg);

	return 1;
}

static inline int
orion_spi_write_read_16bit(struct spi_device *spi,
			   const u16 **tx_buf, u16 **rx_buf)
{
	void __iomem *tx_reg, *rx_reg, *int_reg;
	struct orion_spi *orion_spi;

	orion_spi = spi_master_get_devdata(spi->master);
	tx_reg = spi_reg(orion_spi, ORION_SPI_DATA_OUT_REG);
	rx_reg = spi_reg(orion_spi, ORION_SPI_DATA_IN_REG);
	int_reg = spi_reg(orion_spi, ORION_SPI_INT_CAUSE_REG);

	/* clear the interrupt cause register */
	writel(0x0, int_reg);

	if (tx_buf && *tx_buf)
		writel(__cpu_to_le16(get_unaligned((*tx_buf)++)), tx_reg);
	else
		writel(0, tx_reg);

	if (orion_spi_wait_till_ready(orion_spi) < 0) {
		dev_err(&spi->dev, "TXS timed out\n");
		return -1;
	}

	if (rx_buf && *rx_buf)
		put_unaligned(__le16_to_cpu(readl(rx_reg)), (*rx_buf)++);

	return 1;
}

static unsigned int
orion_spi_write_read(struct spi_device *spi, struct spi_transfer *xfer)
{
	unsigned int count;
	int word_len;

	word_len = spi->bits_per_word;
	count = xfer->len;

	if (word_len == 8) {
		const u8 *tx = xfer->tx_buf;
		u8 *rx = xfer->rx_buf;

		do {
			if (orion_spi_write_read_8bit(spi, &tx, &rx) < 0)
				goto out;
			count--;
		} while (count);
	} else if (word_len == 16) {
		const u16 *tx = xfer->tx_buf;
		u16 *rx = xfer->rx_buf;

		do {
			if (orion_spi_write_read_16bit(spi, &tx, &rx) < 0)
				goto out;
			count -= 2;
		} while (count);
	}

out:
	return xfer->len - count;
}

335 336
static int orion_spi_transfer_one_message(struct spi_master *master,
					   struct spi_message *m)
S
Shadi Ammouri 已提交
337
{
338 339 340 341 342 343
	struct orion_spi *orion_spi = spi_master_get_devdata(master);
	struct spi_device *spi = m->spi;
	struct spi_transfer *t = NULL;
	int par_override = 0;
	int status = 0;
	int cs_active = 0;
S
Shadi Ammouri 已提交
344

345 346
	/* Load defaults */
	status = orion_spi_setup_transfer(spi, NULL);
S
Shadi Ammouri 已提交
347

348 349
	if (status < 0)
		goto msg_done;
S
Shadi Ammouri 已提交
350

351 352 353 354 355 356 357 358
	list_for_each_entry(t, &m->transfers, transfer_list) {
		if (par_override || t->speed_hz || t->bits_per_word) {
			par_override = 1;
			status = orion_spi_setup_transfer(spi, t);
			if (status < 0)
				break;
			if (!t->speed_hz && !t->bits_per_word)
				par_override = 0;
S
Shadi Ammouri 已提交
359 360
		}

361 362 363 364
		if (!cs_active) {
			orion_spi_set_cs(orion_spi, 1);
			cs_active = 1;
		}
S
Shadi Ammouri 已提交
365

366 367
		if (t->len)
			m->actual_length += orion_spi_write_read(spi, t);
S
Shadi Ammouri 已提交
368

369 370 371 372 373 374 375
		if (t->delay_usecs)
			udelay(t->delay_usecs);

		if (t->cs_change) {
			orion_spi_set_cs(orion_spi, 0);
			cs_active = 0;
		}
S
Shadi Ammouri 已提交
376 377
	}

378 379 380 381 382 383 384 385
msg_done:
	if (cs_active)
		orion_spi_set_cs(orion_spi, 0);

	m->status = status;
	spi_finalize_current_message(master);

	return 0;
S
Shadi Ammouri 已提交
386 387
}

388
static int orion_spi_reset(struct orion_spi *orion_spi)
S
Shadi Ammouri 已提交
389 390 391 392 393 394 395
{
	/* Verify that the CS is deasserted */
	orion_spi_set_cs(orion_spi, 0);

	return 0;
}

396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416
static const struct orion_spi_dev orion_spi_dev_data = {
	.typ = ORION_SPI,
	.min_divisor = 4,
	.max_divisor = 30,
	.prescale_mask = ORION_SPI_CLK_PRESCALE_MASK,
};

static const struct orion_spi_dev armada_spi_dev_data = {
	.typ = ARMADA_SPI,
	.min_divisor = 1,
	.max_divisor = 1920,
	.prescale_mask = ARMADA_SPI_CLK_PRESCALE_MASK,
};

static const struct of_device_id orion_spi_of_match_table[] = {
	{ .compatible = "marvell,orion-spi", .data = &orion_spi_dev_data, },
	{ .compatible = "marvell,armada-370-spi", .data = &armada_spi_dev_data, },
	{}
};
MODULE_DEVICE_TABLE(of, orion_spi_of_match_table);

417
static int orion_spi_probe(struct platform_device *pdev)
S
Shadi Ammouri 已提交
418
{
419 420
	const struct of_device_id *of_id;
	const struct orion_spi_dev *devdata;
S
Shadi Ammouri 已提交
421 422 423
	struct spi_master *master;
	struct orion_spi *spi;
	struct resource *r;
424
	unsigned long tclk_hz;
S
Shadi Ammouri 已提交
425 426
	int status = 0;

J
Jingoo Han 已提交
427
	master = spi_alloc_master(&pdev->dev, sizeof(*spi));
S
Shadi Ammouri 已提交
428 429 430 431 432 433 434
	if (master == NULL) {
		dev_dbg(&pdev->dev, "master allocation failed\n");
		return -ENOMEM;
	}

	if (pdev->id != -1)
		master->bus_num = pdev->id;
A
Andrew Lunn 已提交
435
	if (pdev->dev.of_node) {
436
		u32 cell_index;
J
Jingoo Han 已提交
437

438 439 440
		if (!of_property_read_u32(pdev->dev.of_node, "cell-index",
					  &cell_index))
			master->bus_num = cell_index;
A
Andrew Lunn 已提交
441
	}
S
Shadi Ammouri 已提交
442

443
	/* we support only mode 0, and no options */
444
	master->mode_bits = SPI_CPHA | SPI_CPOL;
445

446
	master->transfer_one_message = orion_spi_transfer_one_message;
S
Shadi Ammouri 已提交
447
	master->num_chipselect = ORION_NUM_CHIPSELECTS;
448
	master->bits_per_word_mask = SPI_BPW_MASK(8) | SPI_BPW_MASK(16);
449
	master->auto_runtime_pm = true;
S
Shadi Ammouri 已提交
450

451
	platform_set_drvdata(pdev, master);
S
Shadi Ammouri 已提交
452 453 454 455

	spi = spi_master_get_devdata(master);
	spi->master = master;

456 457 458 459
	of_id = of_match_device(orion_spi_of_match_table, &pdev->dev);
	devdata = of_id->data;
	spi->devdata = devdata;

J
Jingoo Han 已提交
460
	spi->clk = devm_clk_get(&pdev->dev, NULL);
461 462 463 464 465
	if (IS_ERR(spi->clk)) {
		status = PTR_ERR(spi->clk);
		goto out;
	}

466 467 468 469
	status = clk_prepare_enable(spi->clk);
	if (status)
		goto out;

470
	tclk_hz = clk_get_rate(spi->clk);
471 472
	master->max_speed_hz = DIV_ROUND_UP(tclk_hz, devdata->min_divisor);
	master->min_speed_hz = DIV_ROUND_UP(tclk_hz, devdata->max_divisor);
S
Shadi Ammouri 已提交
473 474

	r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
475 476 477
	spi->base = devm_ioremap_resource(&pdev->dev, r);
	if (IS_ERR(spi->base)) {
		status = PTR_ERR(spi->base);
478
		goto out_rel_clk;
S
Shadi Ammouri 已提交
479 480
	}

481 482 483 484 485
	pm_runtime_set_active(&pdev->dev);
	pm_runtime_use_autosuspend(&pdev->dev);
	pm_runtime_set_autosuspend_delay(&pdev->dev, SPI_AUTOSUSPEND_TIMEOUT);
	pm_runtime_enable(&pdev->dev);

486 487
	status = orion_spi_reset(spi);
	if (status < 0)
488 489 490 491
		goto out_rel_pm;

	pm_runtime_mark_last_busy(&pdev->dev);
	pm_runtime_put_autosuspend(&pdev->dev);
S
Shadi Ammouri 已提交
492

A
Andrew Lunn 已提交
493
	master->dev.of_node = pdev->dev.of_node;
494
	status = spi_register_master(master);
S
Shadi Ammouri 已提交
495
	if (status < 0)
496
		goto out_rel_pm;
S
Shadi Ammouri 已提交
497 498 499

	return status;

500 501
out_rel_pm:
	pm_runtime_disable(&pdev->dev);
502 503
out_rel_clk:
	clk_disable_unprepare(spi->clk);
S
Shadi Ammouri 已提交
504 505 506 507 508 509
out:
	spi_master_put(master);
	return status;
}


510
static int orion_spi_remove(struct platform_device *pdev)
S
Shadi Ammouri 已提交
511
{
512 513
	struct spi_master *master = platform_get_drvdata(pdev);
	struct orion_spi *spi = spi_master_get_devdata(master);
S
Shadi Ammouri 已提交
514

515
	pm_runtime_get_sync(&pdev->dev);
516 517
	clk_disable_unprepare(spi->clk);

518 519 520
	spi_unregister_master(master);
	pm_runtime_disable(&pdev->dev);

S
Shadi Ammouri 已提交
521 522 523 524 525
	return 0;
}

MODULE_ALIAS("platform:" DRIVER_NAME);

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
#ifdef CONFIG_PM_RUNTIME
static int orion_spi_runtime_suspend(struct device *dev)
{
	struct spi_master *master = dev_get_drvdata(dev);
	struct orion_spi *spi = spi_master_get_devdata(master);

	clk_disable_unprepare(spi->clk);
	return 0;
}

static int orion_spi_runtime_resume(struct device *dev)
{
	struct spi_master *master = dev_get_drvdata(dev);
	struct orion_spi *spi = spi_master_get_devdata(master);

	return clk_prepare_enable(spi->clk);
}
#endif

static const struct dev_pm_ops orion_spi_pm_ops = {
	SET_RUNTIME_PM_OPS(orion_spi_runtime_suspend,
			   orion_spi_runtime_resume,
			   NULL)
};

S
Shadi Ammouri 已提交
551 552 553 554
static struct platform_driver orion_spi_driver = {
	.driver = {
		.name	= DRIVER_NAME,
		.owner	= THIS_MODULE,
555
		.pm	= &orion_spi_pm_ops,
A
Andrew Lunn 已提交
556
		.of_match_table = of_match_ptr(orion_spi_of_match_table),
S
Shadi Ammouri 已提交
557
	},
558
	.probe		= orion_spi_probe,
559
	.remove		= orion_spi_remove,
S
Shadi Ammouri 已提交
560 561
};

562
module_platform_driver(orion_spi_driver);
S
Shadi Ammouri 已提交
563 564 565 566

MODULE_DESCRIPTION("Orion SPI driver");
MODULE_AUTHOR("Shadi Ammouri <shadi@marvell.com>");
MODULE_LICENSE("GPL");