Linux内核API完全参考手册(第2版)
上QQ阅读APP看书,第一时间看更新

2.12 函数:symbol_put_addr( )

文件包含:

          #include <linux/module.h>

函数定义:

在内核源码中的位置:linux-3.19.3/kernel/module.c

函数定义格式:void symbol_put_addr(void *addr)

函数功能描述:

该函数的功能是根据给定的一个内存地址addr,找到该地址所在的模块后,将模块的引用计数减1。它与__symbol_put( )函数(见本章中关于该函数的分析)有类似的功能,但二者的参数不一致。

输入参数说明:

addr:内存地址,通常指某一内核符号的内存地址。

返回参数说明:

该函数无返回值。

实例解析:

编写测试文件:symbol_put_addr.c

头文件及全局变量声明如下:

        #include <linux/module.h>
        #include <linux/init.h>
        MODULE_LICENSE("GPL");
        static int __init symbol_put_addr_init(void);
        static void __exit symbol_put_addr_exit(void);

模块初始化函数:

        int __init symbol_put_addr_init(void)
        {
            const char * symbol_name ;
            const char * mod_name ;
            struct module * fmodule ;
            void * addr;

            symbol_name = "symbol_A";
            addr = __symbol_get( symbol_name );

            if( addr ! = NULL )
            {
                printk("addr : %lx \n", (unsigned long)addr );

                mod_name = "test_module";                 //定义待查找的模块名为“test_module”
                fmodule = find_module( mod_name );        //调用查找模块函数
                if(fmodule ! = NULL )
                {
                    printk("before calling symbol_put_addr, \n");
                    printk("refs of %s is: %d\n", mod_name, module_refcount(fmodule));
                    symbol_put_addr( addr );              //will dec the count
                    printk("after calling symbol_put_addr, \n");
                    printk("refs of %s is: %d\n", mod_name, module_refcount(fmodule));
                }
                else
                {
                        printk("find %s failed! \n", mod_name );
                    }
                }
                else
                {
                    printk("%s isn't found\n", symbol_name);
                }
                return 0;
            }

模块退出函数:

        void __exit symbol_put_addr_exit(void)
        {
            printk("module exit ok! \n");
        }

模块初始化及退出函数调用:

        module_init(symbol_put_addr_init);
        module_exit(symbol_put_addr_exit);

实例运行结果及分析:

首先编译模块,执行命令insmod symbol_put_addr.ko插入模块,然后执行命令dmesg -c,会出现如图2-22所示的结果。

图2-22 插入symbol_put_addr模块后系统输出信息

结果分析:

关于实例程序中内核符号symbol_A和模块test_module的相关信息,可参见图2-10。

从图2-22显示的信息可以看出,模块插入之前模块test_module的引用计数为5,在插入模块之时调用了函数__symbol_get( ),所以在调用symbol_put_addr( )函数之前,模块test_module的引用计数为6,将存在于该模块中的内核符号symbol_A的地址作为实参传递给symbol_put_addr( )函数之后,模块的引用计数变为5。说明函数symbol_put_addr ( )的功能是查找内存地址addr所在的模块,并将该模块的引用计数减1。

实例程序中用到了内核函数__symbol_get( )、f ind_module( )和module_refcount( ),函数__symbol_get( )用来获取内核符号symbol_A的内存地址,该地址存在于test_module模块中;函数f ind_module( )根据模块名查找模块并返回查找到的模块;函数module_refcount( )则是用来获得模块被引用的次数。关于这三个函数的详细说明可以参见本章中关于它们的分析。