1 /***************************************************************************
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
10 * Driver for WM8975 audio codec
12 * Based on code from the ipodlinux project - http://ipodlinux.org/
13 * Adapted for Rockbox in December 2005
15 * Original file: linux/arch/armnommu/mach-ipod/audio.c
17 * Copyright (c) 2003-2005 Bernard Leach (leachbj@bouncycastle.org)
19 * This program is free software; you can redistribute it and/or
20 * modify it under the terms of the GNU General Public License
21 * as published by the Free Software Foundation; either version 2
22 * of the License, or (at your option) any later version.
24 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
25 * KIND, either express or implied.
27 ****************************************************************************/
37 const struct sound_settings_info audiohw_settings
[] = {
38 [SOUND_VOLUME
] = {"dB", 0, 1, -74, 6, -25},
39 [SOUND_BASS
] = {"dB", 0, 1, -6, 9, 0},
40 [SOUND_TREBLE
] = {"dB", 0, 1, -6, 9, 0},
41 [SOUND_BALANCE
] = {"%", 0, 1,-100, 100, 0},
42 [SOUND_CHANNELS
] = {"", 0, 1, 0, 5, 0},
43 [SOUND_STEREO_WIDTH
] = {"%", 0, 5, 0, 250, 100},
45 [SOUND_LEFT_GAIN
] = {"dB", 1, 1, 0, 63, 23},
46 [SOUND_RIGHT_GAIN
] = {"dB", 1, 1, 0, 63, 23},
47 [SOUND_MIC_GAIN
] = {"dB", 1, 1, 0, 63, 0},
51 static unsigned short wm8975_regs
[WM8975_NUM_REGISTERS
] =
53 [LINVOL
] = LINVOL_LZCEN
| 23, /* 0dB */
54 [RINVOL
] = RINVOL_RIVU
| RINVOL_RZCEN
| 23, /* 0dB */
55 [DAPCTRL
] = DAPCTRL_DACMU
,
56 /* This reduces the popping noise during codec powerup
57 noticably, especially with high-impedance loads.
58 We might want to change this for all targets,
59 but it has only been tested on iPod Nano 2G so far. */
61 [PWRMGMT1
] = PWRMGMT1_VMIDSEL_500K
| PWRMGMT1_VREF
,
63 [PWRMGMT1
] = PWRMGMT1_VMIDSEL_5K
| PWRMGMT1_VREF
,
65 [PWRMGMT2
] = PWRMGMT2_DACL
| PWRMGMT2_DACR
| PWRMGMT2_LOUT1
66 | PWRMGMT2_ROUT1
| PWRMGMT2_LOUT2
| PWRMGMT2_ROUT2
,
69 static void wm8975_write(int reg
, unsigned val
)
71 if (WM8975_NUM_REGISTERS
> reg
) {
72 wm8975_regs
[reg
] = val
;
73 wmcodec_write(reg
, val
);
77 static void wm8975_write_and(int reg
, unsigned bits
)
79 wm8975_write(reg
, wm8975_regs
[reg
] & bits
);
82 static void wm8975_write_or(int reg
, unsigned bits
)
84 wm8975_write(reg
, wm8975_regs
[reg
] | bits
);
87 /* convert tenth of dB volume (-730..60) to master volume register value */
88 int tenthdb2master(int db
)
90 /* +6 to -73dB 1dB steps (plus mute == 80levels) 7bits */
91 /* 1111111 == +6dB (0x7f) */
92 /* 1111001 == 0dB (0x79) */
93 /* 0110000 == -73dB (0x30 */
94 /* 0101111..0000000 == mute (0x2f) */
96 if (db
< VOLUME_MIN
) {
99 return((db
/10)+73+0x30);
103 int sound_val2phys(int setting
, int value
)
109 #ifdef HAVE_RECORDING
110 case SOUND_LEFT_GAIN
:
111 case SOUND_RIGHT_GAIN
:
112 result
= ((value
- 23) * 15) / 2;
115 result
= ((value
- 23) * 15) / 2 + 200;
126 static void audiohw_mute(bool mute
)
129 /* Set DACMU = 1 to soft-mute the audio DACs. */
130 wm8975_write_or(DAPCTRL
, DAPCTRL_DACMU
);
132 /* Set DACMU = 0 to soft-un-mute the audio DACs. */
133 wm8975_write_and(DAPCTRL
, ~DAPCTRL_DACMU
);
137 #define IPOD_PCM_LEVEL 0x65 /* -6dB */
139 /* This reduces the popping noise during codec powerup
140 noticably, especially with high-impedance loads.
141 We might want to change this for all targets,
142 but it has only been tested on iPod Nano 2G so far. */
144 void audiohw_preinit(void)
146 wm8975_write(RESET
, RESET_RESET
);
148 wm8975_write(AINTFCE
, AINTFCE_MS
| AINTFCE_LRP_I2S_RLO
149 | AINTFCE_IWL_16BIT
| AINTFCE_FORMAT_I2S
);
151 wm8975_write(LOUTMIX1
, LOUTMIX1_LD2LO
| LOUTMIX1_LI2LOVOL(5));
152 wm8975_write(ROUTMIX2
, ROUTMIX2_RD2RO
| ROUTMIX2_RI2ROVOL(5));
154 wm8975_write(PWRMGMT1
, wm8975_regs
[PWRMGMT1
]);
155 wm8975_write(PWRMGMT2
, wm8975_regs
[PWRMGMT2
]);
158 void audiohw_postinit(void)
160 wm8975_regs
[PWRMGMT1
] &= ~PWRMGMT1_VMIDSEL_MASK
;
161 wm8975_regs
[PWRMGMT1
] |= PWRMGMT1_VMIDSEL_50K
;
162 wm8975_write(PWRMGMT1
, wm8975_regs
[PWRMGMT1
]);
165 #else /* !IPOD_NANO2G */
166 void audiohw_preinit(void)
168 /* POWER UP SEQUENCE */
169 wm8975_write(RESET
, RESET_RESET
);
171 /* 2. Enable Vmid and VREF, quick startup. */
172 wm8975_write(PWRMGMT1
, wm8975_regs
[PWRMGMT1
]);
174 wm8975_regs
[PWRMGMT1
] &= ~PWRMGMT1_VMIDSEL_MASK
;
175 wm8975_write(PWRMGMT1
, wm8975_regs
[PWRMGMT1
] | PWRMGMT1_VMIDSEL_50K
);
177 /* 4. Enable DACs, line and headphone output buffers as required. */
178 wm8975_write(PWRMGMT2
, wm8975_regs
[PWRMGMT2
]);
180 wm8975_write(AINTFCE
, AINTFCE_MS
| AINTFCE_LRP_I2S_RLO
181 | AINTFCE_IWL_16BIT
| AINTFCE_FORMAT_I2S
);
183 wm8975_write(DAPCTRL
, wm8975_regs
[DAPCTRL
] );
185 /* Set sample rate. */
186 wm8975_write(SAMPCTRL
, WM8975_44100HZ
);
188 /* set the volume to -6dB */
189 wm8975_write(LOUT1VOL
, LOUT1VOL_LO1ZC
| IPOD_PCM_LEVEL
);
190 wm8975_write(ROUT1VOL
, ROUT1VOL_RO1VU
| ROUT1VOL_RO1ZC
| IPOD_PCM_LEVEL
);
192 wm8975_write(LOUTMIX1
, LOUTMIX1_LD2LO
| LOUTMIX1_LI2LOVOL(5));
193 wm8975_write(LOUTMIX2
, LOUTMIX2_RI2LOVOL(5));
195 wm8975_write(ROUTMIX1
, ROUTMIX1_LI2ROVOL(5));
196 wm8975_write(ROUTMIX2
, ROUTMIX2_RD2RO
| ROUTMIX2_RI2ROVOL(5));
198 wm8975_write(MOUTMIX1
, 0);
199 wm8975_write(MOUTMIX2
, 0);
202 void audiohw_postinit(void)
208 void audiohw_set_master_vol(int vol_l
, int vol_r
)
210 /* +6 to -73dB 1dB steps (plus mute == 80levels) 7bits */
211 /* 1111111 == +6dB */
213 /* 0110000 == -73dB */
214 /* 0101111 == mute (0x2f) */
217 wm8975_write(LOUT1VOL
, LOUT1VOL_LO1ZC
| vol_l
);
218 wm8975_write(ROUT1VOL
, ROUT1VOL_RO1VU
| ROUT1VOL_RO1ZC
| vol_r
);
221 void audiohw_set_lineout_vol(int vol_l
, int vol_r
)
224 wm8975_write(LOUT2VOL
, LOUT2VOL_LO2ZC
| vol_l
);
225 wm8975_write(ROUT2VOL
, ROUT2VOL_RO2VU
| ROUT2VOL_RO2ZC
| vol_r
);
228 void audiohw_enable_lineout(bool enable
)
232 wm8975_regs
[PWRMGMT2
] |= (PWRMGMT2_LOUT2
| PWRMGMT2_ROUT2
);
234 /* Disable lineout */
235 wm8975_regs
[PWRMGMT2
] &= ~(PWRMGMT2_LOUT2
| PWRMGMT2_ROUT2
);
237 wm8975_write(PWRMGMT2
, wm8975_regs
[PWRMGMT2
]);
240 void audiohw_set_bass(int value
)
242 const int regvalues
[] = {
243 11, 10, 10, 9, 8, 8, 0xf, 6, 6, 5, 4, 4, 3, 2, 2, 1
246 if ((value
>= -6) && (value
<= 9)) {
247 /* We use linear bass control with 200 Hz cutoff */
248 wm8975_write(BASSCTRL
, regvalues
[value
+ 6] | BASSCTRL_BC
);
252 void audiohw_set_treble(int value
)
254 const int regvalues
[] = {
255 11, 10, 10, 9, 8, 8, 0xf, 6, 6, 5, 4, 4, 3, 2, 2, 1
258 if ((value
>= -6) && (value
<= 9)) {
259 /* We use linear treble control with 4 kHz cutoff */
260 wm8975_write(TREBCTRL
, regvalues
[value
+ 6] | TREBCTRL_TC
);
264 /* Nice shutdown of WM8975 codec */
265 void audiohw_close(void)
269 /* 2. Disable all output buffers. */
270 wm8975_write(PWRMGMT2
, 0x0);
272 /* 3. Switch off the power supplies. */
273 wm8975_write(PWRMGMT1
, 0x0);
276 /* Note: Disable output before calling this function */
277 void audiohw_set_frequency(int fsel
)
282 #ifdef HAVE_RECORDING
283 void audiohw_enable_recording(bool source_mic
)
285 wm8975_regs
[PWRMGMT1
] |= PWRMGMT1_AINL
| PWRMGMT1_AINR
286 | PWRMGMT1_ADCL
| PWRMGMT1_ADCR
;
287 wm8975_write(PWRMGMT1
, wm8975_regs
[PWRMGMT1
]);
289 /* NOTE: When switching to digital monitoring we will not want
290 * the DACs disabled. Also the outputs shouldn't be disabled
291 * when recording from line in (dock connector) - needs testing. */
292 wm8975_regs
[PWRMGMT2
] &= ~(PWRMGMT2_LOUT1
| PWRMGMT2_ROUT1
293 | PWRMGMT2_LOUT2
| PWRMGMT2_ROUT2
);
294 wm8975_write(PWRMGMT2
, wm8975_regs
[PWRMGMT2
]);
296 wm8975_write_or(LINVOL
, LINVOL_LINMUTE
);
297 wm8975_write_or(RINVOL
, RINVOL_RINMUTE
);
299 wm8975_write(ADDCTRL3
, ADDCTRL3_VROI
);
302 wm8975_write(ADDCTRL1
, ADDCTRL1_VSEL_LOWBIAS
| ADDCTRL1_DATSEL_RADC
304 wm8975_write(ADCLPATH
, 0);
305 wm8975_write(ADCRPATH
, ADCRPATH_RINSEL_RIN2
| ADCRPATH_RMICBOOST_20dB
);
307 wm8975_write(ADDCTRL1
, ADDCTRL1_VSEL_LOWBIAS
| ADDCTRL1_DATSEL_NORMAL
309 wm8975_write(ADCLPATH
, ADCLPATH_LINSEL_LIN1
| ADCLPATH_LMICBOOST_OFF
);
310 wm8975_write(ADCRPATH
, ADCRPATH_RINSEL_RIN1
| ADCRPATH_RMICBOOST_OFF
);
312 wm8975_write_and(LINVOL
, ~LINVOL_LINMUTE
);
313 wm8975_write_and(RINVOL
, ~RINVOL_RINMUTE
);
316 void audiohw_disable_recording(void)
319 wm8975_write_or(LINVOL
, LINVOL_LINMUTE
);
320 wm8975_write_or(RINVOL
, RINVOL_RINMUTE
);
322 wm8975_write(ADDCTRL3
, 0);
324 wm8975_regs
[PWRMGMT2
] |= PWRMGMT2_DACL
| PWRMGMT2_DACR
325 | PWRMGMT2_LOUT1
| PWRMGMT2_ROUT1
326 | PWRMGMT2_LOUT2
| PWRMGMT2_ROUT2
;
327 wm8975_write(PWRMGMT2
, wm8975_regs
[PWRMGMT2
]);
329 wm8975_regs
[PWRMGMT1
] &= ~(PWRMGMT1_AINL
| PWRMGMT1_AINR
330 | PWRMGMT1_ADCL
| PWRMGMT1_ADCR
);
331 wm8975_write(PWRMGMT1
, wm8975_regs
[PWRMGMT1
]);
334 void audiohw_set_recvol(int left
, int right
, int type
)
338 case AUDIO_GAIN_MIC
: /* Mic uses right ADC */
339 wm8975_regs
[RINVOL
] &= ~RINVOL_MASK
;
340 wm8975_write_or(RINVOL
, left
& RINVOL_MASK
);
342 case AUDIO_GAIN_LINEIN
:
343 wm8975_regs
[LINVOL
] &= ~LINVOL_MASK
;
344 wm8975_write_or(LINVOL
, left
& LINVOL_MASK
);
345 wm8975_regs
[RINVOL
] &= ~RINVOL_MASK
;
346 wm8975_write_or(RINVOL
, right
& RINVOL_MASK
);
353 void audiohw_set_monitor(bool enable
)
356 /* set volume to 0 dB */
357 wm8975_regs
[LOUTMIX1
] &= ~LOUTMIX1_LI2LOVOL_MASK
;
358 wm8975_regs
[LOUTMIX1
] |= LOUTMIX1_LI2LOVOL(2);
359 wm8975_regs
[ROUTMIX2
] &= ~ROUTMIX2_RI2ROVOL_MASK
;
360 wm8975_regs
[ROUTMIX2
] |= ROUTMIX2_RI2ROVOL(2);
361 /* set mux to line input */
362 wm8975_write_and(LOUTMIX1
, ~LOUTMIX1_LMIXSEL_MASK
);
363 wm8975_write_and(ROUTMIX1
, ~ROUTMIX1_RMIXSEL_MASK
);
365 wm8975_write_or(LOUTMIX1
, LOUTMIX1_LI2LO
);
366 wm8975_write_or(ROUTMIX2
, ROUTMIX2_RI2RO
);
369 wm8975_write_and(LOUTMIX1
, ~LOUTMIX1_LI2LO
);
370 wm8975_write_and(ROUTMIX2
, ~ROUTMIX2_RI2RO
);
373 #endif /* HAVE_RECORDING */