第四期训练营-rust for linux 作业
作业1
make x86_64_defconfig
make LLVM=1 menuconfig
报错,解决 sudo apt install lld

配置界面:

接下来编译
make LLVM=1 -j 12

作业2
问题1:
在Kconfig中有相关的配置,具体在:
obj-m := r4l_e1000_demo.o
由于是obj-m 所以会按照模块编译
问题2:
通过insmod,可以将out-of-tree 的module挂载到内核中驱动的管理链表中,就和直接挂载的驱动没有大的区别,可以直接访问内核空间。
禁用C的e1000网卡

挂载e1000网卡,并setup
insmod r4l_e1000_demo.ko
ip link set eth0 up

ifconfig eth0 broadcast 10.0.2.255
ip addr add 10.0.2.15/255.255.255.0 dev eth0
ip route add default via 10.0.2.1
ping 10.0.2.2

作业3


编译后将ko文件放到rootfs下,启动qemu后用insmod挂载就会有下面的输出

作业4
修改部分
- 释放bars
- 注销设备
- 注销中断回调函数(否则第二次注册的时候无法ping通)
释放bars和注销设备
将remove接口修改,方便传入pci::Device 可以从init中看到,最开始的注册都是由这个成员注册的。
插播一下remove的调用原理:
pci::Drive的trait提供了remove接口。
同时该接口会被remove_callback调用。而这个函数在driver被注册的时候:

所以在remove_callback中,会有pci_dev这个成员,而我们也只需要将这个成员传入remove中,就可以在驱动中很方便的使用这个成员,将其注销掉。

然后再提供几个api用来注销掉使用的bar


这样只需要将remove接口中如下实现:

注销中断函数
在init最后可以看到注册了一个中断回调函数 irq_handler
需要在stop中将其注销掉。
irq_handler: AtomicPtr<kernel::irq::Registration<E1000InterruptHandler>>
pub struct Registration<H: Handler>(InternalRegistration<H::Data>);
E1000InterruptHandler提供了drop,但是Registration没有提供
impl<T: PointerWrapper> Drop for InternalRegistration<T> {
fn drop(&mut self) {
unsafe { bindings::free_irq(self.irq, self.data) };
unsafe { T::from_pointer(self.data) };
}
}
并且包裹的元素是private,所以需要为Registration提供一个drop接口用来释放掉内部的元素

最后在stop的函数实现(其中 netif_stop_queue netif_carrier_off 就是仿照open中的注册函数进行注销):

最后的效果:



作业5
得到cicv的主次设备号
ls /dev/cicv -al

可以看到是248 0
在 build_image.sh 脚本中也有对应挂载的语句

/proc/devices中配置将248分配给rust_chrdev

在module!中以后对应的name字段,从而将248分配给了rust_chrdev

修改部分

结果:

测验
mount -t nfs -o nolock host_machine:/home/alan/Project/Rust/OS-4/cicv-r4l-3-lliYanze/r4l_experiment/driver /mnt
按照文档操作后,最终实现

- 现将实验5实现的字符设备拿来用。小修改load_module.sh 直接将主设备号设置成248
- 在sample其实可以看到一个信号量的示例。使用了
CondVar和Mutex来做通信。这可以满足测验的通信要求。
但是最开始将这两个当成员放到了RustFile的结构中,并且open会new一个,这就导致了一个问题,每次打开文件都会创建一个新的信号量,还会有卡死的问题,目前还不知道为什么。

然后就把这两个放到全局使用了。。

只需要在write和read中实现阻塞和通信就行。


最后实现的结果:



浙公网安备 33010602011771号