天天看點

Linux裝置模型 (3)

在上文中,我們介紹到如何使用default attribute。Default attribute使用很友善,但不夠靈活。比如上篇文章在Kobject一節中提到的那個例子,name和val這兩個attribute使用同一個show/store函數來通路,如果attribute非常多,show/store函數裡的分支就會很淩亂。

為了解決這個問題,我們可以參考核心提供的kobj_attribute。在核心裡,kobj_attibute是這樣定義的:

<code>struct</code> <code>kobj_attribute {</code>

<code>    </code><code>struct</code> <code>attribute attr;</code>

<code>    </code><code>ssize_t (*show)(</code><code>struct</code> <code>kobject *kobj,</code><code>struct</code> <code>kobj_attribute *attr,</code>

<code>            </code><code>char</code> <code>*buf);</code>

<code>    </code><code>ssize_t (*store)(</code><code>struct</code> <code>kobject *kobj,</code><code>struct</code> <code>kobj_attribute *attr,</code>

<code>             </code><code>const</code> <code>char</code> <code>*buf,</code><code>size_t</code> <code>count);</code>

<code>};</code>

每一個attribute會對應自己的show/store函數,這樣就極大的提高了靈活性。可是,在上一篇文章中我們的認知是,sysfs是通過kobject裡的kobj_type-&gt;sysfs_ops來讀寫attribute的,那如果要利用kobj_attribute中的show/store來讀寫attribute的話,就必須在kobj_type-&gt;sysfs_ops裡指定。Linux核心提供了一個預設的kobj_type類型dynamic_kobj_ktype來實作上述的操作。

<code>/* default kobject attribute operations */</code>

<code>static</code> <code>ssize_t kobj_attr_show(</code><code>struct</code> <code>kobject *kobj,</code><code>struct</code> <code>attribute *attr,</code>

<code>                  </code><code>char</code> <code>*buf)</code>

<code>{</code>

<code>    </code><code>struct</code> <code>kobj_attribute *kattr;</code>

<code>    </code><code>ssize_t ret = -EIO;</code>

<code>    </code><code>kattr = container_of(attr,</code><code>struct</code> <code>kobj_attribute, attr);</code>

<code>    </code><code>if</code> <code>(kattr-&gt;show)</code>

<code>        </code><code>ret = kattr-&gt;show(kobj, kattr, buf);</code>

<code>    </code><code>return</code> <code>ret;</code>

<code>}</code>

<code>static</code> <code>ssize_t kobj_attr_store(</code><code>struct</code> <code>kobject *kobj,</code><code>struct</code> <code>attribute *attr,</code>

<code>                   </code><code>const</code> <code>char</code> <code>*buf,</code><code>size_t</code> <code>count)</code>

<code>    </code><code>if</code> <code>(kattr-&gt;store)</code>

<code>        </code><code>ret = kattr-&gt;store(kobj, kattr, buf, count);</code>

<code>const</code> <code>struct</code> <code>sysfs_ops kobj_sysfs_ops = {</code>

<code>    </code><code>.show   = kobj_attr_show,</code>

<code>    </code><code>.store  = kobj_attr_store,</code>

<code>static</code> <code>void</code> <code>dynamic_kobj_release(</code><code>struct</code> <code>kobject *kobj)</code>

<code>    </code><code>pr_debug(</code><code>"kobject: (%p): %s\n"</code><code>, kobj, __func__);</code>

<code>    </code><code>kfree(kobj);</code>

<code>static</code> <code>struct</code> <code>kobj_type dynamic_kobj_ktype = {</code>

<code>    </code><code>.release    = dynamic_kobj_release,</code>

<code>    </code><code>.sysfs_ops  = &amp;kobj_sysfs_ops,</code>

kobj_attribute是核心提供給我們的一種更加靈活的處理attribute的方式,但是它還不夠。隻有當我們使用kobject_create來建立kobject時,使用kobj_attribute才比較友善,但大部分情況下,我們是把kobject内嵌到自己的結構裡,此時就無法直接使用核心提供的dynamic_kobj_ktype,是以,我們需要建立自己的kobj_attribute。

首先,我們需要定義自己的attribute:

<code>struct</code> <code>my_attribute {</code>

<code>        </code><code>struct</code> <code>attribute attr;</code>

<code>        </code><code>ssize_t (*show)(</code><code>struct</code> <code>my_kobj *obj,</code><code>struct</code> <code>my_attribute *attr,</code>

<code>                        </code><code>char</code> <code>*buf);</code>

<code>        </code><code>ssize_t (*store)(</code><code>struct</code> <code>my_kobj *obj,</code><code>struct</code> <code>my_attribute *attr,</code>

<code>                        </code><code>const</code> <code>char</code> <code>*buf,</code><code>size_t</code> <code>count);</code>

在my_attribute裡,我們的show/store直接操作my_kobj,這樣更加友善。

參考Linux核心,kobj_type裡的sysfs_ops這樣定義:

<code>static</code> <code>ssize_t my_attr_show(</code><code>struct</code> <code>kobject *kobj,</code><code>struct</code> <code>attribute *attr,</code>

<code>    </code><code>struct</code> <code>my_attribute *my_attr;</code>

<code>    </code><code>my_attr = container_of(attr,</code><code>struct</code> <code>my_attribute, attr);</code>

<code>    </code><code>if</code> <code>(my_attr-&gt;show)</code>

<code>        </code><code>ret = my_attr-&gt;show(container_of(kobj,</code><code>struct</code> <code>my_kobj, kobj),</code>

<code>                </code><code>my_attr, buf);</code>

<code>static</code> <code>ssize_t my_attr_store(</code><code>struct</code> <code>kobject *kobj,</code><code>struct</code> <code>attribute *attr,</code>

<code>    </code><code>if</code> <code>(my_attr-&gt;store)</code>

<code>        </code><code>ret = my_attr-&gt;store(container_of(kobj,</code><code>struct</code> <code>my_kobj, kobj),</code>

<code>                </code><code>my_attr, buf, count);</code>

下面就可以分别對name和val兩個attribute定義自己的show/store。name這個attribute是隻讀的,隻要為它定義show即可。

<code>ssize_t name_show(</code><code>struct</code> <code>my_kobj *obj,</code><code>struct</code> <code>my_attribute *attr,</code><code>char</code> <code>*buffer)</code>

<code>    </code><code>return</code> <code>sprintf</code><code>(buffer,</code><code>"%s\n"</code><code>, kobject_name(&amp;obj-&gt;kobj));</code>

<code>ssize_t val_show(</code><code>struct</code> <code>my_kobj *obj,</code><code>struct</code> <code>my_attribute *attr,</code><code>char</code> <code>*buffer)</code>

<code>    </code><code>return</code> <code>sprintf</code><code>(buffer,</code><code>"%d\n"</code><code>, obj-&gt;val);</code>

<code>ssize_t val_store(</code><code>struct</code> <code>my_kobj *obj,</code><code>struct</code> <code>my_attribute *attr,</code>

<code>        </code><code>const</code> <code>char</code> <code>*buffer,</code><code>size_t</code> <code>size)</code>

<code>    </code><code>sscanf</code><code>(buffer,</code><code>"%d"</code><code>, &amp;obj-&gt;val);</code>

<code>    </code><code>return</code> <code>size;</code>

接下來,利用核心提供的宏__ATTR來初始化my_attribute,并建立attribute數組。

<code>struct</code> <code>my_attribute name_attribute = __ATTR(name, 0444, name_show, NULL);</code>

<code>struct</code> <code>my_attribute val_attribute = __ATTR(val, 0666, val_show, val_store);</code>

<code>struct</code> <code>attribute *my_attrs[] = {</code>

<code>    </code><code>&amp;name_attribute.attr,</code>

<code>    </code><code>&amp;val_attribute.attr,</code>

<code>    </code><code>NULL,</code>

其中,宏__ATTR的定義如下:

<code>#define __ATTR(_name,_mode,_show,_store) { \</code>

<code>    </code><code>.attr = {.name = __stringify(_name), .mode = _mode },   \</code>

<code>    </code><code>.show   = _show,                    \</code>

<code>    </code><code>.store  = _store,                   \</code>

在module_init裡,我們調用sysfs_create_files來把attribute增加到sysfs中。

<code>retval = sysfs_create_files(&amp;obj-&gt;kobj,</code>

<code>            </code><code>(</code><code>const</code> <code>struct</code> <code>attribute **)my_attrs);</code>

<code>if</code> <code>(retval) {</code>

<code>    </code><code>// ...</code>

在kobject對應的目錄裡,還可以建立子目錄,Linux核心裡是用attribute_group來實作。在本例中,我們可以這麼做:

<code>struct</code> <code>attribute_group my_group = {</code>

<code>    </code><code>.name     =</code><code>"mygroup"</code><code>,</code>

<code>    </code><code>.attrs    = my_attrs,</code>

然後在module_init裡調用sysfs_create_group來添加。

<code>retval = sysfs_create_group(&amp;obj-&gt;kobj, &amp;my_group);</code>

本例建立的attribute_group中包含的attribute也是my_attrs,是以在子目錄mygroup下的檔案和mykobj目錄下的檔案完全一緻。

最後我們得到的目錄結構是這樣的。

mykobj/ |-- mygroup |   |-- name |   `-- val |-- name `-- val

完成這個實作之後,你可以用指令echo 2 &gt; /sys/mykobj/val來修改mykobj下的val檔案,可以觀察到/sys/mykobj/mygroup/val的内容也會變成2,反之亦然。

作者:wwang 

出處:http://www.cnblogs.com/wwang 

--------懶人評論(請勿重複點選)--------

繼續閱讀