On Mon, 2008-09-08 at 13:06 +0400, Dmitry Baryshkov wrote: > ASoC and non-ASoC drivers for ACLINK on PXA share lot's of common code. > Move all common code into separate file. > > Signed-off-by: Dmitry Baryshkov <dbaryshkov@xxxxxxxxx> > --- > sound/arm/pxa2xx-ac97-lib.c | 275 +++++++++++++++++++++++++++++++++++++++++++ > sound/arm/pxa2xx-ac97.c | 233 +++--------------------------------- > sound/soc/pxa/pxa2xx-ac97.c | 257 ++-------------------------------------- > 3 files changed, 302 insertions(+), 463 deletions(-) > create mode 100644 sound/arm/pxa2xx-ac97-lib.c > > diff --git a/sound/arm/pxa2xx-ac97-lib.c b/sound/arm/pxa2xx-ac97-lib.c > new file mode 100644 > index 0000000..ee0de4d > --- /dev/null > +++ b/sound/arm/pxa2xx-ac97-lib.c > @@ -0,0 +1,275 @@ > + > +static DEFINE_MUTEX(car_mutex); > +static DECLARE_WAIT_QUEUE_HEAD(gsr_wq); > +static volatile long gsr_bits; > +static struct clk *ac97_clk; > +#ifdef CONFIG_PXA27x > +static struct clk *ac97conf_clk; > +#endif > + > +/* > + * Beware PXA27x bugs: > + * > + * o Slot 12 read from modem space will hang controller. > + * o CDONE, SDONE interrupt fails after any slot 12 IO. > + * > + * We therefore have an hybrid approach for waiting on SDONE (interrupt or > + * 1 jiffy timeout if interrupt never comes). > + */ > + > +static unsigned short pxa2xx_ac97_read(struct snd_ac97 *ac97, unsigned short reg) > +{ > + unsigned short val = -1; > + volatile u32 *reg_addr; > + > + mutex_lock(&car_mutex); > + > + /* set up primary or secondary codec space */ > +#if defined(CONFIG_PXA27x) || defined(CONFIG_PXA3xx) > + reg_addr = (ac97->num & 1) ? &SAC_REG_BASE : &PAC_REG_BASE; > +#else > + if (reg == AC97_GPIO_STATUS) > + reg_addr = ac97->num ? &SMC_REG_BASE : &PMC_REG_BASE; > + else > + reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; > +#endif > + reg_addr += (reg >> 1); > + > + /* start read access across the ac97 link */ > + GSR = GSR_CDONE | GSR_SDONE; > + gsr_bits = 0; > + val = *reg_addr; > + if (reg == AC97_GPIO_STATUS) > + goto out; > + if (wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_SDONE, 1) <= 0 && > + !((GSR | gsr_bits) & GSR_SDONE)) { > + printk(KERN_ERR "%s: read error (ac97_reg=%d GSR=%#lx)\n", > + __func__, reg, GSR | gsr_bits); > + val = -1; > + goto out; > + } > + > + /* valid data now */ > + GSR = GSR_CDONE | GSR_SDONE; > + gsr_bits = 0; > + val = *reg_addr; > + /* but we've just started another cycle... */ > + wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_SDONE, 1); > + > +out: mutex_unlock(&car_mutex); > + return val; > +} > + > +static void pxa2xx_ac97_write(struct snd_ac97 *ac97, unsigned short reg, unsigned short val) > +{ > + volatile u32 *reg_addr; > + > + mutex_lock(&car_mutex); > + > + /* set up primary or secondary codec space */ > +#if defined(CONFIG_PXA27x) || defined(CONFIG_PXA3xx) > + reg_addr = (ac97->num & 1) ? &SAC_REG_BASE : &PAC_REG_BASE; > +#else > + if (reg == AC97_GPIO_STATUS) > + reg_addr = ac97->num ? &SMC_REG_BASE : &PMC_REG_BASE; > + else > + reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; > +#endif > + reg_addr += (reg >> 1); > + > + GSR = GSR_CDONE | GSR_SDONE; > + gsr_bits = 0; > + *reg_addr = val; > + if (wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_CDONE, 1) <= 0 && > + !((GSR | gsr_bits) & GSR_CDONE)) > + printk(KERN_ERR "%s: write error (ac97_reg=%d GSR=%#lx)\n", > + __func__, reg, GSR | gsr_bits); > + > + mutex_unlock(&car_mutex); > +} > + > +static bool pxa2xx_ac97_try_warm_reset(struct snd_ac97 *ac97) > +{ > +#ifdef CONFIG_PXA3xx > + int timeout = 100; > +#endif > + gsr_bits = 0; > + > +#ifdef CONFIG_PXA27x > + /* warm reset broken on Bulverde, > + so manually keep AC97 reset high */ > + pxa_gpio_mode(113 | GPIO_OUT | GPIO_DFLT_HIGH); > + udelay(10); > + GCR |= GCR_WARM_RST; > + pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); > + udelay(500); > +#elif defined(CONFIG_PXA3xx) > + /* Can't use interrupts */ > + GCR |= GCR_WARM_RST; > + while (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR)) && timeout--) > + mdelay(1); > +#else > + GCR |= GCR_WARM_RST | GCR_PRIRDY_IEN | GCR_SECRDY_IEN; > + wait_event_timeout(gsr_wq, gsr_bits & (GSR_PCR | GSR_SCR), 1); > +#endif > + > + if (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR))) > + return false; > + > + return true; > +} > + > +static bool pxa2xx_ac97_try_cold_reset(struct snd_ac97 *ac97) > +{ > +#ifdef CONFIG_PXA3xx > + int timeout = 1000; > + > + /* Hold CLKBPB for 100us */ > + GCR = 0; > + GCR = GCR_CLKBPB; > + udelay(100); > + GCR = 0; > +#endif > + > + GCR &= GCR_COLD_RST; /* clear everything but nCRST */ > + GCR &= ~GCR_COLD_RST; /* then assert nCRST */ > + > + gsr_bits = 0; > +#ifdef CONFIG_PXA27x > + /* PXA27x Developers Manual section 13.5.2.2.1 */ > + clk_enable(ac97conf_clk); > + udelay(5); > + clk_disable(ac97conf_clk); > + GCR = GCR_COLD_RST; > + udelay(50); > +#elif defined(CONFIG_PXA3xx) > + /* Can't use interrupts on PXA3xx */ > + GCR &= ~(GCR_PRIRDY_IEN|GCR_SECRDY_IEN); > + > + GCR = GCR_WARM_RST | GCR_COLD_RST; > + while (!(GSR & (GSR_PCR | GSR_SCR)) && timeout--) > + mdelay(10); > +#else > + GCR = GCR_COLD_RST; > + GCR |= GCR_CDONE_IE|GCR_SDONE_IE; > + wait_event_timeout(gsr_wq, gsr_bits & (GSR_PCR | GSR_SCR), 1); > +#endif > + > + if (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR))) > + return false; > + > + return true; > +} > + > + > +static void pxa2xx_ac97_finish_reset(struct snd_ac97 *ac97) > +{ > + GCR &= ~(GCR_PRIRDY_IEN|GCR_SECRDY_IEN); > + GCR |= GCR_SDONE_IE|GCR_CDONE_IE; > +} > + > +static irqreturn_t pxa2xx_ac97_irq(int irq, void *dev_id) > +{ > + long status; > + > + status = GSR; > + if (status) { > + GSR = status; > + gsr_bits |= status; > + wake_up(&gsr_wq); > + > +#ifdef CONFIG_PXA27x > + /* Although we don't use those we still need to clear them > + since they tend to spuriously trigger when MMC is used > + (hardware bug? go figure)... */ > + MISR = MISR_EOC; > + PISR = PISR_EOC; > + MCSR = MCSR_EOC; > +#endif > + > + return IRQ_HANDLED; > + } > + > + return IRQ_NONE; > +} > + > +#ifdef CONFIG_PM > +static int pxa2xx_ac97_hw_suspend(void) > +{ > + GCR |= GCR_ACLINK_OFF; > + clk_disable(ac97_clk); > + return 0; > +} > + > +static int pxa2xx_ac97_hw_resume(void) > +{ > + pxa_gpio_mode(GPIO31_SYNC_AC97_MD); > + pxa_gpio_mode(GPIO30_SDATA_OUT_AC97_MD); > + pxa_gpio_mode(GPIO28_BITCLK_AC97_MD); > + pxa_gpio_mode(GPIO29_SDATA_IN_AC97_MD); > +#ifdef CONFIG_PXA27x > + /* Use GPIO 113 as AC97 Reset on Bulverde */ > + pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); > +#endif > + clk_enable(ac97_clk); > + return 0; > +} > +#endif > + > +static int __devinit pxa2xx_ac97_hw_probe(struct platform_device *dev) > +{ > + int ret; > + > + ret = request_irq(IRQ_AC97, pxa2xx_ac97_irq, 0, "AC97", NULL); > + if (ret < 0) > + goto err; > + > + pxa_gpio_mode(GPIO31_SYNC_AC97_MD); > + pxa_gpio_mode(GPIO30_SDATA_OUT_AC97_MD); > + pxa_gpio_mode(GPIO28_BITCLK_AC97_MD); > + pxa_gpio_mode(GPIO29_SDATA_IN_AC97_MD); > +#ifdef CONFIG_PXA27x > + /* Use GPIO 113 as AC97 Reset on Bulverde */ > + pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); > + ac97conf_clk = clk_get(&dev->dev, "AC97CONFCLK"); > + if (IS_ERR(ac97conf_clk)) { > + ret = PTR_ERR(ac97conf_clk); > + ac97conf_clk = NULL; > + goto err_irq; > + } > +#endif > + > + ac97_clk = clk_get(&dev->dev, "AC97CLK"); > + if (IS_ERR(ac97_clk)) { > + ret = PTR_ERR(ac97_clk); > + ac97_clk = NULL; > + goto err_irq; > + } > + > + return clk_enable(ac97_clk); > + > +err_irq: > + GCR |= GCR_ACLINK_OFF; > +#ifdef CONFIG_PXA27x > + if (ac97conf_clk) { > + clk_put(ac97conf_clk); > + ac97conf_clk = NULL; > + } > +#endif > + free_irq(IRQ_AC97, NULL); > +err: > + return ret; > +} > + > +static void pxa2xx_ac97_hw_remove(struct platform_device *dev) > +{ > + GCR |= GCR_ACLINK_OFF; > + free_irq(IRQ_AC97, NULL); > +#ifdef CONFIG_PXA27x > + clk_put(ac97conf_clk); > + ac97conf_clk = NULL; > +#endif > + clk_disable(ac97_clk); > + clk_put(ac97_clk); > + ac97_clk = NULL; > +} > diff --git a/sound/arm/pxa2xx-ac97.c b/sound/arm/pxa2xx-ac97.c > index 199cca3..320ac75 100644 > --- a/sound/arm/pxa2xx-ac97.c > +++ b/sound/arm/pxa2xx-ac97.c > @@ -33,177 +33,20 @@ > > #include "pxa2xx-pcm.h" > > - > -static DEFINE_MUTEX(car_mutex); > -static DECLARE_WAIT_QUEUE_HEAD(gsr_wq); > -static volatile long gsr_bits; > -static struct clk *ac97_clk; > -#ifdef CONFIG_PXA27x > -static struct clk *ac97conf_clk; > -#endif > - > -/* > - * Beware PXA27x bugs: > - * > - * o Slot 12 read from modem space will hang controller. > - * o CDONE, SDONE interrupt fails after any slot 12 IO. > - * > - * We therefore have an hybrid approach for waiting on SDONE (interrupt or > - * 1 jiffy timeout if interrupt never comes). > - */ > - > -static unsigned short pxa2xx_ac97_read(struct snd_ac97 *ac97, unsigned short reg) > -{ > - unsigned short val = -1; > - volatile u32 *reg_addr; > - > - mutex_lock(&car_mutex); > - > - /* set up primary or secondary codec space */ > - reg_addr = (ac97->num & 1) ? &SAC_REG_BASE : &PAC_REG_BASE; > - reg_addr += (reg >> 1); > - > - /* start read access across the ac97 link */ > - GSR = GSR_CDONE | GSR_SDONE; > - gsr_bits = 0; > - val = *reg_addr; > - if (reg == AC97_GPIO_STATUS) > - goto out; > - if (wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_SDONE, 1) <= 0 && > - !((GSR | gsr_bits) & GSR_SDONE)) { > - printk(KERN_ERR "%s: read error (ac97_reg=%d GSR=%#lx)\n", > - __func__, reg, GSR | gsr_bits); > - val = -1; > - goto out; > - } > - > - /* valid data now */ > - GSR = GSR_CDONE | GSR_SDONE; > - gsr_bits = 0; > - val = *reg_addr; > - /* but we've just started another cycle... */ > - wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_SDONE, 1); > - > -out: mutex_unlock(&car_mutex); > - return val; > -} > - > -static void pxa2xx_ac97_write(struct snd_ac97 *ac97, unsigned short reg, unsigned short val) > -{ > - volatile u32 *reg_addr; > - > - mutex_lock(&car_mutex); > - > - /* set up primary or secondary codec space */ > - reg_addr = (ac97->num & 1) ? &SAC_REG_BASE : &PAC_REG_BASE; > - reg_addr += (reg >> 1); > - > - GSR = GSR_CDONE | GSR_SDONE; > - gsr_bits = 0; > - *reg_addr = val; > - if (wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_CDONE, 1) <= 0 && > - !((GSR | gsr_bits) & GSR_CDONE)) > - printk(KERN_ERR "%s: write error (ac97_reg=%d GSR=%#lx)\n", > - __func__, reg, GSR | gsr_bits); > - > - mutex_unlock(&car_mutex); > -} > +#include "pxa2xx-ac97-lib.c" > Any reason for not linking with pxa2xx-ac97-lib.o ? > static void pxa2xx_ac97_reset(struct snd_ac97 *ac97) > { > - /* First, try cold reset */ > -#ifdef CONFIG_PXA3xx > - int timeout; > - > - /* Hold CLKBPB for 100us */ > - GCR = 0; > - GCR = GCR_CLKBPB; > - udelay(100); > - GCR = 0; > -#endif > - > - GCR &= GCR_COLD_RST; /* clear everything but nCRST */ > - GCR &= ~GCR_COLD_RST; /* then assert nCRST */ > - > - gsr_bits = 0; > -#ifdef CONFIG_PXA27x > - /* PXA27x Developers Manual section 13.5.2.2.1 */ > - clk_enable(ac97conf_clk); > - udelay(5); > - clk_disable(ac97conf_clk); > - GCR = GCR_COLD_RST; > - udelay(50); > -#elif defined(CONFIG_PXA3xx) > - timeout = 1000; > - /* Can't use interrupts on PXA3xx */ > - GCR &= ~(GCR_PRIRDY_IEN|GCR_SECRDY_IEN); > - > - GCR = GCR_WARM_RST | GCR_COLD_RST; > - while (!(GSR & (GSR_PCR | GSR_SCR)) && timeout--) > - mdelay(10); > -#else > - GCR = GCR_COLD_RST; > - GCR |= GCR_CDONE_IE|GCR_SDONE_IE; > - wait_event_timeout(gsr_wq, gsr_bits & (GSR_PCR | GSR_SCR), 1); > -#endif > - > - if (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR))) { > + if (!pxa2xx_ac97_try_cold_reset(ac97)) { > printk(KERN_INFO "%s: cold reset timeout (GSR=%#lx)\n", > __func__, gsr_bits); > > - /* let's try warm reset */ > - gsr_bits = 0; > -#ifdef CONFIG_PXA27x > - /* warm reset broken on Bulverde, > - so manually keep AC97 reset high */ > - pxa_gpio_mode(113 | GPIO_OUT | GPIO_DFLT_HIGH); > - udelay(10); > - GCR |= GCR_WARM_RST; > - pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); > - udelay(500); > -#elif defined(CONFIG_PXA3xx) > - timeout = 100; > - /* Can't use interrupts */ > - GCR |= GCR_WARM_RST; > - while (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR)) && timeout--) > - mdelay(1); > -#else > - GCR |= GCR_WARM_RST|GCR_PRIRDY_IEN|GCR_SECRDY_IEN; > - wait_event_timeout(gsr_wq, gsr_bits & (GSR_PCR | GSR_SCR), 1); > -#endif > - > - if (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR))) > + if (!pxa2xx_ac97_try_warm_reset(ac97)) > printk(KERN_INFO "%s: warm reset timeout (GSR=%#lx)\n", > __func__, gsr_bits); > } > > - GCR &= ~(GCR_PRIRDY_IEN|GCR_SECRDY_IEN); > - GCR |= GCR_SDONE_IE|GCR_CDONE_IE; > -} > - > -static irqreturn_t pxa2xx_ac97_irq(int irq, void *dev_id) > -{ > - long status; > - > - status = GSR; > - if (status) { > - GSR = status; > - gsr_bits |= status; > - wake_up(&gsr_wq); > - > -#ifdef CONFIG_PXA27x > - /* Although we don't use those we still need to clear them > - since they tend to spuriously trigger when MMC is used > - (hardware bug? go figure)... */ > - MISR = MISR_EOC; > - PISR = PISR_EOC; > - MCSR = MCSR_EOC; > -#endif > - > - return IRQ_HANDLED; > - } > - > - return IRQ_NONE; > + pxa2xx_ac97_finish_reset(ac97); > } > > static struct snd_ac97_bus_ops pxa2xx_ac97_ops = { > @@ -288,17 +131,19 @@ static int pxa2xx_ac97_do_suspend(struct snd_card *card, pm_message_t state) > snd_ac97_suspend(pxa2xx_ac97_ac97); > if (platform_ops && platform_ops->suspend) > platform_ops->suspend(platform_ops->priv); > - GCR |= GCR_ACLINK_OFF; > - clk_disable(ac97_clk); > > - return 0; > + return pxa2xx_ac97_hw_suspend(); > } > > static int pxa2xx_ac97_do_resume(struct snd_card *card) > { > pxa2xx_audio_ops_t *platform_ops = card->dev->platform_data; > + int rc; > + > + rc = pxa2xx_ac97_hw_resume(); > + if (rc) > + return rc; > > - clk_enable(ac97_clk); > if (platform_ops && platform_ops->resume) > platform_ops->resume(platform_ops->priv); > snd_ac97_resume(pxa2xx_ac97_ac97); > @@ -354,40 +199,17 @@ static int __devinit pxa2xx_ac97_probe(struct platform_device *dev) > if (ret) > goto err; > > - ret = request_irq(IRQ_AC97, pxa2xx_ac97_irq, 0, "AC97", NULL); > - if (ret < 0) > - goto err; > - > - pxa_gpio_mode(GPIO31_SYNC_AC97_MD); > - pxa_gpio_mode(GPIO30_SDATA_OUT_AC97_MD); > - pxa_gpio_mode(GPIO28_BITCLK_AC97_MD); > - pxa_gpio_mode(GPIO29_SDATA_IN_AC97_MD); > -#ifdef CONFIG_PXA27x > - /* Use GPIO 113 as AC97 Reset on Bulverde */ > - pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); > - ac97conf_clk = clk_get(&dev->dev, "AC97CONFCLK"); > - if (IS_ERR(ac97conf_clk)) { > - ret = PTR_ERR(ac97conf_clk); > - ac97conf_clk = NULL; > - goto err; > - } > -#endif > - > - ac97_clk = clk_get(&dev->dev, "AC97CLK"); > - if (IS_ERR(ac97_clk)) { > - ret = PTR_ERR(ac97_clk); > - ac97_clk = NULL; > + ret = pxa2xx_ac97_hw_probe(dev); > + if (ret) > goto err; > - } > - clk_enable(ac97_clk); > > ret = snd_ac97_bus(card, 0, &pxa2xx_ac97_ops, NULL, &ac97_bus); > if (ret) > - goto err; > + goto err_remove; > memset(&ac97_template, 0, sizeof(ac97_template)); > ret = snd_ac97_mixer(ac97_bus, &ac97_template, &pxa2xx_ac97_ac97); > if (ret) > - goto err; > + goto err_remove; > > snprintf(card->shortname, sizeof(card->shortname), > "%s", snd_ac97_get_short_name(pxa2xx_ac97_ac97)); > @@ -401,22 +223,11 @@ static int __devinit pxa2xx_ac97_probe(struct platform_device *dev) > return 0; > } > > - err: > +err_remove: > + pxa2xx_ac97_hw_remove(dev); > +err: > if (card) > snd_card_free(card); > - if (ac97_clk) { > - GCR |= GCR_ACLINK_OFF; > - free_irq(IRQ_AC97, NULL); > - clk_disable(ac97_clk); > - clk_put(ac97_clk); > - ac97_clk = NULL; > - } > -#ifdef CONFIG_PXA27x > - if (ac97conf_clk) { > - clk_put(ac97conf_clk); > - ac97conf_clk = NULL; > - } > -#endif > return ret; > } > > @@ -427,15 +238,7 @@ static int __devexit pxa2xx_ac97_remove(struct platform_device *dev) > if (card) { > snd_card_free(card); > platform_set_drvdata(dev, NULL); > - GCR |= GCR_ACLINK_OFF; > - free_irq(IRQ_AC97, NULL); > - clk_disable(ac97_clk); > - clk_put(ac97_clk); > - ac97_clk = NULL; > -#ifdef CONFIG_PXA27x > - clk_put(ac97conf_clk); > - ac97conf_clk = NULL; > -#endif > + pxa2xx_ac97_hw_remove(dev); > } > > return 0; > diff --git a/sound/soc/pxa/pxa2xx-ac97.c b/sound/soc/pxa/pxa2xx-ac97.c > index d94a495..99f096d 100644 > --- a/sound/soc/pxa/pxa2xx-ac97.c > +++ b/sound/soc/pxa/pxa2xx-ac97.c > @@ -34,204 +34,24 @@ > #include "pxa2xx-pcm.h" > #include "pxa2xx-ac97.h" > > -static DEFINE_MUTEX(car_mutex); > -static DECLARE_WAIT_QUEUE_HEAD(gsr_wq); > -static volatile long gsr_bits; > -static struct clk *ac97_clk; > -#ifdef CONFIG_PXA27x > -static struct clk *ac97conf_clk; > -#endif > - > -/* > - * Beware PXA27x bugs: > - * > - * o Slot 12 read from modem space will hang controller. > - * o CDONE, SDONE interrupt fails after any slot 12 IO. > - * > - * We therefore have an hybrid approach for waiting on SDONE (interrupt or > - * 1 jiffy timeout if interrupt never comes). > - */ > - > -static unsigned short pxa2xx_ac97_read(struct snd_ac97 *ac97, > - unsigned short reg) > -{ > - unsigned short val = -1; > - volatile u32 *reg_addr; > - > - mutex_lock(&car_mutex); > - > - /* set up primary or secondary codec/modem space */ > -#if defined(CONFIG_PXA27x) || defined(CONFIG_PXA3xx) > - reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; > -#else > - if (reg == AC97_GPIO_STATUS) > - reg_addr = ac97->num ? &SMC_REG_BASE : &PMC_REG_BASE; > - else > - reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; > -#endif > - reg_addr += (reg >> 1); > - > -#ifndef CONFIG_PXA27x > - if (reg == AC97_GPIO_STATUS) { > - /* read from controller cache */ > - val = *reg_addr; > - goto out; > - } > -#endif > - > - /* start read access across the ac97 link */ > - GSR = GSR_CDONE | GSR_SDONE; > - gsr_bits = 0; > - val = *reg_addr; > - > - wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_SDONE, 1); > - if (!((GSR | gsr_bits) & GSR_SDONE)) { > - printk(KERN_ERR "%s: read error (ac97_reg=%x GSR=%#lx)\n", > - __func__, reg, GSR | gsr_bits); > - val = -1; > - goto out; > - } > - > - /* valid data now */ > - GSR = GSR_CDONE | GSR_SDONE; > - gsr_bits = 0; > - val = *reg_addr; > - /* but we've just started another cycle... */ > - wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_SDONE, 1); > - > -out: mutex_unlock(&car_mutex); > - return val; > -} > - > -static void pxa2xx_ac97_write(struct snd_ac97 *ac97, unsigned short reg, > - unsigned short val) > -{ > - volatile u32 *reg_addr; > - > - mutex_lock(&car_mutex); > - > - /* set up primary or secondary codec/modem space */ > -#if defined(CONFIG_PXA27x) || defined(CONFIG_PXA3xx) > - reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; > -#else > - if (reg == AC97_GPIO_STATUS) > - reg_addr = ac97->num ? &SMC_REG_BASE : &PMC_REG_BASE; > - else > - reg_addr = ac97->num ? &SAC_REG_BASE : &PAC_REG_BASE; > -#endif > - reg_addr += (reg >> 1); > - > - GSR = GSR_CDONE | GSR_SDONE; > - gsr_bits = 0; > - *reg_addr = val; > - wait_event_timeout(gsr_wq, (GSR | gsr_bits) & GSR_CDONE, 1); > - if (!((GSR | gsr_bits) & GSR_CDONE)) > - printk(KERN_ERR "%s: write error (ac97_reg=%x GSR=%#lx)\n", > - __func__, reg, GSR | gsr_bits); > - > - mutex_unlock(&car_mutex); > -} > +#include "../../arm/pxa2xx-ac97-lib.c" > ditto > static void pxa2xx_ac97_warm_reset(struct snd_ac97 *ac97) > { > -#ifdef CONFIG_PXA3xx > - int timeout = 100; > -#endif > - gsr_bits = 0; > - > -#ifdef CONFIG_PXA27x > - /* warm reset broken on Bulverde, > - so manually keep AC97 reset high */ > - pxa_gpio_mode(113 | GPIO_OUT | GPIO_DFLT_HIGH); > - udelay(10); > - GCR |= GCR_WARM_RST; > - pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); > - udelay(500); > -#elif defined(CONFIG_PXA3xx) > - /* Can't use interrupts */ > - GCR |= GCR_WARM_RST; > - while (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR)) && timeout--) > - mdelay(1); > -#else > - GCR |= GCR_WARM_RST | GCR_PRIRDY_IEN | GCR_SECRDY_IEN; > - wait_event_timeout(gsr_wq, gsr_bits & (GSR_PCR | GSR_SCR), 1); > -#endif > - > - if (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR))) > + if (!pxa2xx_ac97_try_warm_reset(ac97)) > printk(KERN_INFO "%s: warm reset timeout (GSR=%#lx)\n", > __func__, gsr_bits); > > - GCR &= ~(GCR_PRIRDY_IEN|GCR_SECRDY_IEN); > - GCR |= GCR_SDONE_IE|GCR_CDONE_IE; > + pxa2xx_ac97_finish_reset(ac97); > } > > static void pxa2xx_ac97_cold_reset(struct snd_ac97 *ac97) > { > -#ifdef CONFIG_PXA3xx > - int timeout = 1000; > - > - /* Hold CLKBPB for 100us */ > - GCR = 0; > - GCR = GCR_CLKBPB; > - udelay(100); > - GCR = 0; > -#endif > - > - GCR &= GCR_COLD_RST; /* clear everything but nCRST */ > - GCR &= ~GCR_COLD_RST; /* then assert nCRST */ > - > - gsr_bits = 0; > -#ifdef CONFIG_PXA27x > - /* PXA27x Developers Manual section 13.5.2.2.1 */ > - clk_enable(ac97conf_clk); > - udelay(5); > - clk_disable(ac97conf_clk); > - GCR = GCR_COLD_RST; > - udelay(50); > -#elif defined(CONFIG_PXA3xx) > - /* Can't use interrupts on PXA3xx */ > - GCR &= ~(GCR_PRIRDY_IEN|GCR_SECRDY_IEN); > - > - GCR = GCR_WARM_RST | GCR_COLD_RST; > - while (!(GSR & (GSR_PCR | GSR_SCR)) && timeout--) > - mdelay(10); > -#else > - GCR = GCR_COLD_RST; > - GCR |= GCR_CDONE_IE|GCR_SDONE_IE; > - wait_event_timeout(gsr_wq, gsr_bits & (GSR_PCR | GSR_SCR), 1); > -#endif > - > - if (!((GSR | gsr_bits) & (GSR_PCR | GSR_SCR))) > + if (!pxa2xx_ac97_try_cold_reset(ac97)) > printk(KERN_INFO "%s: cold reset timeout (GSR=%#lx)\n", > __func__, gsr_bits); > > - GCR &= ~(GCR_PRIRDY_IEN|GCR_SECRDY_IEN); > - GCR |= GCR_SDONE_IE|GCR_CDONE_IE; > -} > - > -static irqreturn_t pxa2xx_ac97_irq(int irq, void *dev_id) > -{ > - long status; > - > - status = GSR; > - if (status) { > - GSR = status; > - gsr_bits |= status; > - wake_up(&gsr_wq); > - > -#ifdef CONFIG_PXA27x > - /* Although we don't use those we still need to clear them > - since they tend to spuriously trigger when MMC is used > - (hardware bug? go figure)... */ > - MISR = MISR_EOC; > - PISR = PISR_EOC; > - MCSR = MCSR_EOC; > -#endif > - > - return IRQ_HANDLED; > - } > - > - return IRQ_NONE; > + pxa2xx_ac97_finish_reset(ac97); > } > > struct snd_ac97_bus_ops soc_ac97_ops = { > @@ -285,24 +105,13 @@ static struct pxa2xx_pcm_dma_params pxa2xx_ac97_pcm_mic_mono_in = { > static int pxa2xx_ac97_suspend(struct platform_device *pdev, > struct snd_soc_dai *dai) > { > - GCR |= GCR_ACLINK_OFF; > - clk_disable(ac97_clk); > - return 0; > + return pxa2xx_ac97_hw_suspend(); > } > > static int pxa2xx_ac97_resume(struct platform_device *pdev, > struct snd_soc_dai *dai) > { > - pxa_gpio_mode(GPIO31_SYNC_AC97_MD); > - pxa_gpio_mode(GPIO30_SDATA_OUT_AC97_MD); > - pxa_gpio_mode(GPIO28_BITCLK_AC97_MD); > - pxa_gpio_mode(GPIO29_SDATA_IN_AC97_MD); > -#ifdef CONFIG_PXA27x > - /* Use GPIO 113 as AC97 Reset on Bulverde */ > - pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); > -#endif > - clk_enable(ac97_clk); > - return 0; > + return pxa2xx_ac97_hw_resume(); > } > > #else > @@ -313,61 +122,13 @@ static int pxa2xx_ac97_resume(struct platform_device *pdev, > static int pxa2xx_ac97_probe(struct platform_device *pdev, > struct snd_soc_dai *dai) > { > - int ret; > - > - ret = request_irq(IRQ_AC97, pxa2xx_ac97_irq, IRQF_DISABLED, "AC97", NULL); > - if (ret < 0) > - goto err; > - > - pxa_gpio_mode(GPIO31_SYNC_AC97_MD); > - pxa_gpio_mode(GPIO30_SDATA_OUT_AC97_MD); > - pxa_gpio_mode(GPIO28_BITCLK_AC97_MD); > - pxa_gpio_mode(GPIO29_SDATA_IN_AC97_MD); > -#ifdef CONFIG_PXA27x > - /* Use GPIO 113 as AC97 Reset on Bulverde */ > - pxa_gpio_mode(113 | GPIO_ALT_FN_2_OUT); > - > - ac97conf_clk = clk_get(&pdev->dev, "AC97CONFCLK"); > - if (IS_ERR(ac97conf_clk)) { > - ret = PTR_ERR(ac97conf_clk); > - ac97conf_clk = NULL; > - goto err_irq; > - } > -#endif > - ac97_clk = clk_get(&pdev->dev, "AC97CLK"); > - if (IS_ERR(ac97_clk)) { > - ret = PTR_ERR(ac97_clk); > - ac97_clk = NULL; > - goto err_irq; > - } > - clk_enable(ac97_clk); > - return 0; > - > - err_irq: > - GCR |= GCR_ACLINK_OFF; > -#ifdef CONFIG_PXA27x > - if (ac97conf_clk) { > - clk_put(ac97conf_clk); > - ac97conf_clk = NULL; > - } > -#endif > - free_irq(IRQ_AC97, NULL); > - err: > - return ret; > + return pxa2xx_ac97_hw_probe(pdev); > } > > static void pxa2xx_ac97_remove(struct platform_device *pdev, > struct snd_soc_dai *dai) > { > - GCR |= GCR_ACLINK_OFF; > - free_irq(IRQ_AC97, NULL); > -#ifdef CONFIG_PXA27x > - clk_put(ac97conf_clk); > - ac97conf_clk = NULL; > -#endif > - clk_disable(ac97_clk); > - clk_put(ac97_clk); > - ac97_clk = NULL; > + pxa2xx_ac97_hw_remove(pdev); > } > > static int pxa2xx_ac97_hw_params(struct snd_pcm_substream *substream, Liam _______________________________________________ Alsa-devel mailing list Alsa-devel@xxxxxxxxxxxxxxxx http://mailman.alsa-project.org/mailman/listinfo/alsa-devel