【ZYNQ】IIC 简介及 EMIO 模拟 IIC 驱动示例

IIC 简介

IIC (Inter-Integrated Circuit) 总线是一种由 NXP (原PHILIPS) 公司开发的两线式串行通信总线,多用于主控制器和从设备之间的主从式通信,且任意时刻只能有一个主机。

IIC 总线由数据线 SDA 和时钟线 SCL 构成,可实现在主控制器与被控设备之间、设备与设备之间双向通信,高速 IIC 总线一般可达 400kbps 以上。

为驱动实验室自研基于 DAC60502 的双通道电流输出模块,基于 ZYNQ 平台,将 PS 端的 GPIO 资源通过 EMIO 引出到 PL 端,使用 PL 端引脚资源连接 IIC 从设备,既使用 EMIO 模拟 IIC 实现主从通信控制。

在这里插入图片描述

模拟 IIC 示例代码
  • iic_io.c
/**
 * Copyright (c) 2022-2023,HelloAlpha
 * 
 * Change Logs:
 * Date           Author       Notes
 */
#include "iic_io.h"

extern XGpioPs GpioPs;

void SCL_OUT(void)
{
    XGpioPs_SetDirectionPin(&GpioPs, SCL_PIN, GPIO_MODEL_OUTPUT);
    XGpioPs_SetOutputEnablePin(&GpioPs, SCL_PIN, GPIO_OUTPUT_ENABLE);
}

void SCL_HIGH(void)
{
    XGpioPs_WritePin(&GpioPs, SCL_PIN, GPIO_SET);
}

void SCL_LOW(void)
{
    XGpioPs_WritePin(&GpioPs, SCL_PIN, GPIO_RESET);
}

void SDA_IN(void)
{
    XGpioPs_SetDirectionPin(&GpioPs, SDA_PIN, GPIO_MODEL_INPUT);
}

uint32_t SDA_READ(void)
{
    return XGpioPs_ReadPin(&GpioPs, SDA_PIN);
}

void SDA_OUT(void)
{
    XGpioPs_SetDirectionPin(&GpioPs, SDA_PIN, GPIO_MODEL_OUTPUT);
    XGpioPs_SetOutputEnablePin(&GpioPs, SDA_PIN, GPIO_OUTPUT_ENABLE);
}

void SDA_HIGH(void)
{
    XGpioPs_WritePin(&GpioPs, SDA_PIN, GPIO_SET);
}

void SDA_LOW(void)
{
    XGpioPs_WritePin(&GpioPs, SDA_PIN, GPIO_RESET);
}

int IIC_Init(void)
{
    SCL_OUT();
    SCL_HIGH();
    SDA_OUT();
    SDA_HIGH();

    return 0;
}
  • iic_io.h
/**
 * Copyright (c) 2022-2023,HelloAlpha
 *
 * Change Logs:
 * Date           Author       Notes
 */
#ifndef __IIC_IO_H__
#define __IIC_IO_H__

#include "xgpiops.h"

#define SCL_PIN           54
#define SDA_PIN           55

#define GPIO_MODEL_INPUT        0
#define GPIO_MODEL_OUTPUT       1

#define GPIO_OUTPUT_DISABLE     0
#define GPIO_OUTPUT_ENABLE      1

#define GPIO_RESET              0
#define GPIO_SET                1

void SCL_OUT(void);
void SCL_HIGH(void);
void SCL_LOW(void);

void SDA_IN(void);
uint32_t SDA_READ(void);

void SDA_OUT(void);
void SDA_HIGH(void);
void SDA_LOW(void);

int IIC_Init(void);

#endif
  • iic_ctrl.c
/**
 * Copyright (c) 2022-2023,HelloAlpha
 *
 * Change Logs:
 * Date           Author       Notes
 */
#include "iic_ctrl.h"
#include "sleep.h"

#define I2C_DELAY(...)	usleep(__VA_ARGS__)

void IIC_Start(void)
{
    SDA_OUT();
    SDA_HIGH();
    SCL_HIGH();
    I2C_DELAY(4);
    SDA_LOW();
    I2C_DELAY(4);
    SCL_LOW();
    I2C_DELAY(2);
}
 
void IIC_Stop(void)
{
    SDA_OUT();
    SCL_LOW();
    SDA_LOW();
    I2C_DELAY(4);
    SCL_HIGH();
    SDA_HIGH();
    I2C_DELAY(4);
}
 
// 1,接收应答失败
// 0,接收应答成功
uint8_t IIC_Wait_Ack(void)
{
    uint8_t ucErrTime=0;

    SDA_OUT();
    SDA_HIGH();
    I2C_DELAY(1);
    SCL_HIGH();
    I2C_DELAY(1);
    SDA_IN();
    while(SDA_READ())
    {
        ucErrTime++;
        if(ucErrTime > 250)
        {
            IIC_Stop();
            return 1;
        }
    }
    SCL_LOW();
    return 0;  
}

//产生ACK应答
void IIC_Ack(void)
{
    SCL_LOW();
    SDA_OUT();
    SDA_LOW();
    I2C_DELAY(2);
    SCL_HIGH();
    I2C_DELAY(2);
    SCL_LOW();
}
//不产生ACK应答
void IIC_NAck(void)
{
    SCL_LOW();
    SDA_OUT();
    SDA_HIGH();
    I2C_DELAY(2);
    SCL_HIGH();
    I2C_DELAY(2);
    SCL_LOW();
}				

void IIC_Write_Byte(uint8_t data)
{
    uint8_t  i = 0;
 
    SDA_OUT();
    SCL_LOW();
    for(i=0; i<8; i++)
    {
        if(data & 0x80 )
            SDA_HIGH();
        else
            SDA_LOW();
        data <<= 1;
        I2C_DELAY(2);
        SCL_HIGH();
        I2C_DELAY(2);
        SCL_LOW();
        I2C_DELAY(2);
    }
}

// ack = 1,发送 ACK
// ack = 0,发送 nACK
uint8_t IIC_Read_Byte(uint8_t ack)
{
    uint8_t i = 0, data = 0;
 
    SDA_IN();
    for(i = 0; i < 8; i++)
    {
        SCL_LOW();
        I2C_DELAY(2);
         SCL_HIGH();
        data <<= 1;
        if(SDA_READ())
            data++;
        I2C_DELAY(1);
    }
    if(!ack)
        IIC_NAck();
    else
        IIC_Ack();
    return data;
}
 
void IIC_Write_UINT16(uint8_t dev_addr, uint8_t wr_addr, uint16_t data)
{
    IIC_Start();
    IIC_Write_Byte(dev_addr);
    IIC_Wait_Ack(); 
    IIC_Write_Byte(wr_addr);
    IIC_Wait_Ack(); 
    IIC_Write_Byte(data >> 8);
    IIC_Wait_Ack(); 
    IIC_Write_Byte(data & 0xFF);
    IIC_Wait_Ack();  	
    IIC_Stop();
    I2C_DELAY(800);
}

uint16_t IIC_Read_UINT16(uint8_t dev_addr, uint8_t rd_addr)
{
    uint16_t data;

    IIC_Start();
    IIC_Write_Byte(dev_addr);
    IIC_Wait_Ack();
    IIC_Write_Byte(rd_addr);
    IIC_Wait_Ack(); 

    IIC_Start();  	
    IIC_Write_Byte(dev_addr + 1);
    IIC_Wait_Ack();
    data = IIC_Read_Byte(1);
    data = (data <<8 ) + IIC_Read_Byte(0);
    IIC_Stop();

    return data;
}
  • iic_ctrl.h
/**
 * Copyright (c) 2022-2023,HelloAlpha
 *
 * Change Logs:
 * Date           Author       Notes
 */
#ifndef __I2C_CTRL_H__
#define __I2C_CTRL_H__

#include "iic_io.h"

void IIC_Start(void);
void IIC_Stop(void);
uint8_t IIC_Wait_Ack(void);
void IIC_Ack(void);	    
void IIC_NAck(void);
void IIC_Write_Byte(uint8_t value );
uint8_t IIC_Read_Byte(uint8_t addr);
void IIC_Write_UINT16(uint8_t dev_addr, uint8_t wr_addr, uint16_t data);
uint16_t IIC_Read_UINT16(uint8_t dev_addr, uint8_t rd_addr);

#endif

软件模拟 IIC 最大的优点就在于方便移植,但相对于硬件 IIC 来说,模拟 IIC 通信速率要慢得多。

  • 自问:那为啥不用硬件 IIC
  • 自答:就想用模拟的嘿嘿嘿
  • 4
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 4
    评论
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Hello阿尔法

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值