<menu id="w8yyk"><menu id="w8yyk"></menu></menu>
  • <dd id="w8yyk"><nav id="w8yyk"></nav></dd>
    <menu id="w8yyk"></menu>
    <menu id="w8yyk"><code id="w8yyk"></code></menu>
    <menu id="w8yyk"></menu>
    <xmp id="w8yyk">
    <xmp id="w8yyk"><nav id="w8yyk"></nav>
  • 網站首頁 > 物聯資訊 > 技術分享

    insmod module_param 模塊參數

    2016-09-28 00:00:00 廣州睿豐德信息科技有限公司 閱讀
    睿豐德科技 專注RFID識別技術和條碼識別技術與管理軟件的集成項目。質量追溯系統、MES系統、金蝶與條碼系統對接、用友與條碼系統對接 模塊參數 引導模塊時,可以向它傳遞參數。要使用模塊參數加載模塊,這樣寫: insmod module.ko [param1=value param2=value ...] 為了使用這些參數的值,要在模塊中聲明變量來保存它們,并在所有函數之外的某個地方使用宏MODULE_PARM(variable, type) 和 MODULE_PARM_DESC(variable, description) 來接收它們。type參數應該是一個格式為 [min[-max]]{b,h,i,l,s} 字符串,其中 min 和 max 是數組的長度限度。如果兩者都忽略了,則默認為 1。最后一個字符是類型說明符: b byte
    h short 
    i int 
    l long 
    s string      

    在用戶態下編程可以通過main()的來傳遞命令行參數,而編寫一個內核模塊則通過module_param ()
    module_param宏是Linux 2.6內核中新增的,該宏被定義在include/linux/moduleparam.h文件中,具體定義如下:
    #define module_param(name, type, perm)
    module_param_named(name, name, type, perm)

    其中使用了 3 個參數:要傳遞的參數變量名, 變量的數據類型, 以及訪問參數的權限。
    module_param (name,type,perm);
    module_param 使用了 3 個參數: 變量名, 它的類型, 以及一個權限掩碼用來做一個輔助的 sysfs 入口(啥意思). 這個宏定義應當放在任何函數之外, 典型地是出現在源文件的前面.定義如:
    static char *whom = "world";
    static int howmany = 1;
    module_param (howmany, int, S_IRUGO);
    module_param (whom, charp, S_IRUGO);
    模塊參數支持許多類型:
    bool
    invbool
    一個布爾型( true 或者 false)值(相關的變量應當是 int 類型). invbool 類型顛倒了值, 所以真值變成 false, 反之亦然.
    charp
    一個字符指針值. 內存為用戶提供的字串分配, 指針因此設置.
    int
    long
    short
    uint
    ulong
    ushort
    基本的變長整型值. 以 u 開頭的是無符號值.

    數組參數, 用逗號間隔的列表提供的值, 模塊加載者也支持. 聲明一個數組參數, 使用:

    module_param _array(name,type,num,perm);
    這里 name 是你的數組的名子(也是參數名),
    type 是數組元素的類型,
    num 是一個整型變量,
    perm 是通常的權限值.
    如果數組參數在加載時設置, num 被設置成提供的數的個數. 模塊加載者拒絕比數組能放下的多的值.

    perm參數的作用是什么?
    最后的 module_param 字段是一個權限值; 你應當使用 中定義的值. 這個值控制誰可以存取這些模塊參數在 sysfs 中的表示.如果 perm 被設為 0, 就根本沒有 sysfs 項. 否則, 它出現在 /sys/module下面, 帶有給定的權限. 使用 S_IRUGO 作為參數可以被所有人讀取, 但是不能改變; S_IRUGO|S_IWUSR 允許 root 來改變參數. 注意, 如果一個參數被 sysfs 修改, 你的模塊看到的參數值也改變了, 但是你的模塊沒有任何其他的通知. 你應當不要使模塊參數可寫, 除非你準備好檢測這個改變并且因而作出反應.
    perm參數的作用是什么?
    最后的 module_param 字段是一個權限值,表示此參數在sysfs文件系統中所對應的文件節點的屬性。你應當使用 中定義的值. 這個值控制誰可以存取這些模塊參數在 sysfs 中的表示.當perm為0時,表示此參數不存在 sysfs文件系統下對應的文件節點。 否則, 模塊被加載后,在/sys/module/ 目錄下將出現以此模塊名命名的目錄, 帶有給定的權限.。
    權限在include/linux/stat.h中有定義
    比如:
    #define S_IRWXU 00700
    #define S_IRUSR 00400
    #define S_IWUSR 00200
    #define S_IXUSR 00100
    #define S_IRWXG 00070
    #define S_IRGRP 00040
    #define S_IWGRP 00020
    #define S_IXGRP 00010
    #define S_IRWXO 00007
    #define S_IROTH 00004
    #define S_IWOTH 00002
    #define S_IXOTH 00001
    使用 S_IRUGO 作為參數可以被所有人讀取, 但是不能改變; S_IRUGO|S_IWUSR 允許 root 來改變參數. 注意, 如果一個參數被 sysfs 修改, 你的模塊看到的參數值也改變了, 但是你的模塊沒有任何其他的通知. 你應當不要使模塊參數可寫, 除非你準備好檢測這個改變并且因而作出反應.

    通過宏MODULE_PARM_DESC()對參數進行說明:
    static unsigned short size = 1;
    module_param(size, ushort, 0644);
    MODULE_PARM_DESC(size, “The size in inches of the fishing pole”
    “connected to this computer.” );

     

     

     

    module_param() 和module_param_array() 的作用就是讓那些全局變量對insmod 可見,使模塊裝載時可重新賦值。

    module_param_array() 宏的第三個參數用來記錄用戶insmod 時提供的給這個數組的元素個數,NULL 表示不關心用戶提供的個數

    module_param() 和module_param_array() 最后一個參數權限值不能包含讓普通用戶也有寫權限,否則編譯報錯。這點可參考linux/moduleparam.h 中__module_param_call() 宏的定義。

    字符串數組中的字符串似乎不能包含逗號,否則一個字符串會被解析成兩個 

     

    一個測試用例:parm_hello.c

     

     

     

    #include<linux/module.h>
    #include<linux/moduleparam.h>
    #include<linux/kernel.h>


    #defineMAX_ARRAY 6

    staticintint_var=0;
    staticconstchar*str_var="default";
    static int int_array[6];
    intnarr;

    module_param(int_var,int,0644);
    MODULE_PARM_DESC(int_var,"A integer variable");


    module_param(str_var,charp,0644);
    MODULE_PARM_DESC(str_var,"A string variable");

    module_param_array(int_array,int,&narr,0644);
    MODULE_PARM_DESC(int_array,"A integer array");
     

    staticint__init hello_init(void)
    {
           inti;
           printk(KERN_ALERT"Hello, world./n");
           printk(KERN_ALERT"int_var %d./n",int_var);
           printk(KERN_ALERT"str_var %s./n",str_var);

           for(i=0;i<narr;i++){
                   printk("int_array[%d] = %d/n",i,int_array[i]);
           }
           return0;
    }

    staticvoid__exit hello_exit(void)
    {
           printk(KERN_ALERT"Bye, world./n");
    }
    module_init(hello_init);
    module_exit(hello_exit);
    MODULE_LICENSE("GPL");
    MODULE_AUTHOR("kelly");
    MODULE_DEION("This module is a example.");

     

    Linux中EXPORT_SYMBOL的用法


    EXPORT_SYMBOL標簽內定義的函數對全部內核代碼公開,不用修改內核代碼就可以在您的內核模塊中直接調用。您還可以手工修改內核源代碼來導出另外的函數,用于重新編譯并加載新內核后的測試。


    Linux symbol export method:

    [1] If we want export the symbol in a module, just use the EXPORT_SYMBOL(xxxx) in the C or H file.
        And compile the module by adding the compile flag -DEXPORT_SYMTAB.
        Then we can use the xxxx in the other module.


    [2] If we want export some symbol in Kernel that is not in a module such as xxxx in the /arch/ppc/fec.c.
        Firstly, define the xxxx in the fec.c;
        Secondly, make a new file which contain the "extern" define the xxxx(for example, extern int xxxx);
        Lastly, in the ppc_ksyms.c we includes the new file, and add the EXPORT_SYMBOL(xxxx).
        Then we can use the xxxx.

     

     

     

     

     

     

     

     

     

     

    對于如何向模塊傳遞參數,Linux kernel 提供了一個簡單的框架。其允許驅動程序聲明參數,并且用戶在系統啟動或模塊裝載時為參數指定相應值,在驅動程序里,參數的用法如同全局變量。

    使用下面的宏時需要包含頭文件。

     

        通過宏module_param()定義一個模塊參數:
    module_param(name, type, perm);
    name既是用戶看到的參數名,又是模塊內接受參數的變量; 

    type表示參數的數據類型,是下列之一:byte, short, ushort, int, uint, long, ulong, charp, bool, invbool;     

    perm指定了在sysfs中相應文件的訪問權限。訪問權限與linux文件愛你訪問權限相同的方式管理,如0644,或使用stat.h中的宏如S_IRUGO表示。

    0表示完全關閉在sysfs中相對應的項。


    這些宏不會聲明變量,因此在使用宏之前,必須聲明變量,典型地用法如下:
    static unsigned int int_var = 0;   
    module_param(int_var, uint, S_IRUGO);

    這些必須寫在模塊源文件的開頭部分。即int_var是全局的。也可以使模塊源文件內部的變量名與外部的參數名有不同的名字,通過module_param_named()定義。module_param_named(name, variable, type, perm);其中name是外部可見的參數名,variable是源文件內部的全局變量名,而module_param通過module_param_named實現,只不過name與variable相同。

    例如:
    static unsigned int max_test = 9;
    module_param_name(maximum_line_test, max_test, int, 0);

     

    如果模塊參數是一個字符串時,通常使用charp類型定義這個模塊參數。內核復制用戶提供的字符串到內存,并且相對應的變量指向這個字符串。

    例如:
    static char *name;
    module_param(name, charp, 0);

    另一種方法是通過宏module_param_string()讓內核把字符串直接復制到程序中的字符數組內。
    module_param_string(name, string, len, perm);

    這里,name是外部的參數名,string是內部的變量名,len是以string命名的buffer大小(可以小于buffer的大小,但是沒有意義),perm表示sysfs的訪問權限(或者perm是零,表示完全關閉相對應的sysfs項)。

    例如:
    static char species[BUF_LEN];
    module_param_string(specifies, species, BUF_LEN, 0);

     

    如果需要傳遞多個參數可以通過宏module_param_array()實現。 
    module_param_array(name, type, nump, perm);
    其中,name既是外部模塊的參數名又是程序內部的變量名,type是數據類型,perm是sysfs的訪問權限。指針nump指向一個整數,其值表示有多少個參數存放在數組name中。值得注意是name數組必須靜態分配。

    例如:
    static int finsh[MAX_FISH];
    static int nr_fish;
    module_param_array(fish, int, &nr_fish, 0444); //最終傳遞數組元素個數存在nr_fish中

    通過宏module_param_array_named()使得內部的數組名與外部的參數名有不同的名字。

    例如:
    module_param_array_named(name, array, type, nump, perm);

    通過宏MODULE_PARM_DESC()對參數進行說明:
    static unsigned short size = 1;
    module_param(size, ushort, 0644);
    MODULE_PARM_DESC(size, “The size in inches of the fishing pole”
    “connected to this computer.” );

     

     

     說明:from  http://blog.csdn.net/iczyh/archive/2008/10/26/3149727.aspx

    module_param() 和 module_param_array() 的作用就是讓那些全局變量對 insmod 可見,使模塊裝載時可重新賦值。

    module_param_array() 宏的第三個參數用來記錄用戶 insmod 時提供的給這個數組的元素個數,NULL 表示不關心用戶提供的個數

    module_param() 和 module_param_array() 最后一個參數權限值不能包含讓普通用戶也有寫權限,否則編譯報錯。這點可參考 linux/moduleparam.h 中 __module_param_call() 宏的定義。

    字符串數組中的字符串似乎不能包含逗號,否則一個字符串會被解析成兩個

     

    一個測試用例:parm_hello.c

     

    #include <linux/module.h>
    #include <linux/moduleparam.h>
    #include <linux/kernel.h>


    #define MAX_ARRAY 6

    static int int_var = 0;
    static const char *str_var = "default";
    static int int_array[6];
    int narr;

    module_param(int_var, int, 0644);
    MODULE_PARM_DESC(int_var, "A integer variable");


    module_param(str_var, charp, 0644);
    MODULE_PARM_DESC(str_var, "A string variable");

    module_param_array(int_array, int, &narr, 0644);
    MODULE_PARM_DESC(int_array, "A integer array");
     

    static int __init hello_init(void)
    {
           int i;
           printk(KERN_ALERT "Hello, my LKM.\n");
           printk(KERN_ALERT "int_var %d.\n", int_var);
           printk(KERN_ALERT "str_var %s.\n", str_var);

           for(i = 0; i < narr; i ++){
                   printk("int_array[%d] = %d\n", i, int_array[i]);
           }
           return 0;
    }

    static void __exit hello_exit(void)
    {
           printk(KERN_ALERT "Bye, my LKM.\n");
    }
    module_init(hello_init);
    module_exit(hello_exit);
    MODULE_LICENSE("GPL");
    MODULE_AUTHOR("ydzhang");
    MODULE_DEION("This module is a example.");

     

    測試:insmod parm_hello.ko int_var=100 str_var=hello int_array=100,200

    RFID管理系統集成商 RFID中間件 條碼系統中間層 物聯網軟件集成
    最近免费观看高清韩国日本大全