PIC单片机-普通IO口模拟I2C总线对24C02进行读写操作

2023-12-24 00:08

本文主要是介绍PIC单片机-普通IO口模拟I2C总线对24C02进行读写操作,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

使用PIC单片机的普通IO口模拟I2C总线,对24C02进行读写操作。其中,总线定义为:

#define       IIC_SCLK     RC3      //24C02时钟线
#define      IIC_SDAT     RC4      //24C02数据线

将跳线帽P11短接至24C02方向,将跳线帽P6段接至IIC方向。

I2C 总线协议规定任何将数据传送到总线的器件作为发送器,任何从总线接收数据的器件为接收器,数据传送是由产生串行时钟和所有起始停止信号的主器件控制的,主器件和从器件都可以作为发送器或接收器。

I2C 总线协议定义如下:

1、只有在总线空闲时才允许启动数据传送

2、在数据传送过程中当时钟线为高电平时数据线必须保持稳定状态不允许有跳变时钟线为高电平时数据线的任何电平变化将被看作总线的起始或停止信号

  • 起始信号:时钟线保持高电平期间数据线电平从高到低的跳变作为I2C 总线的起始信号。
  • 停止信号:时钟线保持高电平期间数据线电平从低到高的跳变作为I2C 总线的停止信号。
#include <htc.h>#define uint8 unsigned char
#define uint16 unsigned int__CONFIG(FOSC_HS &WDTE_OFF &BOREN_OFF &PWRTE_OFF &LVP_OFF); //设置配置位
//WDTE_OFF:disable watchdog timer       看门狗禁止
//LVP_OFF:low voltage programming disabled  低电压编程禁止
//FOSC_HS:high speed crystal/resonator  4M以上晶振选择HS高速
//PWRTDIS:disable power up timer
//BOREN_OFF:disable brown out reset/********************定义数据线************************************************/#define     IIC_SCLK    RC3     //24C02时钟线
#define     IIC_SDAT    RC4     //24C02数据线/********************定义全局变量**********************************************/
const uint8 disp[] = {0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8, 0x80, 0x90}; //0~9共阳数码管段码表
uint8 init[8] = {0x59, 0x43, 0x21, 0x11, 0x03, 0x03, 0x09, 0x00};           //EEPROM写初始值数组
uint8 recive[8];    //读EEPROM数据存放数组/********************函数声明**************************************************/void IIC_WriteACK(void);            //返回应答信号
void IIC_Stop(void);                //停止信号
void IIC_Writenoack(void);          //无应答信号
void IIC_Start(void);               //启动信号
void DelayUS(uint8 delay);          //微妙延时
void DelayMS(uint16 delay);         //毫秒延时
uint8 IIC_ReadByte(void);           //IIC读8位字节数据
uint8 IIC_WriteByte(uint8 data);    //IIC写8位字节数据
void Display(uint8 chosebit, uint8 data);
void IIC_Read_Mulit_Data(uint8 commond, uint8 address, uint8 *addrpoint, uint8 bytenum);
void IIC_Write_Mulit_Data(uint8 commond, uint8 reg_addr, uint8 *mdatapointer, uint8 bytenum);/*******************************************************************************
* 函 数 名: DelayUS(uint8 delay)
* 函数功能: 微秒延时    for 20MHZ
* 入口参数: delay
* 返    回: 无
*******************************************************************************/void DelayUS(uint8 delay)
{
    while(--delay);
}/*******************************************************************************
* 函 数 名: DelayMS(uint16 delay)
* 函数功能: 毫秒延时    for20MHZ
* 入口参数: delay
* 返    回: 无
*******************************************************************************/void DelayMS(uint16 delay)
{
    uint16 i;
    for(; delay > 0; delay--)
        for(i = 0; i < 453; i++);
}/******************************************************************************
* 函 数 名: IIC_WriteACK(void)
* 函数功能: 从机接收数据正常时,向主机发送应答信号
* 入口参数: 无
* 返    回: 无
*******************************************************************************/void IIC_WriteACK(void)
{
    IIC_SDAT = 0;           //拉低数据线,给主机发送一个低电平
    IIC_SCLK = 1;           //时钟有效,发送数据
    IIC_SCLK = 0;
    //    IIC_SDAT = 1;
}/******************************************************************************
* 函 数 名: IIC_Writenoack(void)
* 函数功能: 从机接收数据结束时,向主机发送无应答信号
* 入口参数: 无
* 返    回: 无
*******************************************************************************/void IIC_Writenoack(void)
{
    IIC_SCLK = 0;
    IIC_SDAT = 1;         //发送高电平,表示无应答
    IIC_SCLK = 1;         //时钟有效
    IIC_SCLK = 0;
}/******************************************************************************
* 函 数 名: IIC_Start(void)
* 函数功能: 启动EEPROM读写操作
* 入口参数: 无
* 返    回: 无
*******************************************************************************/void IIC_Start(void)
{
    IIC_SDAT = 1;         //数据线置位
    IIC_SCLK = 1;         //时钟有效
    IIC_SDAT = 0;         //数据线复位,在时钟有效期间,数据线产生一个下降沿表示启动EEPROM读写操作
    IIC_SCLK = 0;         //时钟无效
}/******************************************************************************
* 函 数 名: IIC_Stop(void)
* 函数功能: 停止对EEPROM读写操作
* 入口参数: 无
* 返    回: 无
*******************************************************************************/void IIC_Stop(void)
{
    IIC_SDAT = 0;         //数据线复位
    IIC_SCLK = 1;         //时钟有效
    IIC_SDAT = 1;         //数据线复位
    IIC_SCLK = 0;         //数据线置位,在时钟有效期间,数据线产生一个上升沿表示停止EEPROM读写操作
}/******************************************************************************
* 函 数 名: IIC_WriteByte(uint8 wdata)
* 函数功能: 进行8位字节数据写操作
* 入口参数: wdata待写数据
* 返    回: status:0:成功 1:失败
*******************************************************************************/
uint8 IIC_WriteByte(uint8 wdata)
{
    uint8 i, status;
    IIC_SCLK = 0;                   //时钟无效
    for(i = 0; i < 8; i++)              //写8位字节数据
    {
        if((wdata & 0x80) == 0x80)  //读最高位数据
        {
            IIC_SDAT = 1;
        }
        else
        {
            IIC_SDAT = 0;
        }
        IIC_SCLK = 1;           //时钟有效
        wdata <<= 1;            //数据左移一位
        NOP();
        IIC_SCLK = 0;
    }    TRISC4 = 1;         //数据线设置为输入模式
    IIC_SDAT = 1;       //拉高数据线
    NOP();
    IIC_SCLK = 1;       //时钟有效
    NOP();
    if(IIC_SDAT)        //第九个时钟数据为1,则写失败成功,IIC器件没有应答
    {
        status = 1;
    }
    else
    {
        status = 0;
    }
    IIC_SCLK = 0;
    TRISC4 = 0;         //数据线设置回输出模式
    return status;      //返回写成功与否状态
}/******************************************************************************
* 函 数 名: IIC_WriteData(uint8 commond,uint8 reg_addr,*mdatapointer,uint8 bytenum)
* 函数功能: 写多字节数据
* 入口参数: commond:读写指令,reg_addr写操作地址,mdata待写数据,bytenum字节数
* 返    回: 无
*******************************************************************************/
void IIC_Write_Mulit_Data(uint8 commond, uint8 reg_addr, uint8 *mdatapointer, uint8 bytenum)
{
    uint8 i, status;
    IIC_Start();    //启动IIC
    status = IIC_WriteByte(commond);    //写器件地址
    if(!status)
    {
        status = IIC_WriteByte(reg_addr);   //写寄存器地址
    }
    for(i = 0; i < bytenum; i++)
    {
        if(!status)
        {
            status = IIC_WriteByte(*(mdatapointer + i));        //写数据
        }
    }
    IIC_Stop();     //结束IIC
}/******************************************************************************
* 函 数 名: IIC_ReadByte(void)
* 函数功能: 8位数据读操作
* 入口参数: 无
* 返    回: bytedata读取的8位数据
*******************************************************************************/
uint8 IIC_ReadByte(void)
{
    uint8 i, bytedata;
    TRISC4 = 1;     //数据线设置为输入模式
    IIC_SDAT = 1;   //数据线拉高
    IIC_SCLK = 0;
    for(i = 0; i < 8; i++)      //读8位数据
    {
        IIC_SCLK = 1;
        bytedata <<= 1;
        bytedata |= IIC_SDAT;
        IIC_SCLK = 0;
        NOP();
    }
    TRISC4 = 0;     //数据线设置回输出模式
    return(bytedata);//返回数据
}/******************************************************************************
* 函 数 名: IIC_Read_Mulit_Data(uint8 commond,uint8 address,uint8 *addrpoint,uint8 bytenum)
* 函数功能: 对IIC行进多字节读操作
* 入口参数: commond地址读写命令,address读数据的起始地址,addrpoint数据存放地址指针,bytenum字节数
* 返    回: 无
*******************************************************************************/void IIC_Read_Mulit_Data(uint8 commond, uint8 address, uint8 *addrpoint, uint8 bytenum)
{
    uint8 i, status;
    IIC_Start();                        //启动EEPROM读写操作
    status = IIC_WriteByte(commond);     //写从机地址及写操作指令
    if(!status)
    {
        status = IIC_WriteByte(address);    //写寄存器地址
    }
    if(!status)
    {
        IIC_Start();    //重新启动IIC
        status = IIC_WriteByte(commond + 0x01); //写读指令
    }
    for(i = 0; i < bytenum; i++)
    {
        if(!status)
        {
            addrpoint[i] = IIC_ReadByte();  //读取数据存放到数组
            if(i < (bytenum - 1))
            {
                IIC_WriteACK();             //返回应答信号
            }
            else
            {
                IIC_Writenoack();           //读完最后一个数据返回无应答信号,表示结束读操作
            }
        }
    }
    IIC_Stop();                     //停止EEPROM读写操作
    DelayUS(1);
}/*******************************************************************************
* 函 数 名: LED_Display(uint8 chosebit,uint8 wdata)
* 函数功能: 单个数码管显示
* 入口参数: chosebit选择字位,wdata字型
* 返    回: 无
*******************************************************************************/void LED_Display(uint8 chosebit, uint8 wdata)
{
    if((chosebit < 0) || (chosebit > 4))
        return;
    ADCON1 = 0x86;  //10000110  PORTA口设置为数字IO口
    TRISA &= 0x00;
    TRISE &= 0xEF;      //配置PORTD为普通IO口,对PORTD的操作务必加上这句
    TRISD = 0x00;       //数据线配置为输出
    PORTA = (PORTA & 0xE3) | ((chosebit + 3) << 2); //RA2:RA3:RA4:011 选通第一位 100选通第二位 101选通第三位 110选通第四位
    PORTD = wdata;  //送显示数字的断码
    DelayUS(100);       //延时
    PORTD = 0xFF;       //关闭显示
}/******************************************************************************
* 函 数 名: main(void)
* 函数功能: 利用普通IO口模拟IIC总线,对24C02写8字节数据,并读取8个字节数据
* 入口参数: 无
* 返    回: 无
*******************************************************************************/void main(void)
{
    TRISC3 = 0;             //配置时钟线为输出
    TRISC4 = 0;             //配置数据线为输出
    IIC_Write_Mulit_Data(0xa0, 0x00, init, 8);  //写8字节数据
    DelayMS(5);
    while(1)
    {
        IIC_Read_Mulit_Data(0xa0, 0x00, recive, 8); //读8字节数据
        LED_Display(0, disp[init[0] / 16]); //显示要被写入的值
        LED_Display(1, disp[init[0] % 16]);
        LED_Display(2, disp[recive[0] / 16]);   //显示从AT2402读到的值
        LED_Display(3, disp[recive[0] % 16]);
        DelayMS(1);
    }
}

这篇关于PIC单片机-普通IO口模拟I2C总线对24C02进行读写操作的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/529938

相关文章

Nginx中配置使用非默认80端口进行服务的完整指南

《Nginx中配置使用非默认80端口进行服务的完整指南》在实际生产环境中,我们经常需要将Nginx配置在其他端口上运行,本文将详细介绍如何在Nginx中配置使用非默认端口进行服务,希望对大家有所帮助... 目录一、为什么需要使用非默认端口二、配置Nginx使用非默认端口的基本方法2.1 修改listen指令

MySQL按时间维度对亿级数据表进行平滑分表

《MySQL按时间维度对亿级数据表进行平滑分表》本文将以一个真实的4亿数据表分表案例为基础,详细介绍如何在不影响线上业务的情况下,完成按时间维度分表的完整过程,感兴趣的小伙伴可以了解一下... 目录引言一、为什么我们需要分表1.1 单表数据量过大的问题1.2 分表方案选型二、分表前的准备工作2.1 数据评估

MySQL进行分片合并的实现步骤

《MySQL进行分片合并的实现步骤》分片合并是指在分布式数据库系统中,将不同分片上的查询结果进行整合,以获得完整的查询结果,下面就来具体介绍一下,感兴趣的可以了解一下... 目录环境准备项目依赖数据源配置分片上下文分片查询和合并代码实现1. 查询单条记录2. 跨分片查询和合并测试结论分片合并(Shardin

SpringBoot结合Knife4j进行API分组授权管理配置详解

《SpringBoot结合Knife4j进行API分组授权管理配置详解》在现代的微服务架构中,API文档和授权管理是不可或缺的一部分,本文将介绍如何在SpringBoot应用中集成Knife4j,并进... 目录环境准备配置 Swagger配置 Swagger OpenAPI自定义 Swagger UI 底

基于Python Playwright进行前端性能测试的脚本实现

《基于PythonPlaywright进行前端性能测试的脚本实现》在当今Web应用开发中,性能优化是提升用户体验的关键因素之一,本文将介绍如何使用Playwright构建一个自动化性能测试工具,希望... 目录引言工具概述整体架构核心实现解析1. 浏览器初始化2. 性能数据收集3. 资源分析4. 关键性能指

Nginx进行平滑升级的实战指南(不中断服务版本更新)

《Nginx进行平滑升级的实战指南(不中断服务版本更新)》Nginx的平滑升级(也称为热升级)是一种在不停止服务的情况下更新Nginx版本或添加模块的方法,这种升级方式确保了服务的高可用性,避免了因升... 目录一.下载并编译新版Nginx1.下载解压2.编译二.替换可执行文件,并平滑升级1.替换可执行文件

Python进行JSON和Excel文件转换处理指南

《Python进行JSON和Excel文件转换处理指南》在数据交换与系统集成中,JSON与Excel是两种极为常见的数据格式,本文将介绍如何使用Python实现将JSON转换为格式化的Excel文件,... 目录将 jsON 导入为格式化 Excel将 Excel 导出为结构化 JSON处理嵌套 JSON:

python运用requests模拟浏览器发送请求过程

《python运用requests模拟浏览器发送请求过程》模拟浏览器请求可选用requests处理静态内容,selenium应对动态页面,playwright支持高级自动化,设置代理和超时参数,根据需... 目录使用requests库模拟浏览器请求使用selenium自动化浏览器操作使用playwright

一文解密Python进行监控进程的黑科技

《一文解密Python进行监控进程的黑科技》在计算机系统管理和应用性能优化中,监控进程的CPU、内存和IO使用率是非常重要的任务,下面我们就来讲讲如何Python写一个简单使用的监控进程的工具吧... 目录准备工作监控CPU使用率监控内存使用率监控IO使用率小工具代码整合在计算机系统管理和应用性能优化中,监

如何使用Lombok进行spring 注入

《如何使用Lombok进行spring注入》本文介绍如何用Lombok简化Spring注入,推荐优先使用setter注入,通过注解自动生成getter/setter及构造器,减少冗余代码,提升开发效... Lombok为了开发环境简化代码,好处不用多说。spring 注入方式为2种,构造器注入和setter