Основываясь на моем учебном проекте, моя текущая задача состоит в том, чтобы сгенерировать 10 случайных чисел, используя модуль ядра, и моя программа пользовательского пространства (c-программа) должна иметь возможность отображать эти числа. Я изучал пространство ядра и программы пользовательского пространства. И я столкнулся с созданием персональных устройств. Я создал устройство, использующее эту команду.реализация символьного устройства для генерации случайных чисел
mknod /dev/my_device c 222 0
От этого я понял, что это устройство является промежуточным звеном между пользовательским пространством и программами пространства ядра. Таким образом, я создал модуль ядра регистров Wich и отменяет регистрацию мой характер device.Saved в my_dev.c
#include<linux/module.h>
#include<linux/init.h>
#include"my_dev.h"
MODULE_AUTHOR("Krishna");
MODULE_DESCRIPTION("A simple char device");
static int r_init(void);
static void r_cleanup(void);
module_init(r_init);
module_exit(r_cleanup);
static int r_init(void)
{
printk("<1>hi\n");
if(register_chrdev(222,"my_device",&my_fops)){
printk("<1>failed to register");
}
return 0;
}
static void r_cleanup(void)
{
printk("<1>bye\n");
unregister_chrdev(222,"my_device");
return ;
}
Мой файл Make для compling этот модуль является
obj-m += my_dev.o
all:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules
clean:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) clean
Этот модуль ядра компилируется и загружается в память используя команду insmod.
Вот программа, которая записывает и читает какой-либо текст в пользовательский буфер, сохраненный как my_dev.h.
/*
* my device header file
*/
#ifndef _MY_DEVICE_H
#define _MY_DEVICE_H
#include <linux/fs.h>
#include <linux/sched.h>
#include <linux/errno.h>
#include <asm/current.h>
#include <asm/segment.h>
#include <asm/uaccess.h>
char my_data[80]="heloooo"; /* our device */
int my_open(struct inode *inode,struct file *filep);
int my_release(struct inode *inode,struct file *filep);
ssize_t my_read(struct file *filep,char *buff,size_t count,loff_t *offp);
ssize_t my_write(struct file *filep,const char *buff,size_t count,loff_t *offp);
struct file_operations my_fops={
open: my_open,
read: my_read,
write: my_write,
release:my_release,
};
int my_open(struct inode *inode,struct file *filep)
{
/*MOD_INC_USE_COUNT;*/ /* increments usage count of module */
return 0;
}
int my_release(struct inode *inode,struct file *filep)
{
/*MOD_DEC_USE_COUNT;*/ /* decrements usage count of module */
return 0;
}
ssize_t my_read(struct file *filep,char *buff,size_t count,loff_t *offp)
{
/* function to copy kernel space buffer to user space*/
if (copy_to_user(buff,my_data,strlen(my_data)) != 0)
printk("Kernel -> userspace copy failed!\n");
return strlen(my_data);
}
ssize_t my_write(struct file *filep,const char *buff,size_t count,loff_t *offp)
{
/* function to copy user space buffer to kernel space*/
if (copy_from_user(my_data,buff,count) != 0)
printk("Userspace -> kernel copy failed!\n");
return 0;
}
#endif
Вот моя космическая программа acs.c
пользователя, который при запуске печатает «heloooo», читая текст из буфера ядра из приведенной выше программы.
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
int main()
{
int fd=0,ret=0;
char buff[80]="";
fd=open("/dev/my_device",O_RDONLY);
printf("fd :%d\n",fd);
ret=read(fd,buff,10);
buff[ret]='\0';
printf("buff: %s ;length: %d bytes\n",buff,ret);
close(fd);
}
Теперь моя проблема в том, что мне нужно написать программу пользовательского пространства, которая при запуске печатает 10 случайных чисел. Но эти числа должны быть сгенерированы с использованием модуля ядра. Так что в основном выше трех кодов хорошо работает и печатает «helooo». что мне нужно сделать, вместо «helooo» мне нужно получить случайные числа в качестве вывода.
Вот модуль памяти, который генерирует некоторые случайные числа, используя алгоритм линейного конгруэнтного генератора. LCG.c
#include <linux/module.h> /* Needed by all modules */
#include <linux/kernel.h> /* Needed for KERN_INFO */
int init_module(void)
{
int M = 8; //Modulus, M>0
int a = 9; //Multiplier, 0 <= a < M.
int c = 3; //Increment, 0 <= c < M.
int X = 1; //seed value, 0 <= X(0) < M
int i; //iterator, i < M
for(i=0; i<8; i++)
{
X = (a * X + c) % M;
printk(KERN_INFO "%d\n",X);
}
return 0;
}
void cleanup_module(void)
{
printk(KERN_INFO "Task Done ! :D.\n");
}
У меня есть все коды. Но я не знаю, как поместить этот код генератора случайных чисел в код активации моего устройства charecter. Когда я запускаю программу acs.c, мне нужно получить вывод модуля памяти LCG.c
, используя персональное устройство. Пожалуйста, помогите мне найти решение.
'ret = read (fd, buff);' 'read()' принимает 3 параметра, вы забываете последний (nbytes) –
Но эта программа работает даже без третьего параметра – Krishna
Вы уверены? Он не компилируется: 'error: инициализатор массива должен быть списком инициализатора int buff [80] =" ";' –