找回密码
 注册
关于网站域名变更的通知
查看: 545|回复: 3
打印 上一主题 下一主题

linux学习之路_or1200下linux的i2c(二)

[复制链接]
  • TA的每日心情

    2019-11-20 15:22
  • 签到天数: 2 天

    [LV.1]初来乍到

    跳转到指定楼层
    1#
    发表于 2021-8-5 10:26 | 只看该作者 |只看大图 回帖奖励 |倒序浏览 |阅读模式

    EDA365欢迎您登录!

    您需要 登录 才可以下载或查看,没有帐号?注册

    x
    ! N/ Y! @/ x7 g
    与前一节添加GPIO Controller的步骤一样,进入虚拟机中的关于SOC的设备树文件目录openrisc-3.1\arch\openrisc\boot\dts添加关于板子上I2C器件的设备描述
    3 U! |9 |3 s9 W6 i0 h# T
    # }4 a1 S% F+ j% `: |$ H0 ? % Q' x$ \) m( [5 w8 D1 B0 a
    / f5 u& x4 L, \+ a) K3 S
            open之,在其中加入关于板子上的I2C器件的描述,例如我板子上有个AT24c08,所以我就添加eeprom设备,形式如下:8 D# b: o9 V6 H+ ]3 Q$ A
    # t- b' M2 J+ `* C; F/ K/ V, r
    • i2c0: ocores@93000000 {
    •              compatible = "opencores,i2c-ocores";
    •              reg = <0x93000000 0x8>;
    •              interrupts = <5>;
    •              regstep = <1>;
    •              clock-frequency = <40000000>;
    •              #address-cells = <1>;
    •              #size-cells = <0>;
    •              eeprom@54 {
    •                     compatible = "at24";
    •                     reg = <0x54>;
    •     };
    • };) z; q2 Y, A. I7 K5 b5 v
       : H8 R" c7 b8 o' h& G# }
            至于关于其中编写方法后述。
    4 k5 r6 |8 B/ p( _- `4 D6 z) ]: r6 o- N5 j# u
            然后,在命令行中进入linux源码目录,然后需要配置内核加入上面添加的I2C的总线驱动,这是因为关于这个ipcore的驱动社区上的大虾已经编写好,现在先熟悉和使用,在后面再去详细分析这个总线驱动如何编写。7 D) t- u  K( w0 D8 d* j# |
    " ~1 X  i. L. Y4 M3 _7 p( B- s

    7 J- ~: j1 o; v6 m
      q0 D5 {" G; _( B: m: ~+ p' x        内核配置,进入General setup项
    - x+ f5 J. L9 b! P# g- C8 v- s3 l" g
    & A' r& U3 m0 ^

      R# a: j+ j* |  T% G  @        将Prompt for development and/or incomplete code/drivers打开
    % m: |/ J# n# Q2 l. X
    5 n; O" B8 ]% m" n $ a7 d. _" P& X9 P/ x! a
    2 L1 p- S  j. G. s0 L
            Exit后,到Device Drivers项中
    ) |% u# h3 w3 C% i2 R# K0 e2 t
    5 S' }: y# \8 @& u% m
    ( s' ~3 l+ [6 x- A( k& B
    $ h0 s* K3 f% a6 V8 M        将I2C support选择编入内核中
    7 k2 M; [) F7 [6 s
    2 r5 K5 I) i" f& a' r
    7 |& i% w1 @2 q4 P1 I; s" \' [  }% Q
            进入,将I2C device inteRFace选中编入内核,我们第一个应用层的测试文件就基于内核编写好的设备接口文件来写
    & W' |! n6 f+ F8 j, ~! C5 J* O! S" b( K

    ! j  p, V! f. R( Z( W& _+ `1 j: L3 |5 q1 d& a* O; W) F
            然后选择I2C Hardware Bus support中: {8 V" B9 A) L9 Y$ S. a( A$ V# x
    5 M' L% N2 G7 s) q6 J

    & Z, F$ d; l, W) K  }' ~& k
    . A, m4 R: m, s- G% Q( ?  }8 Y: U( @        将OpenCores I2C Controller选择编入内核8 y" Y0 u7 f7 e% P  E
    ! q# Z1 i% O. Z# E. z
    ; R1 o  s0 w, h0 o, H+ z2 D  t0 Z: U) i
    . V' @: }) P6 O
            好,最后exit出来选择保存,重新make
    - f/ x& V0 k( R6 s
    4 z$ h4 l4 v* d; x2 l, t+ k# R
    6 N% D; N' ?+ d1 P0 i
    6 {5 `: _! u8 N% F* ?+ p) y4 Q        之后的步骤按照之前移植linux的步骤一样,生成uImage镜像,然后下载到开发板上面即可。6 p3 {* n! F  q
    ( g" }+ C, f$ H- B
            然后在boot linux的过程和进入/dev目录下可以看到i2c bus driver和生成设备文件i2c-x
    ! [, i% _/ c) S5 U5 X
    ! Y% x, y. Q8 z* ^7 E+ M
    9 N. ?7 P! r6 K
    ! `) O- C# T' j$ u+ E7 v        至此,i2c的总线驱动就添加到内核了。3 H% A' T1 [& i9 I8 E9 ?

    / `2 @4 C) @. z9 q' Y0 q/ v. x6 A" b        根据我自己的理解:* s  U5 z3 C7 m/ R6 M1 y; ^5 h

    6 A6 q0 I& ?/ ~* I        1.对于i2c controller来说,编写的驱动程序称为总线驱动,例如上面在or1200_soc中添加的i2c控制器ipcore,为之编写的驱动程序就是总线驱动。; ]6 _# W$ e$ ], \1 W  y$ J0 a
    $ J+ g& U6 u. ?% \( ^  h/ M# Y
            2.对于挂在i2c总线上的slave device来说,为之编写的驱动程序称为设备驱动,例如我板子上挂在这个ipcore上的AT24c08的eeprom。, [5 ~' E9 u: ?- y5 y3 |
    9 u9 e* a& z3 x+ B& o& B6 h8 \
            3.linux下的i2c子系统中还有一个i2c核心模块来为总线驱动和设备驱动服务,完成注册,删除······功能。& t+ J/ Q( Z/ S+ h
    4 y, I& M& P& [2 J. P0 E* {# T+ Z
            现在说明一下:在openrisc-3.1\drivers\i2c\busses目录下的i2c-ocores.c文件为soc上添加的i2c controller提供了总线驱动,即在刚才的内核配置中已经选择编入了内核,所以可以略过总线驱动的编写,在下节我们着重分析这个总线驱动。
    ! x7 t. p+ v8 \" y, N6 L2 Q. |- W0 S2 P- P' a7 g4 V; B+ `9 _

    / A# }; C$ f7 D! F
    - _- ]/ K+ `  _# y9 N        现在可以先打开这个文件,里面的comment中就有关于device-tree文件如何加入i2c device描述的说明
    + y' Y( U( w) O6 [1 L
    % k) G9 L6 X% Y
    • /*
    • * Device tree configuration:
    • *
    • * Required properties:
    • * - compatible      : "opencores,i2c-ocores"
    • * - reg             : bus address start and address range size of device
    • * - interrupts      : interrupt number
    • * - regstep         : size of device registers in bytes
    • * - clock-frequency : frequency of bus clock in Hz
    • *
    • * Example:
    • *
    • *  i2c0: ocores@a0000000 {
    • *              compatible = "opencores,i2c-ocores";
    • *              reg = <0xa0000000 0x8>;
    • *              interrupts = <10>;
    • *
    • *              regstep = <1>;
    • *              clock-frequency = <20000000>;
    • *
    • * -- Devices connected on this I2C bus get
    • * -- defined here; address- and size-cells
    • * -- apply to these child devices
    • *
    • *              #address-cells = <1>;
    • *              #size-cells = <0>;
    • *
    • *              dummy@60 {
    • *                     compatible = "dummy";
    • *                     reg = <60>;
    • *              };
    • *  };
    • *
    • */
      4 y$ P$ e; e. {1 q! o& w1 _; f

    4 f: v5 T; M# Q& J8 [! t; y% Z
    5 T+ v5 y! B% o' I# f. T# p  O  @$ u* _: S
            在openrisc-3.1\drivers\i2c目录下的i2c-dev.c文件中,linux为我们提供了虚拟的统一的设备文件接口,如果我们懒得去编写设备驱动程序时,可以利用这个接口直接在应用层上完成对i2c总线的slave device操作。
    5 Y! i; R, j6 h0 b- s0 E9 p" `! b- H& \2 c$ G" n5 S

    8 Q7 J* a* `; O# _9 L8 \" j; E9 M, P0 u6 z, G* }$ V' e5 t% g
            所以在这一节当中,我们不编写总线驱动和设备驱动,先运用系统提供的文件接口进行操作,具体的代码主要参考hongtao_liu老师提供的代码。( D6 Q4 x) }  [) B

    # w$ |7 F/ M  b( P( j7 ^2 R* G# B
    • #include <stdio.h>
    • #include <string.h>
    • #include <stdlib.h>
    • #include <fcntl.h> // open()
    • #include <sys/ioctl.h>
    • #include <unistd.h> // read() write() close() usleep()
    • #include <linux/types.h>
    • #include <linux/i2c.h> // i2c_msg
    • #include <linux/i2c-dev.h> // i2c_rdwr_ioctl_data
    • #define DEVICE_NAME "/dev/i2c-0"
    • #define MAX_MSG 2
    • #define EEPROM_ADDR 0x54
    • //--------------------------------------------- main -----------------------------------------------------
    • int main(){
    •     int fd;
    •     int ret;
    •     struct i2c_rdwr_ioctl_data eeprom_data;
    •     /* open device file */
    •     printf("\nstart simple_i2c application test ! \n");
    •     fd = open(DEVICE_NAME, O_RDWR);
    •         if (fd == -1){
    •                 printf("open device %s error !\n",DEVICE_NAME);
    •         }
    •     else
    •         printf("open device file successful, fd = %d\n",fd);
    •     /* set parameter for msg */
    •     eeprom_data.nmsgs = MAX_MSG;
    •     eeprom_data.msgs=(struct i2c_msg *)malloc(eeprom_data.nmsgs * sizeof(struct i2c_msg));
    •     if( !eeprom_data.msgs ){
    •         printf("malloc error...\n");
    •         return -1;
    •     }
    •     /* timeout & retry */
    •     ioctl(fd, I2C_TIMEOUT, 1);
    •     ioctl(fd, I2C_RETRIES, 2);
    •     /* write data to eeprom */
    •     eeprom_data.nmsgs = 1;
    •     (eeprom_data.msgs[0]).len = 2;
    •     (eeprom_data.msgs[0]).addr = EEPROM_ADDR;
    •     (eeprom_data.msgs[0]).flags = 0;
    •     (eeprom_data.msgs[0]).buf = (unsigned char*)malloc(2);
    •     (eeprom_data.msgs[0]).buf[0] = 0x10;
    •     (eeprom_data.msgs[0]).buf[1] = 0x50;
    •     ret = ioctl(fd, I2C_RDWR, (unsigned long)&eeprom_data);
    •     if(ret < 0){
    •         printf("ioctl error...\n");
    •     }
    •     /* read data from eeprom */
    •     eeprom_data.nmsgs = 2;
    •     (eeprom_data.msgs[0]).len = 1;
    •     (eeprom_data.msgs[0]).addr = EEPROM_ADDR;
    •     (eeprom_data.msgs[0]).flags = 0;
    •     (eeprom_data.msgs[0]).buf[0] = 0x10;
    •     (eeprom_data.msgs[1]).len = 1;
    •     (eeprom_data.msgs[1]).addr = EEPROM_ADDR;
    •     (eeprom_data.msgs[1]).flags = I2C_M_RD;
    •     (eeprom_data.msgs[1]).buf = (unsigned char*)malloc(1);
    •     (eeprom_data.msgs[1]).buf[0] = 0;
    •     ret = ioctl(fd, I2C_RDWR, (unsigned long)&eeprom_data);
    •     if(ret < 0){
    •         printf("ioctl error...\n");
    •     }
    •     printf("buff[0]=%x\n",(eeprom_data.msgs[1]).buf[0]);
    •     /* close device file */
    •     ret = close(fd);
    •     printf ("ret=%d\n",ret);
    •     printf ("end simple_i2c test...\n");
    •     /*  */
    •     return 0;
    • }
      : R6 [: t0 I3 Z" B; x& `

    ! H% @3 `8 `/ O) w/ u
      A$ i5 v4 u9 p  s2 Q2 @# |2 D1 P9 O  ~& U% O* W

    , L# y5 U4 }) u+ q3 L4 T        编写完后在虚拟机上编译,老规矩,在板子上lrz后修改文件属性
    & v! C% f; Q0 E$ G. _2 B! n/ J5 [7 U+ Q! O, F0 T  p1 P1 L3 q
    9 U+ ^  ?5 ?: j6 q; d9 M4 `% F" M: |6 p# T
    5 ]) v: e4 r- F5 Y1 h
            然后可以多次修改其中写入地址和数据的代码,看看是否能正确读写& s" x  I; a0 D2 p

    # ]8 v% N. v. f7 L- ?7 Z1 n 5 [( _+ n: a, H$ C
    8 G" w# r2 U8 c5 F* f
            至此,有时间修改一下这个测试文件成比较实用的文件,下节对这个ipcore的i2c bus driver的分析,即对openrisc-3.1\drivers\i2c\busses目录下的i2c-ocores.c的总线驱动稍作分析,当是学习如何编写总线驱动。
      m, {. e- Z# [* a- c

    该用户从未签到

    2#
    发表于 2021-8-5 14:17 | 只看该作者
    or1200下linux的i2c

    该用户从未签到

    3#
    发表于 2021-8-5 18:53 | 只看该作者
    or1200下linux的i2c
  • TA的每日心情
    慵懒
    2020-6-13 15:46
  • 签到天数: 1 天

    [LV.1]初来乍到

    4#
    发表于 2021-8-5 18:54 | 只看该作者
    or1200下linux的i2c
    您需要登录后才可以回帖 登录 | 注册

    本版积分规则

    关闭

    推荐内容上一条 /1 下一条

    EDA365公众号

    关于我们|手机版|EDA365电子论坛网 ( 粤ICP备18020198号-1 )

    GMT+8, 2025-11-24 14:05 , Processed in 0.203125 second(s), 27 queries , Gzip On.

    深圳市墨知创新科技有限公司

    地址:深圳市南山区科技生态园2栋A座805 电话:19926409050

    快速回复 返回顶部 返回列表