📄 twi.s
字号:
.module twi.c
.area vector(rom, abs)
.org 34
rjmp _twi_isr
.area text(rom, con, rel)
.dbfile E:\iccavr\无线+I2C\test1--sr\twi.c
.dbfunc e StopDelay _StopDelay fV
; i -> R16,R17
.even
_StopDelay::
.dbline -1
.dbline 40
;
; /*函数的使用说明:
; 1.该twi模块包含四种工作模式,主发,主收,从收,从发,其中主发由
; twiWriteByte(从机地址,要发送的数据)实现,主收由twiReadByte(从机地址)实现,从机
; 的发送与接收均通过中断实现
; 2.在从机模式下,需调用twi_init()来使能并初始化twi总线
; 3.在主机模式下,调用twiWriteByte(从机地址,要发送的数据)向指定的从机写入数据,
; 调用twiReadByte(从机地址)从指定的从机读取数据,然后可通过查询错误状态error_state
; 的值决定下一步如何操作。
; */
;
; //************************* 头文件 ******************************************
; #include "twi.h"
;
; //TWI同程序的接口参数
; //receive_data,error_state不用设置
; extern char receive_data; // 接收到的数据
; extern char error_state; // twi错误状态
; //以下三个参数根据需求进行设置
; extern char slave_address; // 设置从机地址,Bits 7..1:存放从机地址,Bit 0:最低位为广播识别使能位
; extern char bit_race; // 设置主机模式的比特率,SCL=CPU频率/(16+2*(TWBR)*4TWPS),TWPS在4的指数位置
; extern char send_data; // 从机发送模式下从机向主机发送的数据
;
; //*************************** twi错误状态说明*********************************
; // 0:twi传送无错误
; // 1:在主发模式下,SLA+W已发送,返回NOT ACK
; // 2:在主发模式下,DATA已发送,返回NOT ACK
; // 3:在主发模式下,SLA+W或者数据的仲裁失败
; // 4:在主收模式下,SLA+R或者数据的仲裁失败
; // 5:在主收模式下,SLA+R已发送,返回NOT ACK
; // 6:在主发模式下,START信号发送不成功
; // 7:在主收模式下,START信号发送不成功
; // 8:在主收模式下,数据接受完成
; // 9:在主收模式下,数据接受不成功
; //*****************************************************************************
;
; //*************************** twi stop 信号延时时间 ****************************
; //******************************************************************************
; void StopDelay(void)
; {
.dbline 42
clr R16
clr R17
rjmp L5
L2:
.dbline 42
L3:
.dbline 42
subi R16,255 ; offset = 1
sbci R17,255
L5:
.dbline 42
; unsigned int i;
; for(i = 0;i < 20;i ++);
cpi R16,20
ldi R30,0
cpc R17,R30
brlo L2
.dbline -2
.dbline 43
; }
L1:
.dbline 0 ; func end
ret
.dbsym r i 16 i
.dbend
.dbfunc e start _start fV
.even
_start::
.dbline -1
.dbline 47
; //******************************* twi 启动总线*********************************
; //******************************************************************************
; void start(void)
; {
.dbline 48
; TWCR = (1 << TWINT) | (1 << TWSTA) | (1 << TWEN); //发出start信号
ldi R24,164
out 0x36,R24
.dbline 49
; _NOP();
nop
.dbline 50
; _NOP();
nop
.dbline -2
.dbline 51
; }
L6:
.dbline 0 ; func end
ret
.dbend
.dbfunc e stop _stop fV
.even
_stop::
.dbline -1
.dbline 55
; //************************** twi 释放总线************************************
; //***************************************************************************
; void stop(void)
; {
.dbline 56
; TWCR = (1 << TWINT) |(1 << TWEN)|(1 << TWSTO); //发出stop信号
ldi R24,148
out 0x36,R24
.dbline 57
; StopDelay(); //等待总线恢复
rcall _StopDelay
.dbline 58
; TWCR= 0x45;
ldi R24,69
out 0x36,R24
.dbline -2
.dbline 59
; }
L7:
.dbline 0 ; func end
ret
.dbend
.dbfunc e twi_init _twi_init fV
.even
_twi_init::
.dbline -1
.dbline 71
; //****************************** twi 初始化*************************************
; // TWCR: 控制寄存器,用来控制TWI操作,说明如下:
; // Bit 7-TWINT:中断标志位,Bit 6-TWEA:使能应答位,Bit 5-TWSTA:START状态位
; // Bit 4-TWSTO:STOP状态位,Bit 3-TWWC: 写冲突标志,Bit 2-TWEN:TWI使能位
; // Bit 1-RES:保留,Bit 0-TWIE:中断使能
; // TWSR: 状态寄存器,Bits 7..3:表示了TWI总线的当前状态,读取时需屏蔽低三位的值,Bits 1..0-TWPS:TWI预分频位
; // TWBR: 比特率寄存器,用来设置TWI的工作频率,计算公式为:SCL=CPU频率/(16+2*(TWBR)*4TWPS),TWPS在4的指数位置
; // TWAR: 从机地址寄存器,Bits 7..1:存放从机地址,Bit 0:最低位为广播识别使能位
; // TWDR: 数据寄存器,用来存放接收或要发送的地址和数据
; //******************************************************************************
; void twi_init(void)
; {
.dbline 72
; TWCR= 0x00; //disable twi
clr R2
out 0x36,R2
.dbline 73
; TWBR= bit_race; //set bit rate
lds R2,_bit_race
out 0x0,R2
.dbline 74
; TWSR= 0x00; //set prescale为1
clr R2
out 0x1,R2
.dbline 75
; TWAR= slave_address; //set slave address
lds R2,_slave_address
out 0x2,R2
.dbline 76
; TWCR= 0x45; //enable twi
ldi R24,69
out 0x36,R24
.dbline -2
.dbline 77
; }
L8:
.dbline 0 ; func end
ret
.dbend
.dbfunc e twi_isr _twi_isr fc
; TWSR_state -> R16
.even
_twi_isr::
st -y,R2
st -y,R16
st -y,R17
st -y,R24
st -y,R25
st -y,R30
in R2,0x3f
st -y,R2
.dbline -1
.dbline 190
; //************************** 检测发送时的状态码 ****************************
; //**************************************************************************
; /*void checkstate(void)
; {
; unsigned char TWSR_state;
; TWSR_state = TWSR & 0xf8; //屏蔽第三位读取状态
; switch(TWSR_state)
; {
; case 0x08:error_state=0;break; // START已发送
; case 0x18:error_state=0;break; // SLA+W已发送,接收到ACK
; case 0x20:error_state=1;break; // SLA+W已发送,接收到NOT ACK
; case 0x28:error_state=0;break; // 数据已发送,接收到ACK
; case 0x30:error_state=2;break; // 数据已发送,接收到NOT ACK
; case 0x38:error_state=3;break; // SLA+W或数据的仲裁失败
; default: break;
; }
; TWCR=0x45;
; }
; //********** twi主机发送,向地址为WriteAddress的从机发送1个字节数据************
; // 入口参数1:WriteAddress:从机地址
; // TWDR中存放的高七位为地址,最低位表示读写控制,0为写,1为读
; // 入口参数2:data,向从机写的数据
; //******************************************************************************
; void twiWriteByte(unsigned char WriteAddress,char data)
; {
; unsigned char TWCR_state;
; char i=0;
; error_state=0;
;
; //初始化 twi
; twi_init();
;
; //发送START信号
; start();
; TWCR_state = TWCR & 0x80;
; while((TWCR_state == 0x00))
; {
; i++;
; if(i>=10){error_state=6;break;}
; TWCR_state = TWCR & 0x80; //轮循等待START信号发送完成
; }
;
; //若START发送成功,发送从机地址
; if(error_state==0)
; {
; TWDR = WriteAddress; //发地址
; _NOP();
; _NOP();
;
; TWCR = (1 << TWINT) | (1 << TWEN);
; TWCR_state = TWCR & 0x80;
; while(TWCR_state == 0x00)
; {TWCR_state = TWCR & 0x80;} //轮循查询地址是否发送
; checkstate();
; }
;
; //若从机地址发送成功,向从机发送数据
; if(error_state==0)
; {
; TWDR = data; //发数据
; _NOP();
; _NOP();
; TWCR = (1 << TWINT) | (1 << TWEN);
; TWCR_state = TWCR & 0x80;
; while(TWCR_state == 0x00) TWCR_state = TWCR & 0x80; //轮循查询数据是否发送
; checkstate();
; }
;
; //发送STOP,退出总线控制
; stop();
; }
; //************** twi主机接收,从地址为ReadAaddress的从机接收1个字节*************
; // 入口参数1:ReadAaddress:从机地址
; // TWDR中存放的高七位为地址,最低位表示读写控制,0为写,1为读
; //******************************************************************************
; void twiReadByte(unsigned char ReadAaddress)
; {
; unsigned char TWCR_state;
; char i=0;
; error_state=0;
;
; //初始化 twi
; twi_init();
;
; //发送START信号
; start();
; TWCR_state = TWCR & 0x80;
; while((TWCR_state == 0x00))
; {
; i++;
; if(i>=10){error_state=7;break;}
; TWCR_state = TWCR & 0x80; //轮循等待START信号发送完成
; }
;
; //若START信号发送完成,发送读地址,并读取数据
; if(error_state==0)
; {
; TWDR = ReadAaddress; //发地址
; _NOP();
; _NOP();
; TWCR = (1 << TWINT) | (1 << TWEN)|( 1<<TWIE )|(1 << TWEA);
; while(error_state==0)
; {i++;if(i>150){error_state=9;break;}};
; }
;
; //发送STOP,退出总线控制
; stop();
; }*/
; //************** twi 中断例程,接收数据 ****************************************
; //******************************************************************************
; //twi中断服务程序
; char twi_isr(void)
; {
.dbline 192
; unsigned char TWSR_state;
; TWSR_state = TWSR & 0xf8;
in R16,0x1
andi R16,248
.dbline 193
; switch(TWSR_state)
clr R17
cpi R16,128
ldi R30,0
cpc R17,R30
breq L14
ldi R24,128
ldi R25,0
cp R24,R16
cpc R25,R17
brlt L20
L19:
cpi R16,0
cpc R16,R17
breq L18
X0:
cpi R16,0
ldi R30,0
cpc R17,R30
brlt L11
L21:
cpi R16,96
ldi R30,0
cpc R17,R30
breq L13
rjmp L11
L20:
cpi R16,160
ldi R30,0
cpc R17,R30
breq L16
ldi R24,160
ldi R25,0
cp R24,R16
cpc R25,R17
brlt L23
L22:
cpi R16,136
ldi R30,0
cpc R17,R30
breq L15
rjmp L11
L23:
cpi R16,248
ldi R30,0
cpc R17,R30
breq L11
rjmp L11
X1:
.dbline 194
; {
L13:
.dbline 196
; //从机接收模式下的中断程序
; case 0x60: TWCR=0xc5;break; // SLA+W已经接收,ACK已返回(从机被寻址,等待主机发送数据)
ldi R24,197
out 0x36,R24
.dbline 196
rjmp L11
L14:
.dbline 197
; case 0x80: receive_data = TWDR;TWCR=0xc5;break; // 数据已接收,ACK已返回(接收数据)
in R2,0x3
sts _receive_data,R2
.dbline 197
ldi R24,197
out 0x36,R24
.dbline 197
rjmp L11
L15:
.dbline 198
; case 0x88: TWCR=0xc5;break; // 数据已接收,NOT ACK已返回(下一步:退出总线)
ldi R24,197
out 0x36,R24
.dbline 198
rjmp L11
L16:
.dbline 199
; case 0xA0: TWCR=0xc5;break; // 接收到STOP或重复的START信号(停止接收数据)
ldi R24,197
out 0x36,R24
.dbline 199
rjmp L11
X2:
.dbline 215
;
; /* //主机接收模式下的中断程序
; case 0x38: error_state=4;TWCR=0xc5;break; // SLA+R或NOT ACK仲裁失败(下一步:退出总线)
; case 0x40: TWCR=0xc5; break; // SLA+R已发送,接收到ACK(从机被寻址,主机等待接收数据)
; case 0x48: error_state=5;TWCR=0xc5;break; // SLA+R已发送,接收到NOT ACK(下一步:退出总线)
; case 0x50: receive_data = TWDR;_NOP();_NOP();error_state=8;TWCR = 0xc4;break; // 接收到数据,ACK已返回(接收数据)
; case 0x58: receive_data = TWDR;_NOP();_NOP();error_state=8;TWCR = 0xc4;break; // 接收到数据,NOT ACK已返回(下一步:退出总线)
;
; //从机发送模式
; case 0xA8: TWDR= send_data;_NOP();_NOP();TWCR= 0x85; break; // SLA+R已接收,ACK已返回(从机被寻址,开送发送数据)
; case 0xB8: TWCR= 0xc5;break; // TWDR里的数据已发送,接受到ACK(主机接收到数据,继续发送数据)
; case 0xC0: TWCR= 0xc5;break; // TWDR里的数据已发送,接受到NOT ACK(下一步:退出总线)
; case 0xC8: TWCR= 0xc5;break; // TWEA=0,接收到ACK(停止发送数据,退出发送)
; */
; //其他状态码
; case 0xf8: break; // 等待或进行当前传输
L18:
.dbline 216
; case 0x00: TWCR= 0x95;break; // 非法的START或STOP引起的总线错误(释放总线)
ldi R24,149
out 0x36,R24
.dbline 216
.dbline 217
; default: break;
L11:
.dbline 219
; }
; return receive_data; // 返回接收到的数据
lds R16,_receive_data
.dbline -2
L9:
ld R2,y+
out 0x3f,R2
ld R30,y+
ld R25,y+
ld R24,y+
ld R17,y+
ld R16,y+
ld R2,y+
.dbline 0 ; func end
reti
.dbsym r TWSR_state 16 c
.dbend
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -