Как мой пользовательский модуль в linux 3.2.28 может сделать вызов print_cpu_info?

Я собираюсь реализовать свой пользовательский модуль, в котором информация для процессора печатается с использованием print_cpu_info(), Чтобы вызвать print_cpu_info(), я включил необходимый заголовочный файл, но он не работает. Вот мой модуль.

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <asm/alternative.h>
#include <asm/bugs.h>
#include <asm/processor.h>
#include <asm/mtrr.h>
#include <asm/cacheflush.h>
extern struct cpuinfo_x86 boot_cpu_data;
int cpuinfox86_init(void)
{

    print_cpu_info(&boot_cpu_data);
    return 0;

}

void cpuinfox86_exit(void)
{
    printk("good bye cpu\n");
}

module_init(cpuinfox86_init);
module_exit(cpuinfox86_exit);
MODULE_LICENSE("GPL");

После компиляции этого модуля я получаю

make -C /lib/modules/3.2.28-2009720166/build  SUBDIRS=/home/tracking/1031_oslab modules
make[1]: Entering directory `/usr/src/linux-3.2.28'
  CC [M]  /home/tracking/1031_oslab/module.o
  Building modules, stage 2.
  MODPOST 1 modules
WARNING: "print_cpu_info" [/home/tracking/1031_oslab/module.ko] undefined!
  CC      /home/tracking/1031_oslab/module.mod.o
  LD [M]  /home/tracking/1031_oslab/module.ko
make[1]: Leaving directory `/usr/src/linux-3.2.28'

любая идея?

2 ответа

Решение

"print_cpu_info" не является экспортируемым символом, поэтому он не может использоваться модулями. Однако вы можете использовать экспортируемое имя kallsyms_lookup_name, чтобы получить адрес print_cpu_info и выполнить вызов функции, используя указатель на функцию.

static void* find_sym( const char *sym ) {  // find address kernel symbol sym
   static unsigned long faddr = 0;          // static !!!
   // ----------- nested functions are a GCC extension ---------
   int symb_fn( void* data, const char* sym, struct module* mod, unsigned long addr ) {
      if( 0 == strcmp( (char*)data, sym ) ) {
         faddr = addr;
         return 1;
      }
      else return 0;
   };
   // --------------------------------------------------------
   kallsyms_on_each_symbol( symb_fn, (void*)sym );
   return (void*)faddr;
}

static void (*cpuInfo)(struct cpuinfo_x86 *);

Как это использовать:

unsigned int cpu = 0;
    struct cpuinfo_x86 *c;
    cpuInfo = find_sym("print_cpu_info");
    for_each_online_cpu(cpu)
    {
        c = &cpu_data(cpu);
        cpuInfo(c);
    }
Другие вопросы по тегам