2005-04-17 02:20:36 +04:00
/*
* PMac AWACS lowlevel functions
*
* Copyright ( c ) by Takashi Iwai < tiwai @ suse . de >
* code based on dmasound . c .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <sound/driver.h>
# include <asm/io.h>
# include <asm/nvram.h>
# include <linux/init.h>
# include <linux/delay.h>
# include <linux/slab.h>
# include <sound/core.h>
# include "pmac.h"
# ifdef CONFIG_ADB_CUDA
# define PMAC_AMP_AVAIL
# endif
# ifdef PMAC_AMP_AVAIL
2005-11-17 17:09:46 +03:00
struct awacs_amp {
2005-04-17 02:20:36 +04:00
unsigned char amp_master ;
unsigned char amp_vol [ 2 ] [ 2 ] ;
unsigned char amp_tone [ 2 ] ;
2005-11-17 17:09:46 +03:00
} ;
2005-04-17 02:20:36 +04:00
# define CHECK_CUDA_AMP() (sys_ctrler == SYS_CTRLER_CUDA)
# endif /* PMAC_AMP_AVAIL */
2005-11-17 17:09:46 +03:00
static void snd_pmac_screamer_wait ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
long timeout = 2000 ;
while ( ! ( in_le32 ( & chip - > awacs - > codec_stat ) & MASK_VALID ) ) {
mdelay ( 1 ) ;
if ( ! - - timeout ) {
snd_printd ( " snd_pmac_screamer_wait timeout \n " ) ;
break ;
}
}
}
/*
* write AWACS register
*/
static void
2005-11-17 17:09:46 +03:00
snd_pmac_awacs_write ( struct snd_pmac * chip , int val )
2005-04-17 02:20:36 +04:00
{
long timeout = 5000000 ;
if ( chip - > model = = PMAC_SCREAMER )
snd_pmac_screamer_wait ( chip ) ;
out_le32 ( & chip - > awacs - > codec_ctrl , val | ( chip - > subframe < < 22 ) ) ;
while ( in_le32 ( & chip - > awacs - > codec_ctrl ) & MASK_NEWECMD ) {
if ( ! - - timeout ) {
snd_printd ( " snd_pmac_awacs_write timeout \n " ) ;
break ;
}
}
}
static void
2005-11-17 17:09:46 +03:00
snd_pmac_awacs_write_reg ( struct snd_pmac * chip , int reg , int val )
2005-04-17 02:20:36 +04:00
{
snd_pmac_awacs_write ( chip , val | ( reg < < 12 ) ) ;
chip - > awacs_reg [ reg ] = val ;
}
static void
2005-11-17 17:09:46 +03:00
snd_pmac_awacs_write_noreg ( struct snd_pmac * chip , int reg , int val )
2005-04-17 02:20:36 +04:00
{
snd_pmac_awacs_write ( chip , val | ( reg < < 12 ) ) ;
}
2005-06-28 01:36:34 +04:00
# ifdef CONFIG_PM
2005-04-17 02:20:36 +04:00
/* Recalibrate chip */
2005-11-17 17:09:46 +03:00
static void screamer_recalibrate ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
if ( chip - > model ! = PMAC_SCREAMER )
return ;
/* Sorry for the horrible delays... I hope to get that improved
* by making the whole PM process asynchronous in a future version
*/
snd_pmac_awacs_write_noreg ( chip , 1 , chip - > awacs_reg [ 1 ] ) ;
if ( chip - > manufacturer = = 0x1 )
/* delay for broken crystal part */
2005-07-09 12:53:24 +04:00
msleep ( 750 ) ;
2005-04-17 02:20:36 +04:00
snd_pmac_awacs_write_noreg ( chip , 1 ,
2005-11-17 17:09:46 +03:00
chip - > awacs_reg [ 1 ] | MASK_RECALIBRATE |
MASK_CMUTE | MASK_AMUTE ) ;
2005-04-17 02:20:36 +04:00
snd_pmac_awacs_write_noreg ( chip , 1 , chip - > awacs_reg [ 1 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 6 , chip - > awacs_reg [ 6 ] ) ;
}
# else
# define screamer_recalibrate(chip) /* NOP */
# endif
/*
* additional callback to set the pcm format
*/
2005-11-17 17:09:46 +03:00
static void snd_pmac_awacs_set_format ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
chip - > awacs_reg [ 1 ] & = ~ MASK_SAMPLERATE ;
chip - > awacs_reg [ 1 ] | = chip - > rate_index < < 3 ;
snd_pmac_awacs_write_reg ( chip , 1 , chip - > awacs_reg [ 1 ] ) ;
}
/*
* AWACS volume callbacks
*/
/*
* volumes : 0 - 15 stereo
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_info_volume ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 2 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 15 ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_volume ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int reg = kcontrol - > private_value & 0xff ;
int lshift = ( kcontrol - > private_value > > 8 ) & 0xff ;
int inverted = ( kcontrol - > private_value > > 16 ) & 1 ;
unsigned long flags ;
int vol [ 2 ] ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
vol [ 0 ] = ( chip - > awacs_reg [ reg ] > > lshift ) & 0xf ;
vol [ 1 ] = chip - > awacs_reg [ reg ] & 0xf ;
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
if ( inverted ) {
vol [ 0 ] = 0x0f - vol [ 0 ] ;
vol [ 1 ] = 0x0f - vol [ 1 ] ;
}
ucontrol - > value . integer . value [ 0 ] = vol [ 0 ] ;
ucontrol - > value . integer . value [ 1 ] = vol [ 1 ] ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_volume ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int reg = kcontrol - > private_value & 0xff ;
int lshift = ( kcontrol - > private_value > > 8 ) & 0xff ;
int inverted = ( kcontrol - > private_value > > 16 ) & 1 ;
int val , oldval ;
unsigned long flags ;
int vol [ 2 ] ;
vol [ 0 ] = ucontrol - > value . integer . value [ 0 ] ;
vol [ 1 ] = ucontrol - > value . integer . value [ 1 ] ;
if ( inverted ) {
vol [ 0 ] = 0x0f - vol [ 0 ] ;
vol [ 1 ] = 0x0f - vol [ 1 ] ;
}
vol [ 0 ] & = 0x0f ;
vol [ 1 ] & = 0x0f ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
oldval = chip - > awacs_reg [ reg ] ;
val = oldval & ~ ( 0xf | ( 0xf < < lshift ) ) ;
val | = vol [ 0 ] < < lshift ;
val | = vol [ 1 ] ;
if ( oldval ! = val )
snd_pmac_awacs_write_reg ( chip , reg , val ) ;
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
return oldval ! = reg ;
}
# define AWACS_VOLUME(xname, xreg, xshift, xinverted) \
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER , . name = xname , . index = 0 , \
. info = snd_pmac_awacs_info_volume , \
. get = snd_pmac_awacs_get_volume , \
. put = snd_pmac_awacs_put_volume , \
. private_value = ( xreg ) | ( ( xshift ) < < 8 ) | ( ( xinverted ) < < 16 ) }
/*
* mute master / ogain for AWACS : mono
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_switch ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int reg = kcontrol - > private_value & 0xff ;
int shift = ( kcontrol - > private_value > > 8 ) & 0xff ;
int invert = ( kcontrol - > private_value > > 16 ) & 1 ;
int val ;
unsigned long flags ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
val = ( chip - > awacs_reg [ reg ] > > shift ) & 1 ;
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
if ( invert )
val = 1 - val ;
ucontrol - > value . integer . value [ 0 ] = val ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_switch ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int reg = kcontrol - > private_value & 0xff ;
int shift = ( kcontrol - > private_value > > 8 ) & 0xff ;
int invert = ( kcontrol - > private_value > > 16 ) & 1 ;
int mask = 1 < < shift ;
int val , changed ;
unsigned long flags ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
val = chip - > awacs_reg [ reg ] & ~ mask ;
if ( ucontrol - > value . integer . value [ 0 ] ! = invert )
val | = mask ;
changed = chip - > awacs_reg [ reg ] ! = val ;
if ( changed )
snd_pmac_awacs_write_reg ( chip , reg , val ) ;
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
return changed ;
}
# define AWACS_SWITCH(xname, xreg, xshift, xinvert) \
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER , . name = xname , . index = 0 , \
. info = snd_pmac_boolean_mono_info , \
. get = snd_pmac_awacs_get_switch , \
. put = snd_pmac_awacs_put_switch , \
. private_value = ( xreg ) | ( ( xshift ) < < 8 ) | ( ( xinvert ) < < 16 ) }
# ifdef PMAC_AMP_AVAIL
/*
* controls for perch / whisper extension cards , e . g . G3 desktop
*
* TDA7433 connected via i2c address 0x45 ( = 0x8a ) ,
* accessed through cuda
*/
static void awacs_set_cuda ( int reg , int val )
{
struct adb_request req ;
cuda_request ( & req , NULL , 5 , CUDA_PACKET , CUDA_GET_SET_IIC , 0x8a , reg , val ) ;
while ( ! req . complete )
cuda_poll ( ) ;
}
/*
* level = 0 - 14 , 7 = 0 dB
*/
2005-11-17 17:09:46 +03:00
static void awacs_amp_set_tone ( struct awacs_amp * amp , int bass , int treble )
2005-04-17 02:20:36 +04:00
{
amp - > amp_tone [ 0 ] = bass ;
amp - > amp_tone [ 1 ] = treble ;
if ( bass > 7 )
bass = ( 14 - bass ) + 8 ;
if ( treble > 7 )
treble = ( 14 - treble ) + 8 ;
awacs_set_cuda ( 2 , ( bass < < 4 ) | treble ) ;
}
/*
* vol = 0 - 31 ( attenuation ) , 32 = mute bit , stereo
*/
2005-11-17 17:09:46 +03:00
static int awacs_amp_set_vol ( struct awacs_amp * amp , int index , int lvol , int rvol ,
int do_check )
2005-04-17 02:20:36 +04:00
{
if ( do_check & & amp - > amp_vol [ index ] [ 0 ] = = lvol & &
amp - > amp_vol [ index ] [ 1 ] = = rvol )
return 0 ;
awacs_set_cuda ( 3 + index , lvol ) ;
awacs_set_cuda ( 5 + index , rvol ) ;
amp - > amp_vol [ index ] [ 0 ] = lvol ;
amp - > amp_vol [ index ] [ 1 ] = rvol ;
return 1 ;
}
/*
* 0 = - 79 dB , 79 = 0 dB , 99 = + 20 dB
*/
2005-11-17 17:09:46 +03:00
static void awacs_amp_set_master ( struct awacs_amp * amp , int vol )
2005-04-17 02:20:36 +04:00
{
amp - > amp_master = vol ;
if ( vol < = 79 )
vol = 32 + ( 79 - vol ) ;
else
vol = 32 - ( vol - 79 ) ;
awacs_set_cuda ( 1 , vol ) ;
}
2005-11-17 17:09:46 +03:00
static void awacs_amp_free ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( amp , return ) ;
kfree ( amp ) ;
chip - > mixer_data = NULL ;
chip - > mixer_free = NULL ;
}
/*
* mixer controls
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_info_volume_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 2 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 31 ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_volume_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( amp , return - EINVAL ) ;
snd_assert ( index > = 0 & & index < = 1 , return - EINVAL ) ;
ucontrol - > value . integer . value [ 0 ] = 31 - ( amp - > amp_vol [ index ] [ 0 ] & 31 ) ;
ucontrol - > value . integer . value [ 1 ] = 31 - ( amp - > amp_vol [ index ] [ 1 ] & 31 ) ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_volume_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
int vol [ 2 ] ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( amp , return - EINVAL ) ;
snd_assert ( index > = 0 & & index < = 1 , return - EINVAL ) ;
vol [ 0 ] = ( 31 - ( ucontrol - > value . integer . value [ 0 ] & 31 ) ) | ( amp - > amp_vol [ index ] [ 0 ] & 32 ) ;
vol [ 1 ] = ( 31 - ( ucontrol - > value . integer . value [ 1 ] & 31 ) ) | ( amp - > amp_vol [ index ] [ 1 ] & 32 ) ;
return awacs_amp_set_vol ( amp , index , vol [ 0 ] , vol [ 1 ] , 1 ) ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_switch_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( amp , return - EINVAL ) ;
snd_assert ( index > = 0 & & index < = 1 , return - EINVAL ) ;
ucontrol - > value . integer . value [ 0 ] = ( amp - > amp_vol [ index ] [ 0 ] & 32 ) ? 0 : 1 ;
ucontrol - > value . integer . value [ 1 ] = ( amp - > amp_vol [ index ] [ 1 ] & 32 ) ? 0 : 1 ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_switch_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
int vol [ 2 ] ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( amp , return - EINVAL ) ;
snd_assert ( index > = 0 & & index < = 1 , return - EINVAL ) ;
vol [ 0 ] = ( ucontrol - > value . integer . value [ 0 ] ? 0 : 32 ) | ( amp - > amp_vol [ index ] [ 0 ] & 31 ) ;
vol [ 1 ] = ( ucontrol - > value . integer . value [ 1 ] ? 0 : 32 ) | ( amp - > amp_vol [ index ] [ 1 ] & 31 ) ;
return awacs_amp_set_vol ( amp , index , vol [ 0 ] , vol [ 1 ] , 1 ) ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_info_tone_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 1 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 14 ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_tone_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( amp , return - EINVAL ) ;
snd_assert ( index > = 0 & & index < = 1 , return - EINVAL ) ;
ucontrol - > value . integer . value [ 0 ] = amp - > amp_tone [ index ] ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_tone_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( amp , return - EINVAL ) ;
snd_assert ( index > = 0 & & index < = 1 , return - EINVAL ) ;
if ( ucontrol - > value . integer . value [ 0 ] ! = amp - > amp_tone [ index ] ) {
amp - > amp_tone [ index ] = ucontrol - > value . integer . value [ 0 ] ;
awacs_amp_set_tone ( amp , amp - > amp_tone [ 0 ] , amp - > amp_tone [ 1 ] ) ;
return 1 ;
}
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_info_master_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 1 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 99 ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_master_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( amp , return - EINVAL ) ;
ucontrol - > value . integer . value [ 0 ] = amp - > amp_master ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_master_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
snd_assert ( amp , return - EINVAL ) ;
if ( ucontrol - > value . integer . value [ 0 ] ! = amp - > amp_master ) {
amp - > amp_master = ucontrol - > value . integer . value [ 0 ] ;
awacs_amp_set_master ( amp , amp - > amp_master ) ;
return 1 ;
}
return 0 ;
}
# define AMP_CH_SPK 0
# define AMP_CH_HD 1
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_amp_vol [ ] __initdata = {
2005-04-17 02:20:36 +04:00
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " PC Speaker Playback Volume " ,
. info = snd_pmac_awacs_info_volume_amp ,
. get = snd_pmac_awacs_get_volume_amp ,
. put = snd_pmac_awacs_put_volume_amp ,
. private_value = AMP_CH_SPK ,
} ,
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Headphone Playback Volume " ,
. info = snd_pmac_awacs_info_volume_amp ,
. get = snd_pmac_awacs_get_volume_amp ,
. put = snd_pmac_awacs_put_volume_amp ,
. private_value = AMP_CH_HD ,
} ,
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Tone Control - Bass " ,
. info = snd_pmac_awacs_info_tone_amp ,
. get = snd_pmac_awacs_get_tone_amp ,
. put = snd_pmac_awacs_put_tone_amp ,
. private_value = 0 ,
} ,
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Tone Control - Treble " ,
. info = snd_pmac_awacs_info_tone_amp ,
. get = snd_pmac_awacs_get_tone_amp ,
. put = snd_pmac_awacs_put_tone_amp ,
. private_value = 1 ,
} ,
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Amp Master Playback Volume " ,
. info = snd_pmac_awacs_info_master_amp ,
. get = snd_pmac_awacs_get_master_amp ,
. put = snd_pmac_awacs_put_master_amp ,
} ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_amp_hp_sw __initdata = {
2005-04-17 02:20:36 +04:00
. iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Headphone Playback Switch " ,
. info = snd_pmac_boolean_stereo_info ,
. get = snd_pmac_awacs_get_switch_amp ,
. put = snd_pmac_awacs_put_switch_amp ,
. private_value = AMP_CH_HD ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_amp_spk_sw __initdata = {
2005-04-17 02:20:36 +04:00
. iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " PC Speaker Playback Switch " ,
. info = snd_pmac_boolean_stereo_info ,
. get = snd_pmac_awacs_get_switch_amp ,
. put = snd_pmac_awacs_put_switch_amp ,
. private_value = AMP_CH_SPK ,
} ;
# endif /* PMAC_AMP_AVAIL */
/*
* mic boost for screamer
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_screamer_mic_boost_info ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 1 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 2 ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_screamer_mic_boost_get ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int val ;
unsigned long flags ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
if ( chip - > awacs_reg [ 6 ] & MASK_MIC_BOOST )
val = 2 ;
else if ( chip - > awacs_reg [ 0 ] & MASK_GAINLINE )
val = 1 ;
else
val = 0 ;
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
ucontrol - > value . integer . value [ 0 ] = val ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_screamer_mic_boost_put ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int changed = 0 ;
int val0 , val6 ;
unsigned long flags ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
val0 = chip - > awacs_reg [ 0 ] & ~ MASK_GAINLINE ;
val6 = chip - > awacs_reg [ 6 ] & ~ MASK_MIC_BOOST ;
if ( ucontrol - > value . integer . value [ 0 ] > 0 ) {
val0 | = MASK_GAINLINE ;
if ( ucontrol - > value . integer . value [ 0 ] > 1 )
val6 | = MASK_MIC_BOOST ;
}
if ( val0 ! = chip - > awacs_reg [ 0 ] ) {
snd_pmac_awacs_write_reg ( chip , 0 , val0 ) ;
changed = 1 ;
}
if ( val6 ! = chip - > awacs_reg [ 6 ] ) {
snd_pmac_awacs_write_reg ( chip , 6 , val6 ) ;
changed = 1 ;
}
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
return changed ;
}
/*
* lists of mixer elements
*/
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_mixers [ ] __initdata = {
2005-04-17 02:20:36 +04:00
AWACS_VOLUME ( " Master Playback Volume " , 2 , 6 , 1 ) ,
AWACS_SWITCH ( " Master Capture Switch " , 1 , SHIFT_LOOPTHRU , 0 ) ,
AWACS_VOLUME ( " Capture Volume " , 0 , 4 , 0 ) ,
AWACS_SWITCH ( " CD Capture Switch " , 0 , SHIFT_MUX_CD , 0 ) ,
} ;
/* FIXME: is this correct order?
* screamer ( powerbook G3 pismo ) seems to have different bits . . .
*/
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_mixers2 [ ] __initdata = {
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " Line Capture Switch " , 0 , SHIFT_MUX_LINE , 0 ) ,
AWACS_SWITCH ( " Mic Capture Switch " , 0 , SHIFT_MUX_MIC , 0 ) ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_screamer_mixers2 [ ] __initdata = {
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " Line Capture Switch " , 0 , SHIFT_MUX_MIC , 0 ) ,
AWACS_SWITCH ( " Mic Capture Switch " , 0 , SHIFT_MUX_LINE , 0 ) ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_master_sw __initdata =
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " Master Playback Switch " , 1 , SHIFT_HDMUTE , 1 ) ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_mic_boost [ ] __initdata = {
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " Mic Boost " , 0 , SHIFT_GAINLINE , 0 ) ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_screamer_mic_boost [ ] __initdata = {
2005-04-17 02:20:36 +04:00
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Mic Boost " ,
. info = snd_pmac_screamer_mic_boost_info ,
. get = snd_pmac_screamer_mic_boost_get ,
. put = snd_pmac_screamer_mic_boost_put ,
} ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_speaker_vol [ ] __initdata = {
2005-04-17 02:20:36 +04:00
AWACS_VOLUME ( " PC Speaker Playback Volume " , 4 , 6 , 1 ) ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_speaker_sw __initdata =
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " PC Speaker Playback Switch " , 1 , SHIFT_SPKMUTE , 1 ) ;
/*
* add new mixer elements to the card
*/
2005-11-17 17:09:46 +03:00
static int build_mixers ( struct snd_pmac * chip , int nums , struct snd_kcontrol_new * mixers )
2005-04-17 02:20:36 +04:00
{
int i , err ;
for ( i = 0 ; i < nums ; i + + ) {
if ( ( err = snd_ctl_add ( chip - > card , snd_ctl_new1 ( & mixers [ i ] , chip ) ) ) < 0 )
return err ;
}
return 0 ;
}
/*
* restore all registers
*/
2005-11-17 17:09:46 +03:00
static void awacs_restore_all_regs ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
snd_pmac_awacs_write_noreg ( chip , 0 , chip - > awacs_reg [ 0 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 1 , chip - > awacs_reg [ 1 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 2 , chip - > awacs_reg [ 2 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 4 , chip - > awacs_reg [ 4 ] ) ;
if ( chip - > model = = PMAC_SCREAMER ) {
snd_pmac_awacs_write_noreg ( chip , 5 , chip - > awacs_reg [ 5 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 6 , chip - > awacs_reg [ 6 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 7 , chip - > awacs_reg [ 7 ] ) ;
}
}
2005-06-28 01:36:34 +04:00
# ifdef CONFIG_PM
2005-11-17 17:09:46 +03:00
static void snd_pmac_awacs_suspend ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
snd_pmac_awacs_write_noreg ( chip , 1 , ( chip - > awacs_reg [ 1 ]
| MASK_AMUTE | MASK_CMUTE ) ) ;
}
2005-11-17 17:09:46 +03:00
static void snd_pmac_awacs_resume ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
if ( machine_is_compatible ( " PowerBook3,1 " )
| | machine_is_compatible ( " PowerBook3,2 " ) ) {
2005-07-09 12:53:24 +04:00
msleep ( 100 ) ;
2005-04-17 02:20:36 +04:00
snd_pmac_awacs_write_reg ( chip , 1 ,
chip - > awacs_reg [ 1 ] & ~ MASK_PAROUT ) ;
2005-07-09 12:53:24 +04:00
msleep ( 300 ) ;
2005-04-17 02:20:36 +04:00
}
awacs_restore_all_regs ( chip ) ;
if ( chip - > model = = PMAC_SCREAMER ) {
/* reset power bits in reg 6 */
mdelay ( 5 ) ;
snd_pmac_awacs_write_noreg ( chip , 6 , chip - > awacs_reg [ 6 ] ) ;
}
screamer_recalibrate ( chip ) ;
# ifdef PMAC_AMP_AVAIL
if ( chip - > mixer_data ) {
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
awacs_amp_set_vol ( amp , 0 , amp - > amp_vol [ 0 ] [ 0 ] , amp - > amp_vol [ 0 ] [ 1 ] , 0 ) ;
awacs_amp_set_vol ( amp , 1 , amp - > amp_vol [ 1 ] [ 0 ] , amp - > amp_vol [ 1 ] [ 1 ] , 0 ) ;
awacs_amp_set_tone ( amp , amp - > amp_tone [ 0 ] , amp - > amp_tone [ 1 ] ) ;
awacs_amp_set_master ( amp , amp - > amp_master ) ;
}
# endif
}
2005-06-28 01:36:34 +04:00
# endif /* CONFIG_PM */
2005-04-17 02:20:36 +04:00
# ifdef PMAC_SUPPORT_AUTOMUTE
/*
* auto - mute stuffs
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_detect_headphone ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
return ( in_le32 ( & chip - > awacs - > codec_stat ) & chip - > hp_stat_mask ) ? 1 : 0 ;
}
# ifdef PMAC_AMP_AVAIL
2005-11-17 17:09:46 +03:00
static int toggle_amp_mute ( struct awacs_amp * amp , int index , int mute )
2005-04-17 02:20:36 +04:00
{
int vol [ 2 ] ;
vol [ 0 ] = amp - > amp_vol [ index ] [ 0 ] & 31 ;
vol [ 1 ] = amp - > amp_vol [ index ] [ 1 ] & 31 ;
if ( mute ) {
vol [ 0 ] | = 32 ;
vol [ 1 ] | = 32 ;
}
return awacs_amp_set_vol ( amp , index , vol [ 0 ] , vol [ 1 ] , 1 ) ;
}
# endif
2005-11-17 17:09:46 +03:00
static void snd_pmac_awacs_update_automute ( struct snd_pmac * chip , int do_notify )
2005-04-17 02:20:36 +04:00
{
if ( chip - > auto_mute ) {
# ifdef PMAC_AMP_AVAIL
if ( chip - > mixer_data ) {
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
int changed ;
if ( snd_pmac_awacs_detect_headphone ( chip ) ) {
changed = toggle_amp_mute ( amp , AMP_CH_HD , 0 ) ;
changed | = toggle_amp_mute ( amp , AMP_CH_SPK , 1 ) ;
} else {
changed = toggle_amp_mute ( amp , AMP_CH_HD , 1 ) ;
changed | = toggle_amp_mute ( amp , AMP_CH_SPK , 0 ) ;
}
if ( do_notify & & ! changed )
return ;
} else
# endif
{
int reg = chip - > awacs_reg [ 1 ] | ( MASK_HDMUTE | MASK_SPKMUTE ) ;
if ( snd_pmac_awacs_detect_headphone ( chip ) )
reg & = ~ MASK_HDMUTE ;
else
reg & = ~ MASK_SPKMUTE ;
if ( do_notify & & reg = = chip - > awacs_reg [ 1 ] )
return ;
snd_pmac_awacs_write_reg ( chip , 1 , reg ) ;
}
if ( do_notify ) {
snd_ctl_notify ( chip - > card , SNDRV_CTL_EVENT_MASK_VALUE ,
& chip - > master_sw_ctl - > id ) ;
snd_ctl_notify ( chip - > card , SNDRV_CTL_EVENT_MASK_VALUE ,
& chip - > speaker_sw_ctl - > id ) ;
snd_ctl_notify ( chip - > card , SNDRV_CTL_EVENT_MASK_VALUE ,
& chip - > hp_detect_ctl - > id ) ;
}
}
}
# endif /* PMAC_SUPPORT_AUTOMUTE */
/*
* initialize chip
*/
int __init
2005-11-17 17:09:46 +03:00
snd_pmac_awacs_init ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
int err , vol ;
/* looks like MASK_GAINLINE triggers something, so we set here
* as start - up
*/
chip - > awacs_reg [ 0 ] = MASK_MUX_CD | 0xff | MASK_GAINLINE ;
chip - > awacs_reg [ 1 ] = MASK_CMUTE | MASK_AMUTE ;
/* FIXME: Only machines with external SRS module need MASK_PAROUT */
if ( chip - > has_iic | | chip - > device_id = = 0x5 | |
/*chip->_device_id == 0x8 || */
chip - > device_id = = 0xb )
chip - > awacs_reg [ 1 ] | = MASK_PAROUT ;
/* get default volume from nvram */
// vol = (~nvram_read_byte(0x1308) & 7) << 1;
// vol = ((pmac_xpram_read( 8 ) & 7 ) << 1 );
vol = 0x0f ; /* no, on alsa, muted as default */
vol = vol + ( vol < < 6 ) ;
chip - > awacs_reg [ 2 ] = vol ;
chip - > awacs_reg [ 4 ] = vol ;
if ( chip - > model = = PMAC_SCREAMER ) {
chip - > awacs_reg [ 5 ] = vol ; /* FIXME: screamer has loopthru vol control */
chip - > awacs_reg [ 6 ] = MASK_MIC_BOOST ; /* FIXME: maybe should be vol << 3 for PCMCIA speaker */
chip - > awacs_reg [ 7 ] = 0 ;
}
awacs_restore_all_regs ( chip ) ;
chip - > manufacturer = ( in_le32 ( & chip - > awacs - > codec_stat ) > > 8 ) & 0xf ;
screamer_recalibrate ( chip ) ;
chip - > revision = ( in_le32 ( & chip - > awacs - > codec_stat ) > > 12 ) & 0xf ;
# ifdef PMAC_AMP_AVAIL
if ( chip - > revision = = 3 & & chip - > has_iic & & CHECK_CUDA_AMP ( ) ) {
2006-07-25 17:28:03 +04:00
struct awacs_amp * amp = kzalloc ( sizeof ( * amp ) , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( ! amp )
return - ENOMEM ;
chip - > mixer_data = amp ;
chip - > mixer_free = awacs_amp_free ;
awacs_amp_set_vol ( amp , 0 , 63 , 63 , 0 ) ; /* mute and zero vol */
awacs_amp_set_vol ( amp , 1 , 63 , 63 , 0 ) ;
awacs_amp_set_tone ( amp , 7 , 7 ) ; /* 0 dB */
awacs_amp_set_master ( amp , 79 ) ; /* 0 dB */
}
# endif /* PMAC_AMP_AVAIL */
if ( chip - > hp_stat_mask = = 0 ) {
/* set headphone-jack detection bit */
switch ( chip - > model ) {
case PMAC_AWACS :
chip - > hp_stat_mask = 0x04 ;
break ;
case PMAC_SCREAMER :
switch ( chip - > device_id ) {
case 0x08 :
/* 1 = side jack, 2 = front jack */
chip - > hp_stat_mask = 0x03 ;
break ;
case 0x00 :
case 0x05 :
chip - > hp_stat_mask = 0x04 ;
break ;
default :
chip - > hp_stat_mask = 0x08 ;
break ;
}
break ;
default :
snd_BUG ( ) ;
break ;
}
}
/*
* build mixers
*/
strcpy ( chip - > card - > mixername , " PowerMac AWACS " ) ;
if ( ( err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_awacs_mixers ) ,
snd_pmac_awacs_mixers ) ) < 0 )
return err ;
if ( chip - > model = = PMAC_SCREAMER )
err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_screamer_mixers2 ) ,
snd_pmac_screamer_mixers2 ) ;
else
err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_awacs_mixers2 ) ,
snd_pmac_awacs_mixers2 ) ;
if ( err < 0 )
return err ;
chip - > master_sw_ctl = snd_ctl_new1 ( & snd_pmac_awacs_master_sw , chip ) ;
if ( ( err = snd_ctl_add ( chip - > card , chip - > master_sw_ctl ) ) < 0 )
return err ;
# ifdef PMAC_AMP_AVAIL
if ( chip - > mixer_data ) {
/* use amplifier. the signal is connected from route A
* to the amp . the amp has its headphone and speaker
* volumes and mute switches , so we use them instead of
* screamer registers .
* in this case , it seems the route C is not used .
*/
if ( ( err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_awacs_amp_vol ) ,
snd_pmac_awacs_amp_vol ) ) < 0 )
return err ;
/* overwrite */
chip - > master_sw_ctl = snd_ctl_new1 ( & snd_pmac_awacs_amp_hp_sw , chip ) ;
if ( ( err = snd_ctl_add ( chip - > card , chip - > master_sw_ctl ) ) < 0 )
return err ;
chip - > speaker_sw_ctl = snd_ctl_new1 ( & snd_pmac_awacs_amp_spk_sw , chip ) ;
if ( ( err = snd_ctl_add ( chip - > card , chip - > speaker_sw_ctl ) ) < 0 )
return err ;
} else
# endif /* PMAC_AMP_AVAIL */
{
/* route A = headphone, route C = speaker */
if ( ( err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_awacs_speaker_vol ) ,
snd_pmac_awacs_speaker_vol ) ) < 0 )
return err ;
chip - > speaker_sw_ctl = snd_ctl_new1 ( & snd_pmac_awacs_speaker_sw , chip ) ;
if ( ( err = snd_ctl_add ( chip - > card , chip - > speaker_sw_ctl ) ) < 0 )
return err ;
}
if ( chip - > model = = PMAC_SCREAMER ) {
if ( ( err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_screamer_mic_boost ) ,
snd_pmac_screamer_mic_boost ) ) < 0 )
return err ;
} else {
if ( ( err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_awacs_mic_boost ) ,
snd_pmac_awacs_mic_boost ) ) < 0 )
return err ;
}
/*
* set lowlevel callbacks
*/
chip - > set_format = snd_pmac_awacs_set_format ;
2005-06-28 01:36:34 +04:00
# ifdef CONFIG_PM
2005-04-17 02:20:36 +04:00
chip - > suspend = snd_pmac_awacs_suspend ;
chip - > resume = snd_pmac_awacs_resume ;
# endif
# ifdef PMAC_SUPPORT_AUTOMUTE
if ( ( err = snd_pmac_add_automute ( chip ) ) < 0 )
return err ;
chip - > detect_headphone = snd_pmac_awacs_detect_headphone ;
chip - > update_automute = snd_pmac_awacs_update_automute ;
snd_pmac_awacs_update_automute ( chip , 0 ) ; /* update the status only */
# endif
if ( chip - > model = = PMAC_SCREAMER ) {
snd_pmac_awacs_write_noreg ( chip , 6 , chip - > awacs_reg [ 6 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 0 , chip - > awacs_reg [ 0 ] ) ;
}
return 0 ;
}