ВходНаше всё Теги codebook 无线电组件 Поиск Опросы Закон Четверг
21 ноября
1352892 Топик полностью
Eddy_Em (21.09.2023 08:53, просмотров: 175) ответил ЫЫyкпy на Были как-то глюки в работе ADC и DMA, причем не на всех экземплярах, а где-то 50/50. Вылечилось добавлением барьера DSB сразу после включения тактирования DMA, там где в исходнике "//Dummy read for a pause".
Не, не помогает. Уже что только ни делал - все равно почему-то первые несколько считываний АЦП проходят "мимо" DMA, а потом все стабильно. В итоге порядок плавает. 
void adc_setup(){
    uint32_t ctr = 0;
    // Enable clocking
    RCC->APB2ENR |= RCC_APB2ENR_ADC1EN;
    RCC->AHBENR |= RCC_AHBENR_DMA1EN;
    __DSB();
    RCC->CFGR = (RCC->CFGR & ~(RCC_CFGR_ADCPRE)) | RCC_CFGR_ADCPRE_DIV8; // ADC clock = RCC / 8
    // sampling time - 239.5 cycles for channels 3, 16 and 17
    ADC1->SMPR2 = ADC_SMPR2_SMP3;
    ADC1->SMPR1 = ADC_SMPR1_SMP16 | ADC_SMPR1_SMP17;
    // sequence order: 3 -> 16 -> 17
    ADC1->SQR3 = (3 << 0) | (16<<5) | (17 << 10);
    ADC1->SQR1 = (NUMBER_OF_ADC_CHANNELS - 1) << 20; // amount of conversions
    ADC1->CR1 = ADC_CR1_SCAN; // scan mode
    // continuous mode & DMA; enable vref & Tsens; wake up ADC
    ADC1->CR2 = ADC_CR2_DMA | ADC_CR2_TSVREFE | ADC_CR2_CONT | ADC_CR2_ADON;
    // wait for Tstab - at least 1us
    while(++ctr < 0xff) nop();
    // calibration
    ADC1->CR2 |= ADC_CR2_RSTCAL;
    ctr = 0; while((ADC1->CR2 & ADC_CR2_RSTCAL) && ++ctr < 0xfffff);
    ADC1->CR2 |= ADC_CR2_CAL;
    ctr = 0; while((ADC1->CR2 & ADC_CR2_CAL) && ++ctr < 0xfffff);
    // DMA configuration
    DMA1_Channel1->CPAR = (uint32_t) (&(ADC1->DR));
    DMA1_Channel1->CMAR = (uint32_t)(ADC_array);
    DMA1_Channel1->CNDTR = NUMBER_OF_ADC_CHANNELS * 9;
    DMA1_Channel1->CCR = DMA_CCR_MINC | DMA_CCR_MSIZE_0 | DMA_CCR_PSIZE_0
                          | DMA_CCR_CIRC | DMA_CCR_PL | DMA_CCR_EN;
    __DSB();
    // turn ON ADC
    ADC1->CR2 |= ADC_CR2_ADON;
}

Даже добавлял 0xffff nop'ов перед включением АЦП - не помогает.

eddy-em.livejournal.com github.com/eddyem