In this section, we will create a simple virtual character device, which will be used by a user space process to instruct our kernel module whether it should hijack or restore certain system call. This virtual device will be controlled with ioctl function. For simplicity, I decided not to add read/write handlers to this device as it is not really required for what we are about to do. Although, it is a "nice to have" feature.
Miscellaneous devices are represented with the struct miscdevice
which is declared in /include/linux/miscdevice.h
as
struct miscdevice { int minor; const char *name; const struct file_operations *fops; struct list_head list; struct device *parent; struct device *this_device; const char *nodename; mode_t mode; };
Quite a big one, ha? However, we only should take care of the first three members of the structure:
minor
stands for the minor number of the device. It is preferred to set it to MISC_DYNAMIC_MINOR
(at least in for our module) unless you need some specific number to be used.
name
this is the name of our device as it should appear in the /dev
filesystem.
struct file_operations
is a set of pointers to corresponding implementations of IO functions and a pointer to the owner module. This structure is too big to be presented here, but you may find it in /include/linux/fs.h
So, first of all, add another include file to your code (which we've written in previous article) with
#include <linux/miscdevice.h>
Then, we should add custom handlers for functions and global variables we are interested in, namely - open
, release
, ioctl
and variable in_use
.
/* We will set this variable to 1 in our open handler and erset it back to zero in release handler*/ int in_use = 0; /* This function will be invoked each time a user process attempts to open our device. You should keep in mind that the prototype of this function may change along different kernel versions. */ static int our_open(struct inode *inode, struct file *file) { /* We would not like to allow multiple processes to open this device */ if(in_use) return -EBUSY; in_use++; printk("device has been opened\n"); return 0; } /* This function, in turn, will be called when a process closes our device */ static int our_release(struct inode *inode, struct file *file) { in_use--; printk("device has been closed\n"); return 0; } /* This function will handle ioctl calls performed on our device */ static int our_ioctl(struct file *file, unsigned int cmd, unsigned long arg) { int retval = 0; /* We will fill this function in the Part III of this series */ return retval; }
Now it's time to create the struct file_operations
and struct miscdevice and populate the relevant fields:
static const struct file_operations our_fops =\ { .owner = THIS_MODULE, .open = &our_open, .release = &our_release, .unlocked_ioctl = (void*)&our_ioctl, .compat_ioctl = (void*)&our_ioctl }
A reasonable question would be "Why do we set unlocked_ioctl
and compat_ioctl
with the same value and where the heck is the regular ioctl?". There is nothing special about this. unlocked_ioctl
is used on 64-bit platforms and compat_ioctl
in 32 bit or in compatibility mode, and it is totally normal to make them point at the same location as long as you handler function does not mess the types up. As to ioctl
, it is simply not there anymore...
static struct miscdevice our_device = \ { MISC_DYNAMIC_MINOR, "interceptor", &our_fops };
After all this, we should make a small adjustment to our init_module
function by inserting the following code:
int retval = misc_register(&our_device);
You should also change "return 0;
" to "return retval;
". The code above tells the system to register a miscellaneous device described by the miscdevice
structure with the kernel. In case the minor field is assigned MISC_DYNAMIC_MINOR
(our case exactly), kernel fills it with random (from our point of view) number, otherwise, the requested minor number is used.
Our cleanup_module
function should have this line added:
misc_deregister(&our_device);
in order to unregister our device and remove it from the system.
Important note: functions exported by kernel return 0 upon success or negative error code in case of failure.
By now we have a working kernel module which registers a miscellaneous device when loaded and unregisters it when unloaded. Build it now with the make
command. Load it with insmod
and check the content of the /dev
file system. You will see that there is a new device called "interceptor" with number 10 as major number and what ever has been assigned as minor number. You may unload it now.
If you wish, you may try to open and close the /dev/interceptor
device from a user process and check the log with dmesg | tail
. You will see the lines "device has been opened" and "device has been closed" respectively. You may also try to open the device from two user processes simultaneously, then you will see that only one process may successfully open it.
In the next section we are going to add some code to our module, which would make it possible to actually patch the sys_call_table
and replace original calls with custom wrappers.
Hope this post was helpful. See you at the next one!