PIC入門7,MAX518的IIC通信實(shí)驗(yàn)
實(shí)驗(yàn)一:正弦波產(chǎn)生
程序:
//本程序?qū)⑼ㄟ^(guò)PIC16F877A的I2C方式驅(qū)動(dòng)D/A轉(zhuǎn)換器MAX518,使其D/A0通道輸出
//一個(gè)連續(xù)的正弦波形(注:本程序并沒(méi)對(duì)正弦波的頻率進(jìn)行控制)
//適合3EPIC實(shí)驗(yàn)板
#include
#include
const char table[ ] = {0X80,0X86,0X8D,0X93,0X99,0X9F,0XA5,0XAB, 0XB1,
0XB7,0XBC,0XC2,0XC7,0XCC,0XD1,0XD6,0XDA,0XDF,0XE3,0XE7,
0XEA,0XEE,0XF1,0XF4,0XF6,0XF8,0XFA,0XFC,0XFD,0XFF,0XFF,
0XFF,0XFF,0XFF,0XFF,0XFE,0XFD,0XFB,0XF9,0XF7,0XF5,0XF2,
0XEF,0XEC,0XE9,0XE5,0XE1,0XDD,0XD8,0XD4,0XCF,0XCA,0XC5,
0XBF,0XBA,0XB4,0XAE,0XA8,0XA2,0X9C,0X96,0X90,0X89,0X83,
0X80,0X79,0X72,0X6C,0X66,0X60,0X5A,0X55,0X4E,0X48,0X43,
0X3D,0X38,0X33,0X2E,0X29,0X25,0X20,0X1C,0X18,0X15,0X11,
0X0E,0X0B,0X09,0X07,0X05,0X03,0X02,0X00,0X00,0X00,0X00,
0X00,0X00,0X01,0X02,0X04,0X06,0X08,0X0A,0X0D,0X10,0X13,
0X16,0X1A,0X1E,0X22,0X27,0X2B,0X30,0X35,0X3A,0X40,0X45,
0X4C,0X51,0X57,0X5D,0X63,0X69,0X6F,0X76,0X7C};
//以上的數(shù)組用于存放正弦表,在定義數(shù)組時(shí),前面應(yīng)該加上 const,
//以使數(shù)組存放于ROM中,而不至于占用太多的RAM
unsigned char i;
unsigned char j;
unsigned char n;
//I2C初始化子程序
void i2cint()
{
SSPCON = 0X08; //初始化SSPCON寄存器
TRISC3 =1; //設(shè)置SCL為輸入口
TRISC4 =1; //設(shè)置SDA為輸入口
SSPSTAT=0X80; //初始化SSPSTAT寄存器
SSPADD=0X02; //設(shè)定I2C時(shí)鐘頻率
SSPCON2=0X00; //初始化SSPCON2寄存器
di(); //關(guān)閉總中斷
SSPIF=0; //清SSP中斷標(biāo)志
SSPEN=1; //SSP模塊使能
}
//I2C總線輸出數(shù)據(jù)子程序
void i2cout()
{
SEN=1; //產(chǎn)生I2C啟動(dòng)信號(hào)
for(n=0;n<2;n++)
continue;//給予一定的延時(shí),保證啟動(dòng)
do {
RSEN=1; //產(chǎn)生I2C啟動(dòng)信號(hào)
}while(SSPIF==0); //如果沒(méi)能啟動(dòng),則反復(fù)啟動(dòng),直到啟動(dòng)為止
SSPIF=0; //SSPIF標(biāo)志清0
SSPBUF=0X58; //I2C總線發(fā)送地址字節(jié)
do {
;
}while(SSPIF==0); //等待地址發(fā)送完畢
SSPIF=0; //SSPIF標(biāo)志清0
SSPBUF=0X01; //I2C總線發(fā)送命令字節(jié)
do {
;
}while(SSPIF==0); //等待命令發(fā)送完畢
SSPIF=0; //SSPIF標(biāo)志清0
SSPBUF=j; //I2C總線發(fā)送數(shù)據(jù)字節(jié)
do {
;
}while(SSPIF==0); //等待數(shù)據(jù)發(fā)送完畢
SSPIF=0; //SSPIF標(biāo)志清0
PEN=1; //產(chǎn)生停止條件
do {
;
}while(SSPIF==0); //等待停止條件產(chǎn)生
SSPIF=0; //SSPIF標(biāo)志清0
}
//主程序
main ()
{
i2cint(); //I2C初始化
while(1)
{
for(i=0;i<=127;++i)
{
j=table[i]; //從數(shù)組中得到需要傳輸?shù)臄?shù)據(jù)量
i2cout(); //利用I2C總線方式送出數(shù)據(jù)
}
}
}
實(shí)驗(yàn)二:?jiǎn)纹瑱C(jī)通過(guò)I2C控制MAX518輸出DA
程序:
/****************************************************************
** 功能描述: 單片機(jī)通過(guò)I2C控制MAX518輸出DA
*************************************************************** */
#include
#include
unsigned int DA_Delay_count=0; //* 間隔一定時(shí)間發(fā)送DA(I2C) ,不使DA過(guò)于頻繁 */
unsigned int DA_data=0; //* DA輸出數(shù)據(jù)(數(shù)字量,0XFF對(duì)應(yīng)5V)*/
/* ****************************************************************
** 函 數(shù) 名: initial()
** 功能描述: 系統(tǒng)初始化子程序,放在程序首部
*************************************************************** */
void initial()
{
INTCON=0x00; //* bit7-bit0:關(guān)總中斷 */
ADCON1=0X07; //* 設(shè)置數(shù)字輸入輸出口 */
PIE1=0; //* PIE1 的中斷不使能 */
PIE2=0; //* PIE2 的中斷不使能 */
}
/* **************************************************************
** 函 數(shù) 名: I2C_Initial()
** 功 能:i2c初始化子程序,I2C用于向MAX518輸出DA
****************************************************************** */
void I2C_Initial()
{
SSPCON=0X08; //允許串行口工作,I2C主控工作方式
TRISC=TRISC|0X08;
TRISC=TRISC|0X10; //定義SCL,SDA
SSPADD=7; //定義波特率,用4MHz時(shí),時(shí)鐘=4M/(4×(7+1))=0.125M
SSPSTAT=0X80; //I2C模式下,關(guān)閉標(biāo)準(zhǔn)速度方式(100K和1MK)的回轉(zhuǎn)率控制
SSPCON2=0; //初始化SSPCON2(該寄存器僅用于I2c方式)
SSPEN=1; //使能串行口(SSP模塊)
}
/* **************************************************************
** 函 數(shù) 名: Wait_Ack()
** 功 能:主機(jī)等待從動(dòng)芯片(MAX518)i2c應(yīng)答子程序
****************************************************************** */
void Wait_Ack()
{
do
{
;
}while(SSPIF==0); // 等待應(yīng)答信號(hào)
SSPIF=0; // 清標(biāo)志
}
/* **************************************************************
** 函 數(shù) 名: I2C_OUT()
** 功 能:主機(jī)向從動(dòng)芯片(MAX518)輸出DA數(shù)據(jù)
** 參 數(shù):DA_Out_data:DA轉(zhuǎn)換數(shù)據(jù),address:DA輸出通道0或1(518只有2個(gè)通道)
****************************************************************** */
void I2C_OUT(char DA_Out_data,char address)
{
unsigned char i=0;
I2C_Initial(); //I2C DA輸出初始化
SEN=1; //啟動(dòng)I2C
for(i=1;i<18;i++)i=i; //延時(shí)
do //啟動(dòng)I2C
{
RSEN=1; //重啟動(dòng)I2C
for(i=0;i<28;i++)i=i; //延時(shí)
if(SSPIF==1)break; //啟動(dòng)成功,繼續(xù)執(zhí)行I2C發(fā)送數(shù)據(jù)程序
else return; //啟動(dòng)不成功,本次退出執(zhí)行I2C程序,防止程序死機(jī)
}while(SSPIF==0);
SSPIF=0; //清標(biāo)志
SSPBUF=0x58; //向DA芯片MAX518寫地址0X58
Wait_Ack(); //等待應(yīng)答
SSPBUF=address; //向MAX518寫通道命令字節(jié)DA0通道或DA1通道
Wait_Ack(); //等待應(yīng)答
SSPBUF=DA_Out_data; //向DA芯片MAX518寫DA轉(zhuǎn)換的輸出數(shù)據(jù)
Wait_Ack(); //等待應(yīng)答
PEN=1; //停止I2C
Wait_Ack(); //等待應(yīng)答
}
main()
{
initial(); //系統(tǒng)初始化子程序
I2C_Initial(); // i2c初始化子程序
while(1)
{
if(DA_Delay_count>=0x2f) // 間隔一定時(shí)間啟動(dòng)DA轉(zhuǎn)換,不使DA轉(zhuǎn)換過(guò)于頻繁
{
DA_Delay_count=0; // DA轉(zhuǎn)換間隔時(shí)間延時(shí)清0
DA_data=0x3f; // D/A0通道轉(zhuǎn)換數(shù)據(jù)為0X3F(1.25V)
I2C_OUT(DA_data,0); // D/A0通道發(fā)DA轉(zhuǎn)換數(shù)據(jù)
DA_data=0xbf; // D/A1通道轉(zhuǎn)換數(shù)據(jù)為0XBF(3.75V)
I2C_OUT(DA_data,1); // D/A1通道發(fā)DA轉(zhuǎn)換數(shù)據(jù)
}
else
DA_Delay_count++; // 不到DA轉(zhuǎn)換間隔時(shí)間,繼續(xù)延時(shí)
}
}