FPGA驱动DS18B20温度传感器

FPGA驱动DS18B20温度传感器
先介绍一下DS18B20

DS18B20是一款广泛应用的高精度数字温度传感器。测温范围为-55℃到+125℃,采用1-Wire通信即仅采用一根数据线与微控制器进行通信。

温度存储在ROM中,占据2byte数据,其中第11位用于判断温度正负,其他各位的权重如图所示。正数即为本身,负数为补码加一

温度转换如图,温度值的分辨率是0.0625,将对应的十进制数*0.0625即为实际的摄氏度值

7D0h的十进制是 2000,对应的温度值是 2000 * 0.0625 = 125°C

FE6F的补码 + 1 对应的十进制数是 -401,对应的温度是是 -401 * 0.0625 = -25.0625°C

几个重要的指令,本次实验使用到这几个寄存器,如果有多个设备,建议阅读DS18B20规格书

44H : 温度转换

BEH : 读取温度寄存器

4EH : 写温度寄存器

CCH : 跳过设备选址,只适用于总线上只有一个设备

单总线时序

复位时序

读写时序

实验介绍

本次实验包含三个部分:

PLL锁相环:用于实现1us时钟的分频

DS18B20驱动模块:用于控制DS18B20的温度采集

数码管显示模块

PLL锁相环(此处也可以使用分频器实现)

生成1Mhz时钟,一个时钟周期是1us

DS18B20驱动模块

输入 : 1us时钟,单总线,复位

输出 : 数据位,符号位

驱动过程

开始 → 初始化 → 发送转换命令 → 等待转换 → 重新初始化 → 发送读取命令 → 读取温度数据 → 数据处理 → 结束

1.初始化 : 拉低500us,松开总线,等待70us,检测总线状态(如果总线有设备,总线会被拉低),总线低电平,输出存在从机脉冲flag_pulse,跳转到发送状态

2.发送状态 : 包含跳过ROM(CCh)命令和温度转换命令(44h),(低位在前,即44CC)一共需要发送16个bit。

添加bit计数器bit_cnt,写0/1都是65us,

写0,拉低63us,释放总线2us

写1,拉低15us,释放总线50us

先拉低1us,检测bit位为0/1,0的话一直拉低,1的话释放总线,等待2us,

当bit_cnt == 15,并且cnt_1us计数器等于64,进入等待状态

3.等待状态 : 等待750ms,即计数器记到750_000,进入重新初始化状态

4.重新初始化 : 完进入接收状态

5.接收状态 : 包含跳过ROM(CCH)和温度读取命令(BE),低位在前,即BECC,

之后进入读取温度状态

6.读取温度状态 : 读数据,拉低1us总线,13us处采样电平值,每次将总线的电平值给data_temp的最高位,当读完16bit数据,判断最高位是0/1,即正负,正数不操作,复数取反码加一,符号位同样是读最高位。

7.读完之后回到初始状态

代码如下:

module ds18b20_ctrl ( input wire clk_1us, //1us的时钟 input wire sys_rst_n,//复位 inout wire dq, //单总线,用于和ds18b20交换数据, 双向 output reg sign, // 输出数据的符号 output wire [19:0] data_out //输出的温度数据 ); parameter ERROR = 3'd0; parameter INIT = 3'd1; parameter WRITE = 3'd2; parameter WAIT = 3'd3; parameter INIT_AGAIN= 3'd4; parameter READ = 3'd5; parameter GET_TEMP = 3'd6; parameter MAX_WAIT_TIME = 20'd749_999; //等待时间 parameter TIME_ONE_WRITE_READ = 7'd64; //一次读写的时间,64us parameter WRITE_DATA = 16'h44CC; parameter READ_DATA = 16'hBECC; reg [2:0] State; reg [19:0] cnt_1us; //计数器,每个时钟计数器增加一 reg flag_pulse; //用于判断初始化时有没有收到从机回应 reg [3:0] bit_cnt; // 计算读写的bit数 reg dq_en; //总线使能控制位 reg [15:0] data_temp; //直接读取的寄存器的数据 reg [19:0] data; //判断正负后的数据 reg dq_out; assign dq = (dq_en == 1'b1) ? dq_out : 1'dz; //只有总线使能位为1时,总线才受FGPA控制 assign data_out = data * 10'd625 / 4'd10; //数码管只有六位,只显示小数点后三位,点加在第四个数码管上面 //计数器 cnt_1us //清零情况 1.两种初始化状态结束 2.读写完一次 3.等待时间结束 always @(posedge clk_1us or negedge sys_rst_n) begin if(sys_rst_n == 1'b0) cnt_1us <= 1'b0; else if(((State == INIT || State == INIT_AGAIN) && (cnt_1us == 999)) || ((State == WRITE || State == READ || State == GET_TEMP) && (cnt_1us == TIME_ONE_WRITE_READ)) || (State == WAIT && cnt_1us == MAX_WAIT_TIME)) cnt_1us <= 1'b0; else cnt_1us <= cnt_1us + 1'b1; end //flag_pulse 用于判断初始化时有没有收到从机回应,即dq == 0 //flag_pulse是在1000us时清零 always @(posedge clk_1us or negedge sys_rst_n) begin if(sys_rst_n == 1'b0) flag_pulse <= 1'b0; else if((State == INIT || State == INIT_AGAIN) && cnt_1us == 20'd570 && dq == 1'b0) flag_pulse <= 1'b1; else if(cnt_1us == 999) flag_pulse <= 1'b0; else flag_pulse <= flag_pulse; end //bit_cnt 计算收发的bit位数 always @(posedge clk_1us or negedge sys_rst_n) begin if(sys_rst_n == 1'b0) bit_cnt <= 4'd0; else if((State == WRITE || State == READ || State == GET_TEMP) && cnt_1us == TIME_ONE_WRITE_READ && bit_cnt == 4'd15) //读写完一次清零 bit_cnt <= 4'd0; else if((State == WRITE || State == READ || State == GET_TEMP) && cnt_1us == TIME_ONE_WRITE_READ) bit_cnt <= bit_cnt + 1'b1; else bit_cnt <= bit_cnt; end //表示状态变化 always @(posedge clk_1us or negedge sys_rst_n) begin if(sys_rst_n == 1'b0) State <= INIT; else case (State) INIT: //570us检测到从机存在(flag_pulse == 1)切换到WRITE begin if(cnt_1us == 999 && flag_pulse == 1'b1) State <= WRITE; else State <= INIT; end WRITE: //发送写指令,一共需要发送16bit数据,当全部发完进入等待状态,1bit数据用时65us begin if(bit_cnt == 4'd15 && cnt_1us == TIME_ONE_WRITE_READ) State <= WAIT; else State <= WRITE; end WAIT: begin if(cnt_1us == MAX_WAIT_TIME) State <= INIT_AGAIN; else State <= WAIT; end INIT_AGAIN: begin if(cnt_1us == 999 && flag_pulse == 1'b1) State <= READ; else State <= INIT_AGAIN; end READ: //发送读命令,一共需要发送16bit数据,当全部发完进入读取温度状态,1bit数据用时65us begin if(bit_cnt == 4'd15 && cnt_1us == TIME_ONE_WRITE_READ) State <= GET_TEMP; else State <= READ; end GET_TEMP: //读取温度,读取总线上的温度值,一共读取16bit,当全部发完进入初始化状态,1bit数据用时65us begin if(bit_cnt == 4'd15 && cnt_1us == TIME_ONE_WRITE_READ) State <= INIT; else State <= GET_TEMP; end default: State <= ERROR; endcase end //控制总线 dq,dq_en,data_temp always @(posedge clk_1us or negedge sys_rst_n) begin if(sys_rst_n == 1'b0) begin dq_en <= 1'b0; dq_out <= 1'b0; end else case (State) INIT: begin if(cnt_1us <= 20'd500) //初始化先拉低500us begin dq_en <= 1'b1; dq_out <= 1'b0; end else begin dq_en <= 1'b0; dq_out <= 1'b0; end end WRITE: begin if(cnt_1us <= 20'd1) //写总线需要先拉低1us begin dq_en <= 1'b1; dq_out <= 1'b0; end else if(cnt_1us >= 20'd62) //释放总线,恢复时间,至少1us,此处是2us begin dq_en <= 1'b0; dq_out <= 1'b0; end else if(WRITE_DATA[bit_cnt] == 1'b0) // 写0 begin dq_en <= 1'b1; dq_out <= 1'b0; end else if(WRITE_DATA[bit_cnt] == 1'b1) // 写1 begin dq_en <= 1'b0; dq_out <= 1'b0; end end WAIT: //为适应寄生电容供电 begin dq_en <= 1'b1; dq_out <= 1'b1; end INIT_AGAIN: begin if(cnt_1us <= 20'd500) //初始化先拉低500us begin dq_en <= 1'b1; dq_out <= 1'b0; end else begin dq_en <= 1'b0; dq_out <= 1'b0; end end READ: begin if(cnt_1us <= 20'd1) //写总线需要先拉低1us begin dq_en <= 1'b1; dq_out <= 1'b0; end else if(cnt_1us >= 20'd62) //释放总线,恢复时间,至少1us,此处是2us begin dq_en <= 1'b0; dq_out <= 1'b0; end else if(READ_DATA[bit_cnt] == 1'b0) // 写0 begin dq_en <= 1'b1; dq_out <= 1'b0; end else if(READ_DATA[bit_cnt] == 1'b1) // 写1 begin dq_en <= 1'b0; dq_out <= 1'b0; end end GET_TEMP: begin if(cnt_1us <= 20'd1) //写总线需要先拉低1us begin dq_en <= 1'b1; dq_out <= 1'b0; end else //1us之后就释放总线,在第13us处采样总线值 begin dq_en <= 1'b0; dq_out <= 1'b0; end end default: begin dq_en <= 1'b0; dq_out <= 1'b0; end endcase end //data_temp,data,sign 读取的数据,判断正负的数据,符号位 always @(posedge clk_1us or negedge sys_rst_n) begin if(sys_rst_n == 1'b0) begin data_temp <= 16'd0; end else if(State == GET_TEMP && cnt_1us == 20'd13) data_temp <= {dq,data_temp[15:1]}; else if(State == GET_TEMP && bit_cnt == 4'd15 && cnt_1us == TIME_ONE_WRITE_READ) begin if(data_temp[12] == 1'b1) //负数 begin data <= ~data_temp[10:0] + 1'b1; sign <= 1'b1; end else if(data_temp[12] == 1'b0) begin data <= data_temp[10:0]; sign <= 1'b0; end end end endmodule

数码管显示模块此处不介绍

绑定引脚:

实现现象如图(图中为野火征途开发板) :

Read more

GESP-C++考试(二级)考试重点 (附:【编程题模板】大全)

GESP-C++考试(二级)考试重点 (附:【编程题模板】大全)

一、GESP C++ 二级考试重点 “会写多层 if else  +  多层循环  +  用基础数学函数解决综合问题” (一)理论部分重点(选择 / 判断题高频) 根据大纲,以下内容是必考且高频 GESP大纲: (1)计算机基础 * 存储器 * RAM(内存) vs ROM(只读) * Cache(高速缓存) * 网络 * LAN / MAN / WAN * OSI 七层模型(知道名字) * TCP/IP 四层模型(知道名字) * 程序设计语言 * 机器语言 / 汇编语言 / 高级语言 * C++ 属于高级语言 👉 特点: * 不考计算 * 多为“认识型”“判断型” (2)流程图

By Ne0inhk
《C++ 基础进阶:内存开辟规则、类型转换原理与 IO 流高效使用》

《C++ 基础进阶:内存开辟规则、类型转换原理与 IO 流高效使用》

前引:在 C++ 编程中,内存管理是程序稳定性与性能的基石,而类型转换与 IO 流则是数据处理和交互的核心工具。栈与堆作为内存分配的两大核心区域,其开辟方式直接决定了变量的生命周期、访问效率及内存安全 —— 错误的分配策略可能导致内存泄漏、野指针或栈溢出等致命问题。与此同时,类型转换的合理性关乎类型系统的严谨性,不当转换易引发数据截断、逻辑错误;IO 流作为数据输入输出的桥梁,其正确使用则直接影响程序与外部设备(如控制台、文件)交互的可靠性! 目录 【一】内存完美开辟 (1)栈和堆的本质区别 (2)如何只在栈上开辟空间 (3)如何只在堆上开辟空间 【二】C++的四种类型转换 (1)static_cast (2)reinterpret_cast (3)const_cast (4)dynamic_cast 【三】operator类型转换 (1)

By Ne0inhk
华为OD机试双机位C卷:采购订单 (Py/Java/C/C++/Js/Go)

华为OD机试双机位C卷:采购订单 (Py/Java/C/C++/Js/Go)

采购订单 华为OD机试双机位C卷 - 华为OD上机考试双机位C卷 100分题型 华为OD机试双机位c卷真题目录点击查看: 华为OD机试双机位C卷真题题库目录|机考题库 + 算法考点详解 题目描述 在一个采购系统中,采购申请(PR)需要经过审批后才能生成采购订单(PO)。每个PR包含商品的单价(假设相同商品的单价一定是一样的)及数量信息。系统要求对商品进行分类处理:单价高于100元的商品需要单独处理,单价低于或等于100元的相同商品可以合并到同一采购订单PO中。针对单价低于100的小额订单,如果量大可以打折购买。 具体规则如下: 如果PR状态为"审批通过",则将其商品加入到PO中。如果PR的状态为"审批拒绝"或"待审批",则忽略改PR。 对于单价高于100元的商品,每个商品单独生成一条PO记录。对于单价低于100元的商品,将相同商品的数量合并到一条PO记录中。 如果商品单价<100且商品数量>=100,则单价打9折。 输入描述 第一行包含整数N,

By Ne0inhk