STM32的DAC

STM32 的 DAC 模块(数字/模拟转换模块)是 12 位数字输入电压输出型的DACDAC 可以配置为 或 12 模式,也可以与 DMA 控制器配合使用。DAC工作在 12 位模式时,数据可以设置成左对齐或右对齐DAC 模块有 个输出通道,每个通道都有单独的转换器。在双DAC 模式下,个通道可以独立地进行转换,也可以同时进行转换并同步地更新 个通道的输出。DAC 可以通过引脚输入参考电压 VREF+以获得更精确的转换结果。

STM32 的 DAC 模块主要特点有:

①  个 DAC 转换器:每个转换器对应 个输出通道 

②  位或者 12 位单调输出 

③  12 位模式下数据左对齐或者右对齐 

④  同步更新功能 

⑤  噪声波形生成 

⑥  三角波形生成 

⑦  双 DAC 通道同时或者分别转换

⑧  每个通道都有 DMA 功能 

使用库函数的方法来设置 DAC 模块的通道 来输出模拟电压,其详细设置步骤如下:

1开启 PA 口时钟,设置 PA4 为模拟输入

STM32F103ZET6 的 DAC 通道 在 PA4 上,所以,我们先要使能 PORTA 的时钟,然后设置 PA4 为模拟输入DAC 本身是输出,但是为什么端口要设置为模拟输入模式呢?因为一但使能 DACx 通道之后,相应的 GPIO 引脚(PA4 或者 PA5)会自动与 DAC 的模拟输出相连,设置为输入,是为了避免额外的干扰

使能 GPIOA 时钟:

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE );    //使能 PORTA 时钟

设置 PA1 为模拟输入只需要设置初始化参数即可:

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;        //模拟输入

2使能 DAC1 时钟

同其他外设一样,要想使用,必须先开启相应的时钟。 STM32 的 DAC 模块时钟是由 APB1提供的,所以我们调用函数 RCC_APB1PeriphClockCmd()设置 DAC 模块的时钟使能。 

RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE );    //使能 DAC 通道时钟 

3初始化 DAC,设置 DAC 的工作模式

该部分设置全部通过 DAC_CR 设置实现,包括:DAC 通道 使能、DAC 通道 输出缓存关闭、不使用触发、不使用波形发生器等设置。这里 DMA  初始化是通过函数 DAC_Init 完成的:

void DAC_Init(uint32_t DAC_Channel, DAC_InitTypeDef* DAC_InitStruct) 

参数设置结构体类型 DAC_InitTypeDef 的定义:

typedef struct

{

uint32_t DAC_Trigger; //设置是否使用触发功能

uint32_t DAC_WaveGeneration; //设置是否使用波形发生

uint32_t DAC_LFSRUnmask_TriangleAmplitude; //设置屏蔽/幅值选择器,这个变量只在使用波形发生器的时候才有用

uint32_t DAC_OutputBuffer;  //设置输出缓存控制位

}DAC_InitTypeDef;

实例代码:

DAC_InitTypeDef DAC_InitType;

DAC_InitType.DAC_Trigger = DAC_Trigger_None;  //不使用触发功能  TEN1=0

DAC_InitType.DAC_WaveGeneration = DAC_WaveGeneration_None;//不使用波形发生

DAC_InitType.DAC_LFSRUnmask_TriangleAmplitude = DAC_LFSRUnmask_Bit0;

DAC_InitType.DAC_OutputBuffer = DAC_OutputBuffer_Disable ;  //DAC1 输出缓存关闭 

DAC_Init(DAC_Channel_1,&DAC_InitType);    //初始化 DAC 通道 1

4使能 DAC 转换通道

初始化 DAC 之后,理所当然要使能 DAC 转换通道,库函数方法是:

DAC_Cmd(DAC_Channel_1, ENABLE);   //使能 DAC1

5设置 DAC 的输出值

通过前面 个步骤的设置,DAC 就可以开始工作了,我们使用 12 位右对齐数据格式,所以我们通过设置 DHR12R1,就可以在 DAC 输出引脚(PA4)得到不同的电压值了。库函数的函数是:

DAC_SetChannel1Data(DAC_Align_12b_R, 0);

第一个参数设置对齐方式,可以为 12 位右对齐 DAC_Align_12b_R12 位左对齐DAC_Align_12b_L 以及 位右对齐 DAC_Align_8b_R 方式。第二个参数就是 DAC 的输入值了,这个很好理解,初始化设置为 0

这里,还可以读出 DAC 的数值,函数是:

DAC_GetDataOutputValue(DAC_Channel_1);

点击(此处)折叠或打开

  1. //DAC通道1输出初始化
  2. void Dac1_Init(void)
  3. {
  4.     GPIO_InitTypeDef GPIO_InitStructure;
  5.     DAC_InitTypeDef DAC_InitType;
  6.     
  7.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE );     //使能PORTA通道时钟
  8.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE );     //使能DAC通道时钟
  9.     
  10.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;                 // 端口配置
  11.      GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;          //模拟输入
  12.      GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  13.      GPIO_Init(GPIOA, &GPIO_InitStructure);
  14.     GPIO_SetBits(GPIOA,GPIO_Pin_4)    ;//PA.4 输出高
  15.     
  16.     DAC_InitType.DAC_Trigger=DAC_Trigger_None;    //不使用触发功能 TEN1=0
  17.     DAC_InitType.DAC_WaveGeneration=DAC_WaveGeneration_None;//不使用波形发生
  18.     DAC_InitType.DAC_LFSRUnmask_TriangleAmplitude=DAC_LFSRUnmask_Bit0;//屏蔽、幅值设置
  19.     DAC_InitType.DAC_OutputBuffer=DAC_OutputBuffer_Disable ;    //DAC1输出缓存关闭 BOFF1=1
  20.     DAC_Init(DAC_Channel_1,&DAC_InitType);     //初始化DAC通道1
  21.     
  22.     DAC_Cmd(DAC_Channel_1, ENABLE); //使能DAC1
  23.     DAC_SetChannel1Data(DAC_Align_12b_R, 0); //12位右对齐数据格式设置DAC值
  24. }


  25. //设置通道1输出电压
  26. //vol:0~3300,代表0~3.3V
  27. void Dac1_Set_Vol(u16 vol)
  28. {
  29.     float temp=vol;
  30.     temp/=1000;
  31.     temp=temp*4096/3.3;
  32.     DAC_SetChannel1Data(DAC_Align_12b_R,temp);//12位右对齐数据格式设置DAC值
  33. }
### STM32 DAC 使用教程及示例代码 #### 一、STM32 DAC 模块概述 STM32系列微控制器中的DAC模块具备多种特性,能够满足不同应用场景的需求。该模块拥有两个独立的DAC转换器,支持8位或12位分辨率的数据输出,并提供数据对齐方式的选择——左对齐或是右对齐[^3]。 #### 二、硬件连接说明 对于具体的型号如STM32L151xxxx和STM32L152xxxx而言,它们均内置了DAC组件。当涉及到外部设备接口时,则需注意特定引脚定义;例如,在某些情况下,DAC_OUT1被映射到PA4而DAC_OUT2则位于PA5上[^1]。 #### 三、软件编程指南 为了实现对外部模拟信号的有效控制,下面给出一段用于设置并启动单通道DAC输出的基础C语言程序片段: ```c #include "stm32f1xx_hal.h" void DAC_Config(void){ __HAL_RCC_DAC_CLK_ENABLE(); //使能DAC时钟 DAC_ChannelConfTypeDef sConfig = {0}; hDac.Instance=DAC; hDac.Init.Trigger=DAC_TRIGGER_NONE;//不使用触发源 hDac.Init.Mode=DAC_MODE_NORMAL; //正常工作模式 HAL_DAC_Init(&hDac); sConfig.DAC_Trigger = DAC_TRIGGER_NONE; sConfig.DAC_OutputBuffer = DAC_OUTPUTBUFFER_DISABLE; HAL_DAC_ConfigChannel(&hDac, &sConfig,DAC_CHANNEL_1); } int main(){ /* 初始化 */ HAL_Init(); SystemClock_Config(); DAC_Config(); while (1){ uint32_t value = 2047 ;// 中间电压值 HAL_DAC_SetValue(&hDac, DAC_CHANNEL_1 , DAC_ALIGN_12B_R,value); HAL_Delay(1000); } } ``` 上述代码展示了如何初始化DAC外设以及设定其参数来完成一次简单的数值写入操作。这里采用的是轮询的方式来进行持续性的电平调整演示。 另外还有一份针对TLV5618芯片通过SPI协议进行通信的例子,此部分适用于更复杂的多路DAC应用场合: ```c void TLV5618_Write_B(uint16_t dac_value) { uint16_t data = (0x4000) | (dac_value & 0x0FFF); DAC_CS_L; delay_ms(1); DAC_SCK_H; for(int i = 15; i >= 0; i--) { if(data & (1<<i)){//如果是'1' DAC_SO_H; }else{//如果是'0' DAC_SO_L; } DAC_SCK_L; delay_ms(1); DAC_SCK_H; } DAC_SCK_L; delay_ms(1); DAC_CS_H; } ``` 这段函数实现了向指定地址发送命令字节序列的功能,从而可以精确地改变目标器件上的输出状态[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值