S3C2440之IIC裸机驱动

发布时间:2024-07-18  

花了两天的时间终于把这个搞定了,其实I2C的原理还是比较简单的,只是几个细节性的东西还是需要特别的注意,主要是需要注意一下几点:
1.rIICCON &= ~0x10; 清中断必须要在rIICDS = slvAddr; 和rIICSTAT = 0xf0;  // 主设备,启动  之后

2.延时对于写外部的低速设备来说非常重要,比如while(flag)之后一定要加延时,还有在写数据时发现只能写入基数地址的数据,这也是由于延时导致的

3.开始调试的时候系统总是死在read的函数中,后来发现在数据手册的note中说当读取最后一个数据的时候一定不能返回ACK信号,而我却在程序中使用while(flag)来等待ACK引发中断,这不死才怪呢。。。。所以数据手册中的NOTE部分也是特别重要的

4.在真正对AT24C02A进行读取数据时,在发送带有读命令的从设备地址后,AT24C02A会再返回一个从设备地址信息或从设备内存地址信息作为应答,所以一定要把该字节读取后抛弃,因为它不是我们所要读取的信息。

5.下面是核心代码:

#include "def.h"  
#include "2440addr.h"  
#include "I2C.h"  
#include "uart.h"   
extern void Delay(int time);   
int flag;   //用于标识是否收到应答信号,改标识在终端处理程序中被清0   
void Test_Iic(void)   
{   
    unsigned int i,j,save_E,save_PE;   
    static U8 data[256];   
    uart_printf("nIIC Test(Interrupt) using AT24C02n");   
    save_E   = rGPECON;   
    save_PE  = rGPEUP;   
    rGPEUP  |= 0xc000;                  //Pull-up disable   
    rGPECON |= 0xa00000;                //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   
       
    uart_printf("Write test data into AT24C02n");   
    for(i=0;i<256;i++)   
       { Wr24C080(0xa0,(U8)i,i);   
        Delay(1);   //注意这个延时不能少,否则出现有些数据无法写入的问题   
       }       
    for(i=0;i<256;i++)   
        data[0] = 0;   
    uart_printf("Read test data from AT24C02n");   
       
     for(i=0;i<256;i++)   
         Rd24C080(0xa0,(U8)i,&(data[i]));    
    for(i=0;i<16;i++)   
    {   
        for(j=0;j<16;j++)   
            uart_printf("%2x ",data[i*16+j]);   
        uart_printf("n");   
    }   
    rINTMSK |= BIT_IIC;       
    rGPEUP  = save_PE;   
    rGPECON = save_E;   
}   
void Wr24C080(U32 slvAddr, U32 addr, U8 data)   
{   
    flag=1;  //应答标志   
    rIICDS = slvAddr;    
    rIICSTAT = 0xf0;  // 主设备,启动   
        rIICCON &= ~0x10;            //清中断标志 ,特别注意这条语句的位置,不能放到上条的前面     
    while(flag == 1)   //当发送从地址完成之后会收到ACK信号,在中断处理函数中将该标志置为0   
     Delay(1);   
        
       
     flag =1 ; //readly to translate addr   
     rIICDS = addr;    
     rIICCON &= ~0x10;            //清中断标志   
     while(flag == 1)   //当发送从地址完成之后会收到ACK信号,在中断处理函数中将该标志置为0   
     Delay(1);   
        
     flag =1 ; //readly to translate data   
     rIICDS = data;    
     rIICCON &= ~0x10;            //清中断标志   
     while(flag == 1)   //当发送从地址完成之后会收到ACK信号,在中断处理函数中将该标志置为0   
     Delay(1);   
        
    rIICSTAT = 0xd0;                    //Stop MasTx condition    
    rIICCON  = 0xaf;                    //Resumes IIC operation.    
    Delay(1);   
       
}   
  
void Rd24C080(U32 slvAddr, U32 addr, U8 *data)   
{   
    unsigned char temp;   
    flag=1;  //应答标志   
    rIICDS = slvAddr;    
    rIICSTAT = 0xf0;  // 主设备发送模式用来发送slvAddr和addr,,启动   
    rIICCON &= ~0x10;            //清中断标志   
    while(flag == 1)   //当发送从地址完成之后会收到ACK信号,在中断处理函数中将该标志置为0   
     Delay(1);   
        
    flag =1 ; //readly to translate addr   
     rIICDS = addr;    
     rIICCON &= ~0x10;            //清中断标志   
     while(flag == 1)   //当发送从地址完成之后会收到ACK信号,在中断处理函数中将该标志置为0   
     Delay(1);   
           
    flag=1;   
    rIICDS = slvAddr;    
    rIICSTAT = 0xb0;  // 主设备接收模式用来接收数据,启动   
        rIICCON &= ~0x10;            //清中断标志   
    while(flag == 1)   //当发送从地址完成之后会收到ACK信号,在中断处理函数中将该标志置为0   
     Delay(1);   
       
    //注意:读取下面这个字节必须进行,因为在发送带有读命令的从设备地址后,   
    //AT24C02A会再返回一个从设备地址信息或从设备内存地址信息作为应答,所以一定要把该字节读取后抛弃,因为它不是我们所要读取的信息;   
    flag =1 ; //readly to translate addr   
    temp = rIICDS;   // 抛弃第一自己   
    rIICCON &= ~0x10;            //清中断标志   
    while(flag)   
     Delay(1);   
        
    rIICCON = 0x2f;                  //Resumes IIC operation with NOACK.    
    *data = rIICDS;    
    Delay(1);   
     rIICSTAT = 0x90;                    //Stop MasTx condition    
     rIICCON  = 0xaf;                    //Resumes IIC operation.   
           
     Delay(1);   
       
}   
  
//-------------------------------------------------------------------------   
void __irq IicInt(void)   
{   
    
    rSRCPND = BIT_IIC;          //Clear pending bit   
    rINTPND = BIT_IIC;   
    flag = 0;   
      

文章来源于:电子工程世界    原文链接
本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。

我们与500+贴片厂合作,完美满足客户的定制需求。为品牌提供定制化的推广方案、专属产品特色页,多渠道推广,SEM/SEO精准营销以及与公众号的联合推广...详细>>

利用葫芦芯平台的卓越技术服务和新产品推广能力,原厂代理能轻松打入消费物联网(IOT)、信息与通信(ICT)、汽车及新能源汽车、工业自动化及工业物联网、装备及功率电子...详细>>

充分利用其强大的电子元器件采购流量,创新性地为这些物料提供了一个全新的窗口。我们的高效数字营销技术,不仅可以助你轻松识别与连接到需求方,更能够极大地提高“闲置物料”的处理能力,通过葫芦芯平台...详细>>

我们的目标很明确:构建一个全方位的半导体产业生态系统。成为一家全球领先的半导体互联网生态公司。目前,我们已成功打造了智能汽车、智能家居、大健康医疗、机器人和材料等五大生态领域。更为重要的是...详细>>

我们深知加工与定制类服务商的价值和重要性,因此,我们倾力为您提供最顶尖的营销资源。在我们的平台上,您可以直接接触到100万的研发工程师和采购工程师,以及10万的活跃客户群体...详细>>

凭借我们强大的专业流量和尖端的互联网数字营销技术,我们承诺为原厂提供免费的产品资料推广服务。无论是最新的资讯、技术动态还是创新产品,都可以通过我们的平台迅速传达给目标客户...详细>>

我们不止于将线索转化为潜在客户。葫芦芯平台致力于形成业务闭环,从引流、宣传到最终销售,全程跟进,确保每一个potential lead都得到妥善处理,从而大幅提高转化率。不仅如此...详细>>