Skip to content
Snippets Groups Projects
Select Git revision
  • c7c4a8439735ff7617f78b5b6c0cdfb05ae88656
  • wip-bootstrap default
  • dualcore
  • ch3/leds
  • ch3/time
  • master
6 results

audio.c

Blame
  • audio.c 8.97 KiB
    #include <stdint.h>
    #include <string.h>
    
    #include "stm32f4xx_dac.h"
    
    #include "nlr.h"
    #include "misc.h"
    #include "mpconfig.h"
    #include "qstr.h"
    #include "parse.h"
    #include "obj.h"
    #include "runtime.h"
    
    #include "audio.h"
    
    STATIC void TIM7_Config(uint freq) {
        // TIM7 clock enable
        RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM7, ENABLE);
    
        // reset TIM7
        TIM_DeInit(TIM7);
    
        // Compute the prescaler value so TIM7 triggers at freq-Hz
        uint16_t period = (uint16_t) ((SystemCoreClock / 2) / freq) - 1;
    
        // Time base configuration
        TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
        TIM_TimeBaseStructure.TIM_Period = period; // timer triggers with this period
        TIM_TimeBaseStructure.TIM_Prescaler = 0; // timer runs at SystemCoreClock / 2
        TIM_TimeBaseStructure.TIM_ClockDivision = 0; // unused for TIM7
        TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; // unused for TIM7
        TIM_TimeBaseInit(TIM7, &TIM_TimeBaseStructure);
    
        // TIM7 TRGO selection
        TIM_SelectOutputTrigger(TIM7, TIM_TRGOSource_Update);
    
        // TIM7 enable counter
        TIM_Cmd(TIM7, ENABLE);
    }
    
    /******************************************************************************/
    // Micro Python bindings
    
    typedef struct _pyb_audio_t {
        mp_obj_base_t base;
        uint dac_channel; // DAC_Channel_1 or DAC_Channel_2
        DMA_Stream_TypeDef *dma_stream; // DMA1_Stream6 or DMA1_Stream7
    } pyb_audio_t;
    
    mp_obj_t pyb_audio_noise(mp_obj_t self_in, mp_obj_t freq) {
        pyb_audio_t *self = self_in;
    
        // set TIM7 to trigger the DAC at the given frequency
        TIM7_Config(mp_obj_get_int(freq));
    
        DAC_Cmd(self->dac_channel, DISABLE);
    
        DAC_InitTypeDef DAC_InitStructure;
        DAC_InitStructure.DAC_Trigger = DAC_Trigger_T7_TRGO;
        DAC_InitStructure.DAC_WaveGeneration = DAC_WaveGeneration_Noise;
        DAC_InitStructure.DAC_LFSRUnmask_TriangleAmplitude = DAC_LFSRUnmask_Bits10_0;
        DAC_InitStructure.DAC_OutputBuffer = DAC_OutputBuffer_Enable;
        DAC_Init(self->dac_channel, &DAC_InitStructure);
    
        DAC_Cmd(self->dac_channel, ENABLE);
    
        if (self->dac_channel == DAC_Channel_1) {
            DAC_SetChannel1Data(DAC_Align_12b_L, 0x7ff0);
        } else {
            DAC_SetChannel2Data(DAC_Align_12b_L, 0x7ff0);
        }
    
        return mp_const_none;
    }
    
    mp_obj_t pyb_audio_triangle(mp_obj_t self_in, mp_obj_t freq) {
        pyb_audio_t *self = self_in;
    
        // set TIM7 to trigger the DAC at the given frequency
        TIM7_Config(mp_obj_get_int(freq));
    
        DAC_Cmd(self->dac_channel, DISABLE);
    
        DAC_InitTypeDef DAC_InitStructure;
        DAC_InitStructure.DAC_Trigger = DAC_Trigger_T7_TRGO;
        DAC_InitStructure.DAC_WaveGeneration = DAC_WaveGeneration_Triangle;
        DAC_InitStructure.DAC_LFSRUnmask_TriangleAmplitude = DAC_TriangleAmplitude_1023;
        DAC_InitStructure.DAC_OutputBuffer = DAC_OutputBuffer_Enable;
        DAC_Init(self->dac_channel, &DAC_InitStructure);
    
        DAC_Cmd(self->dac_channel, ENABLE);
    
        // set base value of triangle wave
        if (self->dac_channel == DAC_Channel_1) {
            DAC_SetChannel1Data(DAC_Align_12b_R, 0x100);
        } else {
            DAC_SetChannel2Data(DAC_Align_12b_R, 0x100);
        }
    
        return mp_const_none;
    }
    
    // direct access to DAC
    mp_obj_t pyb_audio_dac(mp_obj_t self_in, mp_obj_t val) {
        pyb_audio_t *self = self_in;
        if (self->dac_channel == DAC_Channel_1) {
            DAC_SetChannel1Data(DAC_Align_8b_R, mp_obj_get_int(val));
        } else {
            DAC_SetChannel2Data(DAC_Align_8b_R, mp_obj_get_int(val));
        }
        return mp_const_none;
    }
    
    #define DAC_DHR8R1_ADDRESS (DAC_BASE + 0x10)
    #define DAC_DHR8R2_ADDRESS (DAC_BASE + 0x1c)
    
    // initiates a burst of RAM->DAC using DMA
    // input data is treated as an array of bytes (8 bit data)
    // TIM7 is used to set the frequency of the transfer
    mp_obj_t pyb_audio_dma(uint n_args, const mp_obj_t *args, mp_map_t *kw_args) {
        pyb_audio_t *self = args[0];
    
        // set TIM7 to trigger the DAC at the given frequency
        TIM7_Config(mp_obj_get_int(args[2]));
    
        mp_obj_type_t *type = mp_obj_get_type(args[1]);
        if (type->buffer_p.get_buffer == NULL) {
            nlr_raise(mp_obj_new_exception_msg(&mp_type_TypeError, "buffer argument must support buffer protocol"));
        }
        mp_buffer_info_t bufinfo;
        type->buffer_p.get_buffer(args[1], &bufinfo, MP_BUFFER_READ);
    
        RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
    
        DMA_Cmd(self->dma_stream, DISABLE);
        while (DMA_GetCmdStatus(self->dma_stream) != DISABLE) {
        }
    
        DAC_Cmd(self->dac_channel, DISABLE);
    
        // DAC channel configuration
        DAC_InitTypeDef DAC_InitStructure;
        DAC_InitStructure.DAC_Trigger = DAC_Trigger_T7_TRGO;
        DAC_InitStructure.DAC_WaveGeneration = DAC_WaveGeneration_None;
        DAC_InitStructure.DAC_LFSRUnmask_TriangleAmplitude = DAC_TriangleAmplitude_1; // unused, but need to set it to a valid value
        DAC_InitStructure.DAC_OutputBuffer = DAC_OutputBuffer_Enable;
        DAC_Init(self->dac_channel, &DAC_InitStructure);
    
        // DMA1_Stream[67] channel7 configuration
        DMA_DeInit(self->dma_stream);
        DMA_InitTypeDef DMA_InitStructure;
        DMA_InitStructure.DMA_Channel = DMA_Channel_7;
        if (self->dac_channel == DAC_Channel_1) {
            DMA_InitStructure.DMA_PeripheralBaseAddr = DAC_DHR8R1_ADDRESS;
        } else {
            DMA_InitStructure.DMA_PeripheralBaseAddr = DAC_DHR8R2_ADDRESS;
        }
        DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)bufinfo.buf;
        DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral;
        DMA_InitStructure.DMA_BufferSize = bufinfo.len;
        DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
        DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
        DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
        DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
        mp_map_elem_t *kw_mode = mp_map_lookup(kw_args, MP_OBJ_NEW_QSTR(qstr_from_str("mode")), MP_MAP_LOOKUP);
        DMA_InitStructure.DMA_Mode = kw_mode == NULL ? DMA_Mode_Normal : mp_obj_get_int(kw_mode->value); // normal = 0, circular = 0x100
        DMA_InitStructure.DMA_Priority = DMA_Priority_High;
        DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
        DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;
        DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
        DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
        DMA_Init(self->dma_stream, &DMA_InitStructure);
    
        // enable DMA stream
        DMA_Cmd(self->dma_stream, ENABLE);
        while (DMA_GetCmdStatus(self->dma_stream) == DISABLE) {
        }
    
        // enable DAC channel
        DAC_Cmd(self->dac_channel, ENABLE);
    
        // enable DMA for DAC channel
        DAC_DMACmd(self->dac_channel, ENABLE);
    
        //printf("DMA: %p %lu\n", bufinfo.buf, bufinfo.len);
    
        return mp_const_none;
    }
    
    STATIC MP_DEFINE_CONST_FUN_OBJ_2(pyb_audio_noise_obj, pyb_audio_noise);
    STATIC MP_DEFINE_CONST_FUN_OBJ_2(pyb_audio_triangle_obj, pyb_audio_triangle);
    STATIC MP_DEFINE_CONST_FUN_OBJ_2(pyb_audio_dac_obj, pyb_audio_dac);
    STATIC MP_DEFINE_CONST_FUN_OBJ_KW(pyb_audio_dma_obj, 3, pyb_audio_dma);
    
    STATIC const mp_map_elem_t pyb_audio_locals_dict_table[] = {
        { MP_OBJ_NEW_QSTR(MP_QSTR_noise), (mp_obj_t)&pyb_audio_noise_obj },
        { MP_OBJ_NEW_QSTR(MP_QSTR_triangle), (mp_obj_t)&pyb_audio_triangle_obj },
        { MP_OBJ_NEW_QSTR(MP_QSTR_dac), (mp_obj_t)&pyb_audio_dac_obj },
        { MP_OBJ_NEW_QSTR(MP_QSTR_dma), (mp_obj_t)&pyb_audio_dma_obj },
    };
    
    STATIC MP_DEFINE_CONST_DICT(pyb_audio_locals_dict, pyb_audio_locals_dict_table);
    
    STATIC const mp_obj_type_t pyb_audio_type = {
        { &mp_type_type },
        .name = MP_QSTR_,
        .locals_dict = (mp_obj_t)&pyb_audio_locals_dict,
    };
    
    STATIC const pyb_audio_t pyb_audio_channel_1 = {{&pyb_audio_type}, DAC_Channel_1, DMA1_Stream5};
    STATIC const pyb_audio_t pyb_audio_channel_2 = {{&pyb_audio_type}, DAC_Channel_2, DMA1_Stream6};
    
    // create the audio object
    // currently support either DAC1 on X5 (id = 1) or DAC2 on X6 (id = 2)
    
    STATIC mp_obj_t pyb_Audio(mp_obj_t id) {
        // DAC peripheral clock
        RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE);
    
        int dac_id = mp_obj_get_int(id);
        uint pin;
        const pyb_audio_t *dac_obj;
    
        if (dac_id == 1) {
            pin = GPIO_Pin_4;
            dac_obj = &pyb_audio_channel_1;
        } else {
            pin = GPIO_Pin_5;
            dac_obj = &pyb_audio_channel_2;
        }
    
        // DAC channel configuration
        GPIO_InitTypeDef GPIO_InitStructure;
        GPIO_InitStructure.GPIO_Pin = pin;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;
        GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
        GPIO_Init(GPIOA, &GPIO_InitStructure);
    
        // DAC channel Configuration
        DAC_InitTypeDef DAC_InitStructure;
        DAC_InitStructure.DAC_Trigger = DAC_Trigger_None;
        DAC_InitStructure.DAC_WaveGeneration = DAC_WaveGeneration_None;
        DAC_InitStructure.DAC_LFSRUnmask_TriangleAmplitude = DAC_TriangleAmplitude_1023;
        DAC_InitStructure.DAC_OutputBuffer = DAC_OutputBuffer_Enable;
        DAC_Init(dac_obj->dac_channel, &DAC_InitStructure);
    
        // Enable DAC Channel
        DAC_Cmd(dac_obj->dac_channel, ENABLE);
    
        // from now on use DAC_SetChannel[12]Data to trigger a conversion
    
        // return static object
        return (mp_obj_t)dac_obj;
    }
    
    MP_DEFINE_CONST_FUN_OBJ_1(pyb_Audio_obj, pyb_Audio);