/*----------------------------------------------------------------------------------------------------------
IIC总线读写EEPROM(串行扩展eeprom,24c02)
(STC12C系列单片机自带eeprom,且有另外的eeprom操作方式)
作者:Allen.H(帮同学修改的一个程序)
时间:2010.11.5
----------------------------------------------------------------------------------------------------------*/
#include #include //这里使用了_nop_()函数,所以调用此头文件 #define TRUE 1/*define宏定义一般用大写,宏定义并不会减少最终代码空间 define多行语句时,每一行末尾写上\\,最后一行可以不写, 有时比较短的语句写成一个子函数会牺牲更多的时间, 因为函数调用耗时比较多,这个时候用一个define语句更好*/ #define FALSE 0 typedef unsigned char uchar;//良好的程序风格,不应该用#define //#define uchar unsigned char sbit sda=P2^0; //---------你把sda和scl引脚可能定反了,我换过来了------------------------------- sbit scl=P2^1;//等号对其,变量名长短不一时,注意,且测试等于号\"==\"或者其他双目关系运算符两边都空一格 //----------------------------------------------------------------- void delay(uchar z)//带参数很好 {//大括号所在行不要写代码 uchar i,j;//局部变量中用来自加自减可以用i,j之类的定义,计数建议不要用i,j //局部变量不占内存,函数调用时生成堆栈,不应该定义局部变量时作初始化 //----局部变量命名后空一格,写正式代码 for(i=z;i>0;i--) for(j=100;j>0;j--);//注明多少时间,在调试模式下,看窗口左边的SEC值 } //函数与函数之间空一格 void delay_7nop()//子程序命名最好顾名思义,比如delay_1ms(),这里考虑都是使用7nop,不带参数 {/*程序代码每进一层逻辑就缩进一格TAB键,TAB设置为3,4格, 在keil的view->options里面设置,不要使用几个空格来缩进,统一使用TAB键*/ _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();//这里0-1000多个_nop_都可以 } //delay函数都放在一起,函数顺序不要乱放,相关的放一起, //-------------------------------------------------------------------- void init() { sda=1; delay_7nop(); scl=1; delay_7nop(); } //---SCL线为高电平期间,SDA线由高电平向低电平的变化表示起始信号; //SCL线为高电平期间,SDA线由低电平向高电平的变化表示终止信号。 //但更具体还是得看时序图,下面就没有都先把scl先拉高再去变sda void start() { sda=1; delay_7nop(); //这里sda和第三行的scl信号哪个放上没什么区别,主要起始和停止信号风格保持一致就行了 scl=1; delay_7nop(); sda=0; delay_7nop(); //scl=0; //允许数据变化,传数据的时候拉低才允许数据变化, //但是在开始信号和停止信号scl都为高,这里看时序图就知道了 } void stop() { sda=0; delay_7nop(); scl=1; delay_7nop(); sda=1; delay_7nop(); } bit ask()//应答信号,return是什么类型这里函数就是什么类型, { //每一个被传送的字节后面都必须跟随一位应答位(即一帧共有9位) //如果一段时间内没有收到从机的应答信号,则自动认为从机已正确接收到数据 bit flag;//真假判断,或只有0/1取值的标志位设置为bit sda=1; scl=1; delay_7nop(); flag=sda; delay_7nop(); scl=0; delay_7nop(); if(flag==1) return FALSE;//非应答 else return TRUE;//应答 } //用下面屏蔽的的应答信号也可以,上面的应答信号考虑更周全 /* void ask() //应答 { uchar i; scl=1; delay_7nop(); while((sda==1)&&(i<250))i++; scl=0; delay_7nop(); } */ //---------------------------------------------------- void writedata(uchar dat)//下面是readdata()和readadd()保持程序风格的一致性,命名不该命为writecurrent {//函数参数不要乱用P,q之类的毫无意义的名字,这里用dat,date是关键字,不 能用 uchar i; // scl=0;//此句可有可无 for(i=0;i<8;i++)//按位写 { dat=dat<<1;//左移一位 scl=0; delay_7nop(); sda=CY;//psw位中的CY进位标识位,左移后最高位移入CY delay_7nop(); scl=1;//scl高电平,数据稳定 delay_7nop(); } scl=0; delay_7nop(); sda=1;//总线释放 delay_7nop(); } void writeadd(uchar add,uchar infor) { start(); writedata(0xa0);//器件地址 ask(); writedata(add);//器件内部存储区的地址 ask(); writedata(infor);//数据 ask(); stop(); } //---------------------------------------------------- uchar readdata() { uchar i,dat; scl=0; delay_7nop(); sda=1;//数据总线释放 delay_7nop(); for(i=0;i<8;i++) { scl=1; delay_7nop(); dat=(dat<<1)|(uchar)sda;//此处的强制类型转换表现思维考虑到了 //dat左移一位,最低位为0,此时与sda按位或运算就把sda数据读到了最低位 scl=0; delay_7nop(); } //密切相关的代码紧接着写,不很相关的空一格再写 return dat; } uchar readadd(uchar add) { uchar r=0;//局部变量小写,全局变量首字母大写 start(); writedata(0xa0); ask(); writedata(add); ask(); start(); writedata(0xa1); ask(); r=readdata(); stop(); return r; } //------------------------------------------------------- void main() { while(1)//在keil的调试仿真窗口(Perpherals->I/O-ports->)看不出P2口的变换, //因为这里是外部EEPROM,要仿真芯片或者硬件的支持才能观察结果,本程序测试无误 { init();//这里初始化一下 writeadd(25,0xaa); delay(50);//此处最少要delay(7); P1=readadd(25);//P1还是P2还是P3主要是看你的硬件用哪个来测试 } } //主函数放最后是省去了函数申明,但在工程应用中建议放在最上面 //这样一眼就能看到该工程是做什么的,且功能函数本身就应该在头文件中作申明 //以便其他点C文件能方便调用,每写一个功能函数都在头文件中作申明,这是一个好习惯 //方便其他点C文件随时调用 //------------------------------------------------------- //总结: // 1.你的程序最初可能把scl和sda可能定反了, // 2.你没写ask函数(屏蔽了),应答信号必须写, // 3.你的代码风格,变量名命名,函数名命名,函数排放顺序,无注释, // 书写排版有很大问题,看你的程序很吃力,且不美观 // 4.能用子函数代替的就写成子函数,用那么多nop看上去代码真丑 // 5.有关读写的4个函数你函数名命名风格没统一 //建议: // 1.看时序图的能力和对IIC总线的理解有待加强 // 2.要慢慢形成规范的代码风格 // 3.keil软件对你还有很大学习空间,要学习用更多的keil调试和用protus仿真 //相关提示: // 1.要学些使用下列对内存和存储的理解: // code :程序存储区(64KB) // data :可直接寻址的内部数据存储区(128B) 默认的变量存储区 // idata:不可直接寻址的内部数据存储区(256B) 当全局变量定义太多的时候 // 使用,比如:idata int 或int idata来使用这个区域存储全局变量 // bdata:可位寻址内部数据存储区(16B) // xdata:外部数据存储区(64KB)当使用来了外部RAM时,应用它,keil中options for target 的memory mode可选 // pdata:分页的外部数据存储区 因篇幅问题不能全部显示,请点此查看更多更全内容