#include <linux/fs.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/module.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)
{
	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;
	}
	pr_info("Périphérique de caractère enregistré avec le numéro de majeur %d\n",
		major_number);
	return 0;
}

static void __exit basic_module_exit(void)
{
	unregister_chrdev(major_number, DEVICE_NAME);
	pr_info("Au revoir! Le module est déchargé.\n");
}

static int device_open(struct inode *inode, struct file *file)
{
	pr_info("flodev - Ouverture du périphérique\n");
	return 0;
}

static int device_release(struct inode *inode, struct file *file)
{
	pr_info("flodev - Fermeture du périphérique\n");
	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;
	pr_info("flodev - Lecture de %d bytes\n", bytes_read);
	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;
	pr_info("flodev - Message reçu: %s\n", msg);
	return len;
}

module_init(basic_module_init);
module_exit(basic_module_exit);

MODULE_LICENSE("MIT License");
MODULE_AUTHOR("Florian RICHER <florian.richer@protonmail.com>");
MODULE_DESCRIPTION("Un module noyau avec un périphérique de caractère");
MODULE_VERSION("1.0");