// Copyright 2014 Olivier Gillet. // // Author: Olivier Gillet (ol.gillet@gmail.com) // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. // // See http://creativecommons.org/licenses/MIT/ for more information. // // ----------------------------------------------------------------------------- // // WM8371 Codec support. #ifndef ELEMENTS_DRIVERS_CODEC_H_ #define ELEMENTS_DRIVERS_CODEC_H_ #include #include "stmlib/stmlib.h" #include "stmlib/utils/ring_buffer.h" namespace elements { enum CodecProtocol { CODEC_PROTOCOL_PHILIPS = I2S_Standard_Phillips, CODEC_PROTOCOL_MSB_FIRST = I2S_Standard_MSB, CODEC_PROTOCOL_LSB_FIRST = I2S_Standard_LSB }; enum CodecFormat { CODEC_FORMAT_16_BIT = I2S_DataFormat_16b, CODEC_FORMAT_24_BIT = I2S_DataFormat_24b, CODEC_FORMAT_32_BIT = I2S_DataFormat_32b }; enum CodecSettings { CODEC_INPUT_0_DB = 0x17, CODEC_HEADPHONES_MUTE = 0x00, CODEC_MIC_BOOST = 0x1, CODEC_MIC_MUTE = 0x2, CODEC_ADC_MIC = 0x4, CODEC_ADC_LINE = 0x0, CODEC_OUTPUT_DAC_ENABLE = 0x10, CODEC_OUTPUT_MONITOR = 0x20, CODEC_DEEMPHASIS_NONE = 0x00, CODEC_DEEMPHASIS_32K = 0x01, CODEC_DEEMPHASIS_44K = 0x02, CODEC_DEEMPHASIS_48K = 0x03, CODEC_SOFT_MUTE = 0x01, CODEC_ADC_HPF = 0x00, CODEC_POWER_DOWN_LINE_IN = 0x01, CODEC_POWER_DOWN_MIC = 0x02, CODEC_POWER_DOWN_ADC = 0x04, CODEC_POWER_DOWN_DAC = 0x08, CODEC_POWER_DOWN_LINE_OUT = 0x10, CODEC_POWER_DOWN_OSCILLATOR = 0x20, CODEC_POWER_DOWN_CLOCK_OUTPUT = 0x40, CODEC_POWER_DOWN_EVERYTHING = 0x80, CODEC_PROTOCOL_MASK_MSB_FIRST = 0x00, CODEC_PROTOCOL_MASK_LSB_FIRST = 0x01, CODEC_PROTOCOL_MASK_PHILIPS = 0x02, CODEC_PROTOCOL_MASK_DSP = 0x03, CODEC_FORMAT_MASK_16_BIT = 0x00 << 2, CODEC_FORMAT_MASK_20_BIT = 0x01 << 2, CODEC_FORMAT_MASK_24_BIT = 0x02 << 2, CODEC_FORMAT_MASK_32_BIT = 0x03 << 2, CODEC_FORMAT_LR_SWAP = 0x20, CODEC_FORMAT_MASTER = 0x40, CODEC_FORMAT_SLAVE = 0x00, CODEC_FORMAT_INVERT_CLOCK = 0x80, CODEC_RATE_48K_48K = 0x00 << 2, CODEC_RATE_96K_96K = 0x07 << 2, CODEC_RATE_32K_32K = 0x06 << 2, CODEC_RATE_44K_44K = 0x18 << 2, }; // Size of an audio chunk in frames (stereo samples). // The DMA buffer is 2 chunks long. const size_t kAudioChunkSize = 16; // Size of the additional FIFOs in chunks. This gives an additional layer of // Buffering to make things more robust to jitter (for example if there are // occasional expensive calls in the UI). // Use 0 for using naive double-buffering. const size_t kNumFIFOChunks = 0; class Codec { public: Codec() { } ~Codec() { } typedef struct { short l; short r; } Frame; typedef void (*FillBufferCallback)(Frame* rx, Frame* tx, size_t size); bool Init(uint32_t sample_rate, CodecProtocol protocol, CodecFormat format); bool Start() { // No callback - the caller is supposed to poll with available() return Start(NULL); } bool Start(FillBufferCallback callback); void Stop(); void Fill(size_t offset); static Codec* GetInstance() { return instance_; } // When naive double-buffering is used: // 1. Call available() to know if a half-buffer is available. // 2. Call Grab(&input_ptr, &output_ptr) to retrieve pointer on the // available rx/tx half-buffers, along with their size. inline bool available() const { return transmitted_ != processed_; } inline size_t Grab(Frame** input, Frame** output) { ++processed_; *input = client_rx_; *output = client_tx_; return kAudioChunkSize; } // When the extra FIFOs are used: // 1. Call writable() and readable() to know how much data can be read // and written from/to the FIFO. // 2. Call ImmedidateRead() and Overwrite() to read/write from/to the FIFO. size_t writable() const { return tx_buffer_.writable(); } size_t readable() const { return rx_buffer_.readable(); } void ImmediateRead(Frame* destination, size_t size) { rx_buffer_.ImmediateRead(destination, size); } void Overwrite(Frame* source, size_t size) { tx_buffer_.Overwrite(source, size); } private: bool InitializeGPIO(); bool InitializeControlInterface(); bool InitializeAudioInterface(uint32_t, CodecProtocol, CodecFormat); bool WriteControlRegister(uint8_t address, uint16_t data); bool InitializeCodec(uint32_t, CodecProtocol, CodecFormat); bool InitializeDMA(); static Codec* instance_; bool use_buffering_; size_t transmitted_; size_t processed_; Frame* client_tx_; Frame* client_rx_; FillBufferCallback callback_; DMA_InitTypeDef dma_init_tx_; DMA_InitTypeDef dma_init_rx_; Frame tx_dma_buffer_[kAudioChunkSize * 2]; Frame rx_dma_buffer_[kAudioChunkSize * 2]; stmlib::RingBuffer rx_buffer_; stmlib::RingBuffer tx_buffer_; DISALLOW_COPY_AND_ASSIGN(Codec); }; } // namespace elements #endif // ELEMENTS_DRIVERS_CODEC_H_