You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1075 lines
36 KiB
1075 lines
36 KiB
/** |
|
****************************************************************************** |
|
* @file wm8994.c |
|
* @author MCD Application Team |
|
* @brief This file provides the WM8994 Audio Codec driver. |
|
****************************************************************************** |
|
* @attention |
|
* |
|
* <h2><center>© COPYRIGHT(c) 2016 STMicroelectronics</center></h2> |
|
* |
|
* Redistribution and use in source and binary forms, with or without modification, |
|
* are permitted provided that the following conditions are met: |
|
* 1. Redistributions of source code must retain the above copyright notice, |
|
* this list of conditions and the following disclaimer. |
|
* 2. Redistributions in binary form must reproduce the above copyright notice, |
|
* this list of conditions and the following disclaimer in the documentation |
|
* and/or other materials provided with the distribution. |
|
* 3. Neither the name of STMicroelectronics nor the names of its contributors |
|
* may be used to endorse or promote products derived from this software |
|
* without specific prior written permission. |
|
* |
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE |
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR |
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER |
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, |
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
* |
|
****************************************************************************** |
|
*/ |
|
|
|
/* Includes ------------------------------------------------------------------*/ |
|
#include "wm8994.h" |
|
|
|
/** @addtogroup BSP |
|
* @{ |
|
*/ |
|
|
|
/** @addtogroup Components |
|
* @{ |
|
*/ |
|
|
|
/** @addtogroup wm8994 |
|
* @brief This file provides a set of functions needed to drive the |
|
* WM8994 audio codec. |
|
* @{ |
|
*/ |
|
|
|
/** @defgroup WM8994_Private_Types |
|
* @{ |
|
*/ |
|
|
|
/** |
|
* @} |
|
*/ |
|
|
|
/** @defgroup WM8994_Private_Defines |
|
* @{ |
|
*/ |
|
/* Uncomment this line to enable verifying data sent to codec after each write |
|
operation (for debug purpose) */ |
|
#if !defined (VERIFY_WRITTENDATA) |
|
/*#define VERIFY_WRITTENDATA*/ |
|
#endif /* VERIFY_WRITTENDATA */ |
|
/** |
|
* @} |
|
*/ |
|
|
|
/** @defgroup WM8994_Private_Macros |
|
* @{ |
|
*/ |
|
|
|
/** |
|
* @} |
|
*/ |
|
|
|
/** @defgroup WM8994_Private_Variables |
|
* @{ |
|
*/ |
|
|
|
/* Audio codec driver structure initialization */ |
|
AUDIO_DrvTypeDef wm8994_drv = |
|
{ |
|
wm8994_Init, |
|
wm8994_DeInit, |
|
wm8994_ReadID, |
|
|
|
wm8994_Play, |
|
wm8994_Pause, |
|
wm8994_Resume, |
|
wm8994_Stop, |
|
|
|
wm8994_SetFrequency, |
|
wm8994_SetVolume, |
|
wm8994_SetMute, |
|
wm8994_SetOutputMode, |
|
|
|
wm8994_Reset |
|
}; |
|
|
|
static uint32_t outputEnabled = 0; |
|
static uint32_t inputEnabled = 0; |
|
static uint8_t ColdStartup = 1; |
|
|
|
/** |
|
* @} |
|
*/ |
|
|
|
/** @defgroup WM8994_Function_Prototypes |
|
* @{ |
|
*/ |
|
static uint8_t CODEC_IO_Write(uint8_t Addr, uint16_t Reg, uint16_t Value); |
|
/** |
|
* @} |
|
*/ |
|
|
|
|
|
/** @defgroup WM8994_Private_Functions |
|
* @{ |
|
*/ |
|
|
|
/** |
|
* @brief Initializes the audio codec and the control interface. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @param OutputInputDevice: can be OUTPUT_DEVICE_SPEAKER, OUTPUT_DEVICE_HEADPHONE, |
|
* OUTPUT_DEVICE_BOTH, OUTPUT_DEVICE_AUTO, INPUT_DEVICE_DIGITAL_MICROPHONE_1, |
|
* INPUT_DEVICE_DIGITAL_MICROPHONE_2, INPUT_DEVICE_DIGITAL_MIC1_MIC2, |
|
* INPUT_DEVICE_INPUT_LINE_1 or INPUT_DEVICE_INPUT_LINE_2. |
|
* @param Volume: Initial volume level (from 0 (Mute) to 100 (Max)) |
|
* @param AudioFreq: Audio Frequency |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_Init(uint16_t DeviceAddr, uint16_t OutputInputDevice, uint8_t Volume, uint32_t AudioFreq) |
|
{ |
|
uint32_t counter = 0; |
|
uint16_t output_device = OutputInputDevice & 0xFF; |
|
uint16_t input_device = OutputInputDevice & 0xFF00; |
|
uint16_t power_mgnt_reg_1 = 0; |
|
|
|
/* Initialize the Control interface of the Audio Codec */ |
|
AUDIO_IO_Init(); |
|
/* wm8994 Errata Work-Arounds */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x102, 0x0003); |
|
counter += CODEC_IO_Write(DeviceAddr, 0x817, 0x0000); |
|
counter += CODEC_IO_Write(DeviceAddr, 0x102, 0x0000); |
|
|
|
/* Enable VMID soft start (fast), Start-up Bias Current Enabled */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x39, 0x006C); |
|
|
|
/* Enable bias generator, Enable VMID */ |
|
if (input_device > 0) |
|
{ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x01, 0x0013); |
|
} |
|
else |
|
{ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x01, 0x0003); |
|
} |
|
|
|
/* Add Delay */ |
|
AUDIO_IO_Delay(50); |
|
|
|
/* Path Configurations for output */ |
|
if (output_device > 0) |
|
{ |
|
outputEnabled = 1; |
|
|
|
switch (output_device) |
|
{ |
|
case OUTPUT_DEVICE_SPEAKER: |
|
/* Enable DAC1 (Left), Enable DAC1 (Right), |
|
Disable DAC2 (Left), Disable DAC2 (Right)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0C0C); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x601, 0x0000); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x602, 0x0000); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Left) to DAC 2 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x604, 0x0002); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Right) to DAC 2 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x605, 0x0002); |
|
break; |
|
|
|
case OUTPUT_DEVICE_HEADPHONE: |
|
/* Disable DAC1 (Left), Disable DAC1 (Right), |
|
Enable DAC2 (Left), Enable DAC2 (Right)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0303); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x601, 0x0001); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x602, 0x0001); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Left) to DAC 2 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x604, 0x0000); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Right) to DAC 2 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x605, 0x0000); |
|
break; |
|
|
|
case OUTPUT_DEVICE_BOTH: |
|
if (input_device == INPUT_DEVICE_DIGITAL_MIC1_MIC2) |
|
{ |
|
/* Enable DAC1 (Left), Enable DAC1 (Right), |
|
also Enable DAC2 (Left), Enable DAC2 (Right)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0303 | 0x0C0C); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 1 (Left) mixer path |
|
Enable the AIF1 Timeslot 1 (Left) to DAC 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x601, 0x0003); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 1 (Right) mixer path |
|
Enable the AIF1 Timeslot 1 (Right) to DAC 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x602, 0x0003); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 2 (Left) mixer path |
|
Enable the AIF1 Timeslot 1 (Left) to DAC 2 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x604, 0x0003); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 2 (Right) mixer path |
|
Enable the AIF1 Timeslot 1 (Right) to DAC 2 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x605, 0x0003); |
|
} |
|
else |
|
{ |
|
/* Enable DAC1 (Left), Enable DAC1 (Right), |
|
also Enable DAC2 (Left), Enable DAC2 (Right)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0303 | 0x0C0C); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x601, 0x0001); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x602, 0x0001); |
|
|
|
/* Enable the AIF1 Timeslot 1 (Left) to DAC 2 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x604, 0x0002); |
|
|
|
/* Enable the AIF1 Timeslot 1 (Right) to DAC 2 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x605, 0x0002); |
|
} |
|
break; |
|
|
|
case OUTPUT_DEVICE_AUTO : |
|
default: |
|
/* Disable DAC1 (Left), Disable DAC1 (Right), |
|
Enable DAC2 (Left), Enable DAC2 (Right)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0303); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x601, 0x0001); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x602, 0x0001); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Left) to DAC 2 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x604, 0x0000); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Right) to DAC 2 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x605, 0x0000); |
|
break; |
|
} |
|
} |
|
else |
|
{ |
|
outputEnabled = 0; |
|
} |
|
|
|
/* Path Configurations for input */ |
|
if (input_device > 0) |
|
{ |
|
inputEnabled = 1; |
|
switch (input_device) |
|
{ |
|
case INPUT_DEVICE_DIGITAL_MICROPHONE_2 : |
|
/* Enable AIF1ADC2 (Left), Enable AIF1ADC2 (Right) |
|
* Enable DMICDAT2 (Left), Enable DMICDAT2 (Right) |
|
* Enable Left ADC, Enable Right ADC */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x04, 0x0C30); |
|
|
|
/* Enable AIF1 DRC2 Signal Detect & DRC in AIF1ADC2 Left/Right Timeslot 1 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x450, 0x00DB); |
|
|
|
/* Disable IN1L, IN1R, IN2L, IN2R, Enable Thermal sensor & shutdown */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x02, 0x6000); |
|
|
|
/* Enable the DMIC2(Left) to AIF1 Timeslot 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x608, 0x0002); |
|
|
|
/* Enable the DMIC2(Right) to AIF1 Timeslot 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x609, 0x0002); |
|
|
|
/* GPIO1 pin configuration GP1_DIR = output, GP1_FN = AIF1 DRC2 signal detect */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x700, 0x000E); |
|
break; |
|
|
|
case INPUT_DEVICE_INPUT_LINE_1 : |
|
/* IN1LN_TO_IN1L, IN1LP_TO_VMID, IN1RN_TO_IN1R, IN1RP_TO_VMID */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x28, 0x0011); |
|
|
|
/* Disable mute on IN1L_TO_MIXINL and +30dB on IN1L PGA output */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x29, 0x0035); |
|
|
|
/* Disable mute on IN1R_TO_MIXINL, Gain = +30dB */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x2A, 0x0035); |
|
|
|
/* Enable AIF1ADC1 (Left), Enable AIF1ADC1 (Right) |
|
* Enable Left ADC, Enable Right ADC */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x04, 0x0303); |
|
|
|
/* Enable AIF1 DRC1 Signal Detect & DRC in AIF1ADC1 Left/Right Timeslot 0 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x440, 0x00DB); |
|
|
|
/* Enable IN1L and IN1R, Disable IN2L and IN2R, Enable Thermal sensor & shutdown */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x02, 0x6350); |
|
|
|
/* Enable the ADCL(Left) to AIF1 Timeslot 0 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x606, 0x0002); |
|
|
|
/* Enable the ADCR(Right) to AIF1 Timeslot 0 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x607, 0x0002); |
|
|
|
/* GPIO1 pin configuration GP1_DIR = output, GP1_FN = AIF1 DRC1 signal detect */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x700, 0x000D); |
|
break; |
|
|
|
case INPUT_DEVICE_DIGITAL_MICROPHONE_1 : |
|
/* Enable AIF1ADC1 (Left), Enable AIF1ADC1 (Right) |
|
* Enable DMICDAT1 (Left), Enable DMICDAT1 (Right) |
|
* Enable Left ADC, Enable Right ADC */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x04, 0x030C); |
|
|
|
/* Enable AIF1 DRC2 Signal Detect & DRC in AIF1ADC1 Left/Right Timeslot 0 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x440, 0x00DB); |
|
|
|
/* Disable IN1L, IN1R, IN2L, IN2R, Enable Thermal sensor & shutdown */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x02, 0x6350); |
|
|
|
/* Enable the DMIC2(Left) to AIF1 Timeslot 0 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x606, 0x0002); |
|
|
|
/* Enable the DMIC2(Right) to AIF1 Timeslot 0 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x607, 0x0002); |
|
|
|
/* GPIO1 pin configuration GP1_DIR = output, GP1_FN = AIF1 DRC1 signal detect */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x700, 0x000D); |
|
break; |
|
case INPUT_DEVICE_DIGITAL_MIC1_MIC2 : |
|
/* Enable AIF1ADC1 (Left), Enable AIF1ADC1 (Right) |
|
* Enable DMICDAT1 (Left), Enable DMICDAT1 (Right) |
|
* Enable Left ADC, Enable Right ADC */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x04, 0x0F3C); |
|
|
|
/* Enable AIF1 DRC2 Signal Detect & DRC in AIF1ADC2 Left/Right Timeslot 1 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x450, 0x00DB); |
|
|
|
/* Enable AIF1 DRC2 Signal Detect & DRC in AIF1ADC1 Left/Right Timeslot 0 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x440, 0x00DB); |
|
|
|
/* Disable IN1L, IN1R, Enable IN2L, IN2R, Thermal sensor & shutdown */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x02, 0x63A0); |
|
|
|
/* Enable the DMIC2(Left) to AIF1 Timeslot 0 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x606, 0x0002); |
|
|
|
/* Enable the DMIC2(Right) to AIF1 Timeslot 0 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x607, 0x0002); |
|
|
|
/* Enable the DMIC2(Left) to AIF1 Timeslot 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x608, 0x0002); |
|
|
|
/* Enable the DMIC2(Right) to AIF1 Timeslot 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x609, 0x0002); |
|
|
|
/* GPIO1 pin configuration GP1_DIR = output, GP1_FN = AIF1 DRC1 signal detect */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x700, 0x000D); |
|
break; |
|
case INPUT_DEVICE_INPUT_LINE_2 : |
|
default: |
|
/* Actually, no other input devices supported */ |
|
counter++; |
|
break; |
|
} |
|
} |
|
else |
|
{ |
|
inputEnabled = 0; |
|
} |
|
|
|
/* Clock Configurations */ |
|
switch (AudioFreq) |
|
{ |
|
case AUDIO_FREQUENCY_8K: |
|
/* AIF1 Sample Rate = 8 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0003); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_16K: |
|
/* AIF1 Sample Rate = 16 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0033); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_32K: |
|
/* AIF1 Sample Rate = 32 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0063); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_48K: |
|
/* AIF1 Sample Rate = 48 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0083); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_96K: |
|
/* AIF1 Sample Rate = 96 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x00A3); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_11K: |
|
/* AIF1 Sample Rate = 11.025 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0013); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_22K: |
|
/* AIF1 Sample Rate = 22.050 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0043); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_44K: |
|
/* AIF1 Sample Rate = 44.1 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0073); |
|
break; |
|
|
|
default: |
|
/* AIF1 Sample Rate = 48 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0083); |
|
break; |
|
} |
|
|
|
if(input_device == INPUT_DEVICE_DIGITAL_MIC1_MIC2) |
|
{ |
|
/* AIF1 Word Length = 16-bits, AIF1 Format = DSP mode */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x300, 0x4018); |
|
} |
|
else |
|
{ |
|
/* AIF1 Word Length = 16-bits, AIF1 Format = I2S (Default Register Value) */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x300, 0x4010); |
|
} |
|
|
|
/* slave mode */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x302, 0x0000); |
|
|
|
/* Enable the DSP processing clock for AIF1, Enable the core clock */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x208, 0x000A); |
|
|
|
/* Enable AIF1 Clock, AIF1 Clock Source = MCLK1 pin */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x200, 0x0001); |
|
|
|
if (output_device > 0) /* Audio output selected */ |
|
{ |
|
if (output_device == OUTPUT_DEVICE_HEADPHONE) |
|
{ |
|
/* Select DAC1 (Left) to Left Headphone Output PGA (HPOUT1LVOL) path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x2D, 0x0100); |
|
|
|
/* Select DAC1 (Right) to Right Headphone Output PGA (HPOUT1RVOL) path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x2E, 0x0100); |
|
|
|
/* Startup sequence for Headphone */ |
|
if(ColdStartup) |
|
{ |
|
counter += CODEC_IO_Write(DeviceAddr,0x110,0x8100); |
|
|
|
ColdStartup=0; |
|
/* Add Delay */ |
|
AUDIO_IO_Delay(300); |
|
} |
|
else /* Headphone Warm Start-Up */ |
|
{ |
|
counter += CODEC_IO_Write(DeviceAddr,0x110,0x8108); |
|
/* Add Delay */ |
|
AUDIO_IO_Delay(50); |
|
} |
|
|
|
/* Soft un-Mute the AIF1 Timeslot 0 DAC1 path L&R */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x420, 0x0000); |
|
} |
|
/* Analog Output Configuration */ |
|
|
|
/* Enable SPKRVOL PGA, Enable SPKMIXR, Enable SPKLVOL PGA, Enable SPKMIXL */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x03, 0x0300); |
|
|
|
/* Left Speaker Mixer Volume = 0dB */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x22, 0x0000); |
|
|
|
/* Speaker output mode = Class D, Right Speaker Mixer Volume = 0dB ((0x23, 0x0100) = class AB)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x23, 0x0000); |
|
|
|
/* Unmute DAC2 (Left) to Left Speaker Mixer (SPKMIXL) path, |
|
Unmute DAC2 (Right) to Right Speaker Mixer (SPKMIXR) path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x36, 0x0300); |
|
|
|
/* Enable bias generator, Enable VMID, Enable SPKOUTL, Enable SPKOUTR */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x01, 0x3003); |
|
|
|
/* Headphone/Speaker Enable */ |
|
|
|
if (input_device == INPUT_DEVICE_DIGITAL_MIC1_MIC2) |
|
{ |
|
/* Enable Class W, Class W Envelope Tracking = AIF1 Timeslots 0 and 1 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x51, 0x0205); |
|
} |
|
else |
|
{ |
|
/* Enable Class W, Class W Envelope Tracking = AIF1 Timeslot 0 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x51, 0x0005); |
|
} |
|
|
|
/* Enable bias generator, Enable VMID, Enable HPOUT1 (Left) and Enable HPOUT1 (Right) input stages */ |
|
/* idem for Speaker */ |
|
power_mgnt_reg_1 |= 0x0303 | 0x3003; |
|
counter += CODEC_IO_Write(DeviceAddr, 0x01, power_mgnt_reg_1); |
|
|
|
/* Enable HPOUT1 (Left) and HPOUT1 (Right) intermediate stages */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x60, 0x0022); |
|
|
|
/* Enable Charge Pump */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x4C, 0x9F25); |
|
|
|
/* Add Delay */ |
|
AUDIO_IO_Delay(15); |
|
|
|
/* Select DAC1 (Left) to Left Headphone Output PGA (HPOUT1LVOL) path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x2D, 0x0001); |
|
|
|
/* Select DAC1 (Right) to Right Headphone Output PGA (HPOUT1RVOL) path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x2E, 0x0001); |
|
|
|
/* Enable Left Output Mixer (MIXOUTL), Enable Right Output Mixer (MIXOUTR) */ |
|
/* idem for SPKOUTL and SPKOUTR */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x03, 0x0030 | 0x0300); |
|
|
|
/* Enable DC Servo and trigger start-up mode on left and right channels */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x54, 0x0033); |
|
|
|
/* Add Delay */ |
|
AUDIO_IO_Delay(257); |
|
|
|
/* Enable HPOUT1 (Left) and HPOUT1 (Right) intermediate and output stages. Remove clamps */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x60, 0x00EE); |
|
|
|
/* Unmutes */ |
|
|
|
/* Unmute DAC 1 (Left) */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x610, 0x00C0); |
|
|
|
/* Unmute DAC 1 (Right) */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x611, 0x00C0); |
|
|
|
/* Unmute the AIF1 Timeslot 0 DAC path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x420, 0x0010); |
|
|
|
/* Unmute DAC 2 (Left) */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x612, 0x00C0); |
|
|
|
/* Unmute DAC 2 (Right) */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x613, 0x00C0); |
|
|
|
/* Unmute the AIF1 Timeslot 1 DAC2 path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x422, 0x0010); |
|
|
|
/* Volume Control */ |
|
wm8994_SetVolume(DeviceAddr, Volume); |
|
} |
|
|
|
if (input_device > 0) /* Audio input selected */ |
|
{ |
|
if ((input_device == INPUT_DEVICE_DIGITAL_MICROPHONE_1) || (input_device == INPUT_DEVICE_DIGITAL_MICROPHONE_2)) |
|
{ |
|
/* Enable Microphone bias 1 generator, Enable VMID */ |
|
power_mgnt_reg_1 |= 0x0013; |
|
counter += CODEC_IO_Write(DeviceAddr, 0x01, power_mgnt_reg_1); |
|
|
|
/* ADC oversample enable */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x620, 0x0002); |
|
|
|
/* AIF ADC2 HPF enable, HPF cut = voice mode 1 fc=127Hz at fs=8kHz */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x411, 0x3800); |
|
} |
|
else if(input_device == INPUT_DEVICE_DIGITAL_MIC1_MIC2) |
|
{ |
|
/* Enable Microphone bias 1 generator, Enable VMID */ |
|
power_mgnt_reg_1 |= 0x0013; |
|
counter += CODEC_IO_Write(DeviceAddr, 0x01, power_mgnt_reg_1); |
|
|
|
/* ADC oversample enable */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x620, 0x0002); |
|
|
|
/* AIF ADC1 HPF enable, HPF cut = voice mode 1 fc=127Hz at fs=8kHz */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x410, 0x1800); |
|
|
|
/* AIF ADC2 HPF enable, HPF cut = voice mode 1 fc=127Hz at fs=8kHz */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x411, 0x1800); |
|
} |
|
else if ((input_device == INPUT_DEVICE_INPUT_LINE_1) || (input_device == INPUT_DEVICE_INPUT_LINE_2)) |
|
{ |
|
|
|
/* Disable mute on IN1L, IN1L Volume = +0dB */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x18, 0x000B); |
|
|
|
/* Disable mute on IN1R, IN1R Volume = +0dB */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x1A, 0x000B); |
|
|
|
/* AIF ADC1 HPF enable, HPF cut = hifi mode fc=4Hz at fs=48kHz */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x410, 0x1800); |
|
} |
|
/* Volume Control */ |
|
wm8994_SetVolume(DeviceAddr, Volume); |
|
} |
|
/* Return communication control value */ |
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Deinitializes the audio codec. |
|
* @param None |
|
* @retval None |
|
*/ |
|
void wm8994_DeInit(void) |
|
{ |
|
/* Deinitialize Audio Codec interface */ |
|
AUDIO_IO_DeInit(); |
|
} |
|
|
|
/** |
|
* @brief Get the WM8994 ID. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @retval The WM8994 ID |
|
*/ |
|
uint32_t wm8994_ReadID(uint16_t DeviceAddr) |
|
{ |
|
/* Initialize the Control interface of the Audio Codec */ |
|
AUDIO_IO_Init(); |
|
|
|
return ((uint32_t)AUDIO_IO_Read(DeviceAddr, WM8994_CHIPID_ADDR)); |
|
} |
|
|
|
/** |
|
* @brief Start the audio Codec play feature. |
|
* @note For this codec no Play options are required. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_Play(uint16_t DeviceAddr, uint16_t* pBuffer, uint16_t Size) |
|
{ |
|
uint32_t counter = 0; |
|
|
|
/* Resumes the audio file playing */ |
|
/* Unmute the output first */ |
|
counter += wm8994_SetMute(DeviceAddr, AUDIO_MUTE_OFF); |
|
|
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Pauses playing on the audio codec. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_Pause(uint16_t DeviceAddr) |
|
{ |
|
uint32_t counter = 0; |
|
|
|
/* Pause the audio file playing */ |
|
/* Mute the output first */ |
|
counter += wm8994_SetMute(DeviceAddr, AUDIO_MUTE_ON); |
|
|
|
/* Put the Codec in Power save mode */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x02, 0x01); |
|
|
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Resumes playing on the audio codec. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_Resume(uint16_t DeviceAddr) |
|
{ |
|
uint32_t counter = 0; |
|
|
|
/* Resumes the audio file playing */ |
|
/* Unmute the output first */ |
|
counter += wm8994_SetMute(DeviceAddr, AUDIO_MUTE_OFF); |
|
|
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Stops audio Codec playing. It powers down the codec. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @param CodecPdwnMode: selects the power down mode. |
|
* - CODEC_PDWN_SW: only mutes the audio codec. When resuming from this |
|
* mode the codec keeps the previous initialization |
|
* (no need to re-Initialize the codec registers). |
|
* - CODEC_PDWN_HW: Physically power down the codec. When resuming from this |
|
* mode, the codec is set to default configuration |
|
* (user should re-Initialize the codec in order to |
|
* play again the audio stream). |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_Stop(uint16_t DeviceAddr, uint32_t CodecPdwnMode) |
|
{ |
|
uint32_t counter = 0; |
|
|
|
if (outputEnabled != 0) |
|
{ |
|
/* Mute the output first */ |
|
counter += wm8994_SetMute(DeviceAddr, AUDIO_MUTE_ON); |
|
|
|
if (CodecPdwnMode == CODEC_PDWN_SW) |
|
{ |
|
/* Only output mute required*/ |
|
} |
|
else /* CODEC_PDWN_HW */ |
|
{ |
|
/* Mute the AIF1 Timeslot 0 DAC1 path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x420, 0x0200); |
|
|
|
/* Mute the AIF1 Timeslot 1 DAC2 path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x422, 0x0200); |
|
|
|
/* Disable DAC1L_TO_HPOUT1L */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x2D, 0x0000); |
|
|
|
/* Disable DAC1R_TO_HPOUT1R */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x2E, 0x0000); |
|
|
|
/* Disable DAC1 and DAC2 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0000); |
|
|
|
/* Reset Codec by writing in 0x0000 address register */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x0000, 0x0000); |
|
|
|
outputEnabled = 0; |
|
} |
|
} |
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Sets higher or lower the codec volume level. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @param Volume: a byte value from 0 to 255 (refer to codec registers |
|
* description for more details). |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_SetVolume(uint16_t DeviceAddr, uint8_t Volume) |
|
{ |
|
uint32_t counter = 0; |
|
uint8_t convertedvol = VOLUME_CONVERT(Volume); |
|
|
|
/* Output volume */ |
|
if (outputEnabled != 0) |
|
{ |
|
if(convertedvol > 0x3E) |
|
{ |
|
/* Unmute audio codec */ |
|
counter += wm8994_SetMute(DeviceAddr, AUDIO_MUTE_OFF); |
|
|
|
/* Left Headphone Volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x1C, 0x3F | 0x140); |
|
|
|
/* Right Headphone Volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x1D, 0x3F | 0x140); |
|
|
|
/* Left Speaker Volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x26, 0x3F | 0x140); |
|
|
|
/* Right Speaker Volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x27, 0x3F | 0x140); |
|
} |
|
else if (Volume == 0) |
|
{ |
|
/* Mute audio codec */ |
|
counter += wm8994_SetMute(DeviceAddr, AUDIO_MUTE_ON); |
|
} |
|
else |
|
{ |
|
/* Unmute audio codec */ |
|
counter += wm8994_SetMute(DeviceAddr, AUDIO_MUTE_OFF); |
|
|
|
/* Left Headphone Volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x1C, convertedvol | 0x140); |
|
|
|
/* Right Headphone Volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x1D, convertedvol | 0x140); |
|
|
|
/* Left Speaker Volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x26, convertedvol | 0x140); |
|
|
|
/* Right Speaker Volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x27, convertedvol | 0x140); |
|
} |
|
} |
|
|
|
/* Input volume */ |
|
if (inputEnabled != 0) |
|
{ |
|
convertedvol = VOLUME_IN_CONVERT(Volume); |
|
|
|
/* Left AIF1 ADC1 volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x400, convertedvol | 0x100); |
|
|
|
/* Right AIF1 ADC1 volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x401, convertedvol | 0x100); |
|
|
|
/* Left AIF1 ADC2 volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x404, convertedvol | 0x100); |
|
|
|
/* Right AIF1 ADC2 volume */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x405, convertedvol | 0x100); |
|
} |
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Enables or disables the mute feature on the audio codec. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @param Cmd: AUDIO_MUTE_ON to enable the mute or AUDIO_MUTE_OFF to disable the |
|
* mute mode. |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_SetMute(uint16_t DeviceAddr, uint32_t Cmd) |
|
{ |
|
uint32_t counter = 0; |
|
|
|
if (outputEnabled != 0) |
|
{ |
|
/* Set the Mute mode */ |
|
if(Cmd == AUDIO_MUTE_ON) |
|
{ |
|
/* Soft Mute the AIF1 Timeslot 0 DAC1 path L&R */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x420, 0x0200); |
|
|
|
/* Soft Mute the AIF1 Timeslot 1 DAC2 path L&R */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x422, 0x0200); |
|
} |
|
else /* AUDIO_MUTE_OFF Disable the Mute */ |
|
{ |
|
/* Unmute the AIF1 Timeslot 0 DAC1 path L&R */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x420, 0x0010); |
|
|
|
/* Unmute the AIF1 Timeslot 1 DAC2 path L&R */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x422, 0x0010); |
|
} |
|
} |
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Switch dynamically (while audio file is played) the output target |
|
* (speaker or headphone). |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @param Output: specifies the audio output target: OUTPUT_DEVICE_SPEAKER, |
|
* OUTPUT_DEVICE_HEADPHONE, OUTPUT_DEVICE_BOTH or OUTPUT_DEVICE_AUTO |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_SetOutputMode(uint16_t DeviceAddr, uint8_t Output) |
|
{ |
|
uint32_t counter = 0; |
|
|
|
switch (Output) |
|
{ |
|
case OUTPUT_DEVICE_SPEAKER: |
|
/* Enable DAC1 (Left), Enable DAC1 (Right), |
|
Disable DAC2 (Left), Disable DAC2 (Right)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0C0C); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x601, 0x0000); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x602, 0x0000); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Left) to DAC 2 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x604, 0x0002); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Right) to DAC 2 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x605, 0x0002); |
|
break; |
|
|
|
case OUTPUT_DEVICE_HEADPHONE: |
|
/* Disable DAC1 (Left), Disable DAC1 (Right), |
|
Enable DAC2 (Left), Enable DAC2 (Right)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0303); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x601, 0x0001); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x602, 0x0001); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Left) to DAC 2 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x604, 0x0000); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Right) to DAC 2 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x605, 0x0000); |
|
break; |
|
|
|
case OUTPUT_DEVICE_BOTH: |
|
/* Enable DAC1 (Left), Enable DAC1 (Right), |
|
also Enable DAC2 (Left), Enable DAC2 (Right)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0303 | 0x0C0C); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x601, 0x0001); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x602, 0x0001); |
|
|
|
/* Enable the AIF1 Timeslot 1 (Left) to DAC 2 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x604, 0x0002); |
|
|
|
/* Enable the AIF1 Timeslot 1 (Right) to DAC 2 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x605, 0x0002); |
|
break; |
|
|
|
default: |
|
/* Disable DAC1 (Left), Disable DAC1 (Right), |
|
Enable DAC2 (Left), Enable DAC2 (Right)*/ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x05, 0x0303); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Left) to DAC 1 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x601, 0x0001); |
|
|
|
/* Enable the AIF1 Timeslot 0 (Right) to DAC 1 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x602, 0x0001); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Left) to DAC 2 (Left) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x604, 0x0000); |
|
|
|
/* Disable the AIF1 Timeslot 1 (Right) to DAC 2 (Right) mixer path */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x605, 0x0000); |
|
break; |
|
} |
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Sets new frequency. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @param AudioFreq: Audio frequency used to play the audio stream. |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_SetFrequency(uint16_t DeviceAddr, uint32_t AudioFreq) |
|
{ |
|
uint32_t counter = 0; |
|
|
|
/* Clock Configurations */ |
|
switch (AudioFreq) |
|
{ |
|
case AUDIO_FREQUENCY_8K: |
|
/* AIF1 Sample Rate = 8 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0003); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_16K: |
|
/* AIF1 Sample Rate = 16 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0033); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_32K: |
|
/* AIF1 Sample Rate = 32 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0063); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_48K: |
|
/* AIF1 Sample Rate = 48 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0083); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_96K: |
|
/* AIF1 Sample Rate = 96 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x00A3); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_11K: |
|
/* AIF1 Sample Rate = 11.025 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0013); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_22K: |
|
/* AIF1 Sample Rate = 22.050 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0043); |
|
break; |
|
|
|
case AUDIO_FREQUENCY_44K: |
|
/* AIF1 Sample Rate = 44.1 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0073); |
|
break; |
|
|
|
default: |
|
/* AIF1 Sample Rate = 48 (KHz), ratio=256 */ |
|
counter += CODEC_IO_Write(DeviceAddr, 0x210, 0x0083); |
|
break; |
|
} |
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Resets wm8994 registers. |
|
* @param DeviceAddr: Device address on communication Bus. |
|
* @retval 0 if correct communication, else wrong communication |
|
*/ |
|
uint32_t wm8994_Reset(uint16_t DeviceAddr) |
|
{ |
|
uint32_t counter = 0; |
|
|
|
/* Reset Codec by writing in 0x0000 address register */ |
|
counter = CODEC_IO_Write(DeviceAddr, 0x0000, 0x0000); |
|
outputEnabled = 0; |
|
inputEnabled=0; |
|
|
|
return counter; |
|
} |
|
|
|
/** |
|
* @brief Writes/Read a single data. |
|
* @param Addr: I2C address |
|
* @param Reg: Reg address |
|
* @param Value: Data to be written |
|
* @retval None |
|
*/ |
|
static uint8_t CODEC_IO_Write(uint8_t Addr, uint16_t Reg, uint16_t Value) |
|
{ |
|
uint32_t result = 0; |
|
|
|
AUDIO_IO_Write(Addr, Reg, Value); |
|
|
|
#ifdef VERIFY_WRITTENDATA |
|
/* Verify that the data has been correctly written */ |
|
result = (AUDIO_IO_Read(Addr, Reg) == Value)? 0:1; |
|
#endif /* VERIFY_WRITTENDATA */ |
|
|
|
return result; |
|
} |
|
|
|
/** |
|
* @} |
|
*/ |
|
|
|
/** |
|
* @} |
|
*/ |
|
|
|
/** |
|
* @} |
|
*/ |
|
|
|
/** |
|
* @} |
|
*/ |
|
|
|
/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/
|
|
|