2010-04-24 12:58:00

注:如果要读24c128或264,地址位为16位的。现在的地址位为8位。

protues仿真图

源程序如下:

#include <iom16v.h>
#include <macros.h>
#include "delay.h"
#define I2C_DDR     DDRB
#define I2C_PORT    PORTB
#define I2C_PIN     PINB
#define SDA         PB1
#define SCL         PB0

#define SDA_OUT()  
(I2C_DDR |= (1<<SDA))
#define SCL_OUT()   (I2C_DDR |= (1<<SCL))
// 释放SDA,上拉输入
#define SDA_RLS()   do{ I2C_PORT |= (1<<SDA); I2C_DDR &=
~(1<<SDA); }while(0)  
#define SCL_RLS()   do{ I2C_PORT |= (1<<SCL); I2C_DDR &=
~(1<<SCL); }while(0)
#define SDA_SET()   (I2C_PORT |= (1<<SDA))
#define SDA_CLR()   (I2C_PORT &= ~(1<<SDA))
#define SCL_SET()   (I2C_PORT |= (1<<SCL))
#define SCL_CLR()   (I2C_PORT &= ~(1<<SCL))
#define READ_SDA()  ((I2C_PIN & (1<<SDA)) ? 1 : 0)

#define
i2c_delay() delay_us(1)       // 根据系统时钟进行调整
#define AT24C02_WRITE_ADDR  0xA0
#define AT24C02_READ_ADDR   0xA1
#define PAGE_SIZE   8
#define PAGE_MASK   (PAGE_SIZE-1)
// 发送起始信号
void i2c_start(void)
{
        SDA_SET();
  i2c_delay();
        SCL_SET();
        i2c_delay();
        SDA_CLR();
        i2c_delay();
        SCL_CLR();
  //i2c_delay();
}

// 发送停止信号
void i2c_stop(void)
{
    SDA_CLR();
    i2c_delay();
    SCL_SET();
    i2c_delay();
    SDA_SET();
   // i2c_delay();
}

// 向总线写1字节,并返回从机有无应答
unsigned char i2c_write(unsigned char byte)
{
    unsigned char i, ack;
    ack = 0;
    for( i = 0; i < 8; ++i ){
        if ( byte & 0x80)
            SDA_SET();
        else
               
SDA_CLR();
        byte <<= 1;
        SCL_SET();
        i2c_delay();
        SCL_CLR();
    }

SDA_RLS();
    i2c_delay();
    SCL_SET();
    i2c_delay();
    ack = READ_SDA();
    SCL_CLR();
    i2c_delay();
 SDA_OUT();
    return ack;
}

// 读1字节,ack=1时,发送ACK,ack=0,发送nACK
unsigned char i2c_read(unsigned char ack)
{
    unsigned char i, r = 0;

SDA_RLS();
    for( i = 0; i < 8; i++ ){
        SCL_SET();
        i2c_delay();
        r <<= 1;
        if ( READ_SDA() == 1)
        {
            r |= 1;
        }
        SCL_CLR();
        i2c_delay();
    }
   
    SDA_OUT();
    // Send ACK
    if (!ack)
       
SDA_SET();      // Send nACK
    else
       
SDA_CLR();      // Send ACK
    i2c_delay();
    SCL_SET();
    i2c_delay();
    SCL_CLR();
    i2c_delay();
    return r;
}

void at24c02_write1byte(unsigned
int byte_addr, unsigned char byte)
{
    i2c_start();
    i2c_write(AT24C02_WRITE_ADDR);
 //i2c_write(byte_addr>>8);      //输入15-9高位
    i2c_write(byte_addr);
    i2c_write(byte);
    i2c_stop();
    delay_ms(20);      // 等待10ms,保证24C02内部写操作完成
}

void at24c02_page_write(unsigned
char byte_addr, unsigned char *buff, unsigned char num)
{
    unsigned char i;
    i2c_start();
    i2c_write(AT24C02_WRITE_ADDR);
    i2c_write(byte_addr);
    for( i = 1; i <= num; ++i )
    {
        i2c_write(*buff);
        buff++;
    }
    i2c_stop();
    delay_ms(20);
}

void at24c02_writenbytes(unsigned
char byte_addr, unsigned char *buff, unsigned char num)
{
    unsigned char n_tmp;
    n_tmp = PAGE_SIZE - (unsigned char)(byte_addr &
PAGE_MASK); // 本页内剩余空间数量
    if ((num > n_tmp) && (n_tmp != 0))
    {
        // 先将本页剩余空间写满
        at24c02_page_write(byte_addr, buff,
n_tmp);
        byte_addr += n_tmp;
        num -= n_tmp;
        buff += n_tmp;
    }
    while( num >= PAGE_SIZE ){
        at24c02_page_write(byte_addr, buff,
PAGE_SIZE);
        byte_addr += PAGE_SIZE;
        num -= PAGE_SIZE;
        buff += PAGE_SIZE;
    }
    if (num != 0)
    {
        at24c02_page_write(byte_addr, buff,
num);
    }
}

unsigned char
at24c02_read_curr_addr(void)
{
    unsigned char data;
    i2c_start();
    i2c_write(AT24C02_READ_ADDR);
    data = i2c_read(0);     // 读1字节,返回nACK
    i2c_stop();
    return data;
}

unsigned char
at24c02_read1byte(unsigned char byte_addr)
{
    unsigned char data;
    i2c_start();
    i2c_write(AT24C02_WRITE_ADDR);
 //i2c_write(byte_addr>>8);       //输入15-9高位
    i2c_write(byte_addr);
    i2c_start();
    i2c_write(AT24C02_READ_ADDR);
    data = i2c_read(0);     // 读1字节,返回nACK
    i2c_stop();
    return data;

}

void at24c02_readnbytes(unsigned
char byte_addr, unsigned char *buff, unsigned char num)
{
    unsigned char i;
    i2c_start();
    i2c_write(AT24C02_WRITE_ADDR);
    i2c_write(byte_addr);
    i2c_start();
    i2c_write(AT24C02_READ_ADDR);
    for( i = 1; i <= num-1; ++i )
    {
        *buff =
i2c_read(1);        // 读1字节返回ack
        buff++;
    }
    *buff = i2c_read(0);
    i2c_stop();
}

void port_init(void)
{
 PORTA=0xff;
 DDRA=0xff;
 SDA_OUT();
 SCL_OUT();
}

void main(void)
{
    unsigned char temp1[8]={1,2,3,4,5,6,7,8};
    unsigned char temp2[8]={0};
 port_init();
    at24c02_writenbytes(0x03,temp1,8);// write the byte 01h at
address 03h
    at24c02_readnbytes(0x03,temp2,8);  // read the
byte 01h at address 03h
     //SDA_RLS() ;
  //SCL_RLS() ;
    while (1)
    {     
  if(temp2[0]==1)
     {
      PORTA=0XFE;
      delay_ms(100);
      PORTA=0XFF;
   delay_ms(50);
     }

if(temp2[1]==2)
     {
      PORTA=0XFD;
      delay_ms(100);
      PORTA=0XFF;
   delay_ms(50);
     }
    
      if(temp2[2]==3)
     {
      PORTA=0XFB;
      delay_ms(100);
      PORTA=0XFF;
   delay_ms(50);
     }
    
      if(temp2[3]==4)
     {
      PORTA=0XF7;
      delay_ms(100);
      PORTA=0XFF;
   delay_ms(50);
     }
    
      if(temp2[4]==5)
     {
      PORTA=0XEF;
      delay_ms(100);
      PORTA=0XFF;
   delay_ms(50);
     }
    
      if(temp2[5]==6)
     {
      PORTA=0XDF;
      delay_ms(100);
      PORTA=0XFF;
   delay_ms(50);
     }
    
       if(temp2[6]==7)
     {
      PORTA=0XBF;
      delay_ms(100);
      PORTA=0XFF;
   delay_ms(50);
     }
    
      if(temp2[7]==8)
     {
      PORTA=0X7F;
      delay_ms(100);
      PORTA=0XFF;
   delay_ms(50);
     }
  }
}

移植并修改成功的模拟iic读写EEPROM at24c02的更多相关文章

  1. STM32F10x_模拟I2C读写EEPROM

    Ⅰ.写在前面 说到IIC,大家都应该不会陌生,我们初学单片机的时候或多或少都知道或了解过,甚至使用I2C控制过器件.但是,有多少人真正去深入理解,或者深入研究过I2C通信协议呢? 1.我们有必要学习I ...

  2. MSP430F149模拟IIC读写24C02程序

    板子上设置了EEPROM存储器,型号为AT24C02.板子的硬件连接为:SCL--->P2.4,SDA--->P2.5.直接了当,贴上程序! ======================= ...

  3. 软件模拟IIC实现EEPROM

    ....妈的太难.    反正就是控制引脚的高低电平 实现数据的读取....参考 I2C的协议层和物理层的那个几个图,个个信号产生的电平 自己看源码去把. 头疼

  4. C51—模拟IIC总线实现EEPROM存取数据

    a - 什么是IIC总线 -什么是EEPROM -IIC总线的通信格式 模块化设计注解 整体代码 - 什么是IIC总线 IIC总线是同步通信的一种特殊形式,具有接线口少.控制简单.器件封装形式小.通信 ...

  5. STM32F10x_模拟I2C读写_硬件I2C读写

    STM32F10x_模拟I2C读写EEPROM STM32F10x_硬件I2C读写EEPROM(标准外设库版本) STM32F10x_硬件I2C主从通信(轮询发送,中断接收)

  6. STM32F10x_硬件I2C读写EEPROM(标准外设库版本)

    Ⅰ.写在前面 上一篇文章是“STM32F10x_模拟I2C读写EEPROM”,讲述使用IO口模拟I2C总线通信,对EEPROM(AT24Xxx)进行读写操作的过程. 上一篇文章主要内容:I2C协议.模 ...

  7. IIC协议建模——读写EEPROM

    案例采用明德扬设计思想完成.IIC协议是非常常用的接口协议,在电子类岗位招聘要求中经常出现它的身影.关于IIC协议这里只做简要介绍,详细信息请自行百度或查阅相关Datasheet,网上资料非常多.该篇 ...

  8. 第十六章 IIC协议详解+UART串口读写EEPROM

    十六.IIC协议详解+Uart串口读写EEPROM 本文由杭电网友曾凯峰根据小梅哥FPGA IIC协议基本概念公开课内容整理并最终编写Verilog代码实现使用串口读写EEPROM的功能. 以下为原文 ...

  9. stm32软件模拟IIC读取PX4FLOW光流传感器数据

    这段时间在做全国光电设计大赛,用到了px4的px4flow光流传感器,用软件模拟iic读取数据不定期会导致px4flow死机,查了资料和光流的源码,发现这个光流用了stm32的硬件iic,所以对软件模 ...

随机推荐

  1. MongoDB的客户端管理工具--nosqlbooster 查询工具使用

    连接我的MongoDB 看到这样 打开db1数据库里面user集合,看到user集合里面的数据,他会自带查询语句 看这里以tree方式显示 可以以table方式显示 还可以json方式显示 按照自己的 ...

  2. 关于ARMv8另外几个问题

    版权声明:本文为博主原创文章.未经博主同意不得转载. https://blog.csdn.net/qianlong4526888/article/details/27510675 问题1:支持ARMv ...

  3. 20170915 linux系统管理培训

    进程管理 程序:通常为二进制程序放在存储媒介中(如光盘.硬盘.软盘.磁带等),以物理文件的形式存在: 进程:正在运行当中的程序,程序被触发后,执行者的权限与属性.程序的程序代码与所有数据等都会被加载到 ...

  4. 并发编程---线程queue---进程池线程池---异部调用(回调机制)

    线程 队列:先进先出 堆栈:后进先出 优先级:数字越小优先级越大,越先输出 import queue q = queue.Queue(3) # 先进先出-->队列 q.put('first') ...

  5. vue 常用问题

    缺少style-loader 或者 stylus-loader 等问题 在文件[package.json]分支:[devDependencies]添加 版本号: "stylus": ...

  6. 算法 -- 四种方法获取的最长“回文串”,并对时间复杂进行分析对比&PHP

    https://blog.csdn.net/hongyuancao/article/details/82962382 “回文串”是一个正读和反读都一样的字符串,比如“level”或者“noon”等等就 ...

  7. Sql Server 主键 外键约束

    主键约束 表通常具有包含唯一标识表中每一行的值的一列或一组列. 这样的一列或多列称为表的主键 (PK),用于强制表的实体完整性. 由于主键约束可保证数据的唯一性,因此经常对标识列定义这种约束. 如果为 ...

  8. git cherry-pick 报错 fatal: bad object

    场景:程序员A提交了一个commit到gerrit上,我们叫他为commit_id1,但是还没有review,那就是没有入库,程序员B想再本地拿到这个commitd_id1,既然这个提交没有入库,很明 ...

  9. ROC曲线,AUC面积

    AUC(Area under Curve):Roc曲线下的面积,介于0.1和1之间.Auc作为数值可以直观的评价分类器的好坏,值越大越好. 首先AUC值是一个概率值,当你随机挑选一个正样本以及负样本, ...

  10. 梯度下降法(BGD、SGD)、牛顿法、拟牛顿法(DFP、BFGS)、共轭梯度法

    一.梯度下降法 梯度:如果函数是一维的变量,则梯度就是导数的方向:      如果是大于一维的,梯度就是在这个点的法向量,并指向数值更高的等值线,这就是为什么求最小值的时候要用负梯度 梯度下降法(Gr ...