4412 移植mpu9250尝试

4412的板子IO都是1.8v的。只有I2C6是用了电平转换到了3.3v。所以我准备使用I2C6来驱动mpu9250

一、首先去掉占用的模块

menuconfig中去掉触摸的驱动

  • Device Drivers  --->
  • Input device support  --->
  • Touchscreens  --->
  • FT5X0X based touchscreens(去掉)

然后是去掉RC522的驱动(SPI占用I2C了)

  • Device Drivers  --->
  • SPI support   --->
  • < >   RC522 Module driver support(去掉)

-> Networking support (NET [=y])                                
-> CAN bus subsystem support (CAN [=y])                       
 -> CAN Device Drivers                                         
-> Platform CAN drivers with Netlink support (CAN_DEV [=y])

    < >   Microchip MCP251x SPI CAN controllers

二、在mach-itop4412.c中添加设备

static struct i2c_board_info i2c_devs6[] __initdata = {{I2C_BOARD_INFO("mpu9250", MPU9250_ADDRESS),},
};

这里的MPU9250_ADDRESS应该是7位的,如果写0XD0,就是MPU9250_ADDRESS>>1

然后内核编译后,烧录进开发板

cat /sys/bus/i2c/devices下就会有6-0068,这个文件了

 

写了一个空的I2C模版:

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include #include #include 
#include 
#include #define I2C6_9250_NAME "mpu9250"#define I2C_SDA6    EXYNOS4_GPC1(3)
#define I2C_SCL6    EXYNOS4_GPC1(4)static int i2c_mpu9250_probe(struct i2c_client *client, const struct i2c_device_id *id)
{printk("==%s: \n", __FUNCTION__);return 0;
}static int __devexit i2c_mpu9250_remove(struct i2c_client *client)
{i2c_set_clientdata(client, NULL);               //设置client为NULLprintk("==%s: \n", __FUNCTION__);return 0;
}static const struct i2c_device_id i2c_mpu9250_id[] = {{ I2C6_9250_NAME, 0 },{ }
};static struct i2c_driver i2c_mpu9250_driver = {.probe = i2c_mpu9250_probe,.remove = __devexit_p(i2c_mpu9250_remove),.id_table = i2c_mpu9250_id,.driver = {.name  = I2C6_9250_NAME,.owner = THIS_MODULE,},
};static void i2c_io_init()
{int ret;ret = gpio_request(I2C_SCL6, "I2C_SCL6");if(ret) {printk(KERN_ERR "failed to request TP1_EN for I2C control\n");}gpio_direction_output(I2C_SCL6, 1);s3c_gpio_cfgpin(I2C_SCL6, S3C_GPIO_OUTPUT);gpio_free(I2C_SCL6);mdelay(5);ret = gpio_request(I2C_SDA6, "I2C_SDA6");if(ret) {gpio_free(I2C_SDA6);ret = gpio_request(I2C_SDA6, "I2C_SDA6");if(ret) {printk("i2c_io_test: Fialed to request I2C_SDA6 \n");}}gpio_direction_output(I2C_SDA6, 0);mdelay(200);gpio_direction_output(I2C_SDA6, 1);s3c_gpio_cfgpin(I2C_SDA6, S3C_GPIO_OUTPUT);gpio_free(I2C_SDA6);msleep(300);printk("==%s: \n", __FUNCTION__);
}static int __init i2c_mpu9250_init(void)
{printk("==%s: \n", __FUNCTION__);i2c_io_init();return i2c_add_driver(&i2c_mpu9250_driver);
}static void __exit i2c_mpu9250_exit(void)
{printk("==%s: \n", __FUNCTION__);i2c_del_driver(&i2c_mpu9250_driver);
}late_initcall(i2c_mpu9250_init);       //延迟加载
module_exit(i2c_mpu9250_exit);MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("mpu9250");
MODULE_AUTHOR("ChenTuo");
i2c_9250.c

 

三、I2C架构层次分类  

  • 第一层:提供i2c adapter的硬件驱动,探测、初始化i2c adapter(如申请i2c的io地址和中断号),驱动soc控制的i2c adapter在硬件上产生信号(start、stop、ack)以及处理i2c中断。覆盖图中的硬件实现层  
  • 第二层:提供i2c adapter的algorithm,用具体适配器的xxx_xferf()函数来填充i2c_algorithm的master_xfer函数指针,并把赋值后的i2c_algorithm再赋值给i2c_adapter的algo指针。覆盖图中的访问抽象层、i2c核心层  
  • 第三层:实现i2c设备驱动中的i2c_driver接口,用具体的i2c device设备的attach_adapter()、detach_adapter()方法赋值给i2c_driver的成员函数指针。实现设备device与总线(或者叫adapter)的挂接。覆盖图中的driver驱动层  
  • 第四层:实现i2c设备所对应的具体device的驱动,i2c_driver只是实现设备与总线的挂接,而挂接在总线上的设备则是千差万别的,所以要实现具体设备device的write()、read()、ioctl()等方法,赋值给file_operations,然后注册字符设备(多数是字符设备)。覆盖图中的driver驱动层。
  • --------------------- 本文来自 zqixiao_09 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/zqixiao_09/article/details/50916916?utm_source=copy

四、Linux下I2C驱动体系结构三部分详细分析

4.1 IIC核心

  IIC核心提供了IIC总线驱动和设别驱动的注册、注销方法。在LInux驱动的I2C文件夹下有alogs,busses,chips三个文件夹,另外还有i2c-core.c和i2c-dev.c两个文件。

4.2 IIC总线驱动

  IIC总线驱动是对IIC硬件的,适配器可由CPU控制,IIC直接集成在CPU内部。IIC驱动包括IIC适配器数据结构体i2c_adapter、IIC适配器的algorithm数据结构i2c-algorithm和控制器产生通信信号的函数。i2c_algorithm里有iic_xfer就是i2c的低层读写实现。

4.3 IIC设备驱动

  IIC设备驱动主要包含了数据结构i2c_driver和i2c_client,我们需要根据具体设备实现其中的成员函数。

i2c-dev.c文件中实现了I2Cdriver,包括实现open,release,read,write以及ioctl等标准文件操作的接口函数。

通过I2Cdriver提供的通用方法可以访问任何一个I2C设备。

五、一些相关的数据结构

i2c_msg:

struct i2c_msg {__u16 addr;    /* slave address            */__u16 flags;
#define I2C_M_TEN        0x0010    /* this is a ten bit chip address */
#define I2C_M_RD        0x0001    /* read data, from slave to master */
#define I2C_M_NOSTART        0x4000    /* if I2C_FUNC_PROTOCOL_MANGLING */
#define I2C_M_REV_DIR_ADDR    0x2000    /* if I2C_FUNC_PROTOCOL_MANGLING */
#define I2C_M_IGNORE_NAK    0x1000    /* if I2C_FUNC_PROTOCOL_MANGLING */
#define I2C_M_NO_RD_ACK        0x0800    /* if I2C_FUNC_PROTOCOL_MANGLING */
#define I2C_M_RECV_LEN        0x0400    /* length will be first received byte */__u16 len;        /* msg length                */__u8 *buf;        /* pointer to msg data            */
};
struct i2c_msg

I2C_M_TEN:I2C默认就是8位的,如果i2c_msg的flags没有配置I2C_M_TEN的话

I2C_M_RD:标识这是一个读操作

I2C_M_NOSTART:没有起始位

I2C_M_REV_DIR_ADDR:读写标识位反转

I2C_M_IGNORE_NAK:忽略ACK和NACK

I2C_M_NO_RD_ACK:读时忽略ACK

 

转载于:https://www.cnblogs.com/ch122633/p/9686711.html


本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!

相关文章

立即
投稿

微信公众账号

微信扫一扫加关注

返回
顶部