我在我的 Ubunut 14.04 机器上使用 Sensoray 的专有 PCI 卡(我打算很快升级到 18.04)。该卡附带驱动程序的源代码和用于构建和安装驱动程序的 Makefile。与驱动程序相关的 Makefile 部分在这里:
######################################################################
# for kernel modeule level driver:
# Kernel directory
KDIR := /lib/modules/$(shell uname -r)/build
# Module directory
MODDIR := /lib/modules/$(shell uname -r)/kernel/drivers/sensoray
# System values
PWD := $(shell pwd)
KERNEL_24 := $(if $(wildcard $(KDIR)/Rules.make),1,0)
# Target file
obj-m := s626.o
# Source files
ifeq ($(KERNEL_24),0) # > 2.4
s626-objs := s626drv.o
else # <= 2.4
s626-objs := s626drv.o
endif
.PHONY: all clean modules_install
ifeq ($(KERNEL_24),0) # > 2.4
ifeq ($(KERNELRELEASE),)
all:
$(MAKE) -C $(KDIR) M=$(PWD) SUBDIRS=$(PWD)
clean modules_install:
$(MAKE) -C $(KDIR) M=$(PWD) SUBDIRS=$(PWD) $@
endif # KERNELRELEASE
else # <= 2.4
ifneq ($(KERNELRELEASE),)
include $(KDIR)/Rules.make
s626.o: $(s626-objs)
$(Q)$(LD) $(LD_RFLAG) -r -o $@ $(s626-objs)
else
all:
$(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules
clean:
rm -f *.ko *.o .*.cmd .*.o.flags *.mod.c
endif # KERNELRELEASE
endif # KERNEL_24
ifeq ($(KERNEL_24),1) # <= 2.4
install: s626.o
@if [ -d /lib/modules/$(shell uname -r)/kernel/drivers/sensoray/ ];\
then rm -f /lib/modules/$(shell uname -r)/kernel/drivers/sensoray/s626.*;\
fi
@if [ -d /lib/modules/$(shell uname -r)/extra/ ];\
then rm -f /lib/modules/$(shell uname -r)/extra/s626.*;\
fi
su -c "set -x;./MAKEDEV;mkdir -p $(MODDIR);cp -v s626.o $(MODDIR);depmod -a"
else
install: s626.ko
@if [ -d /lib/modules/$(shell uname -r)/kernel/drivers/sensoray/ ];\
then rm -f /lib/modules/$(shell uname -r)/kernel/drivers/sensoray/s626.*; \
fi
@if [ -d /lib/modules/$(shell uname -r)/extra/ ];\
then rm -f /lib/modules/$(shell uname -r)/extra/s626.*;\
fi
@if [ -d /lib/modules/$(shell uname -r)/kernel/drivers/staging/comedi/drivers ];\
then rm -f /lib/modules/$(shell uname -r)/kernel/drivers/staging/comedi/drivers/s626.*;\
fi
su -c "set -x;./MAKEDEV;mkdir -p $(MODDIR);cp -v s626.ko $(MODDIR);install -m 444 s626.ko $(MODDIR);depmod -a"
endif # KERNEL > 2.4
在 Makefile 的末尾,看起来.ko
文件一旦创建,它就被简单地复制到/lib/modules/$(shell uname -r)/kernel/drivers/sensoray
目录中。还有一个自定义的 MAKEDEV shell 脚本正在执行以创建设备文件。该脚本在这里给出:
#!/bin/bash
function makedev () {
for dev in 0 1 2 3; do
echo "/dev/$1$dev: char $2 $dev"
rm -f /dev/$1$dev
mknod /dev/$1$dev c $2 $dev
chmod 666 /dev/$1$dev
done
# symlink for default device
rm -f /dev/$1
ln -s /dev/${1}0 /dev/$1
}
makedev s626a 146
问题是每当系统重新启动时。看起来驱动程序已正确加载,但设备文件/dev
消失了。我对驱动程序开发知之甚少,但我对这个问题进行了很多研究,我发现有关如何在启动时创建这些设备文件的信息相互矛盾。有人说创建 udev 规则是最好的,其他人说 udev 规则不应该用于创建丢失的设备文件,或者 udev 不再负责设备文件的创建,因为它引用了新的 devtempfs。我的问题是:解决此问题的正确方法是什么?我已经尝试过 udev 规则方法,在该方法中我基本上从 Sensoray 调用自定义 MAKEDEV 脚本,但这仅在我的规则不引用 pci 卡的任何特定属性时才有效。例如,以下规则有效:
ACTION=="add", SUBSYSTEM=="pci", RUN+="/home/kpopek/MAKEDEV"
该规则不
ACTION=="add", SUBSYSTEM=="pci", ATTR{vendor}=="0x1131", RUN+="/home/kpopek/MAKEDEV"
在启动时似乎没有任何uevent
与 Sensoray pci 卡匹配的东西,这可能是缺少设备文件的根本原因。我还没有弄清楚如何uevents
在启动时记录以验证这一点。如果 udev 规则是正确的方法,我真的想要一个特定于卡的规则,而不是由于uevent
来自另一台设备而碰巧运行脚本的通用规则。
看起来自定义驱动程序没有生成所需的 udev 事件(必须查看驱动程序来确认这一点)。
因此,正确的方法是修复驱动程序以生成事件。在最简单的情况下,您将拥有类似于this的代码。在更复杂的情况下,您可以拥有额外的 udev 规则来完成更复杂的事情。
如果您不能或不想修改内核模块,恕我直言,任何可行的解决方案都是允许的。理想情况下,您希望有一个依赖项,该依赖项在加载模块时创建设备,并在卸载时删除它,但您需要 udev 事件(见上文)。所以下一个最好的事情是在启动时创建它,这意味着我只需添加一个 init.d 或 systemd 脚本。
如果您可以为您的硬件劫持现有的 udev 事件,那也很好,但看起来您已经尝试过但没有找到。