📄 wpcf8574.c
字号:
//参考程序A: WPCF8574.C
//程序功能:PCF8574是带SUMBUS总线的串口转并口的数据转换芯片.8位输入口连结8个//开关(K1-K8)作输入,二个8位输出锁存口分别连结8个LED灯,作二个8位输出..
// 程序安排二个8位输出口锁存的数据电平相反.
#include <c8051f020.h> // SFR declarations
#include <intrins.h>
#define WRITE 0x00 // 写位标志
#define READ 0x01 // 读位标致
#define CHIP_B 0x70
#define SMB_START 0x08 // (MT & MR)主收发器发送起始位成功.再将从机写地
//址送SMBUS
#define SMB_RP_START 0x10 // (MT & MR)主收发器重复发送起始位成功,再将从机
//读地址送SMB0DAT
#define SMB_MTADDACK 0x18 // (MT) 主收发器发送从地址+W成功;收到 //ACK(从机应答)
#define SMB_MTDBACK 0x28 // (MT)主收发器发送数据字节成功;收到ACK(从机//应答)
#define SMB_MRADDACK 0x40 // (MR) 主收发器发送从地址+R成功;收到 //ACK(从机应答)
#define SMB_MRDBNACK 0x58 // (MR) 主收发器接收数据成功;主机发送NACK
char COMMAND; // 保存从地址用与中断程序
char WORD; // 保存接收到的数据并把他发送出去.
unsigned char xdata SENDMODE;
bit SM_BUSY; // 在收/发数据时该位置1,中断完成后被清另.
void SYSCLK_Init (void);
char SLA_READ(char chip_select);
void SLA_SEND(char chip_select, char wr_data);
void sleep_ms(unsigned int count)
{
unsigned char ii,jj;
for(ii=0;ii<count;ii++)
{
for(jj=0;jj<250;jj++)
_nop_();
}
}
void MAIN (void)
{
unsigned char i,temp;
WDTCN = 0xde; // 禁止看们狗
WDTCN = 0xad;
SYSCLK_Init(); // 开外部晶振
XBR0 = 0x01; // 设置交叉开关,配置IO口
XBR2 = 0x40; // 使能交叉开关和弱上拉.
SMB0CN = 0x44; // 使能SMBus带低电平响应 (AA = 1)
SMB0CR = 0xc9; // 设置SMBus 速率= 100 kHz,系统时钟为11.0582MHZ
EIE1 |= 2; // SMBus 中断使能
EA = 1; // 全局中断使能
SM_BUSY = 0; // 释放总线,用于第一次数据传送
SI = 0;
while(1){
for(i=0;i<40;i++){ //延时
sleep_ms(200);
}
temp = SLA_READ(0x70); //芯片U1读取开关状态
sleep_ms(200);
SLA_SEND(0x72,temp); //芯片U2输出开关状态
sleep_ms(200);
sleep_ms(200);
SLA_SEND(0x74,~temp); //芯片U3输出开关状态的反码
sleep_ms(200);
sleep_ms(200);
sleep_ms(200);
sleep_ms(200);
}
}
void SYSCLK_Init (void) //时钟初始化
{
int i; // 延时计数器 i
OSCXCN = 0x67; // 起动外部时钟(11.0592MHZ)
for (i=0; i < 256; i++) ; // 时间间隔 (>1ms)
while (!(OSCXCN & 0x80)) ; // 等待时钟丢失允许位(MSCLKE)位置1
OSCICN = 0x88; // 选择外部晶振作系统时钟
// 允许时钟丢失检测
}
//数据输出到从器件的函数:
// chip_select = 器件从地址
void SLA_SEND(char chip_select, char wr_data)
{
SENDMODE=0x01;
while(SM_BUSY); // 当总线忙碌的时后就等待
SM_BUSY = 1; // SMBus忙碌标志位置1 .
SMB0CN = 0x44; // SMBus 使能, 响应信号为低电平
COMMAND = (chip_select | WRITE); // COMMAND = 7位地址加"写"位
WORD = wr_data; // 输出数据在WORD中
STO = 0;
STA = 1; // 起动数据传输
while(SM_BUSY); // 等待中断结束
}
//读取从器件输出数据的函数:
char SLA_READ(char chip_select){
SENDMODE=0;
while(SM_BUSY); // 总线忙碌时要等待
SM_BUSY = 1; // 总线忙碌标志位置1.
SMB0CN = 0x44; // SMBus 使能, 响应信号为低电平
COMMAND = (chip_select | READ); // COMMAND = 7位地址加"读"位
STO = 0;
STA = 1; // 开始传送
while(SM_BUSY); // 等待传送完成
return WORD; // 返回接收到的数据
}
// SMBus中断服务程序
void SMBUS_ISR (void) interrupt 7
{
switch (SMB0STA){ // SMBus中断状态 SMB0STA register)
//SMB_START ; (SMB0STA =08H)
//主发送器/接收器发送起始位成功.再将从机写地址送SMB0DAT并清除START位.
case SMB_START: //0x08
SMB0DAT = COMMAND ;
STA = 0; // 人工清除STA
SI = 0; // 人工清除SI
break;
//SMB_RP_START ;(SMB0STA =10H)
//主发送器/接收器发送重复起始位成功.再将从机读地址送SMB0DAT并清除START位.
case SMB_RP_START: //0x10
SMB0DAT = COMMAND;
STA = 0; // 人工清除STA
SI = 0;
break;
//SMB_MTADDACK (SMB0STA =18H)
//主发送器(MT)发送从地址+W成功;收到 ACK(从机应答),再将要发的子地址送入SMB0DAT.
case SMB_MTADDACK: //0x18
SMB0DAT = WORD;
SI = 0; // 清除中断标志
break;
//SMB_MTDBACK SMB0STA =28H)
//主发送器(MT)发送数据字节成功;收到ACK(从机应答)
//检查BYTE_SENT:如果为1,说明刚发出的是存储器地址;为0,刚发出的是数据字节
case SMB_MTDBACK: //0x28
STO = 1;
SM_BUSY=0;
SI = 0;
break;
//SMB_MRADDACK ;(SMB0STA =40H)
//(MR)主收发器发送从地址+R成功;收到 ACK(从机应答).主机发送NACK.
case SMB_MRADDACK: //0x40
AA = 0; // 主机在在回应周期发送NACK
SI = 0;
break;
//SMB_MRDBNACK ;(SMB0STA =58H)
//(MR)主收发器接收数据成功;主机发送NACK
//读操作完成。读数据寄存器并发送STOP
case SMB_MRDBNACK: //0x58
WORD= SMB0DAT;
STO = 1;
SM_BUSY = 0;
AA = 1; // 主机发送ACK,准备下一次传输
SI = 0;
break;
}
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -