驅動13.i2c設備驅動程式

来源:http://www.cnblogs.com/Lwd-linux/archive/2017/01/30/6358158.html
-Advertisement-
Play Games

1 分析i2c設備的識別過程i2c_add_driver i2c_register_driver driver->driver.bus = &i2c_bus_type; driver_register(&driver->driver); list_for_each_entry(adapter, &a ...


1 分析i2c設備的識別過程
i2c_add_driver
    i2c_register_driver
        driver->driver.bus = &i2c_bus_type;
        driver_register(&driver->driver);
        
        list_for_each_entry(adapter, &adapters, list) {
            driver->attach_adapter(adapter);
                        i2c_probe(adapter, &addr_data, eeprom_detect);
                            i2c_probe_address // 發出S信號,發出設備地址(來自addr_data)
                                i2c_smbus_xfer
                                    i2c_smbus_xfer_emulated
                                        i2c_transfer
                                            adap->algo->master_xfer // s3c24xx_i2c_xfer
                                            
        
2 怎麼寫I2C設備驅動程式?
2.1 分配一個i2c_driver結構體
2.2 設置
      attach_adapter // 它直接調用 i2c_probe(adap, 設備地址, 發現這個設備後要調用的函數);
      detach_client  // 卸載這個驅動後,如果之前發現能夠支持的設備,則調用它來清理
      
2.3 註冊:i2c_add_driver

3 寫代碼

  1 #include <linux/kernel.h>
  2 #include <linux/init.h>
  3 #include <linux/module.h>
  4 #include <linux/slab.h>
  5 #include <linux/jiffies.h>
  6 #include <linux/i2c.h>
  7 #include <linux/mutex.h>
  8 #include <linux/fs.h>
  9 #include <asm/uaccess.h>
 10 
 11 static unsigned short ignore[]      = { I2C_CLIENT_END };
 12 static unsigned short normal_addr[] = { 0x50, I2C_CLIENT_END }; /* 地址值是7位 */
 13                                         /* 改為0x60的話, 由於不存在設備地址為0x60的設備, 所以at24cxx_detect不被調用 */
 14 
 15 static unsigned short force_addr[] = {ANY_I2C_BUS, 0x60, I2C_CLIENT_END};
 16 static unsigned short * forces[] = {force_addr, NULL};
 17                                         
 18 static struct i2c_client_address_data addr_data = {
 19     .normal_i2c    = normal_addr,  /* 要發出S信號和設備地址並得到ACK信號,才能確定存在這個設備 */
 20     .probe        = ignore,
 21     .ignore        = ignore,
 22     //.forces     = forces, /* 強制認為存在這個設備 */
 23 };
 24 
 25 static struct i2c_driver at24cxx_driver;
 26 
 27 
 28 static int major;
 29 static struct class *cls;
 30 struct i2c_client *at24cxx_client;
 31 
 32 static ssize_t at24cxx_read(struct file *file, char __user *buf, size_t size, loff_t * offset)
 33 {
 34     unsigned char address;
 35     unsigned char data;
 36     struct i2c_msg msg[2];
 37     int ret;
 38     
 39     /* address = buf[0] 
 40      * data    = buf[1]
 41      */
 42     if (size != 1)
 43         return -EINVAL;
 44     
 45     copy_from_user(&address, buf, 1);
 46 
 47     /* 數據傳輸三要素: 源,目的,長度 */
 48 
 49     /* 讀AT24CXX時,要先把要讀的存儲空間的地址發給它 */
 50     msg[0].addr  = at24cxx_client->addr;  /* 目的 */
 51     msg[0].buf   = &address;              /**/
 52     msg[0].len   = 1;                     /* 地址=1 byte */
 53     msg[0].flags = 0;                     /* 表示寫 */
 54 
 55     /* 然後啟動讀操作 */
 56     msg[1].addr  = at24cxx_client->addr;  /**/
 57     msg[1].buf   = &data;                 /* 目的 */
 58     msg[1].len   = 1;                     /* 數據=1 byte */
 59     msg[1].flags = I2C_M_RD;                     /* 表示讀 */
 60 
 61 
 62     ret = i2c_transfer(at24cxx_client->adapter, msg, 2);
 63     if (ret == 2)
 64     {
 65         copy_to_user(buf, &data, 1);
 66         return 1;
 67     }
 68     else
 69         return -EIO;
 70 }
 71 
 72 static ssize_t at24cxx_write(struct file *file, const char __user *buf, size_t size, loff_t *offset)
 73 {
 74     unsigned char val[2];
 75     struct i2c_msg msg[1];
 76     int ret;
 77     
 78     /* address = buf[0] 
 79      * data    = buf[1]
 80      */
 81     if (size != 2)
 82         return -EINVAL;
 83     
 84     copy_from_user(val, buf, 2);
 85 
 86     /* 數據傳輸三要素: 源,目的,長度 */
 87     msg[0].addr  = at24cxx_client->addr;  /* 目的 */
 88     msg[0].buf   = val;                   /**/
 89     msg[0].len   = 2;                     /* 地址+數據=2 byte */
 90     msg[0].flags = 0;                     /* 表示寫 */
 91 
 92     ret = i2c_transfer(at24cxx_client->adapter, msg, 1);
 93     if (ret == 1)
 94         return 2;
 95     else
 96         return -EIO;
 97 }
 98 
 99 
100 static struct file_operations at24cxx_fops = {
101     .owner = THIS_MODULE,
102     .read  = at24cxx_read,
103     .write = at24cxx_write,
104 };
105 
106 static int at24cxx_detect(struct i2c_adapter *adapter, int address, int kind)
107 {    
108     printk("at24cxx_detect\n");
109 
110     /* 構構一個i2c_client結構體: 以後收改數據時會用到它 */
111     at24cxx_client = kzalloc(sizeof(struct i2c_client), GFP_KERNEL);
112     at24cxx_client->addr    = address;
113     at24cxx_client->adapter = adapter;
114     at24cxx_client->driver  = &at24cxx_driver;
115     strcpy(at24cxx_client->name, "at24cxx");
116     i2c_attach_client(at24cxx_client);
117     
118     major = register_chrdev(0, "at24cxx", &at24cxx_fops);
119 
120     cls = class_create(THIS_MODULE, "at24cxx");
121     class_device_create(cls, NULL, MKDEV(major, 0), NULL, "at24cxx"); /* /dev/at24cxx */
122     
123     return 0;
124 }
125 
126 static int at24cxx_attach(struct i2c_adapter *adapter)
127 {
128     return i2c_probe(adapter, &addr_data, at24cxx_detect);
129 }
130 
131 static int at24cxx_detach(struct i2c_client *client)
132 {
133     printk("at24cxx_detach\n");
134     class_device_destroy(cls, MKDEV(major, 0));
135     class_destroy(cls);
136     unregister_chrdev(major, "at24cxx");
137 
138     i2c_detach_client(client);
139     kfree(i2c_get_clientdata(client));
140 
141     return 0;
142 }
143 
144 
145 /* 1. 分配一個i2c_driver結構體 */
146 /* 2. 設置i2c_driver結構體 */
147 static struct i2c_driver at24cxx_driver = {
148     .driver = {
149         .name    = "at24cxx",
150     },
151     .attach_adapter = at24cxx_attach,
152     .detach_client  = at24cxx_detach,
153 };
154 
155 static int at24cxx_init(void)
156 {
157     i2c_add_driver(&at24cxx_driver);
158     return 0;
159 }
160 
161 static void at24cxx_exit(void)
162 {
163     i2c_del_driver(&at24cxx_driver);
164 }
165 
166 module_init(at24cxx_init);
167 module_exit(at24cxx_exit);
168 
169 MODULE_LICENSE("GPL");
i2c設備驅動程式

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 用了一段時間的MONO,現在MONO也支持了ENTITY FRAMEWORK 6。但是實際上在LINUX環境里用MYSQL還是會有很多坑。並且之前在網路游戲服務端SCUT上擴展一些功能時候也遇到了一些因為MONO和.NET行為方式不一致的坑耗掉了不少時間。使用mono雖然可以節約性能開銷,但是犧牲掉 ...
  • 今天在騰訊雲領取了一個免費試用的windows伺服器,我在deepin下想使用遠程桌面來連接windows,找到了這個工具rdesktop,感覺挺好用的所以分享一下。 安裝rdesktop 使用方法 ...
  • 安裝 啟動 ...
  • 驅動程式的調試一. 列印: printk, 自製proc文件UBOOT傳入console=ttySAC0(串口) console=tty1(LCD)1. 內核處理UBOOT傳入的參數console_setup add_preferred_console // 我想用名為"ttySAC0"的控制台,先 ...
  • 第一、安裝好Win10系統,不需要安裝其他激活工具。第二、是刪除預設序列號,打開命令提示符(管理員),運行 slmgr.vbs -upk,可提示已卸載了序列號。 slmgr /ipk W269N-WFGWX-YVC9B-4J6C9-T83GXslmgr /skms kms.xspace.inslmg ...
  • 給自家的Ubuntu下載軟體速度有點慢,畢竟是從國外下載軟體,就想更換到國內比較好的更新源(就是這些軟體所在的伺服器),一般直接百度Ubuntu更新源就能出來一大堆,這時候最好是找和自己Ubuntu版本一致的更新源,我的Ubuntu版本是16.04,下麵是我找到的一個比較好的更新源 http://w ...
  • 1. 點燈法 可直接使用bl led_flicker來使用該程式。2. 串口列印及棧初步分析2.1 使用的前提是串口已經初始化完畢且可以正常使用 直接在需要定位錯誤的地方加上printk語句,看串口是否有相應的輸出即可。2.2 nand_setup的分析 a.棧和局部變數是臨時生成的 b.局部變數的 ...
  • 原本的字元設備只能有255個驅動程式,原因是一個主設備號占用了0~255的次設備號 把register_chrdev展開可得到一下幾個部分:register_chrdev_region/alloc_chrdev_region,cdev_init,cdev_add 參照register_chrdev的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...