623 lines
14 KiB
C
623 lines
14 KiB
C
/*
|
|
* Copyright (C) 2008 Sascha Hauer, Pengutronix
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of
|
|
* the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
*
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <init.h>
|
|
#include <driver.h>
|
|
#include <spi/spi.h>
|
|
#include <xfuncs.h>
|
|
#include <io.h>
|
|
#include <of.h>
|
|
#include <errno.h>
|
|
#include <malloc.h>
|
|
#include <gpio.h>
|
|
#include <of_gpio.h>
|
|
#include <mach/spi.h>
|
|
#include <mach/generic.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/err.h>
|
|
|
|
#define CSPI_0_0_RXDATA 0x00
|
|
#define CSPI_0_0_TXDATA 0x04
|
|
#define CSPI_0_0_CTRL 0x08
|
|
#define CSPI_0_0_INT 0x0C
|
|
#define CSPI_0_0_DMA 0x18
|
|
#define CSPI_0_0_STAT 0x0C
|
|
#define CSPI_0_0_PERIOD 0x14
|
|
#define CSPI_0_0_TEST 0x10
|
|
#define CSPI_0_0_RESET 0x1C
|
|
|
|
#define CSPI_0_0_CTRL_ENABLE (1 << 10)
|
|
#define CSPI_0_0_CTRL_MASTER (1 << 11)
|
|
#define CSPI_0_0_CTRL_XCH (1 << 9)
|
|
#define CSPI_0_0_CTRL_LOWPOL (1 << 5)
|
|
#define CSPI_0_0_CTRL_PHA (1 << 6)
|
|
#define CSPI_0_0_CTRL_SSCTL (1 << 7)
|
|
#define CSPI_0_0_CTRL_HIGHSSPOL (1 << 8)
|
|
#define CSPI_0_0_CTRL_CS(x) (((x) & 0x3) << 19)
|
|
#define CSPI_0_0_CTRL_BITCOUNT(x) (((x) & 0x1f) << 0)
|
|
#define CSPI_0_0_CTRL_DATARATE(x) (((x) & 0x7) << 14)
|
|
|
|
#define CSPI_0_0_CTRL_MAXDATRATE 0x10
|
|
#define CSPI_0_0_CTRL_DATAMASK 0x1F
|
|
#define CSPI_0_0_CTRL_DATASHIFT 14
|
|
|
|
#define CSPI_0_0_STAT_TE (1 << 0)
|
|
#define CSPI_0_0_STAT_TH (1 << 1)
|
|
#define CSPI_0_0_STAT_TF (1 << 2)
|
|
#define CSPI_0_0_STAT_RR (1 << 4)
|
|
#define CSPI_0_0_STAT_RH (1 << 5)
|
|
#define CSPI_0_0_STAT_RF (1 << 6)
|
|
#define CSPI_0_0_STAT_RO (1 << 7)
|
|
|
|
#define CSPI_0_0_PERIOD_32KHZ (1 << 15)
|
|
|
|
#define CSPI_0_0_TEST_LBC (1 << 14)
|
|
|
|
#define CSPI_0_0_RESET_START (1 << 0)
|
|
|
|
#define CSPI_0_7_RXDATA 0x00
|
|
#define CSPI_0_7_TXDATA 0x04
|
|
#define CSPI_0_7_CTRL 0x08
|
|
#define CSPI_0_7_CTRL_ENABLE (1 << 0)
|
|
#define CSPI_0_7_CTRL_MASTER (1 << 1)
|
|
#define CSPI_0_7_CTRL_XCH (1 << 2)
|
|
#define CSPI_0_7_CTRL_POL (1 << 4)
|
|
#define CSPI_0_7_CTRL_PHA (1 << 5)
|
|
#define CSPI_0_7_CTRL_SSCTL (1 << 6)
|
|
#define CSPI_0_7_CTRL_SSPOL (1 << 7)
|
|
#define CSPI_0_7_CTRL_CS_SHIFT 12
|
|
#define CSPI_0_7_CTRL_DR_SHIFT 16
|
|
#define CSPI_0_7_CTRL_BL_SHIFT 20
|
|
#define CSPI_0_7_STAT 0x14
|
|
#define CSPI_0_7_STAT_RR (1 << 3)
|
|
|
|
#define CSPI_2_3_RXDATA 0x00
|
|
#define CSPI_2_3_TXDATA 0x04
|
|
#define CSPI_2_3_CTRL 0x08
|
|
#define CSPI_2_3_CTRL_ENABLE (1 << 0)
|
|
#define CSPI_2_3_CTRL_XCH (1 << 2)
|
|
#define CSPI_2_3_CTRL_MODE(cs) (1 << ((cs) + 4))
|
|
#define CSPI_2_3_CTRL_POSTDIV_OFFSET 8
|
|
#define CSPI_2_3_CTRL_PREDIV_OFFSET 12
|
|
#define CSPI_2_3_CTRL_CS(cs) ((cs) << 18)
|
|
#define CSPI_2_3_CTRL_BL_OFFSET 20
|
|
|
|
#define CSPI_2_3_CONFIG 0x0c
|
|
#define CSPI_2_3_CONFIG_SCLKPHA(cs) (1 << ((cs) + 0))
|
|
#define CSPI_2_3_CONFIG_SCLKPOL(cs) (1 << ((cs) + 4))
|
|
#define CSPI_2_3_CONFIG_SBBCTRL(cs) (1 << ((cs) + 8))
|
|
#define CSPI_2_3_CONFIG_SSBPOL(cs) (1 << ((cs) + 12))
|
|
|
|
#define CSPI_2_3_INT 0x10
|
|
#define CSPI_2_3_INT_TEEN (1 << 0)
|
|
#define CSPI_2_3_INT_RREN (1 << 3)
|
|
|
|
#define CSPI_2_3_STAT 0x18
|
|
#define CSPI_2_3_STAT_RR (1 << 3)
|
|
|
|
struct imx_spi {
|
|
struct spi_master master;
|
|
int *cs_array;
|
|
void __iomem *regs;
|
|
struct clk *clk;
|
|
|
|
unsigned int (*xchg_single)(struct imx_spi *imx, u32 data);
|
|
void (*chipselect)(struct spi_device *spi, int active);
|
|
};
|
|
|
|
struct spi_imx_devtype_data {
|
|
unsigned int (*xchg_single)(struct imx_spi *imx, u32 data);
|
|
void (*chipselect)(struct spi_device *spi, int active);
|
|
void (*init)(struct imx_spi *imx);
|
|
};
|
|
|
|
static int imx_spi_setup(struct spi_device *spi)
|
|
{
|
|
struct imx_spi *imx = container_of(spi->master, struct imx_spi, master);
|
|
|
|
imx->chipselect(spi, 0);
|
|
|
|
debug("%s mode 0x%08x bits_per_word: %d speed: %d\n",
|
|
__FUNCTION__, spi->mode, spi->bits_per_word,
|
|
spi->max_speed_hz);
|
|
return 0;
|
|
}
|
|
|
|
static unsigned int imx_spi_maybe_reverse_bits(struct spi_device *spi, unsigned int word)
|
|
{
|
|
unsigned int result = word;
|
|
|
|
if (spi->mode & SPI_LSB_FIRST) {
|
|
size_t bits_left = spi->bits_per_word - 1;
|
|
|
|
for (word >>= 1; word; word >>= 1) {
|
|
result <<= 1;
|
|
result |= word & 1;
|
|
bits_left--;
|
|
}
|
|
|
|
result <<= bits_left;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
static unsigned int cspi_0_0_xchg_single(struct imx_spi *imx, unsigned int data)
|
|
{
|
|
void __iomem *base = imx->regs;
|
|
|
|
unsigned int cfg_reg = readl(base + CSPI_0_0_CTRL);
|
|
|
|
writel(data, base + CSPI_0_0_TXDATA);
|
|
|
|
cfg_reg |= CSPI_0_0_CTRL_XCH;
|
|
|
|
writel(cfg_reg, base + CSPI_0_0_CTRL);
|
|
|
|
while (!(readl(base + CSPI_0_0_INT) & CSPI_0_0_STAT_RR));
|
|
|
|
return readl(base + CSPI_0_0_RXDATA);
|
|
}
|
|
|
|
static void cspi_0_0_chipselect(struct spi_device *spi, int is_active)
|
|
{
|
|
struct spi_master *master = spi->master;
|
|
struct imx_spi *imx = container_of(master, struct imx_spi, master);
|
|
void __iomem *base = imx->regs;
|
|
unsigned int cs = 0;
|
|
int gpio = imx->cs_array[spi->chip_select];
|
|
u32 ctrl_reg;
|
|
|
|
if (spi->mode & SPI_CS_HIGH)
|
|
cs = 1;
|
|
|
|
if (!is_active) {
|
|
if (gpio >= 0)
|
|
gpio_direction_output(gpio, !cs);
|
|
return;
|
|
}
|
|
|
|
ctrl_reg = CSPI_0_0_CTRL_BITCOUNT(spi->bits_per_word - 1)
|
|
| CSPI_0_0_CTRL_DATARATE(7) /* FIXME: calculate data rate */
|
|
| CSPI_0_0_CTRL_ENABLE
|
|
| CSPI_0_0_CTRL_MASTER;
|
|
|
|
if (gpio < 0) {
|
|
ctrl_reg |= CSPI_0_0_CTRL_CS(gpio + 32);
|
|
}
|
|
|
|
if (spi->mode & SPI_CPHA)
|
|
ctrl_reg |= CSPI_0_0_CTRL_PHA;
|
|
if (spi->mode & SPI_CPOL)
|
|
ctrl_reg |= CSPI_0_0_CTRL_LOWPOL;
|
|
if (spi->mode & SPI_CS_HIGH)
|
|
ctrl_reg |= CSPI_0_0_CTRL_HIGHSSPOL;
|
|
|
|
writel(ctrl_reg, base + CSPI_0_0_CTRL);
|
|
|
|
if (gpio >= 0)
|
|
gpio_set_value(gpio, cs);
|
|
}
|
|
|
|
static void cspi_0_0_init(struct imx_spi *imx)
|
|
{
|
|
void __iomem *base = imx->regs;
|
|
|
|
writel(CSPI_0_0_RESET_START, base + CSPI_0_0_RESET);
|
|
do {
|
|
} while (readl(base + CSPI_0_0_RESET) & CSPI_0_0_RESET_START);
|
|
}
|
|
|
|
static unsigned int cspi_0_7_xchg_single(struct imx_spi *imx, unsigned int data)
|
|
{
|
|
void __iomem *base = imx->regs;
|
|
|
|
unsigned int cfg_reg = readl(base + CSPI_0_7_CTRL);
|
|
|
|
writel(data, base + CSPI_0_7_TXDATA);
|
|
|
|
cfg_reg |= CSPI_0_7_CTRL_XCH;
|
|
|
|
writel(cfg_reg, base + CSPI_0_7_CTRL);
|
|
|
|
while (!(readl(base + CSPI_0_7_STAT) & CSPI_0_7_STAT_RR))
|
|
;
|
|
|
|
return readl(base + CSPI_0_7_RXDATA);
|
|
}
|
|
|
|
/* MX1, MX31, MX35, MX51 CSPI */
|
|
static unsigned int spi_imx_clkdiv_2(unsigned int fin,
|
|
unsigned int fspi)
|
|
{
|
|
int i, div = 4;
|
|
|
|
for (i = 0; i < 7; i++) {
|
|
if (fspi * div >= fin)
|
|
return i;
|
|
div <<= 1;
|
|
}
|
|
|
|
return 7;
|
|
}
|
|
|
|
static void cspi_0_7_chipselect(struct spi_device *spi, int is_active)
|
|
{
|
|
struct spi_master *master = spi->master;
|
|
struct imx_spi *imx = container_of(master, struct imx_spi, master);
|
|
void __iomem *base = imx->regs;
|
|
unsigned int cs = 0;
|
|
int gpio = imx->cs_array[spi->chip_select];
|
|
unsigned int reg = CSPI_0_7_CTRL_ENABLE | CSPI_0_7_CTRL_MASTER;
|
|
|
|
if (spi->mode & SPI_CS_HIGH)
|
|
cs = 1;
|
|
|
|
if (!is_active) {
|
|
if (gpio >= 0)
|
|
gpio_direction_output(gpio, !cs);
|
|
return;
|
|
}
|
|
|
|
reg |= spi_imx_clkdiv_2(clk_get_rate(imx->clk), spi->max_speed_hz) <<
|
|
CSPI_0_7_CTRL_DR_SHIFT;
|
|
|
|
reg |= (spi->bits_per_word - 1) << CSPI_0_7_CTRL_BL_SHIFT;
|
|
reg |= CSPI_0_7_CTRL_SSCTL;
|
|
|
|
if (spi->mode & SPI_CPHA)
|
|
reg |= CSPI_0_7_CTRL_PHA;
|
|
if (spi->mode & SPI_CPOL)
|
|
reg |= CSPI_0_7_CTRL_POL;
|
|
if (spi->mode & SPI_CS_HIGH)
|
|
reg |= CSPI_0_7_CTRL_SSPOL;
|
|
if (gpio < 0)
|
|
reg |= (gpio + 32) << CSPI_0_7_CTRL_CS_SHIFT;
|
|
|
|
writel(reg, base + CSPI_0_7_CTRL);
|
|
|
|
if (gpio >= 0)
|
|
gpio_set_value(gpio, cs);
|
|
}
|
|
|
|
static void cspi_0_7_init(struct imx_spi *imx)
|
|
{
|
|
void __iomem *base = imx->regs;
|
|
|
|
/* drain receive buffer */
|
|
while (readl(base + CSPI_0_7_STAT) & CSPI_0_7_STAT_RR)
|
|
readl(base + CSPI_0_7_RXDATA);
|
|
}
|
|
|
|
static unsigned int cspi_2_3_xchg_single(struct imx_spi *imx, unsigned int data)
|
|
{
|
|
void __iomem *base = imx->regs;
|
|
|
|
unsigned int cfg_reg = readl(base + CSPI_2_3_CTRL);
|
|
|
|
writel(data, base + CSPI_2_3_TXDATA);
|
|
|
|
cfg_reg |= CSPI_2_3_CTRL_XCH;
|
|
|
|
writel(cfg_reg, base + CSPI_2_3_CTRL);
|
|
|
|
while (!(readl(base + CSPI_2_3_STAT) & CSPI_2_3_STAT_RR));
|
|
|
|
return readl(base + CSPI_2_3_RXDATA);
|
|
}
|
|
|
|
static unsigned int cspi_2_3_clkdiv(unsigned int fin, unsigned int fspi)
|
|
{
|
|
/*
|
|
* there are two 4-bit dividers, the pre-divider divides by
|
|
* $pre, the post-divider by 2^$post
|
|
*/
|
|
unsigned int pre, post;
|
|
|
|
if (unlikely(fspi > fin))
|
|
return 0;
|
|
|
|
post = fls(fin) - fls(fspi);
|
|
if (fin > fspi << post)
|
|
post++;
|
|
|
|
/* now we have: (fin <= fspi << post) with post being minimal */
|
|
|
|
post = max(4U, post) - 4;
|
|
if (unlikely(post > 0xf)) {
|
|
pr_err("%s: cannot set clock freq: %u (base freq: %u)\n",
|
|
__func__, fspi, fin);
|
|
return 0xff;
|
|
}
|
|
|
|
pre = DIV_ROUND_UP(fin, fspi << post) - 1;
|
|
|
|
pr_debug("%s: fin: %u, fspi: %u, post: %u, pre: %u\n",
|
|
__func__, fin, fspi, post, pre);
|
|
return (pre << CSPI_2_3_CTRL_PREDIV_OFFSET) |
|
|
(post << CSPI_2_3_CTRL_POSTDIV_OFFSET);
|
|
}
|
|
|
|
static void cspi_2_3_chipselect(struct spi_device *spi, int is_active)
|
|
{
|
|
struct spi_master *master = spi->master;
|
|
struct imx_spi *imx = container_of(master, struct imx_spi, master);
|
|
void __iomem *base = imx->regs;
|
|
unsigned int cs = spi->chip_select, gpio_cs = 0;
|
|
int gpio = imx->cs_array[spi->chip_select];
|
|
u32 ctrl, cfg = 0;
|
|
|
|
if (spi->mode & SPI_CS_HIGH)
|
|
gpio_cs = 1;
|
|
|
|
if (!is_active) {
|
|
if (gpio >= 0)
|
|
gpio_direction_output(gpio, !gpio_cs);
|
|
return;
|
|
}
|
|
|
|
ctrl = CSPI_2_3_CTRL_ENABLE;
|
|
|
|
/* set master mode */
|
|
ctrl |= CSPI_2_3_CTRL_MODE(cs);
|
|
|
|
/* set clock speed */
|
|
ctrl |= cspi_2_3_clkdiv(clk_get_rate(imx->clk), spi->max_speed_hz);
|
|
|
|
/* set chip select to use */
|
|
ctrl |= CSPI_2_3_CTRL_CS(cs);
|
|
|
|
ctrl |= (spi->bits_per_word - 1) << CSPI_2_3_CTRL_BL_OFFSET;
|
|
|
|
cfg |= CSPI_2_3_CONFIG_SBBCTRL(cs);
|
|
|
|
if (spi->mode & SPI_CPHA)
|
|
cfg |= CSPI_2_3_CONFIG_SCLKPHA(cs);
|
|
|
|
if (spi->mode & SPI_CPOL)
|
|
cfg |= CSPI_2_3_CONFIG_SCLKPOL(cs);
|
|
|
|
if (spi->mode & SPI_CS_HIGH)
|
|
cfg |= CSPI_2_3_CONFIG_SSBPOL(cs);
|
|
|
|
writel(ctrl, base + CSPI_2_3_CTRL);
|
|
writel(cfg, base + CSPI_2_3_CONFIG);
|
|
|
|
if (gpio >= 0)
|
|
gpio_set_value(gpio, gpio_cs);
|
|
}
|
|
|
|
static u32 imx_xchg_single(struct spi_device *spi, u32 tx_val)
|
|
{
|
|
u32 rx_val;
|
|
struct imx_spi *imx = container_of(spi->master, struct imx_spi, master);
|
|
|
|
|
|
tx_val = imx_spi_maybe_reverse_bits(spi, tx_val);
|
|
rx_val = imx->xchg_single(imx, tx_val);
|
|
|
|
return imx_spi_maybe_reverse_bits(spi, rx_val);
|
|
}
|
|
|
|
static void imx_spi_do_transfer(struct spi_device *spi, struct spi_transfer *t)
|
|
{
|
|
unsigned i;
|
|
|
|
if (spi->bits_per_word <= 8) {
|
|
const u8 *tx_buf = t->tx_buf;
|
|
u8 *rx_buf = t->rx_buf;
|
|
u8 rx_val;
|
|
|
|
for (i = 0; i < t->len; i++) {
|
|
rx_val = imx_xchg_single(spi, tx_buf ? tx_buf[i] : 0);
|
|
|
|
if (rx_buf)
|
|
rx_buf[i] = rx_val;
|
|
}
|
|
} else if (spi->bits_per_word <= 16) {
|
|
const u16 *tx_buf = t->tx_buf;
|
|
u16 *rx_buf = t->rx_buf;
|
|
u16 rx_val;
|
|
|
|
for (i = 0; i < t->len >> 1; i++) {
|
|
rx_val = imx_xchg_single(spi, tx_buf ? tx_buf[i] : 0);
|
|
|
|
if (rx_buf)
|
|
rx_buf[i] = rx_val;
|
|
}
|
|
} else if (spi->bits_per_word <= 32) {
|
|
const u32 *tx_buf = t->tx_buf;
|
|
u32 *rx_buf = t->rx_buf;
|
|
u32 rx_val;
|
|
|
|
for (i = 0; i < t->len >> 2; i++) {
|
|
rx_val = imx_xchg_single(spi, tx_buf ? tx_buf[i] : 0);
|
|
|
|
if (rx_buf)
|
|
rx_buf[i] = rx_val;
|
|
}
|
|
}
|
|
}
|
|
|
|
static int imx_spi_transfer(struct spi_device *spi, struct spi_message *mesg)
|
|
{
|
|
struct imx_spi *imx = container_of(spi->master, struct imx_spi, master);
|
|
struct spi_transfer *t;
|
|
|
|
imx->chipselect(spi, 1);
|
|
|
|
mesg->actual_length = 0;
|
|
|
|
list_for_each_entry(t, &mesg->transfers, transfer_list) {
|
|
imx_spi_do_transfer(spi, t);
|
|
mesg->actual_length += t->len;
|
|
}
|
|
|
|
imx->chipselect(spi, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static __maybe_unused struct spi_imx_devtype_data spi_imx_devtype_data_0_0 = {
|
|
.chipselect = cspi_0_0_chipselect,
|
|
.xchg_single = cspi_0_0_xchg_single,
|
|
.init = cspi_0_0_init,
|
|
};
|
|
|
|
static __maybe_unused struct spi_imx_devtype_data spi_imx_devtype_data_0_7 = {
|
|
.chipselect = cspi_0_7_chipselect,
|
|
.xchg_single = cspi_0_7_xchg_single,
|
|
.init = cspi_0_7_init,
|
|
};
|
|
|
|
static __maybe_unused struct spi_imx_devtype_data spi_imx_devtype_data_2_3 = {
|
|
.chipselect = cspi_2_3_chipselect,
|
|
.xchg_single = cspi_2_3_xchg_single,
|
|
};
|
|
|
|
static int imx_spi_dt_probe(struct imx_spi *imx)
|
|
{
|
|
struct device_node *node = imx->master.dev->device_node;
|
|
int ret, i;
|
|
u32 num_cs;
|
|
|
|
if (!node)
|
|
return -ENODEV;
|
|
|
|
ret = of_property_read_u32(node, "fsl,spi-num-chipselects", &num_cs);
|
|
if (ret)
|
|
return ret;
|
|
|
|
imx->master.num_chipselect = num_cs;
|
|
imx->cs_array = xzalloc(sizeof(u32) * num_cs);
|
|
|
|
for (i = 0; i < num_cs; i++) {
|
|
int cs_gpio = of_get_named_gpio(node, "cs-gpios", i);
|
|
imx->cs_array[i] = cs_gpio;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int imx_spi_probe(struct device_d *dev)
|
|
{
|
|
struct spi_master *master;
|
|
struct imx_spi *imx;
|
|
struct spi_imx_master *pdata = dev->platform_data;
|
|
struct spi_imx_devtype_data *devdata = NULL;
|
|
int ret;
|
|
|
|
ret = dev_get_drvdata(dev, (unsigned long *)&devdata);
|
|
if (ret)
|
|
return -ENODEV;
|
|
|
|
imx = xzalloc(sizeof(*imx));
|
|
|
|
master = &imx->master;
|
|
master->dev = dev;
|
|
master->bus_num = dev->id;
|
|
|
|
master->setup = imx_spi_setup;
|
|
master->transfer = imx_spi_transfer;
|
|
|
|
if (pdata) {
|
|
master->num_chipselect = pdata->num_chipselect;
|
|
imx->cs_array = pdata->chipselect;
|
|
} else {
|
|
if (IS_ENABLED(CONFIG_OFDEVICE))
|
|
imx_spi_dt_probe(imx);
|
|
}
|
|
|
|
imx->clk = clk_get(dev, NULL);
|
|
if (IS_ERR(imx->clk)) {
|
|
ret = PTR_ERR(imx->clk);
|
|
goto err_free;
|
|
}
|
|
|
|
imx->chipselect = devdata->chipselect;
|
|
imx->xchg_single = devdata->xchg_single;
|
|
imx->regs = dev_request_mem_region(dev, 0);
|
|
|
|
if (devdata->init)
|
|
devdata->init(imx);
|
|
|
|
spi_register_master(master);
|
|
|
|
return 0;
|
|
|
|
err_free:
|
|
free(imx);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static __maybe_unused struct of_device_id imx_spi_dt_ids[] = {
|
|
#if IS_ENABLED(CONFIG_DRIVER_SPI_IMX_0_0)
|
|
{
|
|
.compatible = "fsl,imx27-cspi",
|
|
.data = (unsigned long)&spi_imx_devtype_data_0_0,
|
|
},
|
|
#endif
|
|
#if IS_ENABLED(CONFIG_DRIVER_SPI_IMX_0_7)
|
|
{
|
|
.compatible = "fsl,imx35-cspi",
|
|
.data = (unsigned long)&spi_imx_devtype_data_0_7,
|
|
},
|
|
#endif
|
|
#if IS_ENABLED(CONFIG_DRIVER_SPI_IMX_2_3)
|
|
{
|
|
.compatible = "fsl,imx51-ecspi",
|
|
.data = (unsigned long)&spi_imx_devtype_data_2_3,
|
|
},
|
|
#endif
|
|
{
|
|
/* sentinel */
|
|
}
|
|
};
|
|
|
|
static struct platform_device_id imx_spi_ids[] = {
|
|
#if IS_ENABLED(CONFIG_DRIVER_SPI_IMX_0_0)
|
|
{
|
|
.name = "imx27-spi",
|
|
.driver_data = (unsigned long)&spi_imx_devtype_data_0_0,
|
|
},
|
|
#endif
|
|
#if IS_ENABLED(CONFIG_DRIVER_SPI_IMX_0_7)
|
|
{
|
|
.name = "imx35-spi",
|
|
.driver_data = (unsigned long)&spi_imx_devtype_data_0_7,
|
|
},
|
|
#endif
|
|
#if IS_ENABLED(CONFIG_DRIVER_SPI_IMX_2_3)
|
|
{
|
|
.name = "imx51-spi",
|
|
.driver_data = (unsigned long)&spi_imx_devtype_data_2_3,
|
|
},
|
|
#endif
|
|
{
|
|
/* sentinel */
|
|
}
|
|
};
|
|
|
|
static struct driver_d imx_spi_driver = {
|
|
.name = "imx_spi",
|
|
.probe = imx_spi_probe,
|
|
.of_compatible = DRV_OF_COMPAT(imx_spi_dt_ids),
|
|
.id_table = imx_spi_ids,
|
|
};
|
|
coredevice_platform_driver(imx_spi_driver);
|