Charakterystyka DAC
STM32F407 został wyposażony w dwa wyjścia sygnału DAC. Zlokalizowane są one w następujących wyprowadzeniach:
- DAC1 - PA4
- DAC2 - PA5
Każdy z kanałów został wyposażony w osobny konwerter. Jeśli wykorzystywany jest tryb dwukanałowy, wtedy dane mogą być wprowadzane osobno bądź jednocześnie.
Przez ten układ akceptowane są trzy rodzaje danych całkowitych: 8 bitowe, 12 bitowe wyrównane do lewej oraz 12 bitowe wyrównane do prawej. Zakres danych 12 bitowych rozpoczyna się od 0x0000, natomiast maksymalnie można wprowadzić 0xFFFF.
Programowanie
Poniżej znajduje się podstawowy program, którego zadaniem jest generacja sygnału analogowego na określonym poziomie.
Jeśli chodzi o strukturę włączająca DAC, do wyboru są cztery pola:
DAC_Trigger - pozwala na wybranie zewnętrznego sygnału generowanego dla wybranego kanału.
- #define DAC_Trigger_None ((uint32_t)0x00000000) //Konwersja automatyczna
- #define DAC_Trigger_T2_TRGO ((uint32_t)0x00000024) //Generowane z TIM2
- #define DAC_Trigger_T4_TRGO ((uint32_t)0x0000002C) //Generowane z TIM4
- #define DAC_Trigger_T5_TRGO ((uint32_t)0x0000001C) //Generowane z TIM5
- #define DAC_Trigger_T6_TRGO ((uint32_t)0x00000004) //Generowane z TIM6
- #define DAC_Trigger_T7_TRGO ((uint32_t)0x00000014) //Generowane z TIM7
- #define DAC_Trigger_T8_TRGO ((uint32_t)0x0000000C) //Generowane z TIM8
- #define DAC_Trigger_Ext_IT9 ((uint32_t)0x00000034) //Generacja z EXTI kanał 9
- #define DAC_Trigger_Software ((uint32_t)0x0000003C) //Rozpoczęcie konwersji poprzez start w systemie
DAC_WaveGeneration - określa jaki sygnał będzie generowany czy sygnał szumu, trójkątny bądź bez generacji.
- #define DAC_WaveGeneration_None ((uint32_t)0x00000000) //Bez generacji
- #define DAC_WaveGeneration_Noise ((uint32_t)0x00000040) //Szum
- #define DAC_WaveGeneration_Triangle ((uint32_t)0x00000080) //Sygnał trójkątny
DAC_LFSRUnmask_TriangleAmplitude - pozwala na określenie maski LFSR dla szumu oraz maksymalnej amplitudy sygnału trójkątnego
- #define DAC_LFSRUnmask_Bit0 ((uint32_t)0x00000000) //Odsloniete LFSR, bit0 dla generacji szumu
- #define DAC_LFSRUnmask_Bits1_0 ((uint32_t)0x00000100) //Odsloniete LFSR, bit[1:0]
- #define DAC_LFSRUnmask_Bits2_0 ((uint32_t)0x00000200) //Odsloniete LFSR, bit[2:0]
- #define DAC_LFSRUnmask_Bits3_0 ((uint32_t)0x00000300) //Odsloniete LFSR, bit[3:0]
- #define DAC_LFSRUnmask_Bits4_0 ((uint32_t)0x00000400) //Odsloniete LFSR, bit[4:0]
- #define DAC_LFSRUnmask_Bits5_0 ((uint32_t)0x00000500) //Odsloniete LFSR, bit[5:0]
- #define DAC_LFSRUnmask_Bits6_0 ((uint32_t)0x00000600) //Odsloniete LFSR, bit[6:0]
- #define DAC_LFSRUnmask_Bits7_0 ((uint32_t)0x00000700) //Odsloniete LFSR, bit[7:0]
- #define DAC_LFSRUnmask_Bits8_0 ((uint32_t)0x00000800) //Odsloniete LFSR, bit[8:0]
- #define DAC_LFSRUnmask_Bits9_0 ((uint32_t)0x00000900) //Odsloniete LFSR, bit[9:0]
- #define DAC_LFSRUnmask_Bits10_0 ((uint32_t)0x00000A00) //Odsloniete LFSR, bit[10:0]
- #define DAC_LFSRUnmask_Bits11_0 ((uint32_t)0x00000B00) //Odsloniete LFSR, bit[11:0]
- #define DAC_TriangleAmplitude_1 ((uint32_t)0x00000000) //Maksymalna amplituda sygnalu trojkatnego 1
- #define DAC_TriangleAmplitude_3 ((uint32_t)0x00000100) //Maksymalna amplituda sygnalu trojkatnego 3
- #define DAC_TriangleAmplitude_7 ((uint32_t)0x00000200) //Maksymalna amplituda sygnalu trojkatnego 7
- #define DAC_TriangleAmplitude_15 ((uint32_t)0x00000300) //Maksymalna amplituda sygnalu trojkatnego 15
- #define DAC_TriangleAmplitude_31 ((uint32_t)0x00000400) //Maksymalna amplituda sygnalu trojkatnego 31
- #define DAC_TriangleAmplitude_63 ((uint32_t)0x00000500) //Maksymalna amplituda sygnalu trojkatnego 63
- #define DAC_TriangleAmplitude_127 ((uint32_t)0x00000600) //Maksymalna amplituda sygnalu trojkatnego 127
- #define DAC_TriangleAmplitude_255 ((uint32_t)0x00000700) //Maksymalna amplituda sygnalu trojkatnego 255
- #define DAC_TriangleAmplitude_511 ((uint32_t)0x00000800) //Maksymalna amplituda sygnalu trojkatnego 511
- #define DAC_TriangleAmplitude_1023 ((uint32_t)0x00000900) //Maksymalna amplituda sygnalu trojkatnego 1023
- #define DAC_TriangleAmplitude_2047 ((uint32_t)0x00000A00) //Maksymalna amplituda sygnalu trojkatnego 2047
- #define DAC_TriangleAmplitude_4095 ((uint32_t)0x00000B00) //Maksymalna amplituda sygnalu trojkatnego 4095
DAC_OutputBuffer - ten parametr pozwala na włączenie bądź wyłączenie bufora wyjściowego
- #define DAC_OutputBuffer_Enable ((uint32_t)0x00000000) //Wlaczony
- #define DAC_OutputBuffer_Disable ((uint32_t)0x00000002) //Wylaczony
Teraz czas przejść do programu głównego. Składa się on z trzech funkcji. Pierwsza inicjuje pracę pinów GPIO oraz DAC, druga ustawia wartość na pinach,
Włączenie pinów oraz DAC wygląda następująco:
- //Deklaracja nazw dla poszczególnych wejsc DAC
- typedef enum {
- DAC1,
- DAC2
- } DACxx;
- void DACInit(DACxx DACx)
- {
- DAC_InitTypeDef DACInit;
- GPIO_InitTypeDef GPIOInit;
- /* Select proper GPIO pin */
- if (DACx == DAC1)
- {
- GPIOInit.GPIO_Pin = GPIO_Pin_4;
- }
- else if(DACx == DAC2)
- {
- GPIOInit.GPIO_Pin = GPIO_Pin_5;
- }
- //Wlaczenie zegara dla portu GPIOA
- //RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
- RCC->AHB1ENR |= RCC_AHB1ENR_GPIOAEN;
- //Pozostale ustawienia wyprowadzen GPIO
- GPIOInit.GPIO_Mode = GPIO_Mode_AN;
- GPIOInit.GPIO_OType = GPIO_OType_PP;
- GPIOInit.GPIO_PuPd = GPIO_PuPd_NOPULL;
- GPIOInit.GPIO_Speed = GPIO_Speed_50MHz;
- GPIO_Init(GPIOA, &GPIOInit);
- //Wlaczenie zegara dla DAC
- //RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE);
- RCC->APB1ENR |= RCC_APB1ENR_DACEN;
- //Ustawienie opcji dla DAC
- DACInit.DAC_Trigger = DAC_Trigger_None;
- DACInit.DAC_WaveGeneration = DAC_WaveGeneration_None;
- DACInit.DAC_OutputBuffer = DAC_OutputBuffer_Enable;
- //Inicjacja i wlaczenie wybranego pinu DAC
- if (DACx == DAC1)
- {
- DAC_Init(DAC_Channel_1, &DACInit);
- //Wlaczenie kanalu pierwszego
- //DAC_Cmd(DAC_Channel_1, ENABLE);
- DAC->CR |= DAC_CR_EN1;
- }
- else
- {
- DAC_Init(DAC_Channel_2, &DACInit);
- //Wlaczenie kanalu drugiego
- //DAC_Cmd(DAC_Channel_2, ENABLE);
- DAC->CR |= DAC_CR_EN2;
- }
- }
Teraz funkcja wprowadzająca wartości do rejestru. Dane w formacie 12 bitowym wyrównane do prawej strony.
- //Ustawienie wartosci cyfrowej, która bedzie zamieniona na analogowa
- void DAC_Set(DACxx DACx, uint32_t digit_val)
- {
- //Wpisanie danych 12 bitowych wyrównanych do prawej
- //do odpowiedniego rejestru
- if (DACx == DAC1)
- {
- DAC->DHR12R1 = digit_val;
- }
- else if(DACx == DAC2)
- {
- DAC->DHR12R2 = digit_val;
- }
- }
Poniżej główna funkcja programu, wywołująca te przedstawione wcześniej z odpowiednimi parametrami.
Dane wyprowadzone oblicza się w następujący sposób:
Wartość wyjściowa [V] = (Wartość wprowadzana) / (Wartość maksymalna dla 12bit) * 3.3V
Wartość wprowadzana do przetwornika, maksymalnie 4096
Wartość maksymalna dla 12 bit = 2^12 = 4096
- int main(void)
- {
- //DAC1 PA4
- DACInit(DAC1);
- //DAC2 PA5
- DACInit(DAC2);
- //Wprowadzenie wartosci do DAC1
- DAC_Set(DAC1, 2000);
- //Wprowadzenie wartosci do DAC2
- DAC_Set(DAC2, 3000);
- while (1)
- {
- }
- }