首先,我們要了解Linux系統提供的I2C操作函數怎麼使用,上篇文章已經提到過,對I2C設備的讀寫,Linux系統提供了多種接口,這些接口可以在內核的 i2c.h 中找到,這裡我主要介紹下面這組讀寫接口:
- extern int i2c_master_send(struct i2c_client *,const char* ,int);
- extern int i2c_master_recv(struct i2c_client *,char* ,int);
第一個參數是 i2c_client 對象指針,第二個參數是要傳輸的數據buffer指針,第三個參數為buffer的大小。
接口函數已經有了,下面我們要解決的問題就是以何種形式/規則去使用這些接口才能正確地讀寫芯片的相關寄存器。
首先,我們需要查詢芯片手冊,找到芯片手冊中,關於寄存器的I2C讀寫時序,其實,大多數芯片的I2C寄存器的讀寫時序都是一樣的,下面我還是以手頭的TVP5158芯片為例。
首先分析寫操作,該芯片的手冊中給出的I2C寄存器寫時序圖如下:
650) this.width=650;" border=0>
從上圖可以看出,真正需要執行寫操作的有兩處,Step4 和 Step6 ,Step4首先寫入寄存器的偏移地址,而Step6則是寫入到該寄存器的值。由此已經很清楚了,對於寫I2C寄存器,我們需要做的就是給 i2c_master_send 函數傳入兩個字節的數據即可,第一個字節為寄存器的地址,第二個字節為要寫入寄存器的數據。示例如下:
- static int tvp5158_i2c_write( struct i2c_client* client,uint8_t reg,uint8_t data)
- {
- unsigned char buffer[2];
- buffer[0] = reg;
- buffer[1] = data;
- if( 2!= i2c_master_send(client,buffer,2) ) {
- printk( KERN_ERR " tvp5158_i2c_write fail! \n" );
- return -1;
- }
- return 0;
- }
其實挺簡單的,沒有什麼復雜的代碼。下面再看看讀時序。
650) this.width=650;" border=0>
由上圖可以,讀時序需要做的操作是,先向I2C總線上寫入需要讀的寄存器地址,然後讀I2C總線上的值。代碼寫起來也不難,示例如下:
- static int tvp5158_i2c_read( struct i2c_client* client,uint8_t reg,uint8_t *data)
- {
- // write reg addr
- if( 1!= i2c_master_send(client,®,1) ) {
- printk( KERN_ERR " tvp5158_i2c_read fail! \n" );
- return -1;
- }
- // wait
- msleep(10);
- // read
- if( 1!= i2c_master_recv(client,data,1) ) {
- printk( KERN_ERR " tvp5158_i2c_read fail! \n" );
- return -1;
- }
- return 0;
- }
到此為止,Linux下具體如何封裝讀寫芯片寄存器的方法已經介紹完畢,其實並不復雜。