花了两天的时间终于把这个搞定了,其实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 AT24C02\n");   
    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 AT24C02\n");   
    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 AT24C02\n");   
       
     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;   
      
}

原文链接:http://www.linuxidc.com/Linux/2011-06/37583p4.htm

S3C2440之IIC裸机驱动的更多相关文章

  1. Exynos4412 IIC总线驱动开发(一)—— IIC 基础概念及驱动架构分析

    关于Exynos4412 IIC 裸机开发请看 :Exynos4412 裸机开发 —— IIC总线 ,下面回顾下 IIC 基础概念 一.IIC 基础概念 IIC(Inter-Integrated Ci ...

  2. JZ2440 裸机驱动 第12章 I2C接口

    本章目标: 了解I2C总线协议: 掌握S3C2410/S3C2440中I2C接口的使用方法: 12.1 I2C总线协议及硬件介绍 12.1.1 I2C总线协议 1 I2C总线的概念 2 I2C总线的信 ...

  3. Tiny6410之按键裸机驱动

    操作步骤: 第一步:查看开发板电路原理图 找到LED 和按键的管脚所对应的寄存器 LED:(见Tiny6410之LED裸机驱动) nLED_1 - GPK4 nLED_2 - GPK5 nLED_3 ...

  4. 基于S3C2440的嵌入式Linux驱动——看门狗(watchdog)驱动解读

    本文将介绍看门狗驱动的实现. 目标平台:TQ2440 CPU:s3c2440 内核版本:2.6.30 1. 看门狗概述 看门狗其实就是一个定时器,当该定时器溢出前必须对看门狗进行"喂狗“,如 ...

  5. I2C裸机驱动程序设计

    ① I2C(Inter-Integrated Circuit)总线是由飞利浦公司开发的两线式串行总线,用于连接微控制器及其外围设备 ② I2C总线有两根双向信号线 (1)SDA:Serial Data ...

  6. s3c2440的IIC控制

    在tq2440和mini2440上都连接着EEPROM 它们作用也不过測试I2C总线能否用. 当中在mini2440上EEPROM型号是 AT24C08,在tq2440上这个型号是 AT24C02A. ...

  7. S3C2440—1.熟悉裸机开发板

    文章目录 一.板载资源介绍 二.安装驱动及上位机 1.USB的驱动及上位机 2.eop驱动安装 3.安装烧录软件oflash 三.烧写开发板 1.预备知识 2.烧写裸板 3.使用u-boot烧写程序 ...

  8. S3C2440上RTC时钟驱动开发实例讲解(转载)

    嵌入式Linux之我行,主要讲述和总结了本人在学习嵌入式linux中的每个步骤.一为总结经验,二希望能给想入门嵌入式Linux的朋友提供方便.如有错误之处,谢请指正. 共享资源,欢迎转载:http:/ ...

  9. 基于KEIL4开发ARM9(S3C2440)的裸机程序

    本文主要介绍如何使用Keil4开发ARM9(S3C2440)裸机程序. 说明: 一.平台: 操作系统:Windows XP系统 KEIL版本:4.73 开发板:ARM9(S3C2440) 二.建立工程 ...

随机推荐

  1. Hack 语言学习/参考---1.3 Summary

    Summary Hack provides the following, non-exhaustive list of features: Ability to annotate function a ...

  2. 使用C语言编写windows服务一般框架

    原文:使用C语言编写windows服务一般框架 编写windows服务和编写windows应用程序一样,有一些回调函数必须填写且向windows 服务管理器(service manager)进行注册, ...

  3. C#创建服务及使用程序自动安装服务

    .NET创建一个即是可执行程序又是Windows服务的exe 不得不说,.NET中安装服务很麻烦,即要创建Service,又要创建ServiceInstall,最后还要弄一堆命令来安装和卸载. 今天给 ...

  4. windows批处理研究_不断更新

    windows批处理脚本(bat),很麻烦,主要原因有: 1.bat脚本编写的风格,太古老,调用方式太奇怪. 2.windows自身运行机制就对批处理脚本有兼容性问题.比如,鼠标双击打开一个bat,与 ...

  5. Qt Creator(编译器MinGW)中使用__attribute__(packed)的问题

    http://www.bttr-software.de/forum/mix_entry.php?id=11767 假设我们从串口中读到一串数据,当我们想要处理这串数据的时候通常是这样做的: 1 将这些 ...

  6. Ubuntu12.04环境搭建遇到的问题和建议(一个)

    后的新公司需要在Ubuntu12.04在结构Android开发环境,在这个过程中,我们还是会遇到很多问题,这里记录.为了方便自己的未来,有人谁需要参考.从网络! 1. Q:在终端: sudo apt- ...

  7. 批处理中set截取字符具体解释

    set截取字符具体解释  在批处理中,set的功能有点繁杂:设置变量.显示环境变量的名及值.做算术运算.等待用户的输入.字符串截取.替换字符串,是我们经常使用的命令之中的一个. 在字符串截取方面,新手 ...

  8. Lamda Action Func Thread 实例

    lamda表达式 格式:( 形参列表 ) => { 函数体 } 作用:简化匿名方法的书写,可用在任何可使用匿名方法和强类型代理的地方: Action是无返回值的泛型委托. Action 表示无参 ...

  9. 学SpringMVC收藏

     一个较完整的SpringMVC工程的配置 2014-01-22 17:17:25 标签:java spring springMVC 配置 springSecurity web.xml 原创作品,允许 ...

  10. 次短路[SPFA]

    Description 贝茜把家搬到了一个小农场,但她常常回到FJ的农场去拜访她的朋友.贝茜很喜欢路边的风景,不想那么快地结束她的旅途,于是她每次回农场,都会选择第二短的路径,而不象我们所习惯的那样, ...