51系列單片機(jī)之I2C協(xié)議
/*----------------------------------------------------
名稱:IIC協(xié)議 PCF8591AD/DA轉(zhuǎn)換
編寫:付新
日期:2012/5/9
平臺:Keil 4, Ly-51S學(xué)習(xí)板
引腳定義如下:
與51連接:
內(nèi)容:函數(shù)是采用軟件延時(shí)的方法產(chǎn)生SCL脈沖,固對高晶振頻率要作 一定的修改....(本例是1us機(jī)器周期,即晶振頻率要小于12MHZ)
-----------------------------------------------------*/
#include
#include
#include "uart.h"
#define AddWr 0x90 //寫數(shù)據(jù)地址
#define AddRd 0x91 //讀數(shù)據(jù)地址
sbit SDA=P2^1;
sbit SCL=P2^0;
bit ack; //應(yīng)答標(biāo)志位
/*啟動(dòng)總線*/
void I2C_start() {
SDA=1; //發(fā)送起始條件的數(shù)據(jù)信號
_nop_();
SCL=1;
_nop_(); //起始條件建立時(shí)間大于4.7us,延時(shí)
_nop_();
_nop_();
_nop_();
_nop_();
SDA=0; //發(fā)送起始信號
_nop_(); //起始條件鎖定時(shí)間大于4μ
_nop_();
_nop_();
_nop_();
_nop_();
SCL=0; //鉗住I2C總線,準(zhǔn)備發(fā)送或接收數(shù)據(jù)
_nop_();
_nop_();
}
/*結(jié)束總線*/
void I2C_stop() {
SDA=0; //發(fā)送結(jié)束條件的數(shù)據(jù)信號
_nop_(); //發(fā)送結(jié)束條件的時(shí)鐘信號
SCL=1; //結(jié)束條件建立時(shí)間大于4μ
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
SDA=1; //發(fā)送I2C總線結(jié)束信號
_nop_();
_nop_();
_nop_();
_nop_();
}
/*字節(jié)數(shù)據(jù)傳送函數(shù)
功能:將數(shù)據(jù)c發(fā)送出去,可以是地址,也可以是數(shù)據(jù),發(fā)完后等待應(yīng)答,并對此狀態(tài)位進(jìn)行操作.
(不應(yīng)答或非應(yīng)答都使ack=0 假)
發(fā)送數(shù)據(jù)正常,ack=1; ack=0表示被控器無應(yīng)答或損壞。
*/
void I2C_send_byte(unsigned char c) {
unsigned char n;
for(n = 0; n < 8; n++) { //要傳送的數(shù)據(jù)長度為8
if((c << n)&0x80) SDA=1; //判斷發(fā)送位
else SDA=0;
_nop_();
SCL=1; //置時(shí)鐘線為高,通知被控器開始接收數(shù)據(jù)位
_nop_();
_nop_(); //保證時(shí)鐘高電平周期大于4μ
_nop_();
_nop_();
_nop_();
SCL=0;
}
_nop_();
_nop_();
SDA=1; //8位發(fā)送完后釋放數(shù)據(jù)線,準(zhǔn)備接收應(yīng)答位
_nop_();
_nop_();
SCL=1;
_nop_();
_nop_();
_nop_();
if(SDA == 1)ack=0;
else ack=1; //判斷是否接收到應(yīng)答信號
SCL=0;
_nop_();
_nop_();
}
/*字節(jié)數(shù)據(jù)傳送函數(shù)
功能: 用來接收從器件傳來的數(shù)據(jù),并判斷總線錯(cuò)誤(不發(fā)應(yīng)答信號),發(fā)完后請用應(yīng)答函數(shù)。
*/
unsigned char I2C_rcv_byte() {
unsigned char dat = 0;
unsigned char n;
SDA=1; //置數(shù)據(jù)線為輸入方式
for(n = 0; n < 8; n++) {
_nop_();
SCL=0; //置時(shí)鐘線為低,準(zhǔn)備接收數(shù)據(jù)位
_nop_();
_nop_(); //時(shí)鐘低電平周期大于4.7us
_nop_();
_nop_();
_nop_();
SCL=1; //置時(shí)鐘線為高使數(shù)據(jù)線上數(shù)據(jù)有效
_nop_();
_nop_();
dat = dat<<1;
if(SDA==1)dat = dat + 1; //讀數(shù)據(jù)位,接收的數(shù)據(jù)位放入retc中
_nop_();
_nop_();
}
SCL=0;
_nop_();
_nop_();
return dat;
}
/*應(yīng)答函數(shù)*/
/*void I2C_ack() {
SDA=0;
_nop_();
_nop_();
_nop_();
SCL=1;
_nop_();
_nop_(); //時(shí)鐘低電平周期大于4μ
_nop_();
_nop_();
_nop_();
SCL=0; //清時(shí)鐘線,鉗住I2C總線以便繼續(xù)接收
_nop_();
_nop_();
}*/
/*非應(yīng)答子函數(shù)*/
void I2C_noack() {
SDA=1;
_nop_();
_nop_();
_nop_();
SCL=1;
_nop_();
_nop_(); //時(shí)鐘低電平周期大于4μ
_nop_();
_nop_();
_nop_();
SCL=0; //清時(shí)鐘線,鉗住I2C總線以便繼續(xù)接收
_nop_();
_nop_();
}
/*讀AD轉(zhuǎn)值程序
輸入?yún)?shù) Chl 表示需要轉(zhuǎn)換的通道,范圍從0-3 返回值范圍0-255
*/
unsigned char read_AD(unsigned char channel)
{
unsigned char val;
I2C_start(); //啟動(dòng)總線
I2C_send_byte(AddWr); //發(fā)送器件地址
if(ack==0) return 0;
I2C_send_byte(0x40 | channel); //發(fā)送器件子地址
if(ack==0)return 0;
I2C_start();
I2C_send_byte(AddWr+1);
if(ack==0)return 0;
val=I2C_rcv_byte();
I2C_noack(); //發(fā)送非應(yīng)位
I2C_stop(); //結(jié)束總線
return val;
}
void main() {
unsigned char num=0;
UART_init();
while(1) {
num = read_AD(0);
UART_send_byte(num);
}
}