打印

十 ARM9(2440)的IIC——理论知识及程序实例

[复制链接]
4098|1
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
mizhongqin|  楼主 | 2011-9-13 13:26 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
概述
S3C2440A RISC 微处理器可以支持一个多主控IIC 总线串行接口。一条专用串行数据线(SDA)和一条专用
串行时钟线(SCL)传递连接到IIC 总线的总线主控和外设之间的信息。SDA 和SCL 线都为双向的。
多主控IIC 总线模式中,多个S3C2440A RISC 微处理器可以发送或接收串行数据来自或到从设备。主机
S3C2440A 可以通过IIC 总线启动和结束数据传输。S3C2440A 中的IIC 总线是使用标准总线仲裁步骤。
为了控制多主控IIC 总线操作,必须写入值到以下寄存器中:
– 多主控IIC 总线控制寄存器,IICCON
– 多主控IIC 总线控制/状态寄存器,IICSTAT
– 多主控IIC 总线Tx/Rx 数据移位寄存器,IICDS
– 多主控IIC 总线地址寄存器,IICADD
当释放了IIC 总线时,SDA 和SCL 线应该都保持为高电平。一个高到低SDA 的变化可以启动一个起始条件。
SCL 稳定保持在高电平时的一个低到高SDA 的变化可以启动一个停止条件。
起始和停止条件通常由主设备产生。第一个数据字节为7 位地址值,其在启动起始条件后放到总线上,可以确
定出主设备要选择的从设备。第8 位是决定传输方向(读或写)。
每个放到SDA 线上的字节都应该总共为8 位。字节可以在总线传输操作期间无限制的发送或接收。数据通常
从最高有效位(MSB)开始始发送,并且每个字节应该立即通过应答(ACK)位跟上。


I2C总线可构成多主和多从系统。在多主系统结构中,系统通过硬件或软件仲裁获得总线控制使用权。应用系统中I2C总线多采用主从结构,即总线上只有一个主控节点,总线上的其他设备都作为从设备。I2C总线上的设备寻址由器件地址接线决定,并且通过访问地址最低位来控制读/写方向。
目前,通用存储器芯片多为EEPROM,其常用的协议主要有两线串行连接协议(I2C)和
三线串行连接协议。带I2C总线接口的EEPROM有许多型号,其中AT24CXX系列使用十分普遍。产品包括AT24C01、AT24C02、AT24C04、AT24C08、AT24C16等。
AT24系列存储器芯片采用CMOS工艺制造,内置有升压电路,可在单电压供电条件下工作。其标准封装为8DIP封装形式。
各引脚的功能说明如下:
SCL:串行时钟。遵循ISO/IEC7816同步协议,漏极开路,需接上拉电阻。在该引脚的上升沿,系统将数据输人到每个EEPROM器件,在下降沿输出。
SDA:串行数据线。漏极开路,需接上拉电阻。双向串行数据线,漏极开路,可与其他开路器件“线或”。
A0、A1、A2:器件/页面寻址地址输人端。在AT24C01和AT24C02中,引脚被硬连接,其他AT24Cxx均可接寻址地址线。
WP:读/写保护。接低电平时可对整片空间进行读/写,高电平时不能读/写,受保护。
Vcc/GND:5V工作电压。

设备地址(DADDR    AT24C04的器件地址是1010

AT24CXX的数据操作格式
I2C总线中,对AT24C04内部存储单元读/写,除了要给出器件的设备地址(DADDR
外,还须指定读/写的页而地址(PADDR)。两者组成操作地址(OPADDR)如下:
                1010 A2 A1 A0-R/W ,一般A2 A1 A0被硬连接,比如是接地,所以设备地址就是0xa0

下面是我摘自别人的**中关于IIC的描述,写的很好
s3c2440内部有一个IIC总线接口,因此为我们连接带有IIC通信模块的外围设备提供了便利。它具有四种操作模式:主设备发送模式、主设备接收模式、从设备发送模式和从设备接收模式。在这里我们只把s3c2440当做IIC总线的主设备来使用,因此只介绍前两种操作模式。在主设备发送模式下,它的工作流程为:首先配置IIC模式,然后把从设备地址写入接收发送数据移位寄存器IICDS中,再把0xF0写入控制状态寄存器IICSTAT中,这时等待从设备发送应答信号,如果想要继续发送数据,那么在接收到应答信号后,再把待发送的数据写入寄存器IICDS中,清除中断标志后,再次等待应答信号;如果不想再发送数据了,那么把0x90写入寄存器IICSTAT中,清除中断标志并等待停止条件后,即完成了一次主设备的发送。在主设备接收模式下,它的工作流程为:首先配置IIC模式,然后把从设备地址写入接收发送数据移位寄存器IICDS中,再把0xB0写入控制状态寄存器IICSTAT中,这时等待从设备发送应答信号,如果想要接收数据,那么在应答信号后,读取寄存器IICDS,清除中断标志;如果不想接收数据了,那么就向寄存器IICSTAT写入0x90,清除中断标志并等待停止条件后,即完成了一次主设备的接收。在完成上述两个模式时,主要用到了控制寄存器IICCON、控制状态寄存器IICSTAT和发送接收数据移位寄存器IICDS。由于我们只把s3c2440当做主设备来用,并且系统的IIC总线上只有这么一个主设备,因此用来设置从设备地址的地址寄存器IICADD,和用于仲裁总线的多主设备线路控制寄存器IICLC都无需配置。寄存器IICCON的第6位和低4位用于设置IIC的时钟频率,因为IIC的时钟线SCL都是由主设备提供的。s3c2440的IIC时钟源为PCLK,当系统的PCLK为50MHz,而从设备最高需要100kHz时,可以将IICCON的第6位置1,IICCON的低4位全为0即可。寄存器IICCON的第7位用于设置是否发出应答信号,第5位用于是否使能发送和接收中断,第4位用于中断的标志,当接收或发送数据后一定要对该位进行清零,以清除中断标志。寄存器IICSTAT的高2位用于设置是哪种操作模式,当向第5位写0或写1时,则表示结束IIC或开始IIC通讯,第4位用于是否使能接收/发送数据。

       由于通讯是双方的事情,在了解了主设备的操作模式后,还要清楚从设备的运行机制,两者要达到完美地结合,才能实现彼此的通讯。在这里,从设备是EEPROM——AT24C02A,要想让s3c2440能够正确地对AT24C02A读写,就必须让s3c2440的时序完全按照AT24C02A的时序。AT24C02A的写操作有两种模式:字节写和页写。字节写是先接收带有写命令的设备地址信息,如果符合就应答,再接收设备内存地址信息,发出应答后,再接收要写入的数据,这样就完成了字节写过程。页写与字节写的区别就是,页写可以一次写多个数据,而字节写只能一次写一个数据。但由于AT24C02A的一页才8个字节,所以页写也最多写8个数据,而且只能在该页内写,不会发生一次页写同时写两页的情况。AT24C02A的读操作有三种模式:当前地址读、随机读和序列读。当前地址读是只能读取当前地址内的数据,它的时序是先接收带有读命令的设备地址信息,如果符合就应答,然后发送当前地址内的数据,在没有接收从主设备发来的应答信号的情况下终止该次操作。随机读的时序是,连续接收带有写命令的设备地址信息和设备内存地址信息,然后主设备重新开启IIC通信,AT24C02A再次接收到带有读命令的设备地址信息,在发出应答信号以后,发送该内存地址的数据,在没有接收到任何应答信号的情况下结束该次通信。当前地址读和随机读一次都只能读取一个数据,而序列读一次可以读取若干个数据,它的时序就是在当前地址读或随机读发出数据后,接收到了应答信号,那么AT24C02A会把下一个内存地址中的数据送出,除非AT24C02A接收不到任何应答信号,否则它会一直把下一个内存地址中的数据送出。序列读没有一页8个字节的限制。

       在介绍完了s3c2440和AT24C02A的IIC通讯方式后,我们就可以写程序了。在这里,我们用UART来实现PC机对AT24C02A的读写。UART的通讯协议是,PC机先发送命令字节:0xC0表示要向AT24C02A写数据,0xC1表示要读取AT24C02A的数据,在命令字节后,紧跟着的是设备内存地址和写入或读取的字节数。如果是要写EEPROM数据,则在这三个字节后是要写入的数据内容。在UART通讯完毕后,s3c2440会根据命令的不同,写入或读取AT24C02A,如果是读取EEPROM,则s3c2440还会利用UART把读取到的数据上传到PC机。在这个程序中,我们只开启了UART的接收中断,而没有开启发送中断,即让s3c2440主动去完成发送任务。并且在与AT24C02A操作中,我们使用的是页写和序列读的模式,这样可以最大程度的完成一次读或写操作,而且我们所编写的页写和序列读子程度也同样可以实现字节写和随机读的模式。在这里我们限制一次读或写的数据量最多为8个字节。
下面是用FL2440开发板的程序分析:

#include <string.h>
#include "2440addr.h"
#include "def.h"
#include "IIC.h"
static U8 _iicData[IICBUFSIZE];
static volatile int _iicDataCount;//发送计数标志
static volatile int _iicStatus;//IIC状态标志
static volatile int _iicMode;//IIC模式标志
static int _iicPt;
extern void Uart_Printf(char *fmt,...);
extern void Uart_Init(int baud);
void Delay(int x);
//===================================================================
//       SMDK2440 IIC configuration
//  GPE15=IICSDA, GPE14=IICSCL
//  "Interrupt mode" for IIC block
//===================================================================
//******************[ Test_Iic ]**************************************
void Test_Iic(void)
{

    unsigned int i,j,save_E,save_PE;
    static U8 data[512];//256
    //Uart_Init(115200);
    Uart_Printf("\nIIC Test(Interrupt) using AT24C02\n");
    save_E   = rGPECON;
    save_PE  = rGPEUP;
    rGPEUP  |= 0xc000;                  //Pull-up disable
   
    rGPECON &=0xfffffff;
    rGPECON |= 0xa0000000;                //GPE15:IICSDA , GPE14:IICSCL
    pISR_IIC = (unsigned)IicInt;
    rINTMSK &= ~(BIT_IIC);
      //Enable ACK, Prescaler IICCLK=PCLK/16, Enable interrupt, Transmit clock value Tx clock=IICCLK/16
      // If PCLK 50.7MHz, IICCLK = 3.17MHz, Tx Clock = 0.198MHz
    rIICCON = (1<<7) | (0<<6) | (1<<5) | (0xf);
    rIICADD  = 0x10;                    //2440 slave address = [7:1] 实验中没有此语句时也没有影响,因为使用的是主机模式
    rIICSTAT = 0x10;                    //IIC bus data output enable(Rx/Tx)
rIICLC = (1<<2)|(1);      // Filter enable, 15 clocks SDA output delay       added by junon
    rIICDS = 0xDD;
    Uart_Printf("Write test data into AT24C02\n");
    for(i=0;i<256;i++)//256
        Wr24C080(0xa0,(U8)i,i);//向地址0--255写入数据0--255
         
    for(i=0;i<256;i++)//256
        data = 0;//数组清零
    Uart_Printf("Read test data from AT24C02\n");
   
    for(i=0;i<256;i++)
        Rd24C080(0xa0,(U8)i,&(data));//将读取的数据存入data数组
        //Line changed 0 ~ f
    for(i=0;i<16;i++)//打印读取的数据  16
    {
        for(j=0;j<16;j++)
            Uart_Printf("%2x ",data[i*16+j]);
        Uart_Printf("\n");
    }
    rINTMSK |= BIT_IIC;//屏蔽中断   
    rGPEUP  = save_PE;//恢复GPE口
    rGPECON = save_E;
    while(1);
}

//*************************[ Wr24C080 ]****************************
void Wr24C080(U32 slvAddr,U32 addr,U8 data)//首地址   内部地址   数据
{
    _iicMode      = WRDATA;//模式标志记为WRDAT
    _iicPt        = 0;//指针记为0
    _iicData[0]   = (U8)addr;//内部地址
    _iicData[1]   = data;// 要写的数据
    _iicDataCount = 2;
   
    rIICDS   = slvAddr;                 //0xa0设备地址
    rIICSTAT = 0xf0;                    //MasTx,Start 启动发送
      //Clearing the pending bit isn't needed because the pending bit has been cleared.
   
    while(_iicDataCount!=-1);//未发送完在此等待
    _iicMode = POLLACK;//发送完后下面等待应答信号ACK
    while(1)
    {
        rIICDS     = slvAddr;
        _iicStatus = 0x100;
        rIICSTAT   = 0xf0;              //MasTx,Start启动
        rIICCON    = 0xaf;              //Resumes IIC operation. 恢复IIC总线
         
        while(_iicStatus==0x100);//未接收到ACk在此等待
         
        if(!(_iicStatus&0x1))
            break;                      //When ACK is received    ACK收到后跳出循环
    }
    rIICSTAT = 0xd0;                    //Stop MasTx condition 停止信号
    rIICCON  = 0xaf;                    //Resumes IIC operation.
    Delay(1);                           //Wait until stop condtion is in effect.恢复
       //Write is completed.
}
      
//**********************[ Rd24C080 ] ***********************************
void Rd24C080(U32 slvAddr,U32 addr,U8 *data)//首地址 内部地址读取的数据存入的地址
{
    _iicMode      = SETRDADDR;//模式设为SETRDADDR
    _iicPt        = 0;
    _iicData[0]   = (U8)addr;//_iicData[0]存内部地址
    _iicDataCount = 1;//计数值
    rIICDS   = slvAddr;//首地址0xa0
    rIICSTAT = 0xf0;                    //MasTx,Start  启动
      //Clearing the pending bit isn't needed because the pending bit has been cleared.
    while(_iicDataCount!=-1);//未读取完成
    _iicMode      = RDDATA;//完成后进入读取数据模式
    _iicPt        = 0;
    _iicDataCount = 1;
   
    rIICDS        = slvAddr;
    rIICSTAT      = 0xb0;               //MasRx,Start设置为主接收模式
    rIICCON       = 0xaf;               //Resumes IIC operation.恢复IIC总线操作  
    while(_iicDataCount!=-1);//读取未完成
    *data = _iicData[1];//1
}

//-------------------------------------------------------------------------
void __irq IicInt(void)
{
    U32 iicSt,i;
   
    rSRCPND = BIT_IIC;          //Clear pending bit
    rINTPND = BIT_IIC;
    iicSt   = rIICSTAT;//控制状态寄存器
   
    if(iicSt & 0x8){}           //When bus arbitration is failed.总线仲裁失败执行空操作
    if(iicSt & 0x4){}           //When a slave address is matched with IICADD从地址匹配执行空操作
    if(iicSt & 0x2){}           //When a slave address is 0000000b
    if(iicSt & 0x1){}           //When ACK isn't received  未收到应答信号执行空操作
    switch(_iicMode)
    {
       case POLLACK:
           _iicStatus = iicSt;//控制状态寄存器的值赋给_iicStatus
           break;
       case RDDATA:
           if((_iicDataCount--)==0)
           {
               _iicData[_iicPt++] = rIICDS;
           
               rIICSTAT = 0x90;                 //Stop MasRx condition
               rIICCON  = 0xaf;                 //Resumes IIC operation.
               Delay(1);                        //Wait until stop condtion is in effect.
                                                //Too long time...
                                                //The pending bit will not be set after issuing stop condition.
               break;   
           }     
           _iicData[_iicPt++] = rIICDS;         //The last data has to be read with no ack.
           if((_iicDataCount)==0)
               rIICCON = 0x2f;                  //Resumes IIC operation with NOACK.  第一次不产生应答信号进行第二次读取并将第二次读取的
                                                 ///////////////数据存入数组中(第一次和第二次读取的数据实际一样)
           else
               rIICCON = 0xaf;                  //Resumes IIC operation with ACK
               break;
        case WRDATA:
            if((_iicDataCount--)==0)//首地址 ,内部地址和数据都发送完毕后
            {
                rIICSTAT = 0xd0;                //Stop MasTx condition 产生停止信号
                rIICCON  = 0xaf;                //Resumes IIC operation.恢复IIC总线
                Delay(1);                       //Wait until stop condtion is in effect.
                       //The pending bit will not be set after issuing stop condition.
                break;   
            }
            rIICDS = _iicData[_iicPt++];        //_iicData[0] has dummy.第一次中断后将内部地址addr的值存入IICDS  第二次存入要写入的数据
            for(i=0;i<10;i++);                  //for setup time until rising edge of IICSCL//延时
            
            rIICCON = 0xaf;                     //resumes IIC operation.恢复IIC总线
            break;
        case SETRDADDR:
//          Uart_Printf("[ S%d ]",_iicDataCount);
            if((_iicDataCount--)==0)
                break;//第二次中断跳出                          //IIC operation is stopped because of IICCON[4]   
            rIICDS = _iicData[_iicPt++];//第一次中断将内部地址addr存入移位寄存器
            for(i=0;i<10;i++);                  //For setup time until rising edge of IICSCL
            rIICCON = 0xaf;                     //Resumes IIC operation.
            break;
        default:
            break;     
    }
}
void Delay(int x)
{
  int k, j;
  while(x)
  {
   for (k=0;k<=0xff;k++)
    for(j=0;j<=0xff;j++);
   
   x--;
  }
}

相关帖子

沙发
aibaolan| | 2011-9-18 10:01 | 只看该作者

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

0

主题

67

帖子

1

粉丝