VS-RK3399 開(kāi)發(fā)板上有 9 個(gè)片上 I2C 控制器,各個(gè) I2C 的使用情況如下表:
本文主要描述如何在該開(kāi)發(fā)板上配置 I2C。 配置 I2C 可分為兩大步驟:
下面以配置 GSL3680 為例。 定義和注冊(cè) I2C 設(shè)備在注冊(cè)I2C設(shè)備時(shí),需要結(jié)構(gòu)體 i2c_client 來(lái)描述 I2C 設(shè)備。然而在標(biāo)準(zhǔn)Linux中,用戶(hù)只需要提供相應(yīng)的 I2C 設(shè)備信息,Linux就會(huì)根據(jù)所提供的信息構(gòu)造 i2c_client 結(jié)構(gòu)體。 用戶(hù)所提供的 I2C 設(shè)備信息以節(jié)點(diǎn)的形式寫(xiě)到 dts 文件中,如下所示: kernel/arch/arm64/boot/dts/rockchip/rk3399-videostrong-board-mipi.dts&i2c4 { status = "okay"; gsl3680: gsl3680@41 { compatible = "gslX680"; reg = <0x41>; screen_max_x = <1536>; screen_max_y = <2048>; touch-gpio = <&gpio1 20 IRQ_TYPE_LEVEL_LOW>; reset-gpio = <&gpio0 12 GPIO_ACTIVE_HIGH>; }; }; 定義和注冊(cè) I2C 驅(qū)動(dòng)定義 I2C 驅(qū)動(dòng) 在定義 I2C 驅(qū)動(dòng)之前,用戶(hù)首先要定義變量 of_device_id 和 i2c_device_id 。 of_device_id 用于在驅(qū)動(dòng)中調(diào)用dts文件中定義的設(shè)備信息,其定義如下所示: static struct of_device_id gsl_ts_ids[] = { {.compatible = "gslX680"}, {} };定義變量 i2c_device_id: static const struct i2c_device_id gsl_ts_id[] = { {GSLX680_I2C_NAME, 0}, {} }; MODULE_DEVICE_TABLE(i2c, gsl_ts_id); i2c_driver 如下所示: static struct i2c_driver gsl_ts_driver = {.driver = { .name = GSLX680_I2C_NAME, .owner = THIS_MODULE, .of_match_table = of_match_ptr(gsl_ts_ids), }, .probe = gsl_ts_probe, .remove = gsl_ts_remove, .id_table = gsl_ts_id, }; 注:變量id_table指示該驅(qū)動(dòng)所支持的設(shè)備。 注冊(cè) I2C 驅(qū)動(dòng) 使用i2c_add_driver函數(shù)注冊(cè) I2C 驅(qū)動(dòng)。 i2c_add_driver(&gsl_ts_driver);在調(diào)用 i2c_add_driver 注冊(cè) I2C 驅(qū)動(dòng)時(shí),會(huì)遍歷 I2C 設(shè)備,如果該驅(qū)動(dòng)支持所遍歷到的設(shè)備,則會(huì)調(diào)用該驅(qū)動(dòng)的 probe 函數(shù)。 通過(guò) I2C 收發(fā)數(shù)據(jù)在注冊(cè)好 I2C 驅(qū)動(dòng)后,即可進(jìn)行 I2C 通訊。
int ret; struct i2c_adapter *adap = client->adapter; struct i2c_msg msg; msg.addr = client->addr; msg.flags = client->flags & I2C_M_TEN; msg.len = count; msg.buf = (char *)buf; ret = i2c_transfer(adap, &msg, 1); /* * If everything went ok (i.e. 1 msg transmitted), return #bytes * transmitted, else error code. */ return (ret == 1) ? count : ret; }
struct i2c_adapter *adap = client->adapter; struct i2c_msg msg; int ret; msg.addr = client->addr; msg.flags = client->flags & I2C_M_TEN; msg.flags |= I2C_M_RD; msg.len = count; msg.buf = buf; ret = i2c_transfer(adap, &msg, 1); /* * If everything went ok (i.e. 1 msg received), return #bytes received, * else error code. */ return (ret == 1) ? count : ret; } EXPORT_SYMBOL(i2c_master_recv); |
歡迎光臨 (http://www.raoushi.com/bbs/) | Powered by Discuz! X3.1 |