一、前言
最近项目上需要使用8路DAC,本来想用STM32自带的DAC,但是STM32自带的DAC好像只有2路,不满足要求,市面上很多的DAC芯片虽然有8路,但是精度只有8位,而我需要一个精度更高的,在选型的时候选到了ADS9226这款芯片,它有8个DAC通道,精度是12位的,当时也只关心这2个指标。
等到使用的时候才发现这个芯片的功能比想象中的更强大,它的每个通道都可以配置为:ADC、DAC、GPIO输入、GPIO输出、85kΩ下拉、开漏、三态,可以配置的功能非常多,然鹅让我没想到的是网上竟然找不到现成的程序可以参考,于是我自己写了一个STM32的测试程序。
手册去ADI的官网可以找到,中文的和英文的都有,但是要注意的是中文手册有点问题,目前发现有些寄存器的地址号写的不对,但是英文手册写的没问题。
我因为只使用DAC功能,因此我只测试了DAC功能,ad5592r.c里面有一个DAC的例程,经过测试是没问题的。下面我附上ad5592r.c和ad5592r.h,这两个文件配合正点原子的库函数版本使用,拿SPI的例程改改就能用。但是SPI初始化函数需要改改,一个是SPI的数据为16位的,一个是第一个边沿采样。代码如下,需要注意的地方我给标红了。
1 void SPI2_Init(void) 2 { 3 GPIO_InitTypeDef GPIO_InitStructure; 4 SPI_InitTypeDef SPI_InitStructure; 5 6 RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOB, ENABLE );//PORTB时钟使能 7 RCC_APB1PeriphClockCmd( RCC_APB1Periph_SPI2, ENABLE );//SPI2时钟使能 8 9 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_15; 10 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //PB13/14/15复用推挽输出 11 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; 12 GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIOB 13 14 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14; 15 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //PB13/14/15复用推挽输出 16 GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIOB 17 18 19 GPIO_SetBits(GPIOB,GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15); //PB13/14/15上拉 20 21 SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工 22 SPI_InitStructure.SPI_Mode = SPI_Mode_Master; //设置SPI工作模式:设置为主SPI 23 SPI_InitStructure.SPI_DataSize = SPI_DataSize_16b; //设置SPI的数据大小:SPI发送接收16位帧结构 24 SPI_InitStructure.SPI_CPOL = SPI_CPOL_High; //串行同步时钟的空闲状态为高电平 25 SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge; //串行同步时钟的第二个跳变沿(上升或下降)数据被采样 26 SPI_InitStructure.SPI_NSS = SPI_NSS_Soft; //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制 27 SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_32; //定义波特率预分频的值:波特率预分频值为256 28 SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始 29 SPI_InitStructure.SPI_CRCPolynomial = 7; //CRC值计算的多项式 30 SPI_Init(SPI2, &SPI_InitStructure); //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器 31 32 SPI_Cmd(SPI2, ENABLE); //使能SPI外设 33 34 //SPI2_ReadWriteByte(0xff);//启动传输 35 }
ad5592r.c
1 #include "ad5592r.h" 2 #include "spi.h" 3 #include "delay.h" 4 #include "usart.h" 5 6 /** 7 * /brief 初始化AD5592R 8 */ 9 void AD5592R_Init(void) 10 { 11 GPIO_InitTypeDef GPIO_InitStructure; 12 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE );//PORTB时钟使能 13 14 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12; // PB12 推挽 15 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出 16 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; 17 GPIO_SetBits(GPIOB,GPIO_Pin_12); 18 GPIO_Init(GPIOB, &GPIO_InitStructure); 19 20 //PB11与AD5592R RESET引脚连接 21 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11; // PB11 推挽 22 GPIO_SetBits(GPIOB, GPIO_Pin_11); 23 GPIO_Init(GPIOB, &GPIO_InitStructure); 24 25 SPI2_Init(); //初始化SPI 26 SPI2_SetSpeed(SPI_BaudRatePrescaler_32); //设置为256分频 27 28 AD5592R_HW_Reset(); 29 } 30 31 /** 32 * \brief AD5592R硬件复位 33 */ 34 void AD5592R_HW_Reset(void) 35 { 36 GPIO_ResetBits(GPIOB, GPIO_Pin_11); 37 delay_us(250); //手册上说复位功能最多需要250us 38 GPIO_SetBits(GPIOB, GPIO_Pin_11); 39 } 40 41 /** 42 * \brief 读取AD5592R的寄存器 43 * 44 * \param[in] addr:寄存器地址 45 * 46 * \note D1和D0用来选择LDAC模式,每当调用一次该函数时,都会将模式设定为DAC即时更 47 * 新模式,需要修改的话,可以用一个变量保存。 48 */ 49 u16 AD5592R_Read_Register(ad5592r_addr_t addr) 50 { 51 u16 data = 0; 52 53 data = (RDBK_LDAC_MODE << 11) | RDBK_LDAC_MODE_EN | (addr << 2); //选择寄存器,使能回读并设定回读寄存器地址 54 55 AD5592R_CS=0; //使能器件 56 SPI2_ReadWriteByte(data); //发送读取状态寄存器命令 57 AD5592R_CS=1; 58 59 delay_us(1); //片选信号拉低到拉高之间的时间间隔最小为20nS 60 61 AD5592R_CS=0; 62 data = SPI2_ReadWriteByte(0X0000); //读取一个字节 63 //delay_us(1); //如果上一行发送的数据是0xFFFF,那么片选信号很快就会拉高,但是发其他数据却不会,原因不明,因此这里加一个延时。 64 AD5592R_CS=1; //取消片选 65 66 return data; 67 } 68 69 70 /** 71 * \brief 读取AD5592R的寄存器 72 * 73 * \note 这个函数主要是用来测试,读出来15个寄存器的值。芯片复位后读取,其中data[6]为0x00FF,data[7]为0x005C,其他寄存器的值都为0,如果读出的结果都为0xFFFF,那可能通信出现问题了 74 */ 75 void AD5592R_Read_All_Register(u16 data[16]) 76 { 77 u8 i; 78 79 for (i = 0; i < 16; i++) { 80 data[i] = AD5592R_Read_Register((ad5592r_addr_t)i); 81 } 82 } 83 84 /** 85 * \brief 通用控制寄存器配置 86 * 87 * \param[in] flag:可以加入如下标志: 88 * #GP_COTL_ADC_BUF_PRECHARGE: ADC缓冲器用于对ADC预充电,不设置则禁用(默认) 89 * #GP_COTL_ADC_BUF_EN: 使能ADC缓冲器,不设置则禁能(默认) 90 * #GP_COTL_LOCK: 不能改变I/Ox引脚配置寄存器的内容,不设置则可以改变(默认) 91 * #GP_COTL_ALL_DAC: (一个数据)写入所有DAC,不设置则由DAC地址决定写入哪个DAC(默认) 92 * #GP_COTL_ADC_RANGE_REFx2: ADC增益为0 V至2 × VREF,不设置则为1倍(默认) 93 * #GP_COTL_DAC_RANGE_REFx2: DAC增益为0 V至2 × VREF,不设置则为1倍(默认) 94 */ 95 void AD5592R_GP_Contol_Set(u16 flag) 96 { 97 u16 data = 0; 98 99 data = (GP_CNTL_REG << 11) | flag; 100 AD5592R_CS = 0; 101 SPI2_ReadWriteByte(data); 102 AD5592R_CS = 1; 103 } 104 105 /** 106 * \brief 将指定的引脚配置为指定的功能 107 * 108 * \param[in] func:设定的功能#ADC_PIN_CONF:配置为ADC引脚 109 * #DAC_PIN_CONF:配置为DAC引脚 110 * #PD_CONF:下拉配置,设定为85kΩ下拉(默认就是这个) 111 * #GPIO_W_CONF:GPIO写配置,意思就是将引脚配置为GPIO输出 112 * #GPIO_R_CONF:GPIO读配置,意思就是将引脚配置为GPIO输入 113 * #GPIO_OD_CONF:GPIO开漏配置,意思就是将引脚配置为开漏输出,否则为推挽输出(默认) 114 * #GPIO_3_STAT_CONF:三态配置 115 * \param[in] pin:选择需要配置的引脚,取值为AD5592R_PIN_0~AD5592R_PIN_7,可以按 116 * 位或。 117 * 118 * 119 * \param[in] flag:根据func的值,可以传入一些特殊标志,具体如下: 120 * 1、如果写入的func值为 GPIO_W_CONF,则传入的flag可以加入 121 * AD5592R_PIN_7_BUSY,表示将PIN7设定为BUSY引脚。 122 * 2、如果写入的func值为GPIO_R_CONF,则传入的flag可以加入 123 * GPIO_R_CONF_EN_RDBK,表示下一个SPI操作输出GPIO引脚的状态, 124 * 并将状态通过返回值返回。 125 * 126 * 127 */ 128 u8 AD5592R_Pin_Config(ad5592r_addr_t func, u8 pin, u16 flag) 129 { 130 u16 data = 0; 131 132 //将PIN_7设定为BUSY引脚,如果传入AD5592R_PIN_7_BUSY,那么D7也必须为1 133 if ((func == GPIO_W_CONF) && (flag & AD5592R_PIN_7_BUSY)) { 134 data |= AD5592R_PIN_7; 135 } 136 137 if ((func == GPIO_W_CONF) && (flag & GPIO_R_CONF_EN_RDBK)) { 138 data |= GPIO_R_CONF_EN_RDBK; 139 } 140 141 data |= (func << 11) | pin; 142 143 AD5592R_CS = 0; 144 SPI2_ReadWriteByte(data); 145 AD5592R_CS = 1; 146 //如果使能GPIO回读,那么下一个SPI操作返回GPIO引脚的状态 147 if ((func == GPIO_W_CONF) && (flag & GPIO_R_CONF_EN_RDBK)) { 148 delay_us(1); 149 AD5592R_CS = 0; 150 data = SPI2_ReadWriteByte(0); 151 AD5592R_CS = 1; 152 return data; 153 } 154 155 return 0; 156 } 157 158 /** 159 * \brief 给8个GPIO写入数据 160 * 161 * \param[in] value:要写入的值 162 */ 163 void AD5592R_GPIO_Write(u8 value) 164 { 165 u16 data = (GPIO_W_DATA << 11) | value; 166 167 AD5592R_CS = 0; 168 SPI2_ReadWriteByte(data); 169 AD5592R_CS = 1; 170 } 171 172 /** 173 * \brief 给DAC通道写入数据 174 * 175 * \param[in] channel:0~7有效,指定DAC通道 176 * \param[in] value:要写入的值,低12位有效 177 */ 178 void AD5592R_DAC_Write_Data(u8 channel, u16 value) 179 { 180 u16 data = 0; 181 182 channel &= 0x07; //只取低3位 183 value &= 0x0FFF; //只取低12位 184 185 data = (1 <<15) | (channel << 12) | value; //最高位要置1,表示写数据 186 187 AD5592R_CS = 0; 188 SPI2_ReadWriteByte(data); 189 AD5592R_CS = 1; 190 } 191 192 /** 193 * \brief 配置DAC通道回读 194 * 195 * \param[in] channel:0~7有效,指定DAC通道 196 * \param[in] value:要写入的值,低12位有效 197 * 198 * \note 指的是当某个DAC通道使能以后,每次使能回读之后的下一个时钟周期读出数据 199 * 并不是每次写DAC数据后的下一个时钟周期读出数据 200 */ 201 u16 AD5592R_DAC_Readback(u8 channel) 202 { 203 u16 data = 0; 204 205 channel &= 0x07; //只取低3位 206 data = (DAC_READBACK << 11) | (0x3 << 3) | channel; 207 AD5592R_CS = 0; 208 SPI2_ReadWriteByte(data); 209 AD5592R_CS = 1; 210 delay_us(1); 211 AD5592R_CS = 0; 212 data = SPI2_ReadWriteByte(0); 213 AD5592R_CS = 0; 214 215 data &= 0xFFF; //取低12位,事实上读取成功后,最高位是1,以表示这是一个数据 216 return data; 217 } 218 219 /** 220 * \brief 设定关断DAC和内部基准电压源 221 * 222 * \param[in] value:#true:关断基准电压源、 DAC和ADC 223 * #false:基准电压源和DAC关断状态由D9和D7至D0决定(默认)。 224 */ 225 void AD5592R_PD_ALL_Set(u8 value) 226 { 227 u16 data = 0; 228 229 data = AD5592R_Read_Register(POW_DOWN_REF_CNTL); //先读出寄存器的值 230 231 data |= (POW_DOWN_REF_CNTL << 11); 232 233 if (value == false) { 234 data &= ~POW_DOWN_REF_CNTL_PD_ALL; 235 } else { 236 data |= POW_DOWN_REF_CNTL_PD_ALL; 237 } 238 239 AD5592R_CS = 0; 240 SPI2_ReadWriteByte(data); 241 AD5592R_CS = 1; 242 } 243 244 /** 245 * \brief 选择基准电压源 246 * 247 * \param[in] value: #REF_INTERNAL:使用内部基准电压源 248 * #REF_EXTERNAL:使用外部基准电压源 249 */ 250 void AD5592R_Ref_Set(ref_select_t value) 251 { 252 u16 data = 0; 253 254 data = AD5592R_Read_Register(POW_DOWN_REF_CNTL); //先读出寄存器的值 255 256 data |= (POW_DOWN_REF_CNTL << 11); 257 258 if (value == REF_INTERNAL) { 259 data &= ~POW_DOWN_REF_CNTL_EN_REF; 260 } else { 261 data |= POW_DOWN_REF_CNTL_EN_REF; 262 } 263 264 AD5592R_CS = 0; 265 SPI2_ReadWriteByte(data); 266 AD5592R_CS = 1; 267 } 268 269 /** 270 * \brief 选择关断DAC 271 * 272 * \param[in] pin:选择需要配置的引脚,取值为AD5592R_PIN_0~AD5592R_PIN_7,可以按 273 * 位或。 274 * \param[in] mode: #MORMAL_MODE:选择的引脚设置为正常模式 275 * #PD_MODE:引脚若配置为DAC则关断 276 */ 277 void AD5592R_PD_DAC(u8 pin, pd_mode_t mode) 278 { 279 u16 data = 0; 280 281 data = AD5592R_Read_Register(POW_DOWN_REF_CNTL); //先读出寄存器的值 282 data |= (POW_DOWN_REF_CNTL << 11); 283 if (mode == MORMAL_MODE) { 284 data &= ~pin; 285 } else { 286 data |= pin; 287 } 288 AD5592R_CS = 0; 289 SPI2_ReadWriteByte(data); 290 AD5592R_CS = 1; 291 } 292 293 /** 294 * \brief ADC序列寄存器设置 295 * 296 * \prarm[in] channel:0~7有效,指定需要加入转换序列的ADC通道 297 * \prarm[in] flag:可以加入如下标志: 298 * #ADC_SEQ_REP:使能序列重复,不设置则禁用序列重复(默认) 299 * #ADC_SEQ_TEMP:使能温度指示器回读,不设置则禁用温度指示器回读(默认) 300 */ 301 void AD5592R_ADC_Sequence_set(u8 channel, u16 flag) 302 { 303 u16 data = 0; 304 305 channel &= 0x07; 306 307 data = (ADC_SEQ_REG << 11) | flag | channel; 308 AD5592R_CS = 0; 309 SPI2_ReadWriteByte(data); 310 AD5592R_CS = 1; 311 } 312 313 314 /** 315 * \brief AD5592R软件复位 316 */ 317 void AD5592R_Soft_Reset(void) 318 { 319 AD5592R_CS = 0; 320 SPI2_ReadWriteByte(0x7DAC); //SPI写入0x7DAC表示软件复位 321 AD5592R_CS = 1; 322 } 323 324 325 #if 0 326 int main(void) 327 { 328 u8 i = 0; 329 u16 data[16] = { 0 }; 330 u16 value = 0; 331 332 delay_init(); //延时函数初始化 333 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级 334 uart_init(115200); //串口初始化为115200 335 printf("Hello World!\r\n"); 336 LED_Init(); //初始化与LED连接的硬件接口 337 338 AD5592R_Init(); 339 //AD5592R_Read_All_Register(data); 340 //for (i = 0; i < 16; i++) { 341 // printf("data[%d] = 0x%04x\r\n", i, data[i]); 342 //} 343 344 AD5592R_Ref_Set(REF_EXTERNAL); //DAC使用外部基准源 345 AD5592R_Pin_Config(DAC_PIN_CONF, AD5592R_PIN_ALL, 0); //将所有引脚配置为DAC 346 AD5592R_DAC_Write_Data(0, 0x888); 347 value = AD5592R_DAC_Readback(0); 348 printf("value = 0x%04x\r\n", value); 349 350 while(1) 351 { 352 delay_ms(500); 353 delay_ms(500); 354 LED0=!LED0;//DS0闪烁 355 } 356 } 357 #endif
ad5592r.h
1 /** 2 * \brief AD5592R的寄存器和数据都是16位的 3 * SYNC引脚在16位数据传输开始之前拉低,16位数据传输完成之后拉高 4 * 高位先传 5 * 空闲时时钟为高电平 6 * 写入DAC时,数据能够以最高50 MHz的速率传输。执行转换或从AD5592R传输数据时, SCLK的最大速度为20 MHz。 7 * 对于AD5592R来说,在时钟的下降沿读取数据,时钟上升沿发送数据,也就是说 8 * 主机应该配置为下降沿发送,上升沿读取 9 * 片选信号拉低到拉高之间的时间间隔最小为30nS 10 */ 11 12 #ifndef __AD5592R_H 13 #define __AD5592R_H 14 #include "sys.h" 15 16 /** 17 * \brief STM32引脚定义 18 * @{ 19 */ 20 #define AD5592R_CS PBout(12) //AD5592R的片选信号 21 /** 22 * @} 23 */ 24 25 /** 26 * \brief AD5592R引脚定义 27 * @{ 28 */ 29 #define AD5592R_PIN_0 (1 << 0) 30 #define AD5592R_PIN_1 (1 << 1) 31 #define AD5592R_PIN_2 (1 << 2) 32 #define AD5592R_PIN_3 (1 << 3) 33 #define AD5592R_PIN_4 (1 << 4) 34 #define AD5592R_PIN_5 (1 << 5) 35 #define AD5592R_PIN_6 (1 << 6) 36 #define AD5592R_PIN_7 (1 << 7) 37 #define AD5592R_PIN_ALL 0xFF 38 39 #define AD5592R_PIN_7_BUSY (1 << 8) 40 #define GPIO_R_CONF_EN_RDBK (1 << 10) 41 /** 42 * @} 43 */ 44 45 #ifndef true 46 #define true 1 47 #endif 48 49 #ifndef false 50 #define false 0 51 #endif 52 53 typedef enum ref_select { 54 REF_INTERNAL = 0, /* 使用内部基准电压源 */ 55 REF_EXTERNAL = 1, /* 使用外部基准电压源 */ 56 } ref_select_t; 57 58 typedef enum pd_mode { 59 MORMAL_MODE = 0, /* 选择的引脚设置为正常模式 */ 60 PD_MODE = 1, /* 通道若配置为DAC则关断 */ 61 } pd_mode_t; 62 63 /** 64 * \brief 定义AD5592R的寄存器地址 65 */ 66 typedef enum ad5592r_addr { 67 NOP = 0, //无操作 68 DAC_READBACK = 1, //DAC回读寄存器 69 ADC_SEQ_REG = 2, //ADC序列寄存器 70 GP_CNTL_REG = 3, //通用控制寄存器 71 ADC_PIN_CONF = 4, //ADC引脚配置 72 DAC_PIN_CONF = 5, //DAC引脚配置 73 PULL_DOWN_CONF = 6, //下拉配置 74 RDBK_LDAC_MODE = 7, //回读和LDAC模式 75 GPIO_W_CONF = 8, //GPIO写配置 76 GPIO_W_DATA = 9, //GPIO写数据 77 GPIO_R_CONF = 10, //GPIO读配置 78 POW_DOWN_REF_CNTL = 11, //断电/基准控制 79 GPIO_OD_CONF = 12, //GPIO开漏配置 80 GPIO_3_STAT_CONF = 13, //三态配置 81 RESERV = 14, //保留 82 SOFT_REST = 15, //软件复位 83 } ad5592r_addr_t; 84 85 /** 86 * \brief 寄存器定义 87 * @{ 88 */ 89 90 /** 91 * \brief 输入移位寄存器 92 * +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+ 93 * |D15|D14|D13|D12|D11|D10|D9 |D8 |D7 |D6 |D5 |D4 |D3 |D2 |D1 |D0 | 94 * +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+ 95 * | 0 | 控制寄存器地址| 0 | 0 | 控制寄存器数据 | 96 * +---+-----------+---+---+---+-----------------------------------+ 97 * | 1 | DAC地址 | 12位DAC数据 | 98 * +---+-----------+-----------------------------------------------+ 99 * 控制寄存器应该写入的数据格式见各个具体寄存器的描述 100 * @{ 101 */ 102 #define CNTL_BIT (0 << 15) //表示控制寄存器 103 #define DATA_BIT (1 << 15) //表示数据寄存器 104 //当D15为1时,[D14:D12]对应DAC通道,[D11:D0]表示DAC数据 105 106 /** 107 * \brief 以下定义为控制地址定义,当B15为0时有效 108 * @{ 109 */ 110 #define CNTL_ADDR_NOP (0 << 11) //无操作 111 112 /** 113 * \brief DAC回读寄存器,[D15:D5]固定为 0000 1000 000,[D4:D3]为11表示使能回读,为00表示禁能回读(默认),[D2:D0]选择DAC通道 114 * 当使能了DAC回读,那么给DAC通道写入数据并且SYNC引脚拉高之后,将SYNC拉低并发送16个时钟就能从SDO引脚读到回读的值。 115 */ 116 #define CNTL_ADDR_DAC_READBACK (1 << 11) //DAC回读 117 118 /** 119 * \brief ADC序列寄存器,[D15:D10]固定为 0001 00,[D7:D0]某一位为1则该ADC通道在转换序列中 120 */ 121 //ADC转换结果,D15为0,[D14:D12]指定ADC通道,[D11:D0]表示转换后的结果 122 #define CNTL_ADDR_ADC_SEQ_REG (2 << 11) //ADC序列寄存器,选择ADC(s)进行转换 123 #define ADC_SEQ_REP (1 << 9) //#0:ADC序列重复禁能(默认)。#1:ADC序列重复使能 124 #define ADC_SEQ_TEMP (1 << 10) //#0:禁用温度指示器回读(默认)。#1:启用温度指示器回读 125 126 #define CNTL_ADDR_GP_CNTL_REG (3 << 11) //通用控制寄存器,ADC和DAC控制寄存器 127 128 /** 129 * \brief ADC引脚配置寄存器,[D15:D8]固定为 0010 0000,[D7:D0]哪一位为1则设定哪个引脚为ADC引脚,如果设置为0(默认), 130 * 则输入输出功能由引脚配置寄存器决定 131 */ 132 #define CNTL_ADDR_ADC_PIN_CONF (4 << 11) //ADC引脚配置,选择哪些引脚是ADC输入 133 134 /** 135 * \brief DAC引脚配置寄存器,[D15:D8]固定为0010 1000,[D7:D0]哪一位为1则设定哪个引脚为DAC引脚,如果设置为0(默认), 136 * 则输入输出功能由引脚配置寄存器决定 137 */ 138 #define CNTL_ADDR_DAC_PIN_CONF (5 << 11) //DAC引脚配置,选择哪些引脚是DAC输出 139 140 /** 141 * \brief 下拉配置寄存器,[D15:D8]固定为0011 0000,[D7:D0]:#0:输入/输出功能由引脚配置寄存器决定。#1:该引脚为85kΩ下拉 142 */ 143 #define CNTL_ADDR_PD_CONF (6 << 11) //下拉配置,选择哪些引脚具有85kΩ下拉电阻到GND 144 145 /** 146 * \brief 回读和LDAC模式寄存器,[D15:D7]固定为0011 1000 0, 147 * [D5:D2]指定要回读的寄存器地址,读取寄存器的值通过该指令 148 * [D1:D0]确定如何处理写入DAC输入寄存器的数据: 149 * #00:写入输入寄存器的数据立即复制到DAC寄存器,DAC输出更新(默认) 150 * #01:写入输入寄存器的数据不会复制到DAC寄存器,DAC输出不更新 151 * #10:写入输入寄存器的数据复制到相应的DAC寄存器,数据传输后DAC输出同时更新 152 */ 153 #define CNTL_ADDR_RDBK_LDAC_MODE (7 << 11) //回读和LDAC模式,选择加载DAC(LDAC)功能的操作和/或回读哪个配置寄存器 154 #define RDBK_LDAC_MODE_EN (1 << 6) //#0:未启用回读。#1:[D5:D2]选择要回读的寄存器,读取结束后D6自动清除 155 #define RDBK_LDAC_MODE_IMMDEDIATELY (0 << 0) //立即更新 156 #define RDBK_LDAC_MODE_NOT (1 << 0) //不更新 157 #define RDBK_LDAC_MODE_AFTER (2 << 0) //数据传输后更新 158 159 /** 160 * \brief GPIO写配置寄存器,[D15:D9]固定为0100 000,D8描述见下,[D7:D0]某位为1表示该引脚为通用输出引脚,否则该引脚功能由引脚配置寄存器决定(默认) 161 */ 162 #define CNTL_ADDR_GPIO_W_CONF (8 << 11) //GPIO写配置,选择哪些引脚是GPIO输出 163 #define GPIO_W_CONF_IO7_BUSY (1 << 8) //#0:表示I/O7未配置为BUSY功能。#1:I/O7引脚配置为BUSY功能,D7也必须设置为1 164 165 /** 166 * \brief GPIO写数据寄存器,[D15:D8]固定为0100 1000,[D7:D0]确定引脚输出电平 167 */ 168 #define CNTL_ADDR_GPIO_W_DATA (9 << 11) //向GPIO写数据 169 170 171 /** 172 * \brief GPIO读配置寄存器,[D15:D11]固定为0101 0,[D9:D8]:固定为0。[D7:D0]某位为0则表示该引脚输入/输出功能由配置寄存器决定,为1表示这是一个通用的输入引脚 173 */ 174 #define CNTL_ADDR_GPIO_R_CONF (10 << 11) //GPIO读配置,选择哪些引脚是GPIO输入 175 #define GPIO_R_CONF_EN_RDBK (1 << 10) //#0:[D7:D0]确定哪些引脚为通用输入,#1:写一个SPI操作会计时(clocks)GPIO引脚的状态 176 177 /** 178 * \brief 断电/参考控制寄存器,[D15:D11]固定为0101 1,D8固定为0,[D7:D0]某位为0则表示通道处于正常工作模式(默认),为1时如果通道配置为DAC,则该通道断电 179 */ 180 #define CNTL_ADDR_POW_DOWN_REF_CNTL (11 << 11) //断电/参考控制,关闭DAC并启用/禁用参考 181 #define POW_DOWN_REF_CNTL_PD_ALL (1 << 10) //#0:参考和DAC断电状态由D9及[D7:D0]确定(默认)。#1:基准、DAC和ADC断电 182 #define POW_DOWN_REF_CNTL_EN_REF (1 << 9) //#0:参考和buffer断电,如果使用外部参考,则设置该位。#1:基准及buffer通电,参考源为VREF引脚 183 184 /** 185 * \brief GPIO开漏配置寄存器,[D15:D8]固定为0110 0000,[D7:D0]某位为1表示该引脚为开漏输出引脚,引脚也必须设置为数字输出引脚。为0则表示改引脚设置为上/下拉 186 */ 187 #define CNTL_ADDR_GPIO_OD_CONF (12 << 11) //GPIO开漏配置,为GPIO选择开漏或上/下拉 188 189 /** 190 * \brief 三态配置寄存器,[D15:D8]固定为0110 1000,[D7:D0]:#0:输入/输出功能由引脚配置寄存器决定。#1:该引脚为三态输出 191 */ 192 #define CNTL_ADDR_GPIO_3_STAT_CONF (13 << 11) //三态配置,选择哪些引脚是三态 193 #define CNTL_ADDR_RESERV (14 << 11) //保留 194 195 /** 196 * \brief 软件复位写入序列:0111 1101 1010 1100,即0x7DAC 197 */ 198 #define CNTL_ADDR_SOFT_REST (15 << 11) //软件复位 199 /** 200 * @} 201 */ 202 203 /** 204 * @} 205 */ 206 207 208 /** 209 * \brief IO引脚配置寄存器 210 * +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+ 211 * |D15|D14|D13|D12|D11|D10|D9 |D8 |D7 |D6 |D5 |D4 |D3 |D2 |D1 |D0 | 212 * +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+ 213 * | 0 | 寄存器地址 | 保留 |IO7|IO6|IO5|IO4|IO3|IO2|IO1|IO0| 214 * +---+---------------+-----------+---+---+---+---+---+---+---+---+ 215 * @{ 216 */ 217 218 /** 219 * \brief IO引脚配置寄存器地址,[D14:D11]选择功能,[D7:D0]选择是否使能该功能 220 * @{ 221 */ 222 #define IO_CONF_ADDR_ADC_PIN (4 << 11) //配置为ADC引脚 223 #define IO_CONF_ADDR_DAC_PIN (5 << 11) //配置为DAC引脚 224 #define IO_CONF_ADDR_PULL_DOWN (6 << 11) //配置为下拉(默认配置为上拉) 225 #define IO_CONF_ADDR_GPIO_W_CONF (8 << 11) //配置为GPIO写 226 #define IO_CONF_ADDR_GPIO_R_CONF (10 << 11) //配置为GPIO读 227 #define IO_CONF_ADDR_GPIO_OD_CONF (12 << 11) //配置为GPIO开漏 228 #define IO_CONF_ADDR_3_STAT_CONF (13 << 11) //配置为三态 229 /** 230 * @} 231 */ 232 233 /** 234 * @} 235 */ 236 237 /** 238 * \brief 通用控置寄存器,[D15:D10] 固定为0 0011 0,[D3:D0]固定为0000 239 * @{ 240 */ 241 #define GP_COTL_ADC_BUF_PRECHARGE (1 << 9) //#0:ADC缓冲器不用于对ADC预充电,如果ADC缓冲区已经启用,则始终通电(默认) 242 //#1:ADC缓冲器用于对ADC预充电,如果ADC缓冲区已经启用,则在转换发生时将其通电,然后断电,直到下一次转换发生 243 #define GP_COTL_ADC_BUF_EN (1 << 8) //#0:ADC缓冲器禁能(默认),#1:ADC缓冲器使能 244 #define GP_COTL_LOCK (1 << 7) //#0:可以更改输入/输出引脚配置寄存器的内容(默认),#1:不能更改…… 245 #define GP_COTL_ALL_DAC (1 << 6) //#0:对于未来的DAC写入,DAC地址位确定写入哪个DAC(默认) 246 //#1:对于未来的DAC写入,DAC地址位将被忽略,配置为DAC的所有通道将使用相同的数据进行更新 247 #define GP_COTL_ADC_RANGE_REFx2 (1 << 5) //#0:ADC增益为0~Vref(默认)。#1:ADC增益为0~2Vref 248 #define GP_COTL_DAC_RANGE_REFx2 (1 << 4) //#0:ADC增益为0~Vref(默认)。#1:DAC增益为0~2Vref 249 /** 250 * @} 251 */ 252 253 254 /** 255 * \brief 函数声明 256 * @{ 257 */ 258 /** 259 * /brief 初始化AD5592R 260 */ 261 extern void AD5592R_Init(void); 262 263 /** 264 * \brief AD5592R硬件复位 265 */ 266 extern void AD5592R_HW_Reset(void); 267 268 /** 269 * \brief 读取AD5592R的寄存器 270 * 271 * \param[in] addr:寄存器地址 272 * 273 * \note D1和D0用来选择LDAC模式,每当调用一次该函数时,都会将模式设定为DAC即时更 274 * 新模式,需要修改的话,可以用一个变量保存。 275 */ 276 extern u16 AD5592R_Read_Register(ad5592r_addr_t addr); 277 278 /** 279 * \brief 读取AD5592R的寄存器 280 * 281 * \note 这个函数主要是用来测试,读出来15个寄存器的值。芯片复位后读取,其中data[6]为0x00FF,data[7]为0x005C,其他寄存器的值都为0,如果读出的结果都为0xFFFF,那可能通信出现问题了 282 */ 283 extern void AD5592R_Read_All_Register(u16 data[16]); 284 285 /** 286 * \brief 通用控制寄存器配置 287 * 288 * \param[in] flag:可以加入如下标志: 289 * #GP_COTL_ADC_BUF_PRECHARGE: ADC缓冲器用于对ADC预充电,不设置则禁用(默认) 290 * #GP_COTL_ADC_BUF_EN: 使能ADC缓冲器,不设置则禁能(默认) 291 * #GP_COTL_LOCK: 不能改变I/Ox引脚配置寄存器的内容,不设置则可以改变(默认) 292 * #GP_COTL_ALL_DAC: (一个数据)写入所有DAC,不设置则由DAC地址决定写入哪个DAC(默认) 293 * #GP_COTL_ADC_RANGE_REFx2: ADC增益为0 V至2 × VREF,不设置则为1倍(默认) 294 * #GP_COTL_DAC_RANGE_REFx2: DAC增益为0 V至2 × VREF,不设置则为1倍(默认) 295 */ 296 extern void AD5592R_GP_Contol_Set(u16 flag); 297 298 299 /** 300 * \brief 将指定的引脚配置为指定的功能 301 * 302 * \param[in] func:设定的功能#ADC_PIN_CONF:配置为ADC引脚 303 * #DAC_PIN_CONF:配置为DAC引脚 304 * #PD_CONF:下拉配置,设定为85kΩ下拉(默认就是这个) 305 * #GPIO_W_CONF:GPIO写配置,意思就是将引脚配置为GPIO输出 306 * #GPIO_R_CONF:GPIO读配置,意思就是将引脚配置为GPIO输入 307 * #GPIO_OD_CONF:GPIO开漏配置,意思就是将引脚配置为开漏输出,否则为推挽输出(默认) 308 * #GPIO_3_STAT_CONF:三态配置 309 * \param[in] pin:选择需要配置的引脚,取值为AD5592R_PIN_0~AD5592R_PIN_7,可以按 310 * 位或。 311 * 312 * 313 * \param[in] flag:根据func的值,可以传入一些特殊标志,具体如下: 314 * 1、如果写入的func值为 GPIO_W_CONF,则传入的flag可以加入 315 * AD5592R_PIN_7_BUSY,表示将PIN7设定为BUSY引脚。 316 * 2、如果写入的func值为GPIO_R_CONF,则传入的flag可以加入 317 * GPIO_R_CONF_EN_RDBK,表示下一个SPI操作输出GPIO引脚的状态, 318 * 并将状态通过返回值返回。 319 * 320 * 321 */ 322 extern u8 AD5592R_Pin_Config(ad5592r_addr_t func, u8 pin, u16 flag); 323 324 /** 325 * \brief 给8个GPIO写入数据 326 * 327 * \param[in] value:要写入的值 328 */ 329 extern void AD5592R_GPIO_Write(u8 value); 330 331 /** 332 * \brief 给DAC通道写入数据 333 * 334 * \param[in] channel:0~7有效,指定DAC通道 335 * \param[in] value:要写入的值,低12位有效 336 */ 337 extern void AD5592R_DAC_Write_Data(u8 channel, u16 value); 338 339 /** 340 * \brief 配置DAC通道回读 341 * 342 * \param[in] channel:0~7有效,指定DAC通道 343 * \param[in] value:要写入的值,低12位有效 344 * 345 * \todo 指的是当某个DAC通道使能以后,每次写DAC数据后的下一个时钟周期读出数据? 346 * 还是说每次使能回读之后的下一个时钟周期读出数据?本函数写的是后者。 347 */ 348 extern u16 AD5592R_DAC_Readback(u8 channel); 349 350 /** 351 * \brief 设定关断DAC和内部基准电压源 352 * 353 * \param[in] value:#true:关断基准电压源、 DAC和ADC 354 * #false:基准电压源和DAC关断状态由D9和D7至D0决定(默认)。 355 */ 356 extern void AD5592R_PD_ALL_Set(u8 value); 357 358 /** 359 * \brief 选择基准电压源 360 * 361 * \param[in] value: #REF_INTERNAL:使用内部基准电压源 362 * #REF_EXTERNAL:使用外部基准电压源 363 */ 364 extern void AD5592R_Ref_Set(ref_select_t value); 365 366 /** 367 * \brief 选择关断DAC 368 * 369 * \param[in] pin:选择需要配置的引脚,取值为AD5592R_PIN_0~AD5592R_PIN_7,可以按 370 * 位或。 371 * \param[in] mode: #MORMAL_MODE:选择的引脚设置为正常模式 372 * #PD_MODE:引脚若配置为DAC则关断 373 */ 374 extern void AD5592R_PD_DAC(u8 pin, pd_mode_t mode); 375 376 /** 377 * \brief ADC序列寄存器设置 378 * 379 * \prarm[in] channel:0~7有效,指定需要加入转换序列的ADC通道 380 * \prarm[in] flag:可以加入如下标志: 381 * #ADC_SEQ_REP:使能序列重复,不设置则禁用序列重复(默认) 382 * #ADC_SEQ_TEMP:使能温度指示器回读,不设置则禁用温度指示器回读(默认) 383 */ 384 extern void AD5592R_ADC_Sequence_set(u8 channel, u16 flag); 385 386 387 /** 388 * \brief AD5592R软件复位 389 */ 390 extern void AD5592R_Soft_Reset(void); 391 392 /** 393 * @} 394 */ 395 396 #endif