Merge remote-tracking branch 'spi/topic/ep93xx' into spi-next
This commit is contained in:
commit
497cb4473a
1 changed files with 82 additions and 273 deletions
|
@ -26,7 +26,6 @@
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/platform_device.h>
|
#include <linux/platform_device.h>
|
||||||
#include <linux/workqueue.h>
|
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
#include <linux/scatterlist.h>
|
#include <linux/scatterlist.h>
|
||||||
#include <linux/spi/spi.h>
|
#include <linux/spi/spi.h>
|
||||||
|
@ -70,19 +69,13 @@
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct ep93xx_spi - EP93xx SPI controller structure
|
* struct ep93xx_spi - EP93xx SPI controller structure
|
||||||
* @lock: spinlock that protects concurrent accesses to fields @running,
|
|
||||||
* @current_msg and @msg_queue
|
|
||||||
* @pdev: pointer to platform device
|
* @pdev: pointer to platform device
|
||||||
* @clk: clock for the controller
|
* @clk: clock for the controller
|
||||||
* @regs_base: pointer to ioremap()'d registers
|
* @regs_base: pointer to ioremap()'d registers
|
||||||
* @sspdr_phys: physical address of the SSPDR register
|
* @sspdr_phys: physical address of the SSPDR register
|
||||||
* @min_rate: minimum clock rate (in Hz) supported by the controller
|
* @min_rate: minimum clock rate (in Hz) supported by the controller
|
||||||
* @max_rate: maximum clock rate (in Hz) supported by the controller
|
* @max_rate: maximum clock rate (in Hz) supported by the controller
|
||||||
* @running: is the queue running
|
|
||||||
* @wq: workqueue used by the driver
|
|
||||||
* @msg_work: work that is queued for the driver
|
|
||||||
* @wait: wait here until given transfer is completed
|
* @wait: wait here until given transfer is completed
|
||||||
* @msg_queue: queue for the messages
|
|
||||||
* @current_msg: message that is currently processed (or %NULL if none)
|
* @current_msg: message that is currently processed (or %NULL if none)
|
||||||
* @tx: current byte in transfer to transmit
|
* @tx: current byte in transfer to transmit
|
||||||
* @rx: current byte in transfer to receive
|
* @rx: current byte in transfer to receive
|
||||||
|
@ -96,30 +89,15 @@
|
||||||
* @tx_sgt: sg table for TX transfers
|
* @tx_sgt: sg table for TX transfers
|
||||||
* @zeropage: dummy page used as RX buffer when only TX buffer is passed in by
|
* @zeropage: dummy page used as RX buffer when only TX buffer is passed in by
|
||||||
* the client
|
* the client
|
||||||
*
|
|
||||||
* This structure holds EP93xx SPI controller specific information. When
|
|
||||||
* @running is %true, driver accepts transfer requests from protocol drivers.
|
|
||||||
* @current_msg is used to hold pointer to the message that is currently
|
|
||||||
* processed. If @current_msg is %NULL, it means that no processing is going
|
|
||||||
* on.
|
|
||||||
*
|
|
||||||
* Most of the fields are only written once and they can be accessed without
|
|
||||||
* taking the @lock. Fields that are accessed concurrently are: @current_msg,
|
|
||||||
* @running, and @msg_queue.
|
|
||||||
*/
|
*/
|
||||||
struct ep93xx_spi {
|
struct ep93xx_spi {
|
||||||
spinlock_t lock;
|
|
||||||
const struct platform_device *pdev;
|
const struct platform_device *pdev;
|
||||||
struct clk *clk;
|
struct clk *clk;
|
||||||
void __iomem *regs_base;
|
void __iomem *regs_base;
|
||||||
unsigned long sspdr_phys;
|
unsigned long sspdr_phys;
|
||||||
unsigned long min_rate;
|
unsigned long min_rate;
|
||||||
unsigned long max_rate;
|
unsigned long max_rate;
|
||||||
bool running;
|
|
||||||
struct workqueue_struct *wq;
|
|
||||||
struct work_struct msg_work;
|
|
||||||
struct completion wait;
|
struct completion wait;
|
||||||
struct list_head msg_queue;
|
|
||||||
struct spi_message *current_msg;
|
struct spi_message *current_msg;
|
||||||
size_t tx;
|
size_t tx;
|
||||||
size_t rx;
|
size_t rx;
|
||||||
|
@ -136,50 +114,36 @@ struct ep93xx_spi {
|
||||||
/**
|
/**
|
||||||
* struct ep93xx_spi_chip - SPI device hardware settings
|
* struct ep93xx_spi_chip - SPI device hardware settings
|
||||||
* @spi: back pointer to the SPI device
|
* @spi: back pointer to the SPI device
|
||||||
* @rate: max rate in hz this chip supports
|
|
||||||
* @div_cpsr: cpsr (pre-scaler) divider
|
|
||||||
* @div_scr: scr divider
|
|
||||||
* @dss: bits per word (4 - 16 bits)
|
|
||||||
* @ops: private chip operations
|
* @ops: private chip operations
|
||||||
*
|
|
||||||
* This structure is used to store hardware register specific settings for each
|
|
||||||
* SPI device. Settings are written to hardware by function
|
|
||||||
* ep93xx_spi_chip_setup().
|
|
||||||
*/
|
*/
|
||||||
struct ep93xx_spi_chip {
|
struct ep93xx_spi_chip {
|
||||||
const struct spi_device *spi;
|
const struct spi_device *spi;
|
||||||
unsigned long rate;
|
|
||||||
u8 div_cpsr;
|
|
||||||
u8 div_scr;
|
|
||||||
u8 dss;
|
|
||||||
struct ep93xx_spi_chip_ops *ops;
|
struct ep93xx_spi_chip_ops *ops;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* converts bits per word to CR0.DSS value */
|
/* converts bits per word to CR0.DSS value */
|
||||||
#define bits_per_word_to_dss(bpw) ((bpw) - 1)
|
#define bits_per_word_to_dss(bpw) ((bpw) - 1)
|
||||||
|
|
||||||
static inline void
|
static void ep93xx_spi_write_u8(const struct ep93xx_spi *espi,
|
||||||
ep93xx_spi_write_u8(const struct ep93xx_spi *espi, u16 reg, u8 value)
|
u16 reg, u8 value)
|
||||||
{
|
{
|
||||||
__raw_writeb(value, espi->regs_base + reg);
|
writeb(value, espi->regs_base + reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline u8
|
static u8 ep93xx_spi_read_u8(const struct ep93xx_spi *spi, u16 reg)
|
||||||
ep93xx_spi_read_u8(const struct ep93xx_spi *spi, u16 reg)
|
|
||||||
{
|
{
|
||||||
return __raw_readb(spi->regs_base + reg);
|
return readb(spi->regs_base + reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void
|
static void ep93xx_spi_write_u16(const struct ep93xx_spi *espi,
|
||||||
ep93xx_spi_write_u16(const struct ep93xx_spi *espi, u16 reg, u16 value)
|
u16 reg, u16 value)
|
||||||
{
|
{
|
||||||
__raw_writew(value, espi->regs_base + reg);
|
writew(value, espi->regs_base + reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline u16
|
static u16 ep93xx_spi_read_u16(const struct ep93xx_spi *spi, u16 reg)
|
||||||
ep93xx_spi_read_u16(const struct ep93xx_spi *spi, u16 reg)
|
|
||||||
{
|
{
|
||||||
return __raw_readw(spi->regs_base + reg);
|
return readw(spi->regs_base + reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ep93xx_spi_enable(const struct ep93xx_spi *espi)
|
static int ep93xx_spi_enable(const struct ep93xx_spi *espi)
|
||||||
|
@ -230,17 +194,13 @@ static void ep93xx_spi_disable_interrupts(const struct ep93xx_spi *espi)
|
||||||
/**
|
/**
|
||||||
* ep93xx_spi_calc_divisors() - calculates SPI clock divisors
|
* ep93xx_spi_calc_divisors() - calculates SPI clock divisors
|
||||||
* @espi: ep93xx SPI controller struct
|
* @espi: ep93xx SPI controller struct
|
||||||
* @chip: divisors are calculated for this chip
|
|
||||||
* @rate: desired SPI output clock rate
|
* @rate: desired SPI output clock rate
|
||||||
*
|
* @div_cpsr: pointer to return the cpsr (pre-scaler) divider
|
||||||
* Function calculates cpsr (clock pre-scaler) and scr divisors based on
|
* @div_scr: pointer to return the scr divider
|
||||||
* given @rate and places them to @chip->div_cpsr and @chip->div_scr. If,
|
|
||||||
* for some reason, divisors cannot be calculated nothing is stored and
|
|
||||||
* %-EINVAL is returned.
|
|
||||||
*/
|
*/
|
||||||
static int ep93xx_spi_calc_divisors(const struct ep93xx_spi *espi,
|
static int ep93xx_spi_calc_divisors(const struct ep93xx_spi *espi,
|
||||||
struct ep93xx_spi_chip *chip,
|
unsigned long rate,
|
||||||
unsigned long rate)
|
u8 *div_cpsr, u8 *div_scr)
|
||||||
{
|
{
|
||||||
unsigned long spi_clk_rate = clk_get_rate(espi->clk);
|
unsigned long spi_clk_rate = clk_get_rate(espi->clk);
|
||||||
int cpsr, scr;
|
int cpsr, scr;
|
||||||
|
@ -248,7 +208,7 @@ static int ep93xx_spi_calc_divisors(const struct ep93xx_spi *espi,
|
||||||
/*
|
/*
|
||||||
* Make sure that max value is between values supported by the
|
* Make sure that max value is between values supported by the
|
||||||
* controller. Note that minimum value is already checked in
|
* controller. Note that minimum value is already checked in
|
||||||
* ep93xx_spi_transfer().
|
* ep93xx_spi_transfer_one_message().
|
||||||
*/
|
*/
|
||||||
rate = clamp(rate, espi->min_rate, espi->max_rate);
|
rate = clamp(rate, espi->min_rate, espi->max_rate);
|
||||||
|
|
||||||
|
@ -263,8 +223,8 @@ static int ep93xx_spi_calc_divisors(const struct ep93xx_spi *espi,
|
||||||
for (cpsr = 2; cpsr <= 254; cpsr += 2) {
|
for (cpsr = 2; cpsr <= 254; cpsr += 2) {
|
||||||
for (scr = 0; scr <= 255; scr++) {
|
for (scr = 0; scr <= 255; scr++) {
|
||||||
if ((spi_clk_rate / (cpsr * (scr + 1))) <= rate) {
|
if ((spi_clk_rate / (cpsr * (scr + 1))) <= rate) {
|
||||||
chip->div_scr = (u8)scr;
|
*div_scr = (u8)scr;
|
||||||
chip->div_cpsr = (u8)cpsr;
|
*div_cpsr = (u8)cpsr;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -319,72 +279,10 @@ static int ep93xx_spi_setup(struct spi_device *spi)
|
||||||
spi_set_ctldata(spi, chip);
|
spi_set_ctldata(spi, chip);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (spi->max_speed_hz != chip->rate) {
|
|
||||||
int err;
|
|
||||||
|
|
||||||
err = ep93xx_spi_calc_divisors(espi, chip, spi->max_speed_hz);
|
|
||||||
if (err != 0) {
|
|
||||||
spi_set_ctldata(spi, NULL);
|
|
||||||
kfree(chip);
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
chip->rate = spi->max_speed_hz;
|
|
||||||
}
|
|
||||||
|
|
||||||
chip->dss = bits_per_word_to_dss(spi->bits_per_word);
|
|
||||||
|
|
||||||
ep93xx_spi_cs_control(spi, false);
|
ep93xx_spi_cs_control(spi, false);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* ep93xx_spi_transfer() - queue message to be transferred
|
|
||||||
* @spi: target SPI device
|
|
||||||
* @msg: message to be transferred
|
|
||||||
*
|
|
||||||
* This function is called by SPI device drivers when they are going to transfer
|
|
||||||
* a new message. It simply puts the message in the queue and schedules
|
|
||||||
* workqueue to perform the actual transfer later on.
|
|
||||||
*
|
|
||||||
* Returns %0 on success and negative error in case of failure.
|
|
||||||
*/
|
|
||||||
static int ep93xx_spi_transfer(struct spi_device *spi, struct spi_message *msg)
|
|
||||||
{
|
|
||||||
struct ep93xx_spi *espi = spi_master_get_devdata(spi->master);
|
|
||||||
struct spi_transfer *t;
|
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
if (!msg || !msg->complete)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
/* first validate each transfer */
|
|
||||||
list_for_each_entry(t, &msg->transfers, transfer_list) {
|
|
||||||
if (t->speed_hz && t->speed_hz < espi->min_rate)
|
|
||||||
return -EINVAL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Now that we own the message, let's initialize it so that it is
|
|
||||||
* suitable for us. We use @msg->status to signal whether there was
|
|
||||||
* error in transfer and @msg->state is used to hold pointer to the
|
|
||||||
* current transfer (or %NULL if no active current transfer).
|
|
||||||
*/
|
|
||||||
msg->state = NULL;
|
|
||||||
msg->status = 0;
|
|
||||||
msg->actual_length = 0;
|
|
||||||
|
|
||||||
spin_lock_irqsave(&espi->lock, flags);
|
|
||||||
if (!espi->running) {
|
|
||||||
spin_unlock_irqrestore(&espi->lock, flags);
|
|
||||||
return -ESHUTDOWN;
|
|
||||||
}
|
|
||||||
list_add_tail(&msg->queue, &espi->msg_queue);
|
|
||||||
queue_work(espi->wq, &espi->msg_work);
|
|
||||||
spin_unlock_irqrestore(&espi->lock, flags);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ep93xx_spi_cleanup() - cleans up master controller specific state
|
* ep93xx_spi_cleanup() - cleans up master controller specific state
|
||||||
* @spi: SPI device to cleanup
|
* @spi: SPI device to cleanup
|
||||||
|
@ -409,39 +307,40 @@ static void ep93xx_spi_cleanup(struct spi_device *spi)
|
||||||
* ep93xx_spi_chip_setup() - configures hardware according to given @chip
|
* ep93xx_spi_chip_setup() - configures hardware according to given @chip
|
||||||
* @espi: ep93xx SPI controller struct
|
* @espi: ep93xx SPI controller struct
|
||||||
* @chip: chip specific settings
|
* @chip: chip specific settings
|
||||||
*
|
* @speed_hz: transfer speed
|
||||||
* This function sets up the actual hardware registers with settings given in
|
* @bits_per_word: transfer bits_per_word
|
||||||
* @chip. Note that no validation is done so make sure that callers validate
|
|
||||||
* settings before calling this.
|
|
||||||
*/
|
*/
|
||||||
static void ep93xx_spi_chip_setup(const struct ep93xx_spi *espi,
|
static int ep93xx_spi_chip_setup(const struct ep93xx_spi *espi,
|
||||||
const struct ep93xx_spi_chip *chip)
|
const struct ep93xx_spi_chip *chip,
|
||||||
|
u32 speed_hz, u8 bits_per_word)
|
||||||
{
|
{
|
||||||
|
u8 dss = bits_per_word_to_dss(bits_per_word);
|
||||||
|
u8 div_cpsr = 0;
|
||||||
|
u8 div_scr = 0;
|
||||||
u16 cr0;
|
u16 cr0;
|
||||||
|
int err;
|
||||||
|
|
||||||
cr0 = chip->div_scr << SSPCR0_SCR_SHIFT;
|
err = ep93xx_spi_calc_divisors(espi, speed_hz, &div_cpsr, &div_scr);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
|
||||||
|
cr0 = div_scr << SSPCR0_SCR_SHIFT;
|
||||||
cr0 |= (chip->spi->mode & (SPI_CPHA|SPI_CPOL)) << SSPCR0_MODE_SHIFT;
|
cr0 |= (chip->spi->mode & (SPI_CPHA|SPI_CPOL)) << SSPCR0_MODE_SHIFT;
|
||||||
cr0 |= chip->dss;
|
cr0 |= dss;
|
||||||
|
|
||||||
dev_dbg(&espi->pdev->dev, "setup: mode %d, cpsr %d, scr %d, dss %d\n",
|
dev_dbg(&espi->pdev->dev, "setup: mode %d, cpsr %d, scr %d, dss %d\n",
|
||||||
chip->spi->mode, chip->div_cpsr, chip->div_scr, chip->dss);
|
chip->spi->mode, div_cpsr, div_scr, dss);
|
||||||
dev_dbg(&espi->pdev->dev, "setup: cr0 %#x", cr0);
|
dev_dbg(&espi->pdev->dev, "setup: cr0 %#x", cr0);
|
||||||
|
|
||||||
ep93xx_spi_write_u8(espi, SSPCPSR, chip->div_cpsr);
|
ep93xx_spi_write_u8(espi, SSPCPSR, div_cpsr);
|
||||||
ep93xx_spi_write_u16(espi, SSPCR0, cr0);
|
ep93xx_spi_write_u16(espi, SSPCR0, cr0);
|
||||||
}
|
|
||||||
|
|
||||||
static inline int bits_per_word(const struct ep93xx_spi *espi)
|
return 0;
|
||||||
{
|
|
||||||
struct spi_message *msg = espi->current_msg;
|
|
||||||
struct spi_transfer *t = msg->state;
|
|
||||||
|
|
||||||
return t->bits_per_word;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ep93xx_do_write(struct ep93xx_spi *espi, struct spi_transfer *t)
|
static void ep93xx_do_write(struct ep93xx_spi *espi, struct spi_transfer *t)
|
||||||
{
|
{
|
||||||
if (bits_per_word(espi) > 8) {
|
if (t->bits_per_word > 8) {
|
||||||
u16 tx_val = 0;
|
u16 tx_val = 0;
|
||||||
|
|
||||||
if (t->tx_buf)
|
if (t->tx_buf)
|
||||||
|
@ -460,7 +359,7 @@ static void ep93xx_do_write(struct ep93xx_spi *espi, struct spi_transfer *t)
|
||||||
|
|
||||||
static void ep93xx_do_read(struct ep93xx_spi *espi, struct spi_transfer *t)
|
static void ep93xx_do_read(struct ep93xx_spi *espi, struct spi_transfer *t)
|
||||||
{
|
{
|
||||||
if (bits_per_word(espi) > 8) {
|
if (t->bits_per_word > 8) {
|
||||||
u16 rx_val;
|
u16 rx_val;
|
||||||
|
|
||||||
rx_val = ep93xx_spi_read_u16(espi, SSPDR);
|
rx_val = ep93xx_spi_read_u16(espi, SSPDR);
|
||||||
|
@ -546,7 +445,7 @@ ep93xx_spi_dma_prepare(struct ep93xx_spi *espi, enum dma_transfer_direction dir)
|
||||||
size_t len = t->len;
|
size_t len = t->len;
|
||||||
int i, ret, nents;
|
int i, ret, nents;
|
||||||
|
|
||||||
if (bits_per_word(espi) > 8)
|
if (t->bits_per_word > 8)
|
||||||
buswidth = DMA_SLAVE_BUSWIDTH_2_BYTES;
|
buswidth = DMA_SLAVE_BUSWIDTH_2_BYTES;
|
||||||
else
|
else
|
||||||
buswidth = DMA_SLAVE_BUSWIDTH_1_BYTE;
|
buswidth = DMA_SLAVE_BUSWIDTH_1_BYTE;
|
||||||
|
@ -610,7 +509,7 @@ ep93xx_spi_dma_prepare(struct ep93xx_spi *espi, enum dma_transfer_direction dir)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (WARN_ON(len)) {
|
if (WARN_ON(len)) {
|
||||||
dev_warn(&espi->pdev->dev, "len = %d expected 0!", len);
|
dev_warn(&espi->pdev->dev, "len = %zu expected 0!", len);
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -708,38 +607,17 @@ static void ep93xx_spi_process_transfer(struct ep93xx_spi *espi,
|
||||||
struct spi_transfer *t)
|
struct spi_transfer *t)
|
||||||
{
|
{
|
||||||
struct ep93xx_spi_chip *chip = spi_get_ctldata(msg->spi);
|
struct ep93xx_spi_chip *chip = spi_get_ctldata(msg->spi);
|
||||||
|
int err;
|
||||||
|
|
||||||
msg->state = t;
|
msg->state = t;
|
||||||
|
|
||||||
/*
|
err = ep93xx_spi_chip_setup(espi, chip, t->speed_hz, t->bits_per_word);
|
||||||
* Handle any transfer specific settings if needed. We use
|
|
||||||
* temporary chip settings here and restore original later when
|
|
||||||
* the transfer is finished.
|
|
||||||
*/
|
|
||||||
if (t->speed_hz || t->bits_per_word) {
|
|
||||||
struct ep93xx_spi_chip tmp_chip = *chip;
|
|
||||||
|
|
||||||
if (t->speed_hz) {
|
|
||||||
int err;
|
|
||||||
|
|
||||||
err = ep93xx_spi_calc_divisors(espi, &tmp_chip,
|
|
||||||
t->speed_hz);
|
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_err(&espi->pdev->dev,
|
dev_err(&espi->pdev->dev,
|
||||||
"failed to adjust speed\n");
|
"failed to setup chip for transfer\n");
|
||||||
msg->status = err;
|
msg->status = err;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (t->bits_per_word)
|
|
||||||
tmp_chip.dss = bits_per_word_to_dss(t->bits_per_word);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Set up temporary new hw settings for this transfer.
|
|
||||||
*/
|
|
||||||
ep93xx_spi_chip_setup(espi, &tmp_chip);
|
|
||||||
}
|
|
||||||
|
|
||||||
espi->rx = 0;
|
espi->rx = 0;
|
||||||
espi->tx = 0;
|
espi->tx = 0;
|
||||||
|
@ -783,9 +661,6 @@ static void ep93xx_spi_process_transfer(struct ep93xx_spi *espi,
|
||||||
ep93xx_spi_cs_control(msg->spi, true);
|
ep93xx_spi_cs_control(msg->spi, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (t->speed_hz || t->bits_per_word)
|
|
||||||
ep93xx_spi_chip_setup(espi, chip);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -838,10 +713,8 @@ static void ep93xx_spi_process_message(struct ep93xx_spi *espi,
|
||||||
espi->fifo_level = 0;
|
espi->fifo_level = 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Update SPI controller registers according to spi device and assert
|
* Assert the chipselect.
|
||||||
* the chipselect.
|
|
||||||
*/
|
*/
|
||||||
ep93xx_spi_chip_setup(espi, spi_get_ctldata(msg->spi));
|
|
||||||
ep93xx_spi_cs_control(msg->spi, true);
|
ep93xx_spi_cs_control(msg->spi, true);
|
||||||
|
|
||||||
list_for_each_entry(t, &msg->transfers, transfer_list) {
|
list_for_each_entry(t, &msg->transfers, transfer_list) {
|
||||||
|
@ -858,50 +731,29 @@ static void ep93xx_spi_process_message(struct ep93xx_spi *espi,
|
||||||
ep93xx_spi_disable(espi);
|
ep93xx_spi_disable(espi);
|
||||||
}
|
}
|
||||||
|
|
||||||
#define work_to_espi(work) (container_of((work), struct ep93xx_spi, msg_work))
|
static int ep93xx_spi_transfer_one_message(struct spi_master *master,
|
||||||
|
struct spi_message *msg)
|
||||||
/**
|
|
||||||
* ep93xx_spi_work() - EP93xx SPI workqueue worker function
|
|
||||||
* @work: work struct
|
|
||||||
*
|
|
||||||
* Workqueue worker function. This function is called when there are new
|
|
||||||
* SPI messages to be processed. Message is taken out from the queue and then
|
|
||||||
* passed to ep93xx_spi_process_message().
|
|
||||||
*
|
|
||||||
* After message is transferred, protocol driver is notified by calling
|
|
||||||
* @msg->complete(). In case of error, @msg->status is set to negative error
|
|
||||||
* number, otherwise it contains zero (and @msg->actual_length is updated).
|
|
||||||
*/
|
|
||||||
static void ep93xx_spi_work(struct work_struct *work)
|
|
||||||
{
|
{
|
||||||
struct ep93xx_spi *espi = work_to_espi(work);
|
struct ep93xx_spi *espi = spi_master_get_devdata(master);
|
||||||
struct spi_message *msg;
|
struct spi_transfer *t;
|
||||||
|
|
||||||
spin_lock_irq(&espi->lock);
|
/* first validate each transfer */
|
||||||
if (!espi->running || espi->current_msg ||
|
list_for_each_entry(t, &msg->transfers, transfer_list) {
|
||||||
list_empty(&espi->msg_queue)) {
|
if (t->speed_hz < espi->min_rate)
|
||||||
spin_unlock_irq(&espi->lock);
|
return -EINVAL;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
msg = list_first_entry(&espi->msg_queue, struct spi_message, queue);
|
|
||||||
list_del_init(&msg->queue);
|
msg->state = NULL;
|
||||||
|
msg->status = 0;
|
||||||
|
msg->actual_length = 0;
|
||||||
|
|
||||||
espi->current_msg = msg;
|
espi->current_msg = msg;
|
||||||
spin_unlock_irq(&espi->lock);
|
|
||||||
|
|
||||||
ep93xx_spi_process_message(espi, msg);
|
ep93xx_spi_process_message(espi, msg);
|
||||||
|
|
||||||
/*
|
|
||||||
* Update the current message and re-schedule ourselves if there are
|
|
||||||
* more messages in the queue.
|
|
||||||
*/
|
|
||||||
spin_lock_irq(&espi->lock);
|
|
||||||
espi->current_msg = NULL;
|
espi->current_msg = NULL;
|
||||||
if (espi->running && !list_empty(&espi->msg_queue))
|
|
||||||
queue_work(espi->wq, &espi->msg_work);
|
|
||||||
spin_unlock_irq(&espi->lock);
|
|
||||||
|
|
||||||
/* notify the protocol driver that we are done with this message */
|
spi_finalize_current_message(master);
|
||||||
msg->complete(msg->context);
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static irqreturn_t ep93xx_spi_interrupt(int irq, void *dev_id)
|
static irqreturn_t ep93xx_spi_interrupt(int irq, void *dev_id)
|
||||||
|
@ -1024,14 +876,24 @@ static int ep93xx_spi_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
info = pdev->dev.platform_data;
|
info = pdev->dev.platform_data;
|
||||||
|
|
||||||
master = spi_alloc_master(&pdev->dev, sizeof(*espi));
|
irq = platform_get_irq(pdev, 0);
|
||||||
if (!master) {
|
if (irq < 0) {
|
||||||
dev_err(&pdev->dev, "failed to allocate spi master\n");
|
dev_err(&pdev->dev, "failed to get irq resources\n");
|
||||||
return -ENOMEM;
|
return -EBUSY;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||||
|
if (!res) {
|
||||||
|
dev_err(&pdev->dev, "unable to get iomem resource\n");
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
|
master = spi_alloc_master(&pdev->dev, sizeof(*espi));
|
||||||
|
if (!master)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
master->setup = ep93xx_spi_setup;
|
master->setup = ep93xx_spi_setup;
|
||||||
master->transfer = ep93xx_spi_transfer;
|
master->transfer_one_message = ep93xx_spi_transfer_one_message;
|
||||||
master->cleanup = ep93xx_spi_cleanup;
|
master->cleanup = ep93xx_spi_cleanup;
|
||||||
master->bus_num = pdev->id;
|
master->bus_num = pdev->id;
|
||||||
master->num_chipselect = info->num_chipselect;
|
master->num_chipselect = info->num_chipselect;
|
||||||
|
@ -1042,14 +904,13 @@ static int ep93xx_spi_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
espi = spi_master_get_devdata(master);
|
espi = spi_master_get_devdata(master);
|
||||||
|
|
||||||
espi->clk = clk_get(&pdev->dev, NULL);
|
espi->clk = devm_clk_get(&pdev->dev, NULL);
|
||||||
if (IS_ERR(espi->clk)) {
|
if (IS_ERR(espi->clk)) {
|
||||||
dev_err(&pdev->dev, "unable to get spi clock\n");
|
dev_err(&pdev->dev, "unable to get spi clock\n");
|
||||||
error = PTR_ERR(espi->clk);
|
error = PTR_ERR(espi->clk);
|
||||||
goto fail_release_master;
|
goto fail_release_master;
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_lock_init(&espi->lock);
|
|
||||||
init_completion(&espi->wait);
|
init_completion(&espi->wait);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1060,55 +921,31 @@ static int ep93xx_spi_probe(struct platform_device *pdev)
|
||||||
espi->min_rate = clk_get_rate(espi->clk) / (254 * 256);
|
espi->min_rate = clk_get_rate(espi->clk) / (254 * 256);
|
||||||
espi->pdev = pdev;
|
espi->pdev = pdev;
|
||||||
|
|
||||||
irq = platform_get_irq(pdev, 0);
|
|
||||||
if (irq < 0) {
|
|
||||||
error = -EBUSY;
|
|
||||||
dev_err(&pdev->dev, "failed to get irq resources\n");
|
|
||||||
goto fail_put_clock;
|
|
||||||
}
|
|
||||||
|
|
||||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
|
||||||
if (!res) {
|
|
||||||
dev_err(&pdev->dev, "unable to get iomem resource\n");
|
|
||||||
error = -ENODEV;
|
|
||||||
goto fail_put_clock;
|
|
||||||
}
|
|
||||||
|
|
||||||
espi->sspdr_phys = res->start + SSPDR;
|
espi->sspdr_phys = res->start + SSPDR;
|
||||||
|
|
||||||
espi->regs_base = devm_ioremap_resource(&pdev->dev, res);
|
espi->regs_base = devm_ioremap_resource(&pdev->dev, res);
|
||||||
if (IS_ERR(espi->regs_base)) {
|
if (IS_ERR(espi->regs_base)) {
|
||||||
error = PTR_ERR(espi->regs_base);
|
error = PTR_ERR(espi->regs_base);
|
||||||
goto fail_put_clock;
|
goto fail_release_master;
|
||||||
}
|
}
|
||||||
|
|
||||||
error = devm_request_irq(&pdev->dev, irq, ep93xx_spi_interrupt,
|
error = devm_request_irq(&pdev->dev, irq, ep93xx_spi_interrupt,
|
||||||
0, "ep93xx-spi", espi);
|
0, "ep93xx-spi", espi);
|
||||||
if (error) {
|
if (error) {
|
||||||
dev_err(&pdev->dev, "failed to request irq\n");
|
dev_err(&pdev->dev, "failed to request irq\n");
|
||||||
goto fail_put_clock;
|
goto fail_release_master;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (info->use_dma && ep93xx_spi_setup_dma(espi))
|
if (info->use_dma && ep93xx_spi_setup_dma(espi))
|
||||||
dev_warn(&pdev->dev, "DMA setup failed. Falling back to PIO\n");
|
dev_warn(&pdev->dev, "DMA setup failed. Falling back to PIO\n");
|
||||||
|
|
||||||
espi->wq = create_singlethread_workqueue("ep93xx_spid");
|
|
||||||
if (!espi->wq) {
|
|
||||||
dev_err(&pdev->dev, "unable to create workqueue\n");
|
|
||||||
error = -ENOMEM;
|
|
||||||
goto fail_free_dma;
|
|
||||||
}
|
|
||||||
INIT_WORK(&espi->msg_work, ep93xx_spi_work);
|
|
||||||
INIT_LIST_HEAD(&espi->msg_queue);
|
|
||||||
espi->running = true;
|
|
||||||
|
|
||||||
/* make sure that the hardware is disabled */
|
/* make sure that the hardware is disabled */
|
||||||
ep93xx_spi_write_u8(espi, SSPCR1, 0);
|
ep93xx_spi_write_u8(espi, SSPCR1, 0);
|
||||||
|
|
||||||
error = spi_register_master(master);
|
error = spi_register_master(master);
|
||||||
if (error) {
|
if (error) {
|
||||||
dev_err(&pdev->dev, "failed to register SPI master\n");
|
dev_err(&pdev->dev, "failed to register SPI master\n");
|
||||||
goto fail_free_queue;
|
goto fail_free_dma;
|
||||||
}
|
}
|
||||||
|
|
||||||
dev_info(&pdev->dev, "EP93xx SPI Controller at 0x%08lx irq %d\n",
|
dev_info(&pdev->dev, "EP93xx SPI Controller at 0x%08lx irq %d\n",
|
||||||
|
@ -1116,12 +953,8 @@ static int ep93xx_spi_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
fail_free_queue:
|
|
||||||
destroy_workqueue(espi->wq);
|
|
||||||
fail_free_dma:
|
fail_free_dma:
|
||||||
ep93xx_spi_release_dma(espi);
|
ep93xx_spi_release_dma(espi);
|
||||||
fail_put_clock:
|
|
||||||
clk_put(espi->clk);
|
|
||||||
fail_release_master:
|
fail_release_master:
|
||||||
spi_master_put(master);
|
spi_master_put(master);
|
||||||
|
|
||||||
|
@ -1133,31 +966,7 @@ static int ep93xx_spi_remove(struct platform_device *pdev)
|
||||||
struct spi_master *master = platform_get_drvdata(pdev);
|
struct spi_master *master = platform_get_drvdata(pdev);
|
||||||
struct ep93xx_spi *espi = spi_master_get_devdata(master);
|
struct ep93xx_spi *espi = spi_master_get_devdata(master);
|
||||||
|
|
||||||
spin_lock_irq(&espi->lock);
|
|
||||||
espi->running = false;
|
|
||||||
spin_unlock_irq(&espi->lock);
|
|
||||||
|
|
||||||
destroy_workqueue(espi->wq);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Complete remaining messages with %-ESHUTDOWN status.
|
|
||||||
*/
|
|
||||||
spin_lock_irq(&espi->lock);
|
|
||||||
while (!list_empty(&espi->msg_queue)) {
|
|
||||||
struct spi_message *msg;
|
|
||||||
|
|
||||||
msg = list_first_entry(&espi->msg_queue,
|
|
||||||
struct spi_message, queue);
|
|
||||||
list_del_init(&msg->queue);
|
|
||||||
msg->status = -ESHUTDOWN;
|
|
||||||
spin_unlock_irq(&espi->lock);
|
|
||||||
msg->complete(msg->context);
|
|
||||||
spin_lock_irq(&espi->lock);
|
|
||||||
}
|
|
||||||
spin_unlock_irq(&espi->lock);
|
|
||||||
|
|
||||||
ep93xx_spi_release_dma(espi);
|
ep93xx_spi_release_dma(espi);
|
||||||
clk_put(espi->clk);
|
|
||||||
|
|
||||||
spi_unregister_master(master);
|
spi_unregister_master(master);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
Loading…
Add table
Reference in a new issue