这是我在16F877,18F1320,18F1220上通过的18B20程序,18B20主要是延时问题,这个解决了,什么都可以通过。 [原作者没有提供所使用的编译器,由于是C程序,所以大同小异,建议使用时,确认是否与你的编译器兼容] C程序 #include <pic18f1220.h> #define uch unsigned char #define unint unsigned int #define DQ RB3 //定义18B20数据端口 #define DQ_DIR TRISB3 //定义18B20D口方向寄存器 #define W1_INPUT 1 #define W1_OUTPUT 0 #define FALSE 0 #define TRUE !FALSE #define DQ_HIGH() DQ_DIR = W1_INPUT #define DQ_LOW() DQ = 0; DQ_DIR = W1_OUTPUT void delay(unint x) { unint d; d=x; while(--d) {;} } bit reset(void) //初始化18B20 { static bit presence; //定义一个应答信号 DQ_LOW(); delay(70); //置总线为低电平并保持至少480us DQ_HIGH(); //等电阻拉高总线并保持15-60us delay(5); presence=DQ; //接受应答信号 delay(20); //延时60-240us return(presence); //返回应答信号 } //*************** 读一位函数******************// bit read_bit(void) { static bit i; DQ_LOW(); DQ_LOW(); DQ_HIGH(); asm("nop"); asm("nop"); asm("nop"); i=DQ; delay(3); return(i); } //*********************写一位函数****************// void write_bit(uch bitval) { DQ_LOW(); delay(1); if (bitval==1) { DQ_HIGH(); } delay(3); DQ_HIGH(); } //************** 从18B20中读一个字节**************// uch read_byte(void) { uch i; uch j; uch value=0; for (i=0;i<8;i++) { j=read_bit(); //调读位函数 if (j) //如果是 1 置1 { value|=(0x01<<i); //先读低位,再读高位 asm("nop"); asm("nop"); asm("nop"); } } //否则置 0 return(value); } //*********************向18B20中 写一个字节**************// void write_byte(uch val) { uch i; uch temp; for (i=0;i<8;i++) { temp=val>>i; temp&=0x01; write_bit(temp); //调写位函数 } asm("nop"); asm("nop"); asm("nop"); } main() { uch teml,temh; GIE=0; OSCCON=0X6E; //这是18F1320的频率选择寄存器 ADCON1=0X7F; do{ ; }while (reset()) ; //复位等待从机应答 write_byte(0XCC); //忽略ROM匹配 write_byte(0X44); //发送温度转化命令 delay(25000); //延时100-300us do { ; }while( reset()); //再次复位,等待从机应答 write_byte(0XCC); //忽略ROM匹配 write_byte(0XBE); //发送读温度命令 teml =read_byte(); //读出温度低8 temh=read_byte(); //读出温度高8位 DQ_HIGH(); //释放总线 }
(责任编辑:admin) |