標(biāo)題: 這個iic讀寫程序是不是有問題,求大神指教 [打印本頁]

作者: xxxxx111    時間: 2015-5-27 10:22
標(biāo)題: 這個iic讀寫程序是不是有問題,求大神指教
void start()
{
sda=1;
scl=1;
delay4us();
sda=0;
delay4us();
scl=0;
}
void stop()
{
sda=0;
scl=1;
delay4us();
sda=1;
delay4us();
scl=0;
}
void init()//初始化
{
sda=1;
delay();
scl=1;
delay();
}

void ack()
{
sda=0;
scl=1;
delay4us();
scl=0;
sda=1;
}
void noack()
{
sda=1;
scl=1;
delay4us();
scl=0;
sda=0;
}
uchar recbyte()
{
uchar i,rd;
rd=0x00;
sda=1;
for(i=0;i<8;i++)
{
scl=1;
rd<<=1;
rd|=sda;
delay4us();
scl=0;
delay4us();
}
scl=0;
delay4us();
return rd;
}
uchar sendbyte(uchar wd)
{
uchar i;
bit ack0;
for(i=0;i<8;i++)
{
sda=(bit)(wd&0x80);
_nop_();
_nop_();
scl=1;
delay4us();
scl=0;
wd<<=1;
}
delay4us();
sda=1;
scl=1;
delay4us();
ack0=!sda;
scl=0;
delay4us();
return ack0;
}
uchar Recstring(uchar slave,uchar subaddr,uchar *buffer,uchar n)
{
uchar i;
start();
if(!sendbyte(slave)) return 0;
if(!sendbyte(subaddr)) return 0;
start();
if(!sendbyte(slave+1)) return 0;
for(i=0;i<n-1;i++)
  {
  buffer[i]=recbyte();
ack();
  }
buffer[n-1]=recbyte();
noack();
stop();
return 1;
}

uchar Sendstring(uchar slave,uchar subaddr,uchar *buffer,uchar n)
{
uchar i;
start();
if(!sendbyte(slave)) return 0;
if(!sendbyte(subaddr)) return 0;
for(i=0;i<n;i++)
{
if(!sendbyte(buffer[i])) return 0;
}
stop();
return 1;
}


作者: admin    時間: 2015-5-27 10:36
這個程序現(xiàn)在有什么現(xiàn)象可以通訊嗎?
作者: xxxxx111    時間: 2015-5-27 12:06
是密碼鎖發(fā)送原始密碼到緩沖區(qū)然后在從緩沖區(qū)讀出來,但是不成功密碼鎖的開機(jī)密碼是6個0而不是之前設(shè)置的123456
作者: yesonjob    時間: 2015-5-28 20:39
假如每次開機(jī)密碼讀到都是6個0,那可能就是沒保存進(jìn)去。
還有,每次讀寫之后,要延時大概10毫秒那樣子再作讀寫,給點(diǎn)時間ic去反應(yīng)。
作者: yesonjob    時間: 2015-5-28 20:43
使用at24c系列的ic嗎?我手頭上在做一個小玩意有相關(guān)i2c協(xié)議可用的程序,有需要可以貼出來供參考下。
作者: xxxxx111    時間: 2015-5-29 14:08
yesonjob 發(fā)表于 2015-5-28 20:43
使用at24c系列的ic嗎?我手頭上在做一個小玩意有相關(guān)i2c協(xié)議可用的程序,有需要可以貼出來供參考下。

嗯是AT24c02芯片的iic,謝謝啦
作者: xxxxx111    時間: 2015-5-29 15:02
yesonjob 發(fā)表于 2015-5-28 20:43
使用at24c系列的ic嗎?我手頭上在做一個小玩意有相關(guān)i2c協(xié)議可用的程序,有需要可以貼出來供參考下。

是的,我用的就是AT24c02的
作者: yesonjob    時間: 2015-5-29 15:51
//**************************************//I2C通信協(xié)議
void nop()
{
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        _nop_();

}


void start()//開始信號
{
        sda=1;
        nop();
        scl=1;
        nop();
        sda=0;
        nop();
}

void stop()//停止
{
        sda=0;
        nop();
        scl=1;
        nop();
        sda=1;
        nop();
}

void respons()//應(yīng)答
{
        unsigned char i;
        scl=1;
        nop();
        while((sda==1)&&(i<250))
        {
                i++;
        }
        scl=0;
        nop();
}

void init()//初始化函數(shù)
{
        sda=1;
        scl=1;
}

void write_byte(unsigned char dat)//寫一字節(jié)
{
        unsigned char i;
        scl=0;
        for(i=0;i<8;i++)
        {
                dat=dat<<1;
                sda=CY;
                scl=1;
                nop();
                scl=0;
                nop();
        }
}

unsigned char read_byte()//讀取一字節(jié)
{
        unsigned char i,k;
        for(i=0;i<8;i++)
        {
                scl=1;
                nop();
                k=(k<<1)|sda;
                scl=0;
                nop();
        }
        return k;
}

void write(unsigned char address,unsigned char dat)//向AT24C02地址中寫數(shù)據(jù)
{
        start();
        write_byte(0xa0);
        respons();
        write_byte(address);
        respons();
        write_byte(dat);
        respons();
        stop();
}

unsigned char read(unsigned char address)//從AT24C02地址中讀出數(shù)據(jù)
{
        unsigned char dat;
        start();
        write_byte(0xa0);
        respons();
        write_byte(address);
        respons();
        start();
        write_byte(0xa1);
        respons();
        dat=read_byte();
        stop();
        return dat;
}
//**************************************//

你直接調(diào)用以上的函數(shù)就可以了。其中at24c02的地址線全部接地,所以器件尋址時a0是寫數(shù)據(jù),a1是讀數(shù)據(jù),具體根據(jù)自己接線修改。
例如read(0x00)就是從0x00地址上讀取數(shù)據(jù)。
write(0x00,0xff)就是把數(shù)據(jù)0xff寫入地址0x00中。
記住讀寫一次之后假如想繼續(xù)讀寫,要延時10ms左右再讀寫。

作者: xxxxx111    時間: 2015-5-29 20:34
yesonjob 發(fā)表于 2015-5-29 15:51
//**************************************//I2C通信協(xié)議
void nop()
{

如果要是想把密碼123456寫進(jìn)去呢
作者: 天空的顏色    時間: 2015-5-29 23:41
怎么不用硬件的IIC
作者: yesonjob    時間: 2015-5-30 08:12
xxxxx111 發(fā)表于 2015-5-29 20:34
如果要是想把密碼123456寫進(jìn)去呢

write(0x00,0x01)就是把1這個密碼存進(jìn)0x00這個地址之中;
然后延時10毫秒
write(0x01,0x02)就是把2這個密碼存進(jìn)0x01這個地址之中;
然后延時10毫秒
以此類推,每個地址只能存一個數(shù)。
讀的時候就是
read(0x00) 把0x00地址的數(shù)(也就是1這個密碼)讀出來;
然后延時10毫秒
以此類推。。。。

作者: xxxxx111    時間: 2015-5-30 10:18
yesonjob 發(fā)表于 2015-5-30 08:12
write(0x00,0x01)就是把1這個密碼存進(jìn)0x00這個地址之中;
然后延時10毫秒
write(0x01,0x02)就是把2這 ...

太謝謝啦

作者: xxxxx111    時間: 2015-5-31 21:46
xxxxx111 發(fā)表于 2015-5-30 10:18
太謝謝啦

你看一下我這樣寫對嗎照著你那個改了以后為什么密碼不是000000了但是也不是123456開鎖不成功啊
uchar code table2[]={123456};
write(0x00,table2[0]);
         delayms(10);
         write(0x01,table2[1]);
         delayms(10);
         write(0x02,table2[2]);
         delayms(10);
         write(0x03,table2[3]);
         delayms(10);
         write(0x04,table2[4]);
         delayms(10);
         write(0x05,table2[5]);
         delayms(10);
         buffer[0]=read(0x00);
         delayms(10);
         buffer[1]=read(0x01);
         delayms(10);
         buffer[2]=read(0x02);
         delayms(10);
         buffer[3]=read(0x03);
         delayms(10);
         buffer[4]=read(0x04);
         delayms(10);
         buffer[5]=read(0x05);
         delayms(10);

case 10: //按A鍵開鎖
                                                  if(cmp(buffer,Userpassword)==0)
                                                        flag=1;
                                                        else flag=0;
                        if (flag==1)
                        { flag=0;
                                         i=0;
                          jdq=0;  //點(diǎn)亮LED
                           clear_password();
                           Display_String("OPEN    OK!      ",0xc0);
                           IS_valid_user = 1;
                                                   j=0;
                                                  
                        }

作者: yesonjob    時間: 2015-6-1 08:13
假如套用之前的程序,i2c器件的硬件電路必須要這樣連接。
有完整的程序嗎,只有一部分不好理解。

QQ圖片20150601081225.png (32.5 KB, 下載次數(shù): 188)

QQ圖片20150601081225.png

作者: xxxxx111    時間: 2015-6-1 10:02
yesonjob 發(fā)表于 2015-6-1 08:13
假如套用之前的程序,i2c器件的硬件電路必須要這樣連接。
有完整的程序嗎,只有一部分不好理解。

我就是這樣接的只不過我接的5k電阻
#include<reg51.h>
#include<string.h>
#include<intrins.h>
#define uchar unsigned char
#define uint unsigned int
#define     LCDIO      P0
//#define delay4us() _nop_();_nop_();_nop_();_nop_();
uchar buffer[6]={0};
sbit sda=P3^7;
sbit scl=P3^6;
sbit beep=P2^7;
sbit jdq=P2^6;
bit flag=0,aa;                                //用戶蹲淵義定時溢出標(biāo)志位
uchar  DSY_BUFFER[16]="                ";
uchar  DSY_BUFFER1[16]="                ";
uchar  Userpassword[6]={0};
sbit rs=P2^0;  
sbit rd=P2^1;
sbit lcden=P2^2;

uchar code table2[]="123456";
uchar code table[]="Your Password...";
void delayms(uint z)
{
        uint x,y;
        for(x=z;x>0;x--)
                for(y=110;y>0;y--);
}

void write_com(uchar com)
{
        rs=0;
        rd=0;
        lcden=0;
        P0=com;
        delayms(5);
        lcden=1;
        delayms(5);
        lcden=0;       
}

void write_date(uchar date)
{
        rs=1;
        rd=0;
        lcden=0;
        P0=date;
        delayms(5);
        lcden=1;
        delayms(5);
        lcden=0;       
}
void Display_String(uchar *p,uchar com)
{  uchar i;
   write_com(com);
   for(i=0;i<16;i++)
   {
   write_date(p);
        }
}
void init_lcd()
{
        lcden=0;
        write_com(0x38);
        write_com(0x0c);
        write_com(0x06);
        write_com(0x01);
        write_com(0x80);
    Display_String(table,0x80);
        Display_String("It is Locked!         ",0xc0);
}


//**************************************//I2C通信協(xié)議
void nop()
{
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        _nop_();

}
  void start()//開始信號
{
        sda=1;
        nop();
        scl=1;
        nop();
        sda=0;
        nop();
}
  void stop()//停止
{
        sda=0;
        nop();
        scl=1;
        nop();
        sda=1;
        nop();
}
void respons()//應(yīng)答
{
        unsigned char i;
        scl=1;
        nop();
        while((sda==1)&&(i<250))
        {
                i++;
        }
        scl=0;
        nop();
}
  void init()//初始化函數(shù)
{
        sda=1;
        scl=1;
}

void write_byte(unsigned char dat)//寫一字節(jié)
{
        unsigned char i;
        scl=0;
        for(i=0;i<8;i++)
        {
                dat=dat<<1;
                sda=CY;
                scl=1;
                nop();
                scl=0;
                nop();
        }
}

unsigned char read_byte()//讀取一字節(jié)
{
        unsigned char i,k;
        for(i=0;i<8;i++)
        {
                scl=1;
                nop();
                k=(k<<1)|sda;
                scl=0;
                nop();
        }
        return k;
}

void write(unsigned char address,unsigned char dat)//向AT24C02地址中寫數(shù)據(jù)
{
        start();
        write_byte(0xa0);
        respons();
        write_byte(address);
        respons();
        write_byte(dat);
        respons();
        stop();
}

unsigned char read(unsigned char address)//從AT24C02地址中讀出數(shù)據(jù)
{
        unsigned char dat;
        start();
        write_byte(0xa0);
        respons();
        write_byte(address);
        respons();
        start();
        write_byte(0xa1);
        respons();
        dat=read_byte();
        stop();
        return dat;
}
//**************************************//

void clear_password()
{        uchar i;
        for(i=0;i<6;i++)
        {
         Userpassword=' ';
        }
        for(i=0;i<16;i++)
        {
           DSY_BUFFER=' ';         
         }
}
uchar Keys_Scan()
{       
        uchar temp,keynum;
        P1=0x0F;
        delayms(5);
        temp=P1^0x0F;
        switch(temp)
        {
                case 1:keynum=0;break;
                case 2:keynum=1;break;
                case 4:keynum=2;break;
                case 8:keynum=3;break;
                break;
        }
        P1=0xF0;
        delayms(5);
        temp=P1>>4^0x0F;
        switch(temp)
        {
                case 1:keynum+=0;break;
                case 2:keynum+=4;break;
                case 4:keynum+=8;break;
                case 8:keynum+=12;break;
                break;
        }
        delayms(600);
   return keynum;
}

void main()
{   uchar temp,i=0,j=0,k=0,n;
        uchar IS_valid_user;
        beep=1;
        init();                  
    init_lcd();
        delayms(5);
         write(0x00,table2[0]);
         delayms(10);
         write(0x01,table2[1]);
         delayms(10);
         write(0x02,table2[2]);
         delayms(10);
         write(0x03,table2[3]);
         delayms(10);
         write(0x04,table2[4]);
         delayms(10);
         write(0x05,table2[5]);
         delayms(10);
         buffer[0]=read(0x00);
         delayms(10);
         buffer[1]=read(0x01);
         delayms(10);
         buffer[2]=read(0x02);
         delayms(10);
         buffer[3]=read(0x03);
         delayms(10);
         buffer[4]=read(0x04);
         delayms(10);
         buffer[5]=read(0x05);
         delayms(10);
         P1=0x0f;          
    while(1)
                {
                 
                  if(P1!=0x0f)
                        {       
                            temp=Keys_Scan();
                            switch(temp)
                 {
                       case 0:  case 1: case 2: case 3: case 4:
                       case 5:  case 6: case 7: case 8: case 9:
                        
                       if (i<=5)             //密碼限制在6位以內(nèi)
                       {
                        Userpassword=temp;
                        DSY_BUFFER='*';
                        Display_String(DSY_BUFFER,0xc0);          
                                                i++;
                         }                                                               
                        break;
               
                        case 10: //按A鍵開鎖
                                    for(k=0;k<=5;k++)
                                                {        flag=flag&&(buffer[k]==Userpassword[k]);}
                                                  
                        if (flag)
                        { flag=0;
                                         i=0;
                          jdq=0;  //點(diǎn)亮LED
                           clear_password();
                           Display_String("OPEN    OK!      ",0xc0);
                           IS_valid_user = 1;
                                                   j=0;
                                                  
                        }
                       else
                        {
                                                 j++;
                         jdq=1;   //關(guān)閉LED
                         clear_password();
                         Display_String("ERROR!Have try   ",0xc0);
                                                 write_com(0xcf);
                                                 write_date(0x30+j);
                         IS_valid_user=0;
                         }
                        i=0;
                        break;
                    
                      case 11: //按B鍵上鎖
                      jdq=1;   
                      clear_password();
                      Display_String(table,0x80);
                      Display_String("Lock OK!         ",0xc0);
                      i=0;                        
                      IS_valid_user=0;               
                      break;
                                   
                      case 12: //按C鍵設(shè)置新密碼
                       //如果是合法用戶則提示輸入新密碼               
                      if ( !IS_valid_user)
                                          {
                                           i=0;
                                           Display_String("No rights !      ",0xc0);
                                           delayms(1000);
                                           Display_String("Your Password...",0x80);
                                           Display_String("Lock OK!         ",0xc0);
                                           }
                       else
                        {
                          i=0;
                          Display_String("New Password:   ",0x80);
                          Display_String("                ",0xc0);
                         }
           
                       break;                        
                    
                      case 13: //按D鍵保存新密碼
                      if ( !IS_valid_user)
                                          {         i=0;
                                             Display_String("No rights !       ",0xc0);
                                                    delayms(1000);
                                                 Display_String("Your Password...",0x80);
                                                 Display_String("Lock OK!         ",0xc0);
                                           }
                       else
                      {i = 0;       
                                           init();
                                           delayms(5);
                                           for(k=0;k<6;k++)
                                           {
                                                   Userpassword[k]=Userpassword[k];
                                           }
                          write(0x00,Userpassword[0]);
                                          delayms(10);
                                          write(0x01,Userpassword[1]);
                                          delayms(10);
                                          write(0x02,Userpassword[2]);
                                          delayms(10);
                                          write(0x03,Userpassword[3]);
                                          delayms(10);
                                          write(0x04,Userpassword[4]);
                                         delayms(10);
                                         write(0x05,Userpassword[5]);
                                         delayms(10);
                                         buffer[0]=read(0x00);
                                         delayms(10);
                                         buffer[1]=read(0x01);
                                         delayms(10);
                                         buffer[2]=read(0x02);
                                         delayms(10);
                                         buffer[3]=read(0x03);
                                         delayms(10);
                                         buffer[4]=read(0x04);
                                         delayms(10);
                                         buffer[5]=read(0x05);
                                         delayms(10);
                       clear_password();                        
                       Display_String(table,0x00);
                       Display_String("Password Saved!   ",0xc0);
                                           delayms(1000);

                                           Display_String("Do lock agian ?   ",0xc0);
                      }               
                      break;                        
      
                      case 14: //按E鍵消除所有輸入
                      i=0;
                      clear_password();
                      Display_String("                ",0xc0);            
                                          break;

                                          case 15:         //清除一位
                                           if(i!=0)i--;
                                           for(n=0;n<i;n++)
                                           {
                                                   DSY_BUFFER1[n]='*';       
                                           }
                                           Display_String(DSY_BUFFER1,0xc0);          
                                          
                                          }
                                          P1=0x0f;
                        }
                        if(j==3)
                        {        Display_String("THIEF!!!THIEF!!!",0xc0);
                                j=0;
                                beep=0;
                        }
                }
}
這是全部程序,改完了以后輸入123456也不對輸入000000也不對了
作者: yesonjob    時間: 2015-6-2 20:54
程序有點(diǎn)長要花點(diǎn)時間才能看懂。
現(xiàn)在可以試試這樣,把一個數(shù)寫進(jìn)at24c02,然后再讀出來,看看讀出來的數(shù)是不是跟寫進(jìn)去的數(shù)一樣,
可以先排除是不是i2c器件的問題。
作者: xxxxx111    時間: 2015-6-2 22:06
yesonjob 發(fā)表于 2015-6-2 20:54
程序有點(diǎn)長要花點(diǎn)時間才能看懂。
現(xiàn)在可以試試這樣,把一個數(shù)寫進(jìn)at24c02,然后再讀出來,看看讀出來的數(shù) ...

我又換了一個新的AT24C02器件還是不行感覺還是接收不到數(shù)據(jù)




歡迎光臨 (http://www.torrancerestoration.com/bbs/) Powered by Discuz! X3.1