kernel_module_learn/02_character_device/test_module.c

84 lines
2.4 KiB
C
Raw Normal View History

2025-02-08 17:01:11 +01:00
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/fs.h>
#include <linux/uaccess.h>
#define DEVICE_NAME "flodev"
#define CLASS_NAME "chardrv"
static int major_number;
static char msg[256] = {0};
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 = {
.read = device_read,
.write = device_write,
.open = device_open,
.release = device_release,
};
static int __init basic_module_init(void) {
2025-02-08 17:20:04 +01:00
pr_info("Bonjour! Le module est chargé.\n");
2025-02-08 17:01:11 +01:00
major_number = register_chrdev(0, DEVICE_NAME, &fops);
if (major_number < 0) {
2025-02-08 17:20:04 +01:00
pr_info("Erreur lors de l'enregistrement du périphérique de caractère\n");
2025-02-08 17:01:11 +01:00
return major_number;
}
2025-02-08 17:20:04 +01:00
pr_info("Périphérique de caractère enregistré avec le numéro de majeur %d\n", major_number);
2025-02-08 17:01:11 +01:00
return 0;
}
static void __exit basic_module_exit(void) {
unregister_chrdev(major_number, DEVICE_NAME);
2025-02-08 17:20:04 +01:00
pr_info("Au revoir! Le module est déchargé.\n");
2025-02-08 17:01:11 +01:00
}
static int device_open(struct inode *inode, struct file *file) {
2025-02-08 17:20:04 +01:00
pr_info("flodev - Ouverture du périphérique\n");
2025-02-08 17:01:11 +01:00
return 0;
}
static int device_release(struct inode *inode, struct file *file) {
2025-02-08 17:20:04 +01:00
pr_info("flodev - Fermeture du périphérique\n");
2025-02-08 17:01:11 +01:00
return 0;
}
static ssize_t device_read(struct file *filp, char *buffer, size_t length, loff_t *offset) {
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;
2025-02-08 17:20:04 +01:00
pr_info("flodev - Lecture de %d bytes\n", bytes_read);
2025-02-08 17:01:11 +01:00
return bytes_read;
}
static ssize_t device_write(struct file *filp, const char *buff, size_t len, loff_t *off) {
if (copy_from_user(msg, buff, len)) {
return -EFAULT;
}
size_of_msg = len;
2025-02-08 17:20:04 +01:00
pr_info("flodev - Message reçu: %s\n", msg);
2025-02-08 17:01:11 +01:00
return len;
}
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");