2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2008-09-10 05:01:17 +04:00
/*
* Based on sound / arm / pxa2xx - ac97 . c and sound / soc / pxa / pxa2xx - ac97 . c
* which contain :
*
* Author : Nicolas Pitre
* Created : Dec 02 , 2004
* Copyright : MontaVista Software Inc .
*/
# include <linux/kernel.h>
# include <linux/platform_device.h>
# include <linux/interrupt.h>
# include <linux/clk.h>
# include <linux/delay.h>
2011-07-15 12:38:28 -04:00
# include <linux/module.h>
2012-03-20 14:33:19 -05:00
# include <linux/io.h>
2013-01-07 13:55:13 -08:00
# include <linux/gpio.h>
2018-06-17 12:50:01 +02:00
# include <linux/of_gpio.h>
2008-09-10 05:01:17 +04:00
# include <sound/pxa2xx-lib.h>
2012-01-03 17:10:17 -06:00
# include <mach/irqs.h>
2008-11-28 14:19:33 +08:00
# include <mach/regs-ac97.h>
2008-09-10 05:01:17 +04:00
# include <mach/audio.h>
static DEFINE_MUTEX ( car_mutex ) ;
static DECLARE_WAIT_QUEUE_HEAD ( gsr_wq ) ;
static volatile long gsr_bits ;
static struct clk * ac97_clk ;
static struct clk * ac97conf_clk ;
2009-03-15 14:10:54 +01:00
static int reset_gpio ;
2008-09-10 05:01:17 +04:00
2013-01-07 13:55:14 -08:00
extern void pxa27x_configure_ac97reset ( int reset_gpio , bool to_gpio ) ;
2010-01-04 16:30:58 +08:00
2008-09-10 05:01:17 +04:00
/*
* 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 ) .
*/
2017-09-02 21:54:06 +02:00
int pxa2xx_ac97_read ( int slot , unsigned short reg )
2008-09-10 05:01:17 +04:00
{
2017-09-02 21:54:06 +02:00
int val = - ENODEV ;
2008-09-10 05:01:17 +04:00
volatile u32 * reg_addr ;
2017-09-02 21:54:06 +02:00
if ( slot > 0 )
return - ENODEV ;
2008-09-10 05:01:17 +04:00
mutex_lock ( & car_mutex ) ;
/* set up primary or secondary codec space */
2008-10-14 09:57:05 +01:00
if ( cpu_is_pxa25x ( ) & & reg = = AC97_GPIO_STATUS )
2017-09-02 21:54:06 +02:00
reg_addr = slot ? & SMC_REG_BASE : & PMC_REG_BASE ;
2008-09-10 05:01:17 +04:00
else
2017-09-02 21:54:06 +02:00
reg_addr = slot ? & SAC_REG_BASE : & PAC_REG_BASE ;
2008-09-10 05:01:17 +04:00
reg_addr + = ( reg > > 1 ) ;
/* start read access across the ac97 link */
GSR = GSR_CDONE | GSR_SDONE ;
gsr_bits = 0 ;
2017-09-02 21:54:06 +02:00
val = ( * reg_addr & 0xffff ) ;
2008-09-10 05:01:17 +04:00
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 ) ;
2017-09-02 21:54:06 +02:00
val = - ETIMEDOUT ;
2008-09-10 05:01:17 +04:00
goto out ;
}
/* valid data now */
GSR = GSR_CDONE | GSR_SDONE ;
gsr_bits = 0 ;
2017-09-02 21:54:06 +02:00
val = ( * reg_addr & 0xffff ) ;
2008-09-10 05:01:17 +04:00
/* 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 ;
}
EXPORT_SYMBOL_GPL ( pxa2xx_ac97_read ) ;
2017-09-02 21:54:06 +02:00
int pxa2xx_ac97_write ( int slot , unsigned short reg , unsigned short val )
2008-09-10 05:01:17 +04:00
{
volatile u32 * reg_addr ;
2017-09-02 21:54:06 +02:00
int ret = 0 ;
2008-09-10 05:01:17 +04:00
mutex_lock ( & car_mutex ) ;
/* set up primary or secondary codec space */
2008-10-14 09:57:05 +01:00
if ( cpu_is_pxa25x ( ) & & reg = = AC97_GPIO_STATUS )
2017-09-02 21:54:06 +02:00
reg_addr = slot ? & SMC_REG_BASE : & PMC_REG_BASE ;
2008-09-10 05:01:17 +04:00
else
2017-09-02 21:54:06 +02:00
reg_addr = slot ? & SAC_REG_BASE : & PAC_REG_BASE ;
2008-09-10 05:01:17 +04:00
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 & &
2017-09-02 21:54:06 +02:00
! ( ( GSR | gsr_bits ) & GSR_CDONE ) ) {
2008-09-10 05:01:17 +04:00
printk ( KERN_ERR " %s: write error (ac97_reg=%d GSR=%#lx) \n " ,
__func__ , reg , GSR | gsr_bits ) ;
2017-09-02 21:54:06 +02:00
ret = - EIO ;
}
2008-09-10 05:01:17 +04:00
mutex_unlock ( & car_mutex ) ;
2017-09-02 21:54:06 +02:00
return ret ;
2008-09-10 05:01:17 +04:00
}
EXPORT_SYMBOL_GPL ( pxa2xx_ac97_write ) ;
2008-09-10 05:01:18 +04:00
# ifdef CONFIG_PXA25x
static inline void pxa_ac97_warm_pxa25x ( void )
2008-09-10 05:01:17 +04:00
{
gsr_bits = 0 ;
2013-10-17 14:01:35 +04:00
GCR | = GCR_WARM_RST ;
2008-09-10 05:01:18 +04:00
}
static inline void pxa_ac97_cold_pxa25x ( void )
{
GCR & = GCR_COLD_RST ; /* clear everything but nCRST */
GCR & = ~ GCR_COLD_RST ; /* then assert nCRST */
gsr_bits = 0 ;
GCR = GCR_COLD_RST ;
}
# endif
2008-09-10 05:01:17 +04:00
# ifdef CONFIG_PXA27x
2008-09-10 05:01:18 +04:00
static inline void pxa_ac97_warm_pxa27x ( void )
{
gsr_bits = 0 ;
2010-01-04 16:30:58 +08:00
/* warm reset broken on Bulverde, so manually keep AC97 reset high */
2013-01-07 13:55:14 -08:00
pxa27x_configure_ac97reset ( reset_gpio , true ) ;
2008-09-10 05:01:17 +04:00
udelay ( 10 ) ;
GCR | = GCR_WARM_RST ;
2013-01-07 13:55:14 -08:00
pxa27x_configure_ac97reset ( reset_gpio , false ) ;
2008-09-10 05:01:17 +04:00
udelay ( 500 ) ;
2008-09-10 05:01:18 +04:00
}
static inline void pxa_ac97_cold_pxa27x ( void )
{
GCR & = GCR_COLD_RST ; /* clear everything but nCRST */
GCR & = ~ GCR_COLD_RST ; /* then assert nCRST */
gsr_bits = 0 ;
/* PXA27x Developers Manual section 13.5.2.2.1 */
2014-06-09 21:59:12 +02:00
clk_prepare_enable ( ac97conf_clk ) ;
2008-09-10 05:01:18 +04:00
udelay ( 5 ) ;
2014-06-09 21:59:12 +02:00
clk_disable_unprepare ( ac97conf_clk ) ;
2013-01-07 13:55:12 -08:00
GCR = GCR_COLD_RST | GCR_WARM_RST ;
2008-09-10 05:01:18 +04:00
}
2008-09-10 05:01:17 +04:00
# endif
2008-09-10 05:01:18 +04:00
# ifdef CONFIG_PXA3xx
static inline void pxa_ac97_warm_pxa3xx ( void )
{
gsr_bits = 0 ;
2008-09-10 05:01:17 +04:00
2008-09-10 05:01:18 +04:00
/* Can't use interrupts */
GCR | = GCR_WARM_RST ;
2008-09-10 05:01:17 +04:00
}
2008-09-10 05:01:18 +04:00
static inline void pxa_ac97_cold_pxa3xx ( void )
2008-09-10 05:01:17 +04:00
{
/* Hold CLKBPB for 100us */
GCR = 0 ;
GCR = GCR_CLKBPB ;
udelay ( 100 ) ;
GCR = 0 ;
GCR & = GCR_COLD_RST ; /* clear everything but nCRST */
GCR & = ~ GCR_COLD_RST ; /* then assert nCRST */
gsr_bits = 0 ;
2008-09-10 05:01:18 +04:00
2008-09-10 05:01:17 +04:00
/* Can't use interrupts on PXA3xx */
GCR & = ~ ( GCR_PRIRDY_IEN | GCR_SECRDY_IEN ) ;
GCR = GCR_WARM_RST | GCR_COLD_RST ;
2008-09-10 05:01:18 +04:00
}
# endif
2017-09-02 21:54:06 +02:00
bool pxa2xx_ac97_try_warm_reset ( void )
2008-09-10 05:01:18 +04:00
{
2009-03-26 13:18:03 +01:00
unsigned long gsr ;
2013-10-17 14:01:35 +04:00
unsigned int timeout = 100 ;
2009-03-26 13:18:03 +01:00
2008-09-10 05:01:18 +04:00
# ifdef CONFIG_PXA25x
2008-10-14 09:57:05 +01:00
if ( cpu_is_pxa25x ( ) )
2008-09-10 05:01:18 +04:00
pxa_ac97_warm_pxa25x ( ) ;
else
2008-09-10 05:01:17 +04:00
# endif
2008-09-10 05:01:18 +04:00
# ifdef CONFIG_PXA27x
if ( cpu_is_pxa27x ( ) )
pxa_ac97_warm_pxa27x ( ) ;
else
# endif
# ifdef CONFIG_PXA3xx
if ( cpu_is_pxa3xx ( ) )
pxa_ac97_warm_pxa3xx ( ) ;
else
# endif
2013-11-05 15:33:40 +01:00
snd_BUG ( ) ;
2013-10-17 14:01:35 +04:00
while ( ! ( ( GSR | gsr_bits ) & ( GSR_PCR | GSR_SCR ) ) & & timeout - - )
mdelay ( 1 ) ;
2009-03-26 13:18:03 +01:00
gsr = GSR | gsr_bits ;
if ( ! ( gsr & ( GSR_PCR | GSR_SCR ) ) ) {
2008-09-10 05:01:18 +04:00
printk ( KERN_INFO " %s: warm reset timeout (GSR=%#lx) \n " ,
2009-03-26 13:18:03 +01:00
__func__ , gsr ) ;
2008-09-10 05:01:18 +04:00
return false ;
}
return true ;
}
EXPORT_SYMBOL_GPL ( pxa2xx_ac97_try_warm_reset ) ;
2017-09-02 21:54:06 +02:00
bool pxa2xx_ac97_try_cold_reset ( void )
2008-09-10 05:01:18 +04:00
{
2009-03-26 13:18:03 +01:00
unsigned long gsr ;
2013-10-17 14:01:35 +04:00
unsigned int timeout = 1000 ;
2009-03-26 13:18:03 +01:00
2008-09-10 05:01:18 +04:00
# ifdef CONFIG_PXA25x
2008-10-14 09:57:05 +01:00
if ( cpu_is_pxa25x ( ) )
2008-09-10 05:01:18 +04:00
pxa_ac97_cold_pxa25x ( ) ;
else
# endif
# ifdef CONFIG_PXA27x
if ( cpu_is_pxa27x ( ) )
pxa_ac97_cold_pxa27x ( ) ;
else
# endif
# ifdef CONFIG_PXA3xx
if ( cpu_is_pxa3xx ( ) )
pxa_ac97_cold_pxa3xx ( ) ;
else
# endif
2013-11-05 15:33:40 +01:00
snd_BUG ( ) ;
2008-09-10 05:01:17 +04:00
2013-10-17 14:01:35 +04:00
while ( ! ( ( GSR | gsr_bits ) & ( GSR_PCR | GSR_SCR ) ) & & timeout - - )
mdelay ( 1 ) ;
2009-03-26 13:18:03 +01:00
gsr = GSR | gsr_bits ;
if ( ! ( gsr & ( GSR_PCR | GSR_SCR ) ) ) {
2008-09-10 05:01:17 +04:00
printk ( KERN_INFO " %s: cold reset timeout (GSR=%#lx) \n " ,
2009-03-26 13:18:03 +01:00
__func__ , gsr ) ;
2008-09-10 05:01:17 +04:00
return false ;
}
return true ;
}
EXPORT_SYMBOL_GPL ( pxa2xx_ac97_try_cold_reset ) ;
2017-09-02 21:54:06 +02:00
void pxa2xx_ac97_finish_reset ( void )
2008-09-10 05:01:17 +04:00
{
GCR & = ~ ( GCR_PRIRDY_IEN | GCR_SECRDY_IEN ) ;
GCR | = GCR_SDONE_IE | GCR_CDONE_IE ;
}
EXPORT_SYMBOL_GPL ( pxa2xx_ac97_finish_reset ) ;
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 ) ;
/* 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 ) . . . */
2008-09-10 05:01:18 +04:00
if ( cpu_is_pxa27x ( ) ) {
MISR = MISR_EOC ;
PISR = PISR_EOC ;
MCSR = MCSR_EOC ;
}
2008-09-10 05:01:17 +04:00
return IRQ_HANDLED ;
}
return IRQ_NONE ;
}
# ifdef CONFIG_PM
int pxa2xx_ac97_hw_suspend ( void )
{
GCR | = GCR_ACLINK_OFF ;
2014-06-09 21:59:12 +02:00
clk_disable_unprepare ( ac97_clk ) ;
2008-09-10 05:01:17 +04:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( pxa2xx_ac97_hw_suspend ) ;
int pxa2xx_ac97_hw_resume ( void )
{
2014-06-09 21:59:12 +02:00
clk_prepare_enable ( ac97_clk ) ;
2008-09-10 05:01:17 +04:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( pxa2xx_ac97_hw_resume ) ;
# endif
2012-12-06 12:35:12 -05:00
int pxa2xx_ac97_hw_probe ( struct platform_device * dev )
2008-09-10 05:01:17 +04:00
{
int ret ;
2009-04-13 11:48:03 +01:00
pxa2xx_audio_ops_t * pdata = dev - > dev . platform_data ;
2009-03-15 14:10:54 +01:00
if ( pdata ) {
switch ( pdata - > reset_gpio ) {
case 95 :
case 113 :
reset_gpio = pdata - > reset_gpio ;
break ;
case 0 :
reset_gpio = 113 ;
break ;
case - 1 :
break ;
default :
2009-03-19 14:08:58 +01:00
dev_err ( & dev - > dev , " Invalid reset GPIO %d \n " ,
2009-03-15 14:10:54 +01:00
pdata - > reset_gpio ) ;
}
2018-06-17 12:50:01 +02:00
} else if ( ! pdata & & dev - > dev . of_node ) {
pdata = devm_kzalloc ( & dev - > dev , sizeof ( * pdata ) , GFP_KERNEL ) ;
if ( ! pdata )
return - ENOMEM ;
pdata - > reset_gpio = of_get_named_gpio ( dev - > dev . of_node ,
" reset-gpios " , 0 ) ;
if ( pdata - > reset_gpio = = - ENOENT )
pdata - > reset_gpio = - 1 ;
else if ( pdata - > reset_gpio < 0 )
return pdata - > reset_gpio ;
reset_gpio = pdata - > reset_gpio ;
2009-03-15 14:10:54 +01:00
} else {
if ( cpu_is_pxa27x ( ) )
reset_gpio = 113 ;
}
2008-09-10 05:01:17 +04:00
2008-09-10 05:01:18 +04:00
if ( cpu_is_pxa27x ( ) ) {
2013-01-07 13:55:13 -08:00
/*
* This gpio is needed for a work - around to a bug in the ac97
* controller during warm reset . The direction and level is set
* here so that it is an output driven high when switching from
* AC97_nRESET alt function to generic gpio .
*/
ret = gpio_request_one ( reset_gpio , GPIOF_OUT_INIT_HIGH ,
" pxa27x ac97 reset " ) ;
if ( ret < 0 ) {
pr_err ( " %s: gpio_request_one() failed: %d \n " ,
__func__ , ret ) ;
goto err_conf ;
}
2013-01-07 13:55:14 -08:00
pxa27x_configure_ac97reset ( reset_gpio , false ) ;
2013-01-07 13:55:13 -08:00
2008-09-10 05:01:18 +04:00
ac97conf_clk = clk_get ( & dev - > dev , " AC97CONFCLK " ) ;
if ( IS_ERR ( ac97conf_clk ) ) {
ret = PTR_ERR ( ac97conf_clk ) ;
ac97conf_clk = NULL ;
2009-01-05 12:58:06 +03:00
goto err_conf ;
2008-09-10 05:01:18 +04:00
}
2008-09-10 05:01:17 +04:00
}
ac97_clk = clk_get ( & dev - > dev , " AC97CLK " ) ;
if ( IS_ERR ( ac97_clk ) ) {
ret = PTR_ERR ( ac97_clk ) ;
ac97_clk = NULL ;
2009-01-05 12:58:06 +03:00
goto err_clk ;
2008-09-10 05:01:17 +04:00
}
2014-06-09 21:59:12 +02:00
ret = clk_prepare_enable ( ac97_clk ) ;
2009-01-05 12:58:06 +03:00
if ( ret )
goto err_clk2 ;
2011-09-22 16:59:20 +08:00
ret = request_irq ( IRQ_AC97 , pxa2xx_ac97_irq , 0 , " AC97 " , NULL ) ;
2009-01-05 12:58:06 +03:00
if ( ret < 0 )
goto err_irq ;
return 0 ;
2008-09-10 05:01:17 +04:00
err_irq :
GCR | = GCR_ACLINK_OFF ;
2009-01-05 12:58:06 +03:00
err_clk2 :
clk_put ( ac97_clk ) ;
ac97_clk = NULL ;
err_clk :
2008-09-10 05:01:17 +04:00
if ( ac97conf_clk ) {
clk_put ( ac97conf_clk ) ;
ac97conf_clk = NULL ;
}
2009-01-05 12:58:06 +03:00
err_conf :
2008-09-10 05:01:17 +04:00
return ret ;
}
EXPORT_SYMBOL_GPL ( pxa2xx_ac97_hw_probe ) ;
void pxa2xx_ac97_hw_remove ( struct platform_device * dev )
{
2013-01-07 13:55:13 -08:00
if ( cpu_is_pxa27x ( ) )
gpio_free ( reset_gpio ) ;
2008-09-10 05:01:17 +04:00
GCR | = GCR_ACLINK_OFF ;
free_irq ( IRQ_AC97 , NULL ) ;
2008-09-10 05:01:18 +04:00
if ( ac97conf_clk ) {
clk_put ( ac97conf_clk ) ;
ac97conf_clk = NULL ;
}
2014-06-09 21:59:12 +02:00
clk_disable_unprepare ( ac97_clk ) ;
2008-09-10 05:01:17 +04:00
clk_put ( ac97_clk ) ;
ac97_clk = NULL ;
}
EXPORT_SYMBOL_GPL ( pxa2xx_ac97_hw_remove ) ;
MODULE_AUTHOR ( " Nicolas Pitre " ) ;
MODULE_DESCRIPTION ( " Intel/Marvell PXA sound library " ) ;
MODULE_LICENSE ( " GPL " ) ;