kernel_module_learn/03_character_device/test_module.c

93 lines
2.3 KiB
C
Raw Permalink Normal View History

2025-02-08 17:01:11 +01:00
#include <linux/fs.h>
2025-02-08 18:57:28 +01:00
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/module.h>
2025-02-08 17:01:11 +01:00
#include <linux/uaccess.h>
#define DEVICE_NAME "flodev"
#define CLASS_NAME "chardrv"
static int major_number;
2025-02-25 13:32:53 +01:00
static char msg[256] = { 0 };
2025-02-08 17:01:11 +01:00
static short size_of_msg;
static int device_open(struct inode *, struct file *);
static int device_release(struct inode *, struct file *);
static ssize_t device_read(struct file *, char *, size_t, loff_t *);
static ssize_t device_write(struct file *, const char *, size_t, loff_t *);
static struct file_operations fops = {
2025-02-23 13:14:54 +01:00
.read = device_read,
.write = device_write,
.open = device_open,
.release = device_release,
2025-02-08 17:01:11 +01:00
};
2025-02-25 13:32:53 +01:00
static int __init basic_module_init(void)
{
2025-02-23 13:14:54 +01:00
pr_info("Bonjour! Le module est chargé.\n");
major_number = register_chrdev(0, DEVICE_NAME, &fops);
if (major_number < 0) {
pr_info("Erreur lors de l'enregistrement du périphérique de caractère\n");
return major_number;
}
2025-02-25 13:32:53 +01:00
pr_info("Périphérique de caractère enregistré avec le numéro de majeur %d\n",
major_number);
2025-02-23 13:14:54 +01:00
return 0;
2025-02-08 17:01:11 +01:00
}
2025-02-25 13:32:53 +01:00
static void __exit basic_module_exit(void)
{
2025-02-23 13:14:54 +01:00
unregister_chrdev(major_number, DEVICE_NAME);
pr_info("Au revoir! Le module est déchargé.\n");
2025-02-08 17:01:11 +01:00
}
2025-02-25 13:32:53 +01:00
static int device_open(struct inode *inode, struct file *file)
{
2025-02-23 13:14:54 +01:00
pr_info("flodev - Ouverture du périphérique\n");
return 0;
2025-02-08 17:01:11 +01:00
}
2025-02-25 13:32:53 +01:00
static int device_release(struct inode *inode, struct file *file)
{
2025-02-23 13:14:54 +01:00
pr_info("flodev - Fermeture du périphérique\n");
return 0;
2025-02-08 17:01:11 +01:00
}
2025-02-25 13:32:53 +01:00
static ssize_t device_read(struct file *filp, char *buffer, size_t length,
loff_t *offset)
{
2025-02-23 13:14:54 +01:00
int bytes_read = 0;
if (*offset >= size_of_msg) {
return 0;
}
if (*offset + length > size_of_msg) {
length = size_of_msg - *offset;
}
if (copy_to_user(buffer, msg + *offset, length)) {
return -EFAULT;
}
*offset += length;
bytes_read = length;
pr_info("flodev - Lecture de %d bytes\n", bytes_read);
return bytes_read;
2025-02-08 17:01:11 +01:00
}
2025-02-25 13:32:53 +01:00
static ssize_t device_write(struct file *filp, const char *buff, size_t len,
loff_t *off)
{
2025-02-23 13:14:54 +01:00
if (copy_from_user(msg, buff, len)) {
return -EFAULT;
}
size_of_msg = len;
pr_info("flodev - Message reçu: %s\n", msg);
return len;
2025-02-08 17:01:11 +01:00
}
module_init(basic_module_init);
module_exit(basic_module_exit);
MODULE_LICENSE("MIT License");
MODULE_AUTHOR("Florian RICHER");
MODULE_DESCRIPTION("Un module noyau avec un périphérique de caractère");
MODULE_VERSION("1.0");