十年专注单片机方案开发的方案公司英锐恩,分享三个源程序(设计风格的比较)。英锐恩现提供服务产品涉及主控芯片:8位单片机、16位单片机、32位单片机及各类运算放大器等。
一:DS1302与PIC16F877接口程序
;================================================ ;DS1302控制子程序 ;================================================= ;程序包括:DS1302初始化,时间数据写入和读取程序 ;DS1302_init ;init ds1302 ;Set_DS1302 ;set time to ds1302 ;Get_DS1302 ;get time from ds1302 ;=================================================
;============== 1302定义 =========================================== #DEFINE T_IO PORTE,2 ;1302 I_O #DEFINE T_CLK PORTE,1 ;1302时钟 #DEFINE T_RST PORTE,0 ;1302使能位 #DEFINE DS1302_RX 20H ;保存接受的1个数据 #DEFINE DS1302_TX 28H ;准备写入到DS1302的一个数据
counter equ xxxh temp_cnt equ xxxh
;************************************************** ; DS1302初始化程序 ;************************************************** DS1302_init ;未写 return ;********************************************************** ;子程序名:Set_DS1302 ;功 能:设置DS1302 初始时间,并启动计时。 ;说 明: ;调 用:Write_byte ;入口参数:初始时间在:W_Second,W_Minute,W_Hour,W_Day,W_Month,W_Week.W_YearL(地址连续) ;出口参数:无 ;消耗资源:counter,SecAddr(预定义) ;设 计:zhengYanbo 日 期:2005.4.21 ;修 改: 日 期: ;********************************************************** Set_DS1302: bcf T_RST bcf T_CLK bsf T_RST movlw 8eh movwf DS1302_TX ;数据发送寄存器 call Write_byte ;发送字节 movlw 00h ;WP=0 movwf DS1302_TX call Write_byte bsf T_CLK bcf T_RST movlw W_Second movwf FSR movlw counter,D'7' movlw 80h movwf SecAddr ;秒写地址 S13021: bcf T_RST bcf T_CLK bsf T_RST movf SecAddr,W movwf DS1302_TX call Write_byte ;写秒地址 movf INDF,W movwf DS1302_TX call Write_byte ;写秒数据 incf FSR incf SecAddr incf SecAddr bsf T_CLK bcf T_RST decfsz counter,F goto S13021 bcf T_RST bcf T_CLK bsf T_RST movlw 8eh ;控制寄存器 movwf DS1302_TX call Write_byte movlw 80h ;控制WP=1,写保护 movwf DS1302_TX call Write_byte bsf T_CLK bcf T_RST return
;********************************************************** ;子程序名:Get_DS1302 ;功 能:从DS1302 读时间 ;说 明: ;调 用:Write_byte,Read_byte ;入口参数:时间保存在:R_Second,R_Minute,R_Hour,R_Day,R_Month,R_Week.R_YearL ;出口参数:无 ;消耗资源: counter,SecAddr(预先定义) ;设 计:zhengYanbo 日 期:2005.4.21 ;修 改: 日 期: ;********************************************************** Get_DS1302: movlw R_Second ;准备地址 movwf FSR movlw D'7' movwf counter movlw 81h ;秒读地址 movwf SecAddr G13021 bcf T_RST bcf T_CLK bsf T_RST movf
SecAddr,W movwf DS1302_TX call Write_byte call Read_byte movf DS1302_RX,W movwf INDF incf FSR incf SecAddr incf SecAddr bsf T_CLK bcf T_RST decfsz counter,F ;接受7个数据 goto G13021 return ;********************************************************** ;功 能:写1302一字节 (内部子程序) ;入口:数据预先在DS1302_TX中 ;出口:无 ;消耗资源:temp_cnt(预先定义) ;********************************************************** Write_byte movlw D'8' movwf tmp_cnt W_shift rrf DS1302_TX,F ;带C移位(低位在前) btfsc STATUS,C goto send_1 bcf T_IO goto send_0 send_1 bsf T_IO send_0 bsf T_CLK ;上升沿写 nop bcf T_CLK decfsz temp_cnt,F goto W_shift return ;********************************************************** ;功 能:读1302一字节 (内部子程序) ;入口:无 ;出口:数据保存在DS1302_RX中 ;消耗资源:temp_cnt(预先定义) ;********************************************************** Read_byte bsf STATUS,RP0 ;bank1 bsf TRISE,2 ;设置串行数据口为输入 bcf STATUS,RP0 ;bank0 movlw D'8' movwf temp_cnt R_shift btfsc T_IO goto get_1 bcf STATUS,C ;clr c goto get_0 get_1 bsf STATUS,C get_0 rrf DS1302_RX,F bsf T_CLK nop bcf T_CLK ;下降沿读 decfsz temp_cnt,F goto R_shift
;reset output bsf STATUS,RP0 ;bank1 bcf TRISE,2 ;设置T_IO为输出 bcf STATUS,RP0 ;bank0 return 这个程序一定要,PIC默认PORTE为A/D口,切记,切记!改为其他口时候请参考PIC的DATASHEET ;============================================= ; 初始化端口 ;============================================= Port_Init bsf STATUS,RP0 ;bank1 bcf ADCON1,PCFG0 bsf ADCON1,PCFG1 bsf ADCON1,PCFG2 bcf ADCON1,PCFG3 ;PORTA and PORTE:digital I/O bcf STATUS,RP0 ;bank0 return 二:红外线发射与接收一例 ; include
;------------------ STATUS equ 0X03 PORTA equ 0x05 PORTB equ 0x06 W equ 0x00 F equ 0x01 Z equ 0x02 C equ 0x00 ;------------------ n55ks equ 0x0C ; nchks equ 0x0D ; avgIR equ 0x0E ; nsamp equ 0x0F ; tdetect equ 0x10 ; ;------------------------------------ org 0x000 goto start org 0x004 start
movlw b'00000001' tris PORTA ; RA0 红外输入 movlw b'00000000' tris PORTB ;RB0 红外输出 ; 2us in goto at end of loop blip call doblip ; +6.248ms = 6.250ms total call seeblip ; 6.250ms total call seeblip ; 6.250ms total call seeblip ; 6.250ms total call seeblip ; 6.250ms total call seeblip ; 6.250ms total call seeblip ; 6.250ms total call seeblip ; 6.250ms total nop nop call doblip call seeblip call seeblip call seeblip call seeblip call seeblip call seeblip call seeblip goto blip ;
;============ 产生 55.5kHz锯形波(346*18us=6.228ms)====== doblip movlw 0xFF movwf n55ks osc nop call make55k ;RB0 红外输出 decfsz n55ks,F goto osc ;-------------- morblip ;6253us DELAY movlw 0x5B movwf n55ks oscmore call make55k ;RB0 红外输出 nop decfsz n55ks,F goto oscmore ;------------- call delay8u nop nop nop nop nop bcf PORTB,2 ; RB2 LED指示 return
;--------------------------- make55k bsf PORTB,0 ;RB0 红外输出 call delay8u bcf PORTB,0 return delay8u nop nop nop nop return ;========= 6.250ms =================== seeblip clrf avgIR movlw 0xF9 movwf nchks chk call chkIR ;// decfsz nchks,F goto chk movf avgIR,W ; sublw 0X78 btfss STATUS,C goto detectd bcf PORTB,0 ;RB0 红外输出 OFF goto chkdone detectd bsf PORTB,0 nop chkdone call delay8u nop nop nop nop return ;======================== chkIR clrf nsamp btfss PORTA,0 incf nsamp,F btfss PORTA,0 ; RA0 红外输入 incf nsamp,F btfss PORTA,0 ; RA0 红外输入计数值存到nsamp incf nsamp,F btfss PORTA,0 incf nsamp,F btfss PORTA,0 incf nsamp,F btfss PORTA,0 ;共六次 nsamp》5,avgIR增一 incf nsamp,F nop movf nsamp,W sublw 0X04 btfss STATUS,C incf avgIR,F return ;============================ end
三:18B20通讯程序
这是我在16F877,18F1320,18F1220上通过的18B20程序,18B20主要是延时问题,这个解决了,什么都可以通过。 # include # 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);>
//*********************向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(); //释放总线 }