Как установить связь между программой пространства пользователя и модулем ядра Linux, чтобы напечатать сообщение, отправленное программой пользователя в ядро - PullRequest
3 голосов
/ 23 октября 2019

Я разработал простой модуль ядра Linux, который я отправлю ему сообщение char от программы пространства пользователя.

Это модуль:

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/device.h>  
#include <linux/kernel.h>


MODULE_LICENSE("GPL");      
MODULE_AUTHOR("Gaston");  
MODULE_DESCRIPTION("A simple Linux char driver"); 
MODULE_VERSION("0.1"); 


static char message[256] = {0};

ssize_t exer_open(struct inode *pinode, struct file *pfile) {

    printk(KERN_INFO "Device has been opened\n");
    return 0;
}



ssize_t exer_read(struct file *pfile, char __user *buffer, size_t length, loff_t *offset) {

    return 0;
}


ssize_t exer_write(struct file *pfile, const char __user *buffer, size_t length, loff_t *offset) {

    printk(KERN_INFO "Received %s characters from the user\n", message);
    return 0;

}


ssize_t exer_close(struct inode *pinode, struct file *pfile) {

    printk(KERN_INFO "Device successfully closed\n");
    return 0;
}


struct file_operations exer_file_operations = { 
    .owner = THIS_MODULE,
    .open = exer_open,
    .read = exer_read,
    .write = exer_write,
    .release = exer_close,
};


int exer_simple_module_init(void) {

    printk(KERN_INFO "Initializing the LKM\n");
    register_chrdev(240, "Simple Char Drv", &exer_file_operations);
    return 0;
}



void exer_simple_module_exit(void) {

    unregister_chrdev(240, "Simple Char Drv");
}

module_init(exer_simple_module_init);
module_exit(exer_simple_module_exit);

И это моя программа на C:

#include<stdio.h>
#include<stdlib.h>
#include<errno.h>
#include<fcntl.h>
#include<string.h>
#include<unistd.h>


#define BUFFER_LENGTH 256 

int main()

{

int ret, fd;
char stringToSend[BUFFER_LENGTH];


fd = open("/dev/char_device", O_RDWR);             // Open the device with read/write access

if (fd < 0)
    {
            perror("Failed to open the device...");
            return errno;
    }


printf("Type in a short string to send to the kernel module:\n");

scanf("%s", stringToSend);                // Read in a string (with spaces)

printf("Writing message to the device [%s].\n", stringToSend);

ret = write(fd, stringToSend, strlen(stringToSend)); // Send the string to the LKM

if (ret < 0)
    {
            perror("Failed to write the message to the device.");
            return errno;
    }

return 0;

}

После вставки модуля с помощью insmod, и когда я запускаю программу и проверяю журналы ядра с помощью команды tail -f /var/log/messages, я вижу:

Oct  1 13:57:37 auth.info login[306]: root login on 'ttyS0'
Oct  1 13:58:22 user warn kernel: exer_simple_char_drv: loading out-of-tree module taints kernel.
Oct  1 13:58:22 user.info kernel: Initializing the LKM
Oct  1 13:58:35 user.info kernel: Device has been opened
Oct  1 13:58:39 user.info kernel: Received  characters from the user
Oct  1 13:58:39 user.info kernel: Device successfully closed

То же самоекогда я запускаю dmesg:

exer_simple_char_drv: loading out-of-tree module taints kernel.
Initializing the LKM
Device has been opened
Received  characters from the user
Device successfully closed

Проблема заключается в том, что я не вижу сообщения, которое я ввел вручную, когда выполняю свою программу на Си. Что мне здесь не хватает, пожалуйста?

1 Ответ

1 голос
/ 23 октября 2019

Первая проблема: вы никогда не изменяете message.

Тогда вы не можете напрямую использовать пользовательскую память в контексте ядра

Вы должны перевести ее раньше: copy_from_user для этого.

Ваша функция записи может выглядеть как

#define MAX 256
/* here, message is defined as 256 bytes +1 one.
   The extra char is here to be compatible with the `%s` formatter */
static char message[MAX+1] ="";

ssize_t exer_write(struct file *pfile, const char __user *buffer, size_t length, loff_t *offset) {
    if (length > MAX)
        return -EINVAL;

    if (copy_from_user(message, buffer, length) != 0)
        return -EFAULT;

    printk(KERN_INFO "Received %s characters from the user\n", message);
    return 0;

}   
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...