Linux下PCI设备驱动程序之注册详解
一 识别方式:
天朝为了方便每个人的管理,搞了一套身份识别的玩意(档案和身份证)。kernel本省就像一个社会,每个进来的设备必须要有它独特的名字和一些档案。
这个工作对PCI来说,它是由pci_device_id这个结构体来进行身份信息保存的:
17 struct pci_device_id {
18 __u32 vendor, device; /* Vendor and device ID or PCI_ANY_ID*/
19 __u32 subvendor, subdevice; /* Subsystem ID's or PCI_ANY_ID */
20 __u32 class, class_mask; /* (class,subclass,prog-if) triplet */
21 kernel_ulong_t driver_data; /* Data private to the driver */
22 };
针对一些英文不好的同学,还是有必要对上面的信息进行翻译一下。vendor,device 分别代表设备商给的设备编号(身份证号)和该设备的名字(姓名)
subverdor,subdevice 分别对应的是该设备是否有别名,绝大多数设备是不需要的。class,class_mask分别对应的是该pci设备属于哪个设备:比如说,属于网络,块设备。
driver_data是属于该设备的私有数据,每个设备都不一样。
二 PCI设备注册和解注册:
人有生老病死,同样,每个驱动也有他们的宿命:出生->生活->死亡(任何中形式)这里,我就介绍一下PCI设备是怎么诞生,怎么死亡的。
诞生:需要注册函数:pci_register_driver 该函数在Android4.2内核中的路径是:kernel/drivers/pci/pci-driver.c
该函数的定义:
int __pci_register_driver(struct pci_driver *drv, struct module *owner,
1106 const char *mod_name)
1107 {
1108 int error;
1109
1110 /* initialize common driver fields */
1111 drv->driver.name = drv->name;
1112 drv->driver.bus = &pci_bus_type;
1113 drv->driver.owner = owner;
1114 drv->driver.mod_name = mod_name;
1115
1116 spin_lock_init(&drv->dynids.lock);
1117 INIT_LIST_HEAD(&drv->dynids.list);
1118
1119 /* register with core */
1120 error = driver_register(&drv->driver);
1121 if (error)
1122 goto out;
1123
1124 error = pci_create_newid_file(drv);/* crate new id file of pci */
1125 if (error)
1126 goto out_newid;
1127
1128 error = pci_create_removeid_file(drv); /* same as above function only different in create different attr */
1129 if (error)
1130 goto out_removeid;
1131 out:
1132 return error;
1133
1134 out_removeid:
1135 pci_remove_newid_file(drv);
1136 out_newid:
1137 driver_unregister(&drv->driver);
1138 goto out;
1139 }
死亡:设备的死亡就意味着是解注册,这里面要做的是删除注册过的文件设备,删除驱动,把该设备从链表中删除等操作。
1151 void
1152 pci_unregister_driver(struct pci_driver *drv)
1153 {
1154 pci_remove_removeid_file(drv);
1155 pci_remove_newid_file(drv);
1156 driver_unregister(&drv->driver);
1157 pci_free_dynids(drv);
1158 }