ARM: OMAP: Allocate McBSP devices dynamically

Based on Chandra's earlier patches in linux-omap tree.

Note that omap1_mcbsp_check and omap2_mcbsp_check are no longer
needed as there's now omap_mcbsp_check_valid_id() defined.

Also some functions can now be marked __init.

Signed-off-by: Chandra Shekhar <x0044955@ti.com>
Signed-off-by: Tony Lindgren <tony@atomide.com>
This commit is contained in:
Chandra Shekhar 2008-10-08 10:01:39 +03:00 committed by Tony Lindgren
parent 25cef22514
commit b4b58f5834
4 changed files with 270 additions and 216 deletions

View File

@ -103,30 +103,6 @@ static inline void omap_mcbsp_clk_init(struct mcbsp_internal_clk *mclk)
{ } { }
#endif #endif
static int omap1_mcbsp_check(unsigned int id)
{
/* REVISIT: Check correctly for number of registered McBSPs */
if (cpu_is_omap730()) {
if (id > OMAP_MAX_MCBSP_COUNT - 2) {
printk(KERN_ERR "OMAP-McBSP: McBSP%d doesn't exist\n",
id + 1);
return -ENODEV;
}
return 0;
}
if (cpu_is_omap15xx() || cpu_is_omap16xx()) {
if (id > OMAP_MAX_MCBSP_COUNT - 1) {
printk(KERN_ERR "OMAP-McBSP: McBSP%d doesn't exist\n",
id + 1);
return -ENODEV;
}
return 0;
}
return -ENODEV;
}
static void omap1_mcbsp_request(unsigned int id) static void omap1_mcbsp_request(unsigned int id)
{ {
/* /*
@ -151,7 +127,6 @@ static void omap1_mcbsp_free(unsigned int id)
} }
static struct omap_mcbsp_ops omap1_mcbsp_ops = { static struct omap_mcbsp_ops omap1_mcbsp_ops = {
.check = omap1_mcbsp_check,
.request = omap1_mcbsp_request, .request = omap1_mcbsp_request,
.free = omap1_mcbsp_free, .free = omap1_mcbsp_free,
}; };
@ -262,6 +237,18 @@ int __init omap1_mcbsp_init(void)
} }
} }
if (cpu_is_omap730())
omap_mcbsp_count = OMAP730_MCBSP_PDATA_SZ;
if (cpu_is_omap15xx())
omap_mcbsp_count = OMAP15XX_MCBSP_PDATA_SZ;
if (cpu_is_omap16xx())
omap_mcbsp_count = OMAP16XX_MCBSP_PDATA_SZ;
mcbsp_ptr = kzalloc(omap_mcbsp_count * sizeof(struct omap_mcbsp *),
GFP_KERNEL);
if (!mcbsp_ptr)
return -ENOMEM;
if (cpu_is_omap730()) if (cpu_is_omap730())
omap_mcbsp_register_board_cfg(omap730_mcbsp_pdata, omap_mcbsp_register_board_cfg(omap730_mcbsp_pdata,
OMAP730_MCBSP_PDATA_SZ); OMAP730_MCBSP_PDATA_SZ);

View File

@ -28,7 +28,7 @@ struct mcbsp_internal_clk {
int n_childs; int n_childs;
}; };
#if defined(CONFIG_ARCH_OMAP24XX) || defined(CONFIG_ARCH_OMAP34XX) #if defined(CONFIG_ARCH_OMAP24XX)
static void omap_mcbsp_clk_init(struct mcbsp_internal_clk *mclk) static void omap_mcbsp_clk_init(struct mcbsp_internal_clk *mclk)
{ {
const char *clk_names[] = { "mcbsp_ick", "mcbsp_fck" }; const char *clk_names[] = { "mcbsp_ick", "mcbsp_fck" };
@ -117,18 +117,8 @@ static void omap2_mcbsp_request(unsigned int id)
omap2_mcbsp2_mux_setup(); omap2_mcbsp2_mux_setup();
} }
static int omap2_mcbsp_check(unsigned int id)
{
if (id > OMAP_MAX_MCBSP_COUNT - 1) {
printk(KERN_ERR "OMAP-McBSP: McBSP%d doesn't exist\n", id + 1);
return -ENODEV;
}
return 0;
}
static struct omap_mcbsp_ops omap2_mcbsp_ops = { static struct omap_mcbsp_ops omap2_mcbsp_ops = {
.request = omap2_mcbsp_request, .request = omap2_mcbsp_request,
.check = omap2_mcbsp_check,
}; };
#ifdef CONFIG_ARCH_OMAP24XX #ifdef CONFIG_ARCH_OMAP24XX
@ -185,7 +175,7 @@ static struct omap_mcbsp_platform_data omap34xx_mcbsp_pdata[] = {
#define OMAP34XX_MCBSP_PDATA_SZ 0 #define OMAP34XX_MCBSP_PDATA_SZ 0
#endif #endif
int __init omap2_mcbsp_init(void) static int __init omap2_mcbsp_init(void)
{ {
int i; int i;
@ -195,14 +185,20 @@ int __init omap2_mcbsp_init(void)
clk_register(&omap_mcbsp_clks[i].clk); clk_register(&omap_mcbsp_clks[i].clk);
} }
if (cpu_is_omap24xx())
omap_mcbsp_count = OMAP24XX_MCBSP_PDATA_SZ;
if (cpu_is_omap34xx())
omap_mcbsp_count = OMAP34XX_MCBSP_PDATA_SZ;
mcbsp_ptr = kzalloc(omap_mcbsp_count * sizeof(struct omap_mcbsp *),
GFP_KERNEL);
if (!mcbsp_ptr)
return -ENOMEM;
if (cpu_is_omap24xx()) if (cpu_is_omap24xx())
omap_mcbsp_register_board_cfg(omap24xx_mcbsp_pdata, omap_mcbsp_register_board_cfg(omap24xx_mcbsp_pdata,
OMAP24XX_MCBSP_PDATA_SZ); OMAP24XX_MCBSP_PDATA_SZ);
if (cpu_is_omap34xx())
omap_mcbsp_register_board_cfg(omap34xx_mcbsp_pdata,
OMAP34XX_MCBSP_PDATA_SZ);
return omap_mcbsp_init(); return omap_mcbsp_init();
} }
arch_initcall(omap2_mcbsp_init); arch_initcall(omap2_mcbsp_init);

View File

@ -81,9 +81,6 @@
#define OMAP_MCBSP_REG_XCERG 0x3A #define OMAP_MCBSP_REG_XCERG 0x3A
#define OMAP_MCBSP_REG_XCERH 0x3C #define OMAP_MCBSP_REG_XCERH 0x3C
#define OMAP_MAX_MCBSP_COUNT 3
#define MAX_MCBSP_CLOCKS 3
#define AUDIO_MCBSP_DATAWRITE (OMAP1510_MCBSP1_BASE + OMAP_MCBSP_REG_DXR1) #define AUDIO_MCBSP_DATAWRITE (OMAP1510_MCBSP1_BASE + OMAP_MCBSP_REG_DXR1)
#define AUDIO_MCBSP_DATAREAD (OMAP1510_MCBSP1_BASE + OMAP_MCBSP_REG_DRR1) #define AUDIO_MCBSP_DATAREAD (OMAP1510_MCBSP1_BASE + OMAP_MCBSP_REG_DRR1)
@ -91,12 +88,14 @@
#define AUDIO_DMA_TX OMAP_DMA_MCBSP1_TX #define AUDIO_DMA_TX OMAP_DMA_MCBSP1_TX
#define AUDIO_DMA_RX OMAP_DMA_MCBSP1_RX #define AUDIO_DMA_RX OMAP_DMA_MCBSP1_RX
#elif defined(CONFIG_ARCH_OMAP24XX) #elif defined(CONFIG_ARCH_OMAP24XX) || defined(CONFIG_ARCH_OMAP34XX)
#define OMAP_MCBSP_REG_DRR2 0x00 #define OMAP_MCBSP_REG_DRR2 0x00
#define OMAP_MCBSP_REG_DRR1 0x04 #define OMAP_MCBSP_REG_DRR1 0x04
#define OMAP_MCBSP_REG_DXR2 0x08 #define OMAP_MCBSP_REG_DXR2 0x08
#define OMAP_MCBSP_REG_DXR1 0x0C #define OMAP_MCBSP_REG_DXR1 0x0C
#define OMAP_MCBSP_REG_DRR 0x00
#define OMAP_MCBSP_REG_DXR 0x08
#define OMAP_MCBSP_REG_SPCR2 0x10 #define OMAP_MCBSP_REG_SPCR2 0x10
#define OMAP_MCBSP_REG_SPCR1 0x14 #define OMAP_MCBSP_REG_SPCR1 0x14
#define OMAP_MCBSP_REG_RCR2 0x18 #define OMAP_MCBSP_REG_RCR2 0x18
@ -124,9 +123,9 @@
#define OMAP_MCBSP_REG_RCERH 0x70 #define OMAP_MCBSP_REG_RCERH 0x70
#define OMAP_MCBSP_REG_XCERG 0x74 #define OMAP_MCBSP_REG_XCERG 0x74
#define OMAP_MCBSP_REG_XCERH 0x78 #define OMAP_MCBSP_REG_XCERH 0x78
#define OMAP_MCBSP_REG_SYSCON 0x8C
#define OMAP_MAX_MCBSP_COUNT 2 #define OMAP_MCBSP_REG_XCCR 0xAC
#define MAX_MCBSP_CLOCKS 2 #define OMAP_MCBSP_REG_RCCR 0xB0
#define AUDIO_MCBSP_DATAWRITE (OMAP24XX_MCBSP2_BASE + OMAP_MCBSP_REG_DXR1) #define AUDIO_MCBSP_DATAWRITE (OMAP24XX_MCBSP2_BASE + OMAP_MCBSP_REG_DXR1)
#define AUDIO_MCBSP_DATAREAD (OMAP24XX_MCBSP2_BASE + OMAP_MCBSP_REG_DRR1) #define AUDIO_MCBSP_DATAREAD (OMAP24XX_MCBSP2_BASE + OMAP_MCBSP_REG_DRR1)
@ -137,10 +136,6 @@
#endif #endif
#define OMAP_MCBSP_READ(base, reg) __raw_readw((base) + OMAP_MCBSP_REG_##reg)
#define OMAP_MCBSP_WRITE(base, reg, val) __raw_writew((val), (base) + OMAP_MCBSP_REG_##reg)
/************************** McBSP SPCR1 bit definitions ***********************/ /************************** McBSP SPCR1 bit definitions ***********************/
#define RRST 0x0001 #define RRST 0x0001
#define RRDY 0x0002 #define RRDY 0x0002
@ -151,6 +146,7 @@
#define DXENA 0x0080 #define DXENA 0x0080
#define CLKSTP(value) ((value)<<11) /* bits 11:12 */ #define CLKSTP(value) ((value)<<11) /* bits 11:12 */
#define RJUST(value) ((value)<<13) /* bits 13:14 */ #define RJUST(value) ((value)<<13) /* bits 13:14 */
#define ALB 0x8000
#define DLB 0x8000 #define DLB 0x8000
/************************** McBSP SPCR2 bit definitions ***********************/ /************************** McBSP SPCR2 bit definitions ***********************/
@ -228,6 +224,17 @@
#define XPABLK(value) ((value)<<5) /* Bits 5:6 */ #define XPABLK(value) ((value)<<5) /* Bits 5:6 */
#define XPBBLK(value) ((value)<<7) /* Bits 7:8 */ #define XPBBLK(value) ((value)<<7) /* Bits 7:8 */
/*********************** McBSP XCCR bit definitions *************************/
#define DILB 0x0020
#define XDMAEN 0x0008
#define XDISABLE 0x0001
/********************** McBSP RCCR bit definitions *************************/
#define RDMAEN 0x0008
#define RDISABLE 0x0001
/********************** McBSP SYSCONFIG bit definitions ********************/
#define SOFTRST 0x0002
/* we don't do multichannel for now */ /* we don't do multichannel for now */
struct omap_mcbsp_reg_cfg { struct omap_mcbsp_reg_cfg {
@ -311,7 +318,6 @@ struct omap_mcbsp_spi_cfg {
struct omap_mcbsp_ops { struct omap_mcbsp_ops {
void (*request)(unsigned int); void (*request)(unsigned int);
void (*free)(unsigned int); void (*free)(unsigned int);
int (*check)(unsigned int);
}; };
struct omap_mcbsp_platform_data { struct omap_mcbsp_platform_data {
@ -353,6 +359,8 @@ struct omap_mcbsp {
struct omap_mcbsp_platform_data *pdata; struct omap_mcbsp_platform_data *pdata;
struct clk *clk; struct clk *clk;
}; };
extern struct omap_mcbsp **mcbsp_ptr;
extern int omap_mcbsp_count;
int omap_mcbsp_init(void); int omap_mcbsp_init(void);
void omap_mcbsp_register_board_cfg(struct omap_mcbsp_platform_data *config, void omap_mcbsp_register_board_cfg(struct omap_mcbsp_platform_data *config,
@ -377,5 +385,6 @@ void omap_mcbsp_set_spi_mode(unsigned int id, const struct omap_mcbsp_spi_cfg *
/* Polled read/write functions */ /* Polled read/write functions */
int omap_mcbsp_pollread(unsigned int id, u16 * buf); int omap_mcbsp_pollread(unsigned int id, u16 * buf);
int omap_mcbsp_pollwrite(unsigned int id, u16 buf); int omap_mcbsp_pollwrite(unsigned int id, u16 buf);
int omap_mcbsp_set_io_type(unsigned int id, omap_mcbsp_io_type_t io_type);
#endif #endif

View File

@ -27,43 +27,65 @@
#include <mach/dma.h> #include <mach/dma.h>
#include <mach/mcbsp.h> #include <mach/mcbsp.h>
static struct omap_mcbsp mcbsp[OMAP_MAX_MCBSP_COUNT]; struct omap_mcbsp **mcbsp_ptr;
int omap_mcbsp_count;
#define omap_mcbsp_check_valid_id(id) (mcbsp[id].pdata && \ void omap_mcbsp_write(void __iomem *io_base, u16 reg, u32 val)
mcbsp[id].pdata->ops && \ {
mcbsp[id].pdata->ops->check && \ if (cpu_class_is_omap1() || cpu_is_omap2420())
(mcbsp[id].pdata->ops->check(id) == 0)) __raw_writew((u16)val, io_base + reg);
else
__raw_writel(val, io_base + reg);
}
int omap_mcbsp_read(void __iomem *io_base, u16 reg)
{
if (cpu_class_is_omap1() || cpu_is_omap2420())
return __raw_readw(io_base + reg);
else
return __raw_readl(io_base + reg);
}
#define OMAP_MCBSP_READ(base, reg) \
omap_mcbsp_read(base, OMAP_MCBSP_REG_##reg)
#define OMAP_MCBSP_WRITE(base, reg, val) \
omap_mcbsp_write(base, OMAP_MCBSP_REG_##reg, val)
#define omap_mcbsp_check_valid_id(id) (id < omap_mcbsp_count)
#define id_to_mcbsp_ptr(id) mcbsp_ptr[id];
static void omap_mcbsp_dump_reg(u8 id) static void omap_mcbsp_dump_reg(u8 id)
{ {
dev_dbg(mcbsp[id].dev, "**** McBSP%d regs ****\n", mcbsp[id].id); struct omap_mcbsp *mcbsp = id_to_mcbsp_ptr(id);
dev_dbg(mcbsp[id].dev, "DRR2: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, DRR2)); dev_dbg(mcbsp->dev, "**** McBSP%d regs ****\n", mcbsp->id);
dev_dbg(mcbsp[id].dev, "DRR1: 0x%04x\n", dev_dbg(mcbsp->dev, "DRR2: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, DRR1)); OMAP_MCBSP_READ(mcbsp->io_base, DRR2));
dev_dbg(mcbsp[id].dev, "DXR2: 0x%04x\n", dev_dbg(mcbsp->dev, "DRR1: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, DXR2)); OMAP_MCBSP_READ(mcbsp->io_base, DRR1));
dev_dbg(mcbsp[id].dev, "DXR1: 0x%04x\n", dev_dbg(mcbsp->dev, "DXR2: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, DXR1)); OMAP_MCBSP_READ(mcbsp->io_base, DXR2));
dev_dbg(mcbsp[id].dev, "SPCR2: 0x%04x\n", dev_dbg(mcbsp->dev, "DXR1: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, SPCR2)); OMAP_MCBSP_READ(mcbsp->io_base, DXR1));
dev_dbg(mcbsp[id].dev, "SPCR1: 0x%04x\n", dev_dbg(mcbsp->dev, "SPCR2: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, SPCR1)); OMAP_MCBSP_READ(mcbsp->io_base, SPCR2));
dev_dbg(mcbsp[id].dev, "RCR2: 0x%04x\n", dev_dbg(mcbsp->dev, "SPCR1: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, RCR2)); OMAP_MCBSP_READ(mcbsp->io_base, SPCR1));
dev_dbg(mcbsp[id].dev, "RCR1: 0x%04x\n", dev_dbg(mcbsp->dev, "RCR2: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, RCR1)); OMAP_MCBSP_READ(mcbsp->io_base, RCR2));
dev_dbg(mcbsp[id].dev, "XCR2: 0x%04x\n", dev_dbg(mcbsp->dev, "RCR1: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, XCR2)); OMAP_MCBSP_READ(mcbsp->io_base, RCR1));
dev_dbg(mcbsp[id].dev, "XCR1: 0x%04x\n", dev_dbg(mcbsp->dev, "XCR2: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, XCR1)); OMAP_MCBSP_READ(mcbsp->io_base, XCR2));
dev_dbg(mcbsp[id].dev, "SRGR2: 0x%04x\n", dev_dbg(mcbsp->dev, "XCR1: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, SRGR2)); OMAP_MCBSP_READ(mcbsp->io_base, XCR1));
dev_dbg(mcbsp[id].dev, "SRGR1: 0x%04x\n", dev_dbg(mcbsp->dev, "SRGR2: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, SRGR1)); OMAP_MCBSP_READ(mcbsp->io_base, SRGR2));
dev_dbg(mcbsp[id].dev, "PCR0: 0x%04x\n", dev_dbg(mcbsp->dev, "SRGR1: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp[id].io_base, PCR0)); OMAP_MCBSP_READ(mcbsp->io_base, SRGR1));
dev_dbg(mcbsp[id].dev, "***********************\n"); dev_dbg(mcbsp->dev, "PCR0: 0x%04x\n",
OMAP_MCBSP_READ(mcbsp->io_base, PCR0));
dev_dbg(mcbsp->dev, "***********************\n");
} }
static irqreturn_t omap_mcbsp_tx_irq_handler(int irq, void *dev_id) static irqreturn_t omap_mcbsp_tx_irq_handler(int irq, void *dev_id)
@ -126,16 +148,18 @@ static void omap_mcbsp_rx_dma_callback(int lch, u16 ch_status, void *data)
*/ */
void omap_mcbsp_config(unsigned int id, const struct omap_mcbsp_reg_cfg *config) void omap_mcbsp_config(unsigned int id, const struct omap_mcbsp_reg_cfg *config)
{ {
struct omap_mcbsp *mcbsp;
void __iomem *io_base; void __iomem *io_base;
if (!omap_mcbsp_check_valid_id(id)) { if (!omap_mcbsp_check_valid_id(id)) {
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return; return;
} }
mcbsp = id_to_mcbsp_ptr(id);
io_base = mcbsp[id].io_base; io_base = mcbsp->io_base;
dev_dbg(mcbsp[id].dev, "Configuring McBSP%d phys_base: 0x%08lx\n", dev_dbg(mcbsp->dev, "Configuring McBSP%d phys_base: 0x%08lx\n",
mcbsp[id].id, mcbsp[id].phys_base); mcbsp->id, mcbsp->phys_base);
/* We write the given config */ /* We write the given config */
OMAP_MCBSP_WRITE(io_base, SPCR2, config->spcr2); OMAP_MCBSP_WRITE(io_base, SPCR2, config->spcr2);
@ -158,23 +182,26 @@ EXPORT_SYMBOL(omap_mcbsp_config);
*/ */
int omap_mcbsp_set_io_type(unsigned int id, omap_mcbsp_io_type_t io_type) int omap_mcbsp_set_io_type(unsigned int id, omap_mcbsp_io_type_t io_type)
{ {
struct omap_mcbsp *mcbsp;
if (!omap_mcbsp_check_valid_id(id)) { if (!omap_mcbsp_check_valid_id(id)) {
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return -ENODEV; return -ENODEV;
} }
mcbsp = id_to_mcbsp_ptr(id);
spin_lock(&mcbsp[id].lock); spin_lock(&mcbsp->lock);
if (!mcbsp[id].free) { if (!mcbsp->free) {
dev_err(mcbsp[id].dev, "McBSP%d is currently in use\n", dev_err(mcbsp->dev, "McBSP%d is currently in use\n",
mcbsp[id].id); mcbsp->id);
spin_unlock(&mcbsp[id].lock); spin_unlock(&mcbsp->lock);
return -EINVAL; return -EINVAL;
} }
mcbsp[id].io_type = io_type; mcbsp->io_type = io_type;
spin_unlock(&mcbsp[id].lock); spin_unlock(&mcbsp->lock);
return 0; return 0;
} }
@ -182,53 +209,55 @@ EXPORT_SYMBOL(omap_mcbsp_set_io_type);
int omap_mcbsp_request(unsigned int id) int omap_mcbsp_request(unsigned int id)
{ {
struct omap_mcbsp *mcbsp;
int err; int err;
if (!omap_mcbsp_check_valid_id(id)) { if (!omap_mcbsp_check_valid_id(id)) {
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return -ENODEV; return -ENODEV;
} }
mcbsp = id_to_mcbsp_ptr(id);
if (mcbsp[id].pdata->ops->request) if (mcbsp->pdata && mcbsp->pdata->ops && mcbsp->pdata->ops->request)
mcbsp[id].pdata->ops->request(id); mcbsp->pdata->ops->request(id);
clk_enable(mcbsp[id].clk); clk_enable(mcbsp->clk);
spin_lock(&mcbsp[id].lock); spin_lock(&mcbsp->lock);
if (!mcbsp[id].free) { if (!mcbsp->free) {
dev_err(mcbsp[id].dev, "McBSP%d is currently in use\n", dev_err(mcbsp->dev, "McBSP%d is currently in use\n",
mcbsp[id].id); mcbsp->id);
spin_unlock(&mcbsp[id].lock); spin_unlock(&mcbsp->lock);
return -1; return -1;
} }
mcbsp[id].free = 0; mcbsp->free = 0;
spin_unlock(&mcbsp[id].lock); spin_unlock(&mcbsp->lock);
if (mcbsp[id].io_type == OMAP_MCBSP_IRQ_IO) { if (mcbsp->io_type == OMAP_MCBSP_IRQ_IO) {
/* We need to get IRQs here */ /* We need to get IRQs here */
err = request_irq(mcbsp[id].tx_irq, omap_mcbsp_tx_irq_handler, err = request_irq(mcbsp->tx_irq, omap_mcbsp_tx_irq_handler,
0, "McBSP", (void *) (&mcbsp[id])); 0, "McBSP", (void *)mcbsp);
if (err != 0) { if (err != 0) {
dev_err(mcbsp[id].dev, "Unable to request TX IRQ %d " dev_err(mcbsp->dev, "Unable to request TX IRQ %d "
"for McBSP%d\n", mcbsp[id].tx_irq, "for McBSP%d\n", mcbsp->tx_irq,
mcbsp[id].id); mcbsp->id);
return err; return err;
} }
init_completion(&(mcbsp[id].tx_irq_completion)); init_completion(&mcbsp->tx_irq_completion);
err = request_irq(mcbsp[id].rx_irq, omap_mcbsp_rx_irq_handler, err = request_irq(mcbsp->rx_irq, omap_mcbsp_rx_irq_handler,
0, "McBSP", (void *) (&mcbsp[id])); 0, "McBSP", (void *)mcbsp);
if (err != 0) { if (err != 0) {
dev_err(mcbsp[id].dev, "Unable to request RX IRQ %d " dev_err(mcbsp->dev, "Unable to request RX IRQ %d "
"for McBSP%d\n", mcbsp[id].rx_irq, "for McBSP%d\n", mcbsp->rx_irq,
mcbsp[id].id); mcbsp->id);
free_irq(mcbsp[id].tx_irq, (void *) (&mcbsp[id])); free_irq(mcbsp->tx_irq, (void *)mcbsp);
return err; return err;
} }
init_completion(&(mcbsp[id].rx_irq_completion)); init_completion(&mcbsp->rx_irq_completion);
} }
return 0; return 0;
@ -237,31 +266,34 @@ EXPORT_SYMBOL(omap_mcbsp_request);
void omap_mcbsp_free(unsigned int id) void omap_mcbsp_free(unsigned int id)
{ {
struct omap_mcbsp *mcbsp;
if (!omap_mcbsp_check_valid_id(id)) { if (!omap_mcbsp_check_valid_id(id)) {
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return; return;
} }
mcbsp = id_to_mcbsp_ptr(id);
if (mcbsp[id].pdata->ops->free) if (mcbsp->pdata && mcbsp->pdata->ops && mcbsp->pdata->ops->free)
mcbsp[id].pdata->ops->free(id); mcbsp->pdata->ops->free(id);
clk_disable(mcbsp[id].clk); clk_disable(mcbsp->clk);
spin_lock(&mcbsp[id].lock); spin_lock(&mcbsp->lock);
if (mcbsp[id].free) { if (mcbsp->free) {
dev_err(mcbsp[id].dev, "McBSP%d was not reserved\n", dev_err(mcbsp->dev, "McBSP%d was not reserved\n",
mcbsp[id].id); mcbsp->id);
spin_unlock(&mcbsp[id].lock); spin_unlock(&mcbsp->lock);
return; return;
} }
mcbsp[id].free = 1; mcbsp->free = 1;
spin_unlock(&mcbsp[id].lock); spin_unlock(&mcbsp->lock);
if (mcbsp[id].io_type == OMAP_MCBSP_IRQ_IO) { if (mcbsp->io_type == OMAP_MCBSP_IRQ_IO) {
/* Free IRQs */ /* Free IRQs */
free_irq(mcbsp[id].rx_irq, (void *) (&mcbsp[id])); free_irq(mcbsp->rx_irq, (void *)mcbsp);
free_irq(mcbsp[id].tx_irq, (void *) (&mcbsp[id])); free_irq(mcbsp->tx_irq, (void *)mcbsp);
} }
} }
EXPORT_SYMBOL(omap_mcbsp_free); EXPORT_SYMBOL(omap_mcbsp_free);
@ -273,6 +305,7 @@ EXPORT_SYMBOL(omap_mcbsp_free);
*/ */
void omap_mcbsp_start(unsigned int id) void omap_mcbsp_start(unsigned int id)
{ {
struct omap_mcbsp *mcbsp;
void __iomem *io_base; void __iomem *io_base;
u16 w; u16 w;
@ -280,11 +313,11 @@ void omap_mcbsp_start(unsigned int id)
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return; return;
} }
mcbsp = id_to_mcbsp_ptr(id);
io_base = mcbsp->io_base;
io_base = mcbsp[id].io_base; mcbsp->rx_word_length = (OMAP_MCBSP_READ(io_base, RCR1) >> 5) & 0x7;
mcbsp->tx_word_length = (OMAP_MCBSP_READ(io_base, XCR1) >> 5) & 0x7;
mcbsp[id].rx_word_length = (OMAP_MCBSP_READ(io_base, RCR1) >> 5) & 0x7;
mcbsp[id].tx_word_length = (OMAP_MCBSP_READ(io_base, XCR1) >> 5) & 0x7;
/* Start the sample generator */ /* Start the sample generator */
w = OMAP_MCBSP_READ(io_base, SPCR2); w = OMAP_MCBSP_READ(io_base, SPCR2);
@ -310,6 +343,7 @@ EXPORT_SYMBOL(omap_mcbsp_start);
void omap_mcbsp_stop(unsigned int id) void omap_mcbsp_stop(unsigned int id)
{ {
struct omap_mcbsp *mcbsp;
void __iomem *io_base; void __iomem *io_base;
u16 w; u16 w;
@ -318,7 +352,8 @@ void omap_mcbsp_stop(unsigned int id)
return; return;
} }
io_base = mcbsp[id].io_base; mcbsp = id_to_mcbsp_ptr(id);
io_base = mcbsp->io_base;
/* Reset transmitter */ /* Reset transmitter */
w = OMAP_MCBSP_READ(io_base, SPCR2); w = OMAP_MCBSP_READ(io_base, SPCR2);
@ -337,6 +372,7 @@ EXPORT_SYMBOL(omap_mcbsp_stop);
/* polled mcbsp i/o operations */ /* polled mcbsp i/o operations */
int omap_mcbsp_pollwrite(unsigned int id, u16 buf) int omap_mcbsp_pollwrite(unsigned int id, u16 buf)
{ {
struct omap_mcbsp *mcbsp;
void __iomem *base; void __iomem *base;
if (!omap_mcbsp_check_valid_id(id)) { if (!omap_mcbsp_check_valid_id(id)) {
@ -344,7 +380,9 @@ int omap_mcbsp_pollwrite(unsigned int id, u16 buf)
return -ENODEV; return -ENODEV;
} }
base = mcbsp[id].io_base; mcbsp = id_to_mcbsp_ptr(id);
base = mcbsp->io_base;
writew(buf, base + OMAP_MCBSP_REG_DXR1); writew(buf, base + OMAP_MCBSP_REG_DXR1);
/* if frame sync error - clear the error */ /* if frame sync error - clear the error */
if (readw(base + OMAP_MCBSP_REG_SPCR2) & XSYNC_ERR) { if (readw(base + OMAP_MCBSP_REG_SPCR2) & XSYNC_ERR) {
@ -366,8 +404,8 @@ int omap_mcbsp_pollwrite(unsigned int id, u16 buf)
(XRST), (XRST),
base + OMAP_MCBSP_REG_SPCR2); base + OMAP_MCBSP_REG_SPCR2);
udelay(10); udelay(10);
dev_err(mcbsp[id].dev, "Could not write to" dev_err(mcbsp->dev, "Could not write to"
" McBSP%d Register\n", mcbsp[id].id); " McBSP%d Register\n", mcbsp->id);
return -2; return -2;
} }
} }
@ -379,14 +417,16 @@ EXPORT_SYMBOL(omap_mcbsp_pollwrite);
int omap_mcbsp_pollread(unsigned int id, u16 *buf) int omap_mcbsp_pollread(unsigned int id, u16 *buf)
{ {
struct omap_mcbsp *mcbsp;
void __iomem *base; void __iomem *base;
if (!omap_mcbsp_check_valid_id(id)) { if (!omap_mcbsp_check_valid_id(id)) {
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return -ENODEV; return -ENODEV;
} }
mcbsp = id_to_mcbsp_ptr(id);
base = mcbsp[id].io_base; base = mcbsp->io_base;
/* if frame sync error - clear the error */ /* if frame sync error - clear the error */
if (readw(base + OMAP_MCBSP_REG_SPCR1) & RSYNC_ERR) { if (readw(base + OMAP_MCBSP_REG_SPCR1) & RSYNC_ERR) {
/* clear error */ /* clear error */
@ -407,8 +447,8 @@ int omap_mcbsp_pollread(unsigned int id, u16 *buf)
(RRST), (RRST),
base + OMAP_MCBSP_REG_SPCR1); base + OMAP_MCBSP_REG_SPCR1);
udelay(10); udelay(10);
dev_err(mcbsp[id].dev, "Could not read from" dev_err(mcbsp->dev, "Could not read from"
" McBSP%d Register\n", mcbsp[id].id); " McBSP%d Register\n", mcbsp->id);
return -2; return -2;
} }
} }
@ -424,6 +464,7 @@ EXPORT_SYMBOL(omap_mcbsp_pollread);
*/ */
void omap_mcbsp_xmit_word(unsigned int id, u32 word) void omap_mcbsp_xmit_word(unsigned int id, u32 word)
{ {
struct omap_mcbsp *mcbsp;
void __iomem *io_base; void __iomem *io_base;
omap_mcbsp_word_length word_length; omap_mcbsp_word_length word_length;
@ -432,10 +473,11 @@ void omap_mcbsp_xmit_word(unsigned int id, u32 word)
return; return;
} }
io_base = mcbsp[id].io_base; mcbsp = id_to_mcbsp_ptr(id);
word_length = mcbsp[id].tx_word_length; io_base = mcbsp->io_base;
word_length = mcbsp->tx_word_length;
wait_for_completion(&(mcbsp[id].tx_irq_completion)); wait_for_completion(&mcbsp->tx_irq_completion);
if (word_length > OMAP_MCBSP_WORD_16) if (word_length > OMAP_MCBSP_WORD_16)
OMAP_MCBSP_WRITE(io_base, DXR2, word >> 16); OMAP_MCBSP_WRITE(io_base, DXR2, word >> 16);
@ -445,6 +487,7 @@ EXPORT_SYMBOL(omap_mcbsp_xmit_word);
u32 omap_mcbsp_recv_word(unsigned int id) u32 omap_mcbsp_recv_word(unsigned int id)
{ {
struct omap_mcbsp *mcbsp;
void __iomem *io_base; void __iomem *io_base;
u16 word_lsb, word_msb = 0; u16 word_lsb, word_msb = 0;
omap_mcbsp_word_length word_length; omap_mcbsp_word_length word_length;
@ -453,11 +496,12 @@ u32 omap_mcbsp_recv_word(unsigned int id)
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return -ENODEV; return -ENODEV;
} }
mcbsp = id_to_mcbsp_ptr(id);
word_length = mcbsp[id].rx_word_length; word_length = mcbsp->rx_word_length;
io_base = mcbsp[id].io_base; io_base = mcbsp->io_base;
wait_for_completion(&(mcbsp[id].rx_irq_completion)); wait_for_completion(&mcbsp->rx_irq_completion);
if (word_length > OMAP_MCBSP_WORD_16) if (word_length > OMAP_MCBSP_WORD_16)
word_msb = OMAP_MCBSP_READ(io_base, DRR2); word_msb = OMAP_MCBSP_READ(io_base, DRR2);
@ -469,6 +513,7 @@ EXPORT_SYMBOL(omap_mcbsp_recv_word);
int omap_mcbsp_spi_master_xmit_word_poll(unsigned int id, u32 word) int omap_mcbsp_spi_master_xmit_word_poll(unsigned int id, u32 word)
{ {
struct omap_mcbsp *mcbsp;
void __iomem *io_base; void __iomem *io_base;
omap_mcbsp_word_length tx_word_length; omap_mcbsp_word_length tx_word_length;
omap_mcbsp_word_length rx_word_length; omap_mcbsp_word_length rx_word_length;
@ -478,10 +523,10 @@ int omap_mcbsp_spi_master_xmit_word_poll(unsigned int id, u32 word)
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return -ENODEV; return -ENODEV;
} }
mcbsp = id_to_mcbsp_ptr(id);
io_base = mcbsp[id].io_base; io_base = mcbsp->io_base;
tx_word_length = mcbsp[id].tx_word_length; tx_word_length = mcbsp->tx_word_length;
rx_word_length = mcbsp[id].rx_word_length; rx_word_length = mcbsp->rx_word_length;
if (tx_word_length != rx_word_length) if (tx_word_length != rx_word_length)
return -EINVAL; return -EINVAL;
@ -496,8 +541,8 @@ int omap_mcbsp_spi_master_xmit_word_poll(unsigned int id, u32 word)
udelay(10); udelay(10);
OMAP_MCBSP_WRITE(io_base, SPCR2, spcr2 | XRST); OMAP_MCBSP_WRITE(io_base, SPCR2, spcr2 | XRST);
udelay(10); udelay(10);
dev_err(mcbsp[id].dev, "McBSP%d transmitter not " dev_err(mcbsp->dev, "McBSP%d transmitter not "
"ready\n", mcbsp[id].id); "ready\n", mcbsp->id);
return -EAGAIN; return -EAGAIN;
} }
} }
@ -517,8 +562,8 @@ int omap_mcbsp_spi_master_xmit_word_poll(unsigned int id, u32 word)
udelay(10); udelay(10);
OMAP_MCBSP_WRITE(io_base, SPCR1, spcr1 | RRST); OMAP_MCBSP_WRITE(io_base, SPCR1, spcr1 | RRST);
udelay(10); udelay(10);
dev_err(mcbsp[id].dev, "McBSP%d receiver not " dev_err(mcbsp->dev, "McBSP%d receiver not "
"ready\n", mcbsp[id].id); "ready\n", mcbsp->id);
return -EAGAIN; return -EAGAIN;
} }
} }
@ -534,6 +579,7 @@ EXPORT_SYMBOL(omap_mcbsp_spi_master_xmit_word_poll);
int omap_mcbsp_spi_master_recv_word_poll(unsigned int id, u32 *word) int omap_mcbsp_spi_master_recv_word_poll(unsigned int id, u32 *word)
{ {
struct omap_mcbsp *mcbsp;
u32 clock_word = 0; u32 clock_word = 0;
void __iomem *io_base; void __iomem *io_base;
omap_mcbsp_word_length tx_word_length; omap_mcbsp_word_length tx_word_length;
@ -545,9 +591,11 @@ int omap_mcbsp_spi_master_recv_word_poll(unsigned int id, u32 *word)
return -ENODEV; return -ENODEV;
} }
io_base = mcbsp[id].io_base; mcbsp = id_to_mcbsp_ptr(id);
tx_word_length = mcbsp[id].tx_word_length; io_base = mcbsp->io_base;
rx_word_length = mcbsp[id].rx_word_length;
tx_word_length = mcbsp->tx_word_length;
rx_word_length = mcbsp->rx_word_length;
if (tx_word_length != rx_word_length) if (tx_word_length != rx_word_length)
return -EINVAL; return -EINVAL;
@ -562,8 +610,8 @@ int omap_mcbsp_spi_master_recv_word_poll(unsigned int id, u32 *word)
udelay(10); udelay(10);
OMAP_MCBSP_WRITE(io_base, SPCR2, spcr2 | XRST); OMAP_MCBSP_WRITE(io_base, SPCR2, spcr2 | XRST);
udelay(10); udelay(10);
dev_err(mcbsp[id].dev, "McBSP%d transmitter not " dev_err(mcbsp->dev, "McBSP%d transmitter not "
"ready\n", mcbsp[id].id); "ready\n", mcbsp->id);
return -EAGAIN; return -EAGAIN;
} }
} }
@ -583,8 +631,8 @@ int omap_mcbsp_spi_master_recv_word_poll(unsigned int id, u32 *word)
udelay(10); udelay(10);
OMAP_MCBSP_WRITE(io_base, SPCR1, spcr1 | RRST); OMAP_MCBSP_WRITE(io_base, SPCR1, spcr1 | RRST);
udelay(10); udelay(10);
dev_err(mcbsp[id].dev, "McBSP%d receiver not " dev_err(mcbsp->dev, "McBSP%d receiver not "
"ready\n", mcbsp[id].id); "ready\n", mcbsp->id);
return -EAGAIN; return -EAGAIN;
} }
} }
@ -610,6 +658,7 @@ EXPORT_SYMBOL(omap_mcbsp_spi_master_recv_word_poll);
int omap_mcbsp_xmit_buffer(unsigned int id, dma_addr_t buffer, int omap_mcbsp_xmit_buffer(unsigned int id, dma_addr_t buffer,
unsigned int length) unsigned int length)
{ {
struct omap_mcbsp *mcbsp;
int dma_tx_ch; int dma_tx_ch;
int src_port = 0; int src_port = 0;
int dest_port = 0; int dest_port = 0;
@ -619,50 +668,51 @@ int omap_mcbsp_xmit_buffer(unsigned int id, dma_addr_t buffer,
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return -ENODEV; return -ENODEV;
} }
mcbsp = id_to_mcbsp_ptr(id);
if (omap_request_dma(mcbsp[id].dma_tx_sync, "McBSP TX", if (omap_request_dma(mcbsp->dma_tx_sync, "McBSP TX",
omap_mcbsp_tx_dma_callback, omap_mcbsp_tx_dma_callback,
&mcbsp[id], mcbsp,
&dma_tx_ch)) { &dma_tx_ch)) {
dev_err(mcbsp[id].dev, " Unable to request DMA channel for " dev_err(mcbsp->dev, " Unable to request DMA channel for "
"McBSP%d TX. Trying IRQ based TX\n", "McBSP%d TX. Trying IRQ based TX\n",
mcbsp[id].id); mcbsp->id);
return -EAGAIN; return -EAGAIN;
} }
mcbsp[id].dma_tx_lch = dma_tx_ch; mcbsp->dma_tx_lch = dma_tx_ch;
dev_err(mcbsp[id].dev, "McBSP%d TX DMA on channel %d\n", mcbsp[id].id, dev_err(mcbsp->dev, "McBSP%d TX DMA on channel %d\n", mcbsp->id,
dma_tx_ch); dma_tx_ch);
init_completion(&(mcbsp[id].tx_dma_completion)); init_completion(&mcbsp->tx_dma_completion);
if (cpu_class_is_omap1()) { if (cpu_class_is_omap1()) {
src_port = OMAP_DMA_PORT_TIPB; src_port = OMAP_DMA_PORT_TIPB;
dest_port = OMAP_DMA_PORT_EMIFF; dest_port = OMAP_DMA_PORT_EMIFF;
} }
if (cpu_class_is_omap2()) if (cpu_class_is_omap2())
sync_dev = mcbsp[id].dma_tx_sync; sync_dev = mcbsp->dma_tx_sync;
omap_set_dma_transfer_params(mcbsp[id].dma_tx_lch, omap_set_dma_transfer_params(mcbsp->dma_tx_lch,
OMAP_DMA_DATA_TYPE_S16, OMAP_DMA_DATA_TYPE_S16,
length >> 1, 1, length >> 1, 1,
OMAP_DMA_SYNC_ELEMENT, OMAP_DMA_SYNC_ELEMENT,
sync_dev, 0); sync_dev, 0);
omap_set_dma_dest_params(mcbsp[id].dma_tx_lch, omap_set_dma_dest_params(mcbsp->dma_tx_lch,
src_port, src_port,
OMAP_DMA_AMODE_CONSTANT, OMAP_DMA_AMODE_CONSTANT,
mcbsp[id].phys_base + OMAP_MCBSP_REG_DXR1, mcbsp->phys_base + OMAP_MCBSP_REG_DXR1,
0, 0); 0, 0);
omap_set_dma_src_params(mcbsp[id].dma_tx_lch, omap_set_dma_src_params(mcbsp->dma_tx_lch,
dest_port, dest_port,
OMAP_DMA_AMODE_POST_INC, OMAP_DMA_AMODE_POST_INC,
buffer, buffer,
0, 0); 0, 0);
omap_start_dma(mcbsp[id].dma_tx_lch); omap_start_dma(mcbsp->dma_tx_lch);
wait_for_completion(&(mcbsp[id].tx_dma_completion)); wait_for_completion(&mcbsp->tx_dma_completion);
return 0; return 0;
} }
@ -671,6 +721,7 @@ EXPORT_SYMBOL(omap_mcbsp_xmit_buffer);
int omap_mcbsp_recv_buffer(unsigned int id, dma_addr_t buffer, int omap_mcbsp_recv_buffer(unsigned int id, dma_addr_t buffer,
unsigned int length) unsigned int length)
{ {
struct omap_mcbsp *mcbsp;
int dma_rx_ch; int dma_rx_ch;
int src_port = 0; int src_port = 0;
int dest_port = 0; int dest_port = 0;
@ -680,50 +731,51 @@ int omap_mcbsp_recv_buffer(unsigned int id, dma_addr_t buffer,
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return -ENODEV; return -ENODEV;
} }
mcbsp = id_to_mcbsp_ptr(id);
if (omap_request_dma(mcbsp[id].dma_rx_sync, "McBSP RX", if (omap_request_dma(mcbsp->dma_rx_sync, "McBSP RX",
omap_mcbsp_rx_dma_callback, omap_mcbsp_rx_dma_callback,
&mcbsp[id], mcbsp,
&dma_rx_ch)) { &dma_rx_ch)) {
dev_err(mcbsp[id].dev, "Unable to request DMA channel for " dev_err(mcbsp->dev, "Unable to request DMA channel for "
"McBSP%d RX. Trying IRQ based RX\n", "McBSP%d RX. Trying IRQ based RX\n",
mcbsp[id].id); mcbsp->id);
return -EAGAIN; return -EAGAIN;
} }
mcbsp[id].dma_rx_lch = dma_rx_ch; mcbsp->dma_rx_lch = dma_rx_ch;
dev_err(mcbsp[id].dev, "McBSP%d RX DMA on channel %d\n", mcbsp[id].id, dev_err(mcbsp->dev, "McBSP%d RX DMA on channel %d\n", mcbsp->id,
dma_rx_ch); dma_rx_ch);
init_completion(&(mcbsp[id].rx_dma_completion)); init_completion(&mcbsp->rx_dma_completion);
if (cpu_class_is_omap1()) { if (cpu_class_is_omap1()) {
src_port = OMAP_DMA_PORT_TIPB; src_port = OMAP_DMA_PORT_TIPB;
dest_port = OMAP_DMA_PORT_EMIFF; dest_port = OMAP_DMA_PORT_EMIFF;
} }
if (cpu_class_is_omap2()) if (cpu_class_is_omap2())
sync_dev = mcbsp[id].dma_rx_sync; sync_dev = mcbsp->dma_rx_sync;
omap_set_dma_transfer_params(mcbsp[id].dma_rx_lch, omap_set_dma_transfer_params(mcbsp->dma_rx_lch,
OMAP_DMA_DATA_TYPE_S16, OMAP_DMA_DATA_TYPE_S16,
length >> 1, 1, length >> 1, 1,
OMAP_DMA_SYNC_ELEMENT, OMAP_DMA_SYNC_ELEMENT,
sync_dev, 0); sync_dev, 0);
omap_set_dma_src_params(mcbsp[id].dma_rx_lch, omap_set_dma_src_params(mcbsp->dma_rx_lch,
src_port, src_port,
OMAP_DMA_AMODE_CONSTANT, OMAP_DMA_AMODE_CONSTANT,
mcbsp[id].phys_base + OMAP_MCBSP_REG_DRR1, mcbsp->phys_base + OMAP_MCBSP_REG_DRR1,
0, 0); 0, 0);
omap_set_dma_dest_params(mcbsp[id].dma_rx_lch, omap_set_dma_dest_params(mcbsp->dma_rx_lch,
dest_port, dest_port,
OMAP_DMA_AMODE_POST_INC, OMAP_DMA_AMODE_POST_INC,
buffer, buffer,
0, 0); 0, 0);
omap_start_dma(mcbsp[id].dma_rx_lch); omap_start_dma(mcbsp->dma_rx_lch);
wait_for_completion(&(mcbsp[id].rx_dma_completion)); wait_for_completion(&mcbsp->rx_dma_completion);
return 0; return 0;
} }
@ -738,12 +790,14 @@ EXPORT_SYMBOL(omap_mcbsp_recv_buffer);
void omap_mcbsp_set_spi_mode(unsigned int id, void omap_mcbsp_set_spi_mode(unsigned int id,
const struct omap_mcbsp_spi_cfg *spi_cfg) const struct omap_mcbsp_spi_cfg *spi_cfg)
{ {
struct omap_mcbsp *mcbsp;
struct omap_mcbsp_reg_cfg mcbsp_cfg; struct omap_mcbsp_reg_cfg mcbsp_cfg;
if (!omap_mcbsp_check_valid_id(id)) { if (!omap_mcbsp_check_valid_id(id)) {
printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1); printk(KERN_ERR "%s: Invalid id (%d)\n", __func__, id + 1);
return; return;
} }
mcbsp = id_to_mcbsp_ptr(id);
memset(&mcbsp_cfg, 0, sizeof(struct omap_mcbsp_reg_cfg)); memset(&mcbsp_cfg, 0, sizeof(struct omap_mcbsp_reg_cfg));
@ -807,6 +861,7 @@ EXPORT_SYMBOL(omap_mcbsp_set_spi_mode);
static int __devinit omap_mcbsp_probe(struct platform_device *pdev) static int __devinit omap_mcbsp_probe(struct platform_device *pdev)
{ {
struct omap_mcbsp_platform_data *pdata = pdev->dev.platform_data; struct omap_mcbsp_platform_data *pdata = pdev->dev.platform_data;
struct omap_mcbsp *mcbsp;
int id = pdev->id - 1; int id = pdev->id - 1;
int ret = 0; int ret = 0;
@ -819,51 +874,58 @@ static int __devinit omap_mcbsp_probe(struct platform_device *pdev)
dev_dbg(&pdev->dev, "Initializing OMAP McBSP (%d).\n", pdev->id); dev_dbg(&pdev->dev, "Initializing OMAP McBSP (%d).\n", pdev->id);
if (id >= OMAP_MAX_MCBSP_COUNT) { if (id >= omap_mcbsp_count) {
dev_err(&pdev->dev, "Invalid McBSP device id (%d)\n", id); dev_err(&pdev->dev, "Invalid McBSP device id (%d)\n", id);
ret = -EINVAL; ret = -EINVAL;
goto exit; goto exit;
} }
spin_lock_init(&mcbsp[id].lock); mcbsp = kzalloc(sizeof(struct omap_mcbsp), GFP_KERNEL);
mcbsp[id].id = id + 1; if (!mcbsp) {
mcbsp[id].free = 1; ret = -ENOMEM;
mcbsp[id].dma_tx_lch = -1; goto exit;
mcbsp[id].dma_rx_lch = -1; }
mcbsp_ptr[id] = mcbsp;
mcbsp[id].phys_base = pdata->phys_base; spin_lock_init(&mcbsp->lock);
mcbsp[id].io_base = ioremap(pdata->phys_base, SZ_4K); mcbsp->id = id + 1;
if (!mcbsp[id].io_base) { mcbsp->free = 1;
mcbsp->dma_tx_lch = -1;
mcbsp->dma_rx_lch = -1;
mcbsp->phys_base = pdata->phys_base;
mcbsp->io_base = ioremap(pdata->phys_base, SZ_4K);
if (!mcbsp->io_base) {
ret = -ENOMEM; ret = -ENOMEM;
goto err_ioremap; goto err_ioremap;
} }
/* Default I/O is IRQ based */ /* Default I/O is IRQ based */
mcbsp[id].io_type = OMAP_MCBSP_IRQ_IO; mcbsp->io_type = OMAP_MCBSP_IRQ_IO;
mcbsp[id].tx_irq = pdata->tx_irq; mcbsp->tx_irq = pdata->tx_irq;
mcbsp[id].rx_irq = pdata->rx_irq; mcbsp->rx_irq = pdata->rx_irq;
mcbsp[id].dma_rx_sync = pdata->dma_rx_sync; mcbsp->dma_rx_sync = pdata->dma_rx_sync;
mcbsp[id].dma_tx_sync = pdata->dma_tx_sync; mcbsp->dma_tx_sync = pdata->dma_tx_sync;
if (pdata->clk_name) if (pdata->clk_name)
mcbsp[id].clk = clk_get(&pdev->dev, pdata->clk_name); mcbsp->clk = clk_get(&pdev->dev, pdata->clk_name);
if (IS_ERR(mcbsp[id].clk)) { if (IS_ERR(mcbsp->clk)) {
dev_err(&pdev->dev, dev_err(&pdev->dev,
"Invalid clock configuration for McBSP%d.\n", "Invalid clock configuration for McBSP%d.\n",
mcbsp[id].id); mcbsp->id);
ret = PTR_ERR(mcbsp[id].clk); ret = PTR_ERR(mcbsp->clk);
goto err_clk; goto err_clk;
} }
mcbsp[id].pdata = pdata; mcbsp->pdata = pdata;
mcbsp[id].dev = &pdev->dev; mcbsp->dev = &pdev->dev;
platform_set_drvdata(pdev, &mcbsp[id]); platform_set_drvdata(pdev, mcbsp);
return 0; return 0;
err_clk: err_clk:
iounmap(mcbsp[id].io_base); iounmap(mcbsp->io_base);
err_ioremap: err_ioremap:
mcbsp[id].free = 0; mcbsp->free = 0;
exit: exit:
return ret; return ret;
} }