Как использовать системный вызов Linux из модуля ядра Linux

У меня возникли трудности с вызовом системного вызова изнутри модуля ядра Linux. Системные вызовы были протестированы и работают должным образом из стандартной программы пространства пользователя c, но я не могу заставить модуль ядра скомпилировать и запустить их.

В моей пользовательской программе я включаю следующий код, и системный вызов работает:

#include <linux/unistd.h>   
#define __NR_sys_mycall 343

extern long int _syscall(long int_sysno,...)__THROW;

//and then a simple call is done as such
long value = syscall(__NR_sys_mycall);

printf("The value is %ld\n",value);

Но когда я пытаюсь сделать то же самое в моем модуле ядра Linux, я получаю кучу ошибок, которые либо говорят об ошибке: неявное объявление функции 'syscall' (если я не включаю определение _syscall), либо длинный список ошибок о синтаксисе, если Я делаю... поэтому я предполагаю, что мне нужна версия пространства ядра для вызова системного вызова. Я прав или нет?

//My LKM code
#include <linux/module.h>
#include <linux/unistd.h>
#define __NR_sys_mycall 343

static int start_init(void)
{
   long value = syscall(__NR_sys_mycall);
   printk("The value is %ld\n",value);

   return 0;
}

static void finish_exit(void)
{
      printk("Done!\n");
}

module_init(start_init);
module_exit(finish_exit);

2 ответа

Решение

Вы можете напрямую вызвать sys_mycall.

#include <linux/module.h>
#include <linux/unistd.h>


static int start_init(void)
{
   long value = sys_mycall (pass_arguments)
   printk("The value is %ld\n",value);

   return 0;
}

static void finish_exit(void)
{
      printk("Done!\n");
}

module_init(start_init);
module_exit(finish_exit);

Большинство системных вызовов использует asmlinkage, что означает поиск аргументов в стеке, а не в регистрах. Убедитесь, что при вызове системного вызова передайте аргументы в стек.

Также много системных вызовов просто использует copy_from_user. Если вы передадите адрес ядра в такой системный вызов, он потерпит неудачу.

Другие вопросы по тегам