class.c 添加中文註釋(2)

来源:http://www.cnblogs.com/hackfun/archive/2016/08/26/5811395.html
-Advertisement-
Play Games

...


  1 /* Class Device Stuff */
  2 
  3 int class_device_create_file(struct class_device * class_dev,
  4                  const struct class_device_attribute * attr)
  5 {
  6     int error = -EINVAL;
  7     
  8     /* [cgw]: class_dev指針不為空 */
  9     if (class_dev)
 10         /* [cgw]: 為class_dev->kobj對象創建一個屬性文件 */
 11         error = sysfs_create_file(&class_dev->kobj, &attr->attr);
 12     return error;
 13 }
 14 
 15 void class_device_remove_file(struct class_device * class_dev,
 16                   const struct class_device_attribute * attr)
 17 {
 18     /* [cgw]: class_dev指針不為空 */
 19     if (class_dev)
 20         /* [cgw]: 刪除class_dev->kobj對象對應的一個屬性文件 */
 21         sysfs_remove_file(&class_dev->kobj, &attr->attr);
 22 }
 23 
 24 int class_device_create_bin_file(struct class_device *class_dev,
 25                  struct bin_attribute *attr)
 26 {
 27     int error = -EINVAL;
 28 
 29     /* [cgw]: class_dev指針不為空 */
 30     if (class_dev)
 31         /* [cgw]: 為class_dev->kobj對象創建一個BIN文件 */
 32         error = sysfs_create_bin_file(&class_dev->kobj, attr);
 33     return error;
 34 }
 35 
 36 void class_device_remove_bin_file(struct class_device *class_dev,
 37                   struct bin_attribute *attr)
 38 {
 39     /* [cgw]: class_dev指針不為空 */
 40     if (class_dev)
 41         /* [cgw]: 刪除class_dev->kobj對象對應的一個BIN文件 */
 42         sysfs_remove_bin_file(&class_dev->kobj, attr);
 43 }
 44 
 45 static ssize_t
 46 class_device_attr_show(struct kobject * kobj, struct attribute * attr,
 47                char * buf)
 48 {
 49     /* [cgw]: 找出包含這個attr的struct class_device_attribute *指針 */
 50     struct class_device_attribute * class_dev_attr = to_class_dev_attr(attr);
 51     /* [cgw]: 找出包含這個kobj的struct class_device *指針
 52       */
 53     struct class_device * cd = to_class_dev(kobj);
 54     ssize_t ret = 0;
 55 
 56     /* [cgw]: class_dev_attr->show指針不為空 */
 57     if (class_dev_attr->show)
 58         /* [cgw]: 調用class_dev_attr->show方法 */
 59         ret = class_dev_attr->show(cd, buf);
 60     return ret;
 61 }
 62 
 63 static ssize_t
 64 class_device_attr_store(struct kobject * kobj, struct attribute * attr,
 65             const char * buf, size_t count)
 66 {
 67     /* [cgw]: 找出包含這個attr的struct class_device_attribute *指針 */
 68     struct class_device_attribute * class_dev_attr = to_class_dev_attr(attr);
 69     /* [cgw]: 找出包含這個kobj的struct class_device *指針
 70       */
 71     struct class_device * cd = to_class_dev(kobj);
 72     ssize_t ret = 0;
 73 
 74     /* [cgw]: class_dev_attr->store指針不為空 */
 75     if (class_dev_attr->store)
 76         /* [cgw]: 調用class_dev_attr->store方法 */
 77         ret = class_dev_attr->store(cd, buf, count);
 78     return ret;
 79 }
 80 
 81 static struct sysfs_ops class_dev_sysfs_ops = {
 82     .show    = class_device_attr_show,
 83     .store    = class_device_attr_store,
 84 };
 85 
 86 static void class_dev_release(struct kobject * kobj)
 87 {
 88     /* [cgw]: 找出包含這個kobj的struct class_device *指針 */
 89     struct class_device *cd = to_class_dev(kobj);
 90     /* [cgw]: cls指向class_device的成員class */
 91     struct class * cls = cd->class;
 92 
 93     pr_debug("device class '%s': release.\n", cd->class_id);
 94 
 95     /* [cgw]: 釋放這個class_device的class_device_attribute記憶體空間
 96           * 並把class_device_attribute指針置為空
 97           */
 98     kfree(cd->devt_attr);
 99     cd->devt_attr = NULL;
100 
101     /* [cgw]: 調用class_device->release 方法,
102           * 釋放這個已分配的struct class_device *的記憶體空間
103           */
104     if (cd->release)
105         cd->release(cd);
106     else if (cls->release)
107         /* [cgw]: 調用class_device->class->release 方法
108               * 釋放這個已分配的struct class *的記憶體空間
109               */
110         cls->release(cd);
111     else {
112         printk(KERN_ERR "Class Device '%s' does not have a release() function, "
113             "it is broken and must be fixed.\n",
114             cd->class_id);
115         WARN_ON(1);
116     }
117 }
118 
119 static struct kobj_type ktype_class_device = {
120     .sysfs_ops    = &class_dev_sysfs_ops,
121     .release    = class_dev_release,
122 };
123 
124 static int class_uevent_filter(struct kset *kset, struct kobject *kobj)
125 {
126     /* [cgw]: 找到kobj所屬的類型ktype */
127     struct kobj_type *ktype = get_ktype(kobj);
128 
129     /* [cgw]: 這個kobj的類型ktype屬於ktype_class_device */
130     if (ktype == &ktype_class_device) {
131         /* [cgw]: 找到包含這個kobj 的class_dev指針
132               */
133         struct class_device *class_dev = to_class_dev(kobj);
134         /* [cgw]: class_dev->class指針不為空 */
135         if (class_dev->class)
136             return 1;
137     }
138     return 0;
139 }
140 
141 static const char *class_uevent_name(struct kset *kset, struct kobject *kobj)
142 {
143     /* [cgw]: 找到包含這個kobj 的class_dev指針
144       */
145     struct class_device *class_dev = to_class_dev(kobj);
146 
147     /* [cgw]: 返回class_dev下class的名字 */
148     return class_dev->class->name;
149 }
150 
151 #ifdef CONFIG_SYSFS_DEPRECATED
152 char *make_class_name(const char *name, struct kobject *kobj)
153 {
154     char *class_name;
155     int size;
156     /* [cgw]: 找出這個kobj的名字
157           * 計算兩個字元串的總長度並+2,
158           */
159     size = strlen(name) + strlen(kobject_name(kobj)) + 2;
160     /* [cgw]: 分配一個size位元組的記憶體空間 */
161     class_name = kmalloc(size, GFP_KERNEL);
162     /* [cgw]: 分配失敗 */
163     if (!class_name)
164         return NULL;
165     /* [cgw] : 把名字name填裝到class_name */
166     strcpy(class_name, name);
167     /* [cgw] : 在名字name末尾添加":" */
168     strcat(class_name, ":");
169     /* [cgw] : 合併字元串,name+":"+kobj->k_name */
170     strcat(class_name, kobject_name(kobj));
171     return class_name;
172 }
173 
174 static int make_deprecated_class_device_links(struct class_device *class_dev)
175 {
176     char *class_name;
177     int error;
178 
179     /* [cgw]: class_dev->dev指針為空 */
180     if (!class_dev->dev)
181         return 0;
182     /* [cgw]: 合併class_dev->class->name和class_dev->kobj->k_name
183           */
184     class_name = make_class_name(class_dev->class->name, &class_dev->kobj);
185     /* [cgw]: 合併成功 */
186     if (class_name)
187         /* [cgw]: 在class_dev->dev->kobj對象目錄下創建一個鏈表
188               * 設置這個鏈表的名字為class_name
189               */
190         error = sysfs_create_link(&class_dev->dev->kobj,
191                       &class_dev->kobj, class_name);
192     else
193         error = -ENOMEM;
194     /* [cgw]: 釋放class_name記憶體空間,class_name在這裡只做臨時作用 */
195     kfree(class_name);
196     return error;
197 }
198 
199 static void remove_deprecated_class_device_links(struct class_device *class_dev)
200 {
201     char *class_name;
202 
203     /* [cgw]: class_dev->dev指針為空 */
204     if (!class_dev->dev)
205         return;
206         
207     /* [cgw]: 合併class_dev->class->name和class_dev->kobj->k_name
208           */
209     class_name = make_class_name(class_dev->class->name, &class_dev->kobj);
210     if (class_name)
211         /* [cgw] : 從class_dev->dev->kobj對象目錄下刪除一個名字為
212               * class_name的鏈表
213               */
214         sysfs_remove_link(&class_dev->dev->kobj, class_name);
215     /* [cgw] : 釋放class_name的記憶體空間 */
216     kfree(class_name);
217 }
218 #else
219 static inline int make_deprecated_class_device_links(struct class_device *cd)
220 { return 0; }
221 static void remove_deprecated_class_device_links(struct class_device *cd)
222 { }
223 #endif
224 
225 static int class_uevent(struct kset *kset, struct kobject *kobj, char **envp,
226              int num_envp, char *buffer, int buffer_size)
227 {
228     /* [cgw]: 找出包含這個kobj的class_device結構體指針 */
229     struct class_device *class_dev = to_class_dev(kobj);
230     struct device *dev = class_dev->dev;
231     int i = 0;
232     int length = 0;
233     int retval = 0;
234 
235     pr_debug("%s - name = %s\n", __FUNCTION__, class_dev->class_id);
236 
237     /* [cgw]: 主設備號不為0 */
238     if (MAJOR(class_dev->devt)) {
239         /* [cgw]: 格式化"MAJOR= MAJOR(class_dev->devt)"填裝到buffer中
240               * 並把buffer指針儲存在envp數組
241               */
242         add_uevent_var(envp, num_envp, &i,
243                    buffer, buffer_size, &length,
244                    "MAJOR=%u", MAJOR(class_dev->devt));
245 
246         /* [cgw]: 格式化"MAJOR= MINOR(class_dev->devt)"填裝到buffer中
247               * 並把buffer指針儲存在envp數組
248               */
249         add_uevent_var(envp, num_envp, &i,
250                    buffer, buffer_size, &length,
251                    "MINOR=%u", MINOR(class_dev->devt));
252     }
253 
254     /* [cgw]: dev指針不為空 */
255     if (dev) {
256         /* [cgw]: 獲得dev->kobj所在的路徑 */
257         const char *path = kobject_get_path(&, GFP_KERNEL);
258         /* [cgw]: 獲取成功 */
259         if (path) {
260             /* [cgw]: 格式化"PHYSDEVPATH=path"填裝到buffer中
261                       * 並把buffer指針儲存在環境變數envp數組,i+1
262                       */
263             add_uevent_var(envp, num_envp, &i,
264                        buffer, buffer_size, &length,
265                        "PHYSDEVPATH=%s", path);
266             /* [cgw]: 釋放path的記憶體空間 */
267             kfree(path);
268         }
269 
270         /* [cgw]: dev->bus指針不為空 */
271         if (dev->bus)
272             /* [cgw]: 格式化"PHYSDEVBUS=dev->bus->name"填裝到buffer中
273                       * 並把buffer指針儲存在環境變數envp數組,i+1
274                       */
275             add_uevent_var(envp, num_envp, &i,
276                        buffer, buffer_size, &length,
277                        "PHYSDEVBUS=%s", dev->bus->name);
278 
279         /* [cgw]: dev->driver指針不為空 */
280         if (dev->driver)
281             /* [cgw]: 格式化"PHYSDEVDRIVER=dev->driver->name"填裝到buffer中
282                       * 並把buffer指針儲存在環境變數envp數組,i+1
283                       */
284             add_uevent_var(envp, num_envp, &i,
285                        buffer, buffer_size, &length,
286                        "PHYSDEVDRIVER=%s", dev->driver->name);
287     }
288 
289     /* terminate, set to next free slot, shrink available space */
290     /* [cgw]: 清空下一個環境變數,envp[i]是一個指針,把這個指針置0 */
291     envp[i] = NULL;
292     /* [cgw]: 重置envp的指針,指向下一個環境變數envp[i] */
293     envp = &envp[i];
294     /* [cgw]: 計算環境變數數組剩下的可用大小 */
295     num_envp -= i;
296     /* [cgw]: 重置buffer指針,指向buffer + length */
297     buffer = &buffer[length];
298     /* [cgw]: 計算環境變數buffer剩下的可用大小 */
299     buffer_size -= length;
300 
301     /* [cgw]: class_dev->uevent指針不為空 */
302     if (class_dev->uevent) {
303         /* have the class device specific function add its stuff */
304         /* [cgw]: 調用class_dev->uevent方法 */
305         retval = class_dev->uevent(class_dev, envp, num_envp,
306                         buffer, buffer_size);
307         if (retval)
308             pr_debug("class_dev->uevent() returned %d\n", retval);
309     } else if (class_dev->class->uevent) {
310         /* have the class specific function add its stuff */
311         /* [cgw]: 調用class_dev->class->uevent方法 */
312         retval = class_dev->class->uevent(class_dev, envp, num_envp,
313                            buffer, buffer_size);
314         if (retval)
315             pr_debug("class->uevent() returned %d\n", retval);
316     }
317 
318     return retval;
319 }
320 
321 static struct kset_uevent_ops class_uevent_ops = {
322     .filter =    class_uevent_filter,
323     .name =        class_uevent_name,
324     .uevent =    class_uevent,
325 };
326 
327 static decl_subsys(class_obj, &ktype_class_device, &class_uevent_ops);
328 
329 
330 static int class_device_add_attrs(struct class_device * cd)
331 {
332     int i;
333     int error = 0;
334     struct class * cls = cd->class;
335 
336     /* [cgw]: cls->class_dev_attrs指針不為空,即分配了class device 的屬性 */
337     if (cls->class_dev_attrs) {
338         /* [cgw]: 歷遍class_dev_attrs[]數組,如果該屬性名字不為空,則
339               * 對應地創建一個屬性文件
340               */
341         for (i = 0; attr_name(cls->class_dev_attrs[i]); i++) {
342             /* [cgw]: 創建一個class device屬性文件 */
343             error = class_device_create_file(cd,
344                              &cls->class_dev_attrs[i]);
345             if (error)
346                 goto Err;
347         }
348     }
349  Done:
350     return error;
351  Err:
352     /* [cgw]: 刪除class device的所有屬性文件 */
353     while (--i >= 0)
354         class_device_remove_file(cd,&cls->class_dev_attrs[i]);
355     goto Done;
356 }
357 
358 static void class_device_remove_attrs(struct class_device * cd)
359 {
360     int i;
361     struct class * cls = cd->class;
362 
363     /* [cgw]: cls->class_dev_attrs指針不為空,即此前已經添加了這個屬性列表 */
364     if (cls->class_dev_attrs) {
365         /* [cgw]: 歷遍class_dev_attrs[]數組,如果該屬性名字不為空,則
366               * 對應地刪除一個屬性文件
367               */
368         for (i = 0; attr_name(cls->class_dev_attrs[i]); i++)
369             /* [cgw]: 刪除一個屬性文件 */
370             class_device_remove_file(cd,&cls->class_dev_attrs[i]);
371     }
372 }
373 
374 static int class_device_add_groups(struct class_device * cd)
375 {
376     int i;
377     int error = 0;
378 
379     /* [cgw]: cd->groups指針不為空 */
380     if (cd->groups) {
381         /* [cgw]: 歷遍groups[],對應每個groups創建一個屬性組 */
382         for (i = 0; cd->groups[i]; i++) {
383             /* [cgw]: 創建一個屬性組 */
384             error = sysfs_create_group(&cd->kobj, cd->groups[i]);
385             /* [cgw]: 創建失敗 */
386             if (error) {
387                 /* [cgw]: 刪除所有屬性組 */
388                 while (--i >= 0)
389                     /* [cgw]: 刪除一個屬性組 */
390                     sysfs_remove_group(&cd->kobj, cd->groups[i]);
391                 goto out;
392             }
393         }
394     }
395 out:
396     return error;
397 }
398 
399 static void class_device_remove_groups(struct class_device * cd)
400 {
401     int i;
402     /* [cgw]: cd->groups指針不為空 */
403     if (cd->groups) {
404         /* [cgw]: 歷遍groups[],刪除所有屬性組 */
405         for (i = 0; cd->groups[i]; i++) {
406             /* [cgw]: 刪除一個屬性組 */
407             sysfs_remove_group(&cd->kobj, cd->groups[i]);
408         }
409     }
410 }
411 
412 static ssize_t show_dev(struct class_device *class_dev, char *buf)
413 {
414     /* [cgw]: 格式化class_dev->devt的主和次設備號為字元串,
415           * 填裝到buf中
416           */
417     return print_dev_t(buf, class_dev->devt);
418 }
419 
420 static ssize_t store_uevent(struct class_device *class_dev,
421                 const char *buf, size_t count)
422 {
423     /* [cgw]: 對class_dev->kobj產生一個KOBJ_ADD事件通知用戶空間 */
424     kobject_uevent(&class_dev->kobj, KOBJ_ADD);
425     return count;
426 }
427 
428 void class_device_initialize(struct class_device *class_dev)
429 {
430     /* [cgw]: 分配class_dev->kobj.kset,指向&class_obj_subsys */
431     kobj_set_kset_s(class_dev, class_obj_subsys);
432     /* [cgw]: 初始化class_dev->kobj */
433     kobject_init(&class_dev->kobj);
434     /* [cgw]: 初始化class_dev->node鏈表 */
435     INIT_LIST_HEAD(&class_dev->node);
436 }
437 
438 int class_device_add(struct class_device *class_dev)
439 {
440     struct class *parent_class = NULL;
441     struct class_device *parent_class_dev = NULL;
442     struct class_interface *class_intf;
443     int error = -EINVAL;
444 
445     /* [cgw]: class_dev->kobj引用計數+1 
446           * 並根據kobj找到包含這個kobj的結構體指針class_dev
447           */
448     class_dev = class_device_get(class_dev);
449     /* [cgw]: class_dev指針為空 */
450     if (!class_dev)
451         return -EINVAL;
452 
453     /* [cgw]: class_dev->class_id長度為0 */
454     if (!strlen(class_dev->class_id))
455         goto out1;
456 

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

-Advertisement-
Play Games
更多相關文章
  • 1、環境準備 1)準備6台ubuntu虛擬機 2)配置好IP(192.168.1.36,192.168.1.37,192.168.1.38, 192.168.1.40,192.168.1.41,192.168.1.42 三台master 三台slave) 3)安裝好openssh-server 2、 ...
  • 表大小 慢的sql select a.city, a.agent_id, a.username, a.real_name, phone, zgy_name, login_count, user_count, count(distinct b.invest_id) user_invested, sum ...
  • 一個學習性任務:每個人有不同次數的成績,統計出每個人的最高成績。 這個問題應該還是相對簡單,其實就用聚合函數就好了。 select id,name,max(score) from Student group by id,name order by name 上邊這種情況只適用id 和name是一一對 ...
  • 1.安裝MySql 目前MySQL有兩種形式的文件,一個是msi格式,一個是zip格式的。msi格式的直接點擊setup.exe就好,按照步驟進行。但是很多人下了zip格式的解壓發現沒有setup.exe,本人下載的也是這樣的,不知道怎麼安裝,點哪裡都沒有反應。只能尋求度娘幫助,然後才瞭解到,這種文 ...
  • 方法一: 直接(手動)去修改資料庫名稱,資料庫表名稱,資料庫列名稱、列屬性 方法二: 使用SQL語句去修改 ...
  • 彙編器構造 一、 彙編器簡介 前面介紹了編譯器構造和靜態鏈接器構造的具體方法,而且我們實現了一個將高級語言轉化為彙編語言的編譯器,同時又實現了一個將多個目標文件鏈接為一個可執行文件的鏈接器。現在需要一個連接這兩個模塊的功能模塊——彙編器,它能將一個單獨的彙編文件轉換為一個可重定位目標文件,如圖1-1 ...
  • file命令及其用法 file:查看文件內容 的類型 du命令及其用法 -s: -h: read命令及其用法 描述GPL,BSD,Apache三個開源協定的大體聯繫及區別 自由軟體 開源協定,版權描述 列出Linux的發行版,並說明其跟Linux內核的關係 Linux,GNU:GNU/Linux, ...
  • UDP和TCP處於同一層網路模型中,也就是運輸層,基於二者之上的應用有很多,常見的基於TCP的有HTTP、Telnet等,基於UDP有DNS、NFS、SNMP等。UDP是無連接,不可靠的數據協議服務,而TCP提供面向流、提供可靠數據服務。註意,UDP和TCP沒有好壞之分,只是二者的適用場景不同罷了。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...