/** * @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_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 cs42l52_Stop(uint16_t DeviceAddr, uint32_t CodecPdwnMode) { uint32_t counter = 0; /* Mute the output first */ counter += cs42l52_SetMute(DeviceAddr, AUDIO_MUTE_ON); AUDIO_IO_Delay(20); /* Power down the DAC and the speaker (PMDAC and PMSPK bits)*/ counter += CODEC_IO_Write(DeviceAddr, 0x02, 0x9F); Is_cs42l52_Stop = 1; 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 cs42l52_Pause(uint16_t DeviceAddr) { uint32_t counter = 0; /* Pause the audio file playing */ /* Mute the output first */ counter += cs42l52_SetMute(DeviceAddr, AUDIO_MUTE_ON); AUDIO_IO_Delay(20); /* 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 cs42l52_Resume(uint16_t DeviceAddr) { uint32_t counter = 0; /* Resumes the audio file playing */ /* Unmute the output first */ counter += cs42l52_SetMute(DeviceAddr, AUDIO_MUTE_OFF); AUDIO_IO_Delay(20); counter += CODEC_IO_Write(DeviceAddr,0x04, OutputDev); /* Exit the Power save mode */ counter += CODEC_IO_Write(DeviceAddr,0x02, 0x9E); return counter; }
/** * @brief Initializes the audio codec and the control interface. * @param DeviceAddr: Device address on communication Bus. * @param OutputDevice: can be OUTPUT_DEVICE_SPEAKER, OUTPUT_DEVICE_HEADPHONE, * OUTPUT_DEVICE_BOTH or OUTPUT_DEVICE_AUTO . * @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 OutputDevice, uint8_t Volume, uint32_t AudioFreq) { uint32_t counter = 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, 0x0064); /* Enable bias generator, Enable VMID */ counter += CODEC_IO_Write(DeviceAddr, 0x01, 0x0003); /* Add Delay */ AUDIO_IO_Delay(50); /* Path Configurations */ switch (OutputDevice) { 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; } /* 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_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; } /* 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); /* Analogue 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); /* Volume Control */ wm8994_SetVolume(DeviceAddr, Volume); /* 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 */ /* 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 */ counter += CODEC_IO_Write(DeviceAddr, 0x01, 0x0303 | 0x3003); /* 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(250); /* 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, 0x0000); /* 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, 0x0000); /* Return communication control value */ return counter; }
/** * @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_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 */ 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: /* 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 : /* 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 : 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_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; } /* 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 */ { /* 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 */ /* 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(250); /* 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, 0x0000); /* 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, 0x0000); /* 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_INPUT_LINE_1) || (input_device == INPUT_DEVICE_INPUT_LINE_2)) { /* Enable normal bias generator, Enable VMID */ power_mgnt_reg_1 |= 0x0003; counter += CODEC_IO_Write(DeviceAddr, 0x01, power_mgnt_reg_1); /* 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); /* Disable mute on IN1L_TO_MIXINL, Gain = +0dB */ counter += CODEC_IO_Write(DeviceAddr, 0x29, 0x0025); /* Disable mute on IN1R_TO_MIXINL, Gain = +0dB */ counter += CODEC_IO_Write(DeviceAddr, 0x2A, 0x0025); /* IN1LN_TO_IN1L, IN1LP_TO_VMID, IN1RN_TO_IN1R, IN1RP_TO_VMID */ counter += CODEC_IO_Write(DeviceAddr, 0x28, 0x0011); /* 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; }