嵌入式培训机构四个月实训课程笔记(完整版)-Linux ARM驱动编程第四天-ARM Linux编程之IIC与uart (物联技术666)

链接:https://pan.baidu.com/s/1V0E9IHSoLbpiWJsncmFgdA?pwd=1688
提取码:1688

教学内容:

1I2C总线:

I2C(Inter-Integrated Circuit),PHILIPS公司开发的两线式半双工同步串行总线;可以用来连接存储器(EEPROM、FLASH)、A/D、D/A转换器、LCD驱动器、传感器等等。

I2C总线有两根信号线:双向数据线(SDA)、时钟线(SCL)。均为双向I/O线,通过上拉电阻接正电源;I2C总线可以连接多个设备,各设备的数据和时钟线均连到SDA、SCL信号线上,主机通过设备地址来区分具体的设备,每个设备有唯一的地址(7位或10位)。

起始和终止信号:

SCL线为高电平期间,SDA线由高电平向低电平的变化表示起始信号;

SCL线为高电平期间,SDA线由低电平向高电平的变化表示终止信号。 

起始和终止信号都是由主机发出的,在起始信号产生后,总线就处于被占用的状态;在终止信号产生后,总线就处于空闲状态。连接到I2C总线上的器件,若具有I2C总线的硬件接口,则很容易检测到起始和终止信号。对于不具备I2C总线硬件接口的有些单片机来说,为了检测起始和终止信号,必须保证在每个时钟周期内对数据线SDA采样两次。接收器件收到一个完整的数据字节后,有可能需要完成一些其它工作,如处理内部中断服务等,可能无法立刻接收下一个字节,这时接收器件可以将SCL线拉成低电平,从而使主机处于等待状态。直到接收器件准备好接收下一个字节时,再释放SCL线使之为高电平,从而使数据传送可以继续进行。

       在起始信号后必须传送一个从机的地址(7位),第8位是数据的传送方向位(R/),用“0”表示主机发送数据(T),“1”表示主机接收数据(R)。每次数据传送总是由主机产生的终止信号结束。但是,若主机希望继续占用总线进行新的数据传送,则可以不产生终止信号,马上再次发出起始信号对另一从机进行寻址。

a、主机向从机发送数据,数据传送方向在整个传送过程中不变:

注:有阴影部分表示数据由主机向从机传送,无阴影部分则表示数据由从机向主机传送。

    A表示应答, 表示非应答(高电平)。S表示起始信号,P表示终止信号。。

b、主机在第一个字节后,立即由从机读数据

c、在传送过程中,当需要改变传送方向时,起始信号和从机地址都被重复产生一次,但两次读/写方向位正好反相。

模拟过程:(以AT24C02为外接设备)

//*************************************************

#define  IIC_CLK_SET()  

#define  IIC_CLK_CLR()

#define  IIC_DAT_SET()

#define  IIC_DAT_CLR()

#define  IIC_DAT_GET()

//IIC启动信号,CLK为高电平的时,DAT由高变低

void I2CStart()

{  

       IIC_DAT_SET();

    I2CDelay();

       IIC_CLK_SET();

       I2CDelay();  

    IIC_DAT_CLR();

    I2CDelay();

    IIC_CLK_CLR();

    I2CDelay();

}

//停止信号,CLK为高电平时,DAT信号由低变高

void I2CStop()

{

   IIC_DAT_CLR();

   I2CDelay();

   IIC_CLK_SET();

   I2CDelay();

   IIC_DAT_SET();

   I2CDelay();

   IIC_CLK_CLR();

   I2CDelay();

}

//IIc初始化函数 CLK信号置低,发送停止信号

void I2CInit()

{

   IIC_CLK_CLR();

   I2CStop();

}

//在CLK为高电平时读取1位总线数据与应答信号

bit I2CClock() //return SDA while SCL is HIGH

{

   unsigned char sample;

   IIC_CLK_SET();

   I2CDelay();

   sample = IIC_DAT_GET();

   IIC_CLK_CLR();

   I2CDelay();

   return sample;

}

// 发送一个字节的数据,并返回应答信号(1表示接收正常,0表示接收器件无应答或损坏)

unsigned int I2CSend(BYTE I2CData)

{

   BYTE i;

   for(i=0;i<8;i++)

   {

       if( (I2CData & 0x80) == 0x80 )

             IIC_DAT_SET();    

       else

             IIC_DAT_CLR();

       I2CData=I2CData << 1;

      

       IIC_CLK_SET();

       I2C_Delay();

       IIC_CLK_CLR();

       I2C_Delay();

   }

   IIC_DAT_SET();

   return (~I2CClock());

}

//主机应答信号,a=1时表示应答信号, a=0时表示非应答信号

void I2CAck(unsigned int a)

{

       if(a==0)

            IIC_DAT_CLR();

     else

           IIC_DAT_SET();

     IIC_CLK_SET();

       I2C_Delay();

       IIC_CLK_CLR();

       I2C_Delay();

}

//延时程序

void I2CDelay()

{

   BYTE ll;

   for(ll=0;ll<100;ll++)

   {

      ;

   }

}

//IIC总线器件接受一个字节的数据

BYTE I2CReceive()

{

   BYTE I2CData=0;

   BYTE kk;

   for(kk=0;kk<8;kk++)

   {

      I2CData = I2CData <<1;

      if(I2CClock())

      {

         I2CData= I2CData + 0x01;

      }

   }

   return I2CData;

}

//向指定器件发送指定字节的数据,包含子设备地址

int SendByte(unsigned char sla,unsigned char c ,int num )

{

       //启动IIC设备

       //发送器件地址,

       //检查应答信号

       //发送一个字节数据

       //结束总线

}

//想从指定的器件接收指定字节的数据,返回0代表失败,返回1代表成功

 unsigned int RevByte(unsigned char sla, char *c ,int num)

 {

       //启动IIC设备

       //发送器件地址

       //检查应答信号

       //接收一个字节数据

       //发送应答信号

      //结束总线

 }

//***************************************************

  • 28
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一段标准的嵌入式LinuxIIC驱动代码,供参考: ``` #include <linux/module.h> #include <linux/i2c.h> #include <linux/init.h> static int i2c_driver_probe(struct i2c_client *client, const struct i2c_device_id *id) { printk(KERN_INFO "I2C driver probe function called\n"); // TODO: Add initialization code here return 0; } static int i2c_driver_remove(struct i2c_client *client) { printk(KERN_INFO "I2C driver remove function called\n"); // TODO: Add cleanup code here return 0; } // Define the list of I2C devices that this driver supports static const struct i2c_device_id i2c_driver_id[] = { { "my_i2c_device", 0 }, {}, }; MODULE_DEVICE_TABLE(i2c, i2c_driver_id); // Define the I2C driver structure static struct i2c_driver i2c_driver = { .driver = { .name = "my_i2c_driver", .owner = THIS_MODULE, }, .probe = i2c_driver_probe, .remove = i2c_driver_remove, .id_table = i2c_driver_id, }; // Register the I2C driver static int __init i2c_driver_init(void) { int ret; printk(KERN_INFO "I2C driver initialization function called\n"); ret = i2c_add_driver(&i2c_driver); if (ret < 0) { printk(KERN_ERR "Failed to register I2C driver\n"); return ret; } return 0; } // Unregister the I2C driver static void __exit i2c_driver_exit(void) { printk(KERN_INFO "I2C driver exit function called\n"); i2c_del_driver(&i2c_driver); } module_init(i2c_driver_init); module_exit(i2c_driver_exit); MODULE_LICENSE("GPL"); MODULE_AUTHOR("Your Name"); MODULE_DESCRIPTION("I2C driver for my device"); ``` 需要注意的是,这段代码仅为示例代码,具体实现要根据具体的I2C设备和需求进行修改。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值