Forum: Mikrocontroller und Digitale Elektronik STM32 DMA ADC Circular Continous mode werte in verschiedenen Empfangsvariablen


von mr_willem (Gast)


Lesenswert?

Hallo liebe Mikrocontroller.net Forumsmitglieder.

Ich möchte zwei ADC Werte (ADC Channel 4 und ADC Channel 5) des STM32 
zirkulär per DMA in das Array ADCBuffer schreiben.
Nun ist es so, dass manchmal das Ergebnis des ADC Channel 4 in 
ADCBuffer[0] steht und manchmal in ADCBuffer[1].
Ich denke das ich da wohl irgendwo die Initialisierung falsch mache. 
Also vielleicht die Reihenfolge oder ähnliches.
Ich poste unten mal den Code, den ich benutze.
Das Ergebnis sollte immer im gleichen ADCBuffer stehen. Also ADC Channel 
4 immer in ADCBuffer[0] und ADC Channel 5 immer in ADCBuffer[1]
1
void pushbuttoninit(void)
2
{
3
  GPIO_InitTypeDef GPIO_InitStructure;
4
  ADC_InitTypeDef ADC_InitStructure;
5
  DMA_InitTypeDef DMA_InitStructure;
6
  NVIC_InitTypeDef NVIC_InitStructure;
7
8
9
        // Read the ADC Value boundaries of the buttons from FRAM
10
  eeprom_read_buttonvals(&buttons);
11
12
  GPIO_InitStructure.GPIO_Pin = SETUPBUTTON_PIN;
13
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
14
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
15
  GPIO_Init(SETUPBUTTON_GPIO, &GPIO_InitStructure);
16
17
  ADCBuffer[0] = ADCBuffer[1]  = 0;
18
19
  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
20
21
  DMA_InitStructure.DMA_BufferSize = 2;
22
  DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
23
  DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
24
  DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADCBuffer;
25
  DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
26
  DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
27
  DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
28
  DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;
29
  DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
30
  DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
31
  DMA_InitStructure.DMA_Priority = DMA_Priority_Low;
32
  DMA_Init(DMA1_Channel1, &DMA_InitStructure);
33
34
  DMA_Cmd(DMA1_Channel1, ENABLE);
35
  DMA_ITConfig(DMA1_Channel1, DMA_IT_TC, DISABLE);
36
  DMA_ITConfig(DMA1_Channel1, DMA_IT_HT, DISABLE);
37
  DMA_ITConfig(DMA1_Channel1, DMA_IT_TE, DISABLE);
38
39
  NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn;
40
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3;
41
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
42
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
43
  NVIC_Init(&NVIC_InitStructure);
44
45
  RCC_ADCCLKConfig(RCC_PCLK2_Div6);
46
47
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO | RCC_APB2Periph_ADC1, ENABLE);
48
49
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
50
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
51
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
52
  GPIO_Init(GPIOA, &GPIO_InitStructure);
53
54
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
55
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
56
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
57
  GPIO_Init(GPIOA, &GPIO_InitStructure);
58
59
  ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
60
  ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
61
  ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
62
  ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
63
  ADC_InitStructure.ADC_NbrOfChannel = 2;
64
  ADC_InitStructure.ADC_ScanConvMode = ENABLE;
65
  ADC_Init(ADC1, &ADC_InitStructure);
66
67
  ADC_RegularChannelConfig(ADC1, ADC_Channel_4, 1, ADC_SampleTime_239Cycles5);
68
  ADC_RegularChannelConfig(ADC1, ADC_Channel_5, 2, ADC_SampleTime_239Cycles5);
69
70
  ADC_Cmd(ADC1, ENABLE);
71
72
73
74
  ADC_ResetCalibration(ADC1);
75
  while(ADC_GetResetCalibrationStatus(ADC1));
76
  ADC_StartCalibration(ADC1);
77
  while(ADC_GetCalibrationStatus(ADC1));
78
79
  ADC_DMACmd(ADC1, ENABLE);
80
81
  ADC_SoftwareStartConvCmd(ADC1, ENABLE);
82
}

von Mw E. (Firma: fritzler-avr.de) (fritzler)


Lesenswert?

Sieht irgendwie so aus als würdeste ADC_DMACmd zu spät aufrufen.
Der ADC sampelt doch schon los und dann wird der DMA eingeschalten.
Dann landet es eben am zufällign Platz im Array.

Der DMA muss schon ab dem ersten Sample lauschen.

von mr_willem (Gast)


Lesenswert?

Ich danke dir.

Das wird es gewesen sein.
Ich dachte der ADC wird erst durch
1
ADC_SoftwareStartConvCmd(ADC1, ENABLE);

aktiviert.

Ich habe nun alles vorher deaktiviert und gestoppt.
1
ADC_Cmd(ADC1, DISABLE);
2
ADC_DMACmd(ADC1, DISABLE);
3
DMA_Cmd(DMA1_Channel1, DISABLE);
4
ADC_DeInit(ADC1);

und dann am Ende der Funktion
1
ADC_DMACmd(ADC1, ENABLE);
2
3
ADC_Cmd(ADC1, ENABLE);
4
5
ADC_ResetCalibration(ADC1);
6
while(ADC_GetResetCalibrationStatus(ADC1));
7
ADC_StartCalibration(ADC1);
8
while(ADC_GetCalibrationStatus(ADC1));
9
10
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
den ADC in dieser Reihenfolge aktiviert.

Jetzt scheint es zu laufen.

Vielen

Bitte melde dich an um einen Beitrag zu schreiben. Anmeldung ist kostenlos und dauert nur eine Minute.
Bestehender Account
Schon ein Account bei Google/GoogleMail? Keine Anmeldung erforderlich!
Mit Google-Account einloggen
Noch kein Account? Hier anmelden.