


Dua rangka kerja pemacu peranti khas dalam kernel Linux: subsistem misc dan rangka kerja pemacu pengecaman peranti 3+2+1
Subsistem misc ialah rangka kerja pemacu peranti yang mudah dan fleksibel dalam kernel Linux Ia boleh digunakan untuk melaksanakan beberapa pemacu peranti yang bukan milik subsistem lain, seperti peranti aksara, peranti maya, peranti hibrid, dsb. Kelebihan subsistem misc ialah ia mudah dan mudah digunakan Ia tidak memerlukan menulis banyak kod dan hanya perlu melaksanakan beberapa fungsi operasi asas. Walau bagaimanapun, subsistem misc juga mempunyai beberapa kelemahan, seperti ketidakupayaan untuk menyokong berbilang kejadian peranti, pepohon peranti dan pertukaran panas. Untuk menyelesaikan masalah ini, artikel ini akan memperkenalkan rangka kerja pemacu peranti baharu: rangka kerja pemacu pengecaman peranti 3+2+1, yang berdasarkan gabungan subsistem misc dan subsistem platform, serta boleh mencapai lebih banyak fungsi dan ciri.
Penggunaan pelbagai
Terdapat tiga kategori utama peranti dalam Linux: peranti watak, rangkaian dan blok Setiap peranti dibahagikan kepada banyak kategori Sebagai contoh, peranti aksara diprabahagikan kepada banyak kategori dan peranti utama yang digunakan oleh kategori ini ditandakan. dalam fail. Nombor peranti, tetapi walaupun begitu, terdapat berpuluh-puluh juta perkakasan, dan sentiasa ada ikan yang tergelincir melalui jaring Untuk peranti aksara ini yang sukar diklasifikasikan, Linux menggunakan peranti "campuran" untuk menerangkannya secara seragam , dan berikan mereka peranti utama biasa No. 10, hanya gunakan nombor peranti ini untuk membezakan peranti, , peranti ini terutamanya termasuk penjana nombor rawak, LCD, penjana jam, dsb. Selain itu, seperti banyak subsistem yang turut merangkum semula cdev, misc juga akan membuat fail peranti secara automatik, menghapuskan keperluan untuk menggunakan class_create() dan device_create() setiap kali anda menulis antara muka cdev.
objek lain yang disediakan dalam kernel:
//include/linux/miscdevice.h 55 struct miscdevice { 56 int minor; 57 const char *name; 58 const struct file_operations *fops; 59 struct list_head list; 60 struct device *parent; 61 struct device *this_device; 62 const char *nodename; 63 umode_t mode; 64 };
fops seperti peranti aksara dan memberikannya minor Jika minor menggunakan makro MISC_DYNAMIC_MINOR (sebenarnya 255), kernel akan menetapkan nombor peranti kecil dan kernel lain secara automatik. telah pun melaksanakan perjanjian Untuk nombor peranti kecil, sila rujuk **"include/linux/miscdevice.h"**. Setelah semuanya sedia, hanya gunakan API berikut untuk mendaftar/log keluar ke kernel
178 int misc_register(struct miscdevice * misc) 238 int misc_deregister(struct miscdevice *misc)
analisis misc
Adakah mudah untuk menggunakan misc? Tetapi walaupun Sparrow kecil dan serba lengkap, justru kerana struktur misc yang diperkemas, kita boleh dengan mudah memahami idea berlapis yang terkandung di dalamnya. Kaedah reka bentuk misc dicerminkan dalam banyak subsistem yang menggunakan cdev sebagai antara muka, dan pembahagian yang jelas antaranya ialah Idea lapisan adalah salah satu daripada dua tonggak pemacu Linux (yang satu lagi ialah pemisahan).Kami boleh belajar daripada idea reka bentuk dan meningkatkan kualiti pemandu kami. Seterusnya, kami menganalisis secara ringkas mekanisme dalaman misc.
misc_init
Sebagai subsistem Linux, subsistem misc akan menyelesaikan kerja penyediaan semasa proses permulaan Linux, yang terutamanya termasukmemulakan struktur data, mencipta kelas yang sepadan, mencipta, memulakan dan mendaftarkan objek cdev ke kernel, dsb. . Dengan asas-asas ini, kita boleh memprogramkan menggunakan pelbagai manfaat misc.
//drivers/char/misc.c 56 static const struct file_operations misc_fops = { 157 .owner = THIS_MODULE, 158 .open = misc_open, 159 .llseek = noop_llseek, 160 }; 268 static int __init misc_init(void) 269 { 272 #ifdef CONFIG_PROC_FS 273 proc_create("misc", 0, NULL, &misc_proc_fops); 274 #endif 275 misc_class = class_create(THIS_MODULE, "misc"); 281 if (register_chrdev(MISC_MAJOR,"misc",&misc_fops)) 282 goto fail_printk; 283 misc_class->devnode = misc_devnode; 284 return 0; 292 } 293 subsys_initcall(misc_init);
“
misc_init() –293–>Subsistem misc akan dimulakan semasa sistem dimulakan
–273–>Bergantung pada konfigurasi sistem, antara muka /proc mungkin perlu disediakan
–275–>Buat kelas dalam /sysfs bernama misc
–281–>Menggunakan nombor peranti utama statik (10) dan set kaedah terkapsul misc_fops, register_chrdev() akan mencipta objek cdev secara dalaman dan menggunakan kedua-dua parameter ini untuk memulakannya dan mendaftarkannya kepada kernel Objek cdev ini akan bertanggungjawab untuk semua Nombor peranti peranti rambang. Untuk perhubungan antara objek cdev dan nombor peranti, lihat cdev_map.
–158–>fops yang digunakan oleh objek cdev misc Jelas sekali, proses panggilan adalah sama seperti peranti aksara biasa, chrdev_open()->misc_open().
”
misc_register
接下来,老规矩,我们从”XXX_register”开始分析,在Linux内核中,这些”XXX_register”往往就是一个设备对象注册到内核的接口,是研究当相应对象注册进去之后内核动作的最佳入口。
178 int misc_register(struct miscdevice * misc) 179 { 180 dev_t dev; 187 if (misc->minor == MISC_DYNAMIC_MINOR) { 188 int i = find_first_zero_bit(misc_minors, DYNAMIC_MINORS); 193 misc->minor = DYNAMIC_MINORS - i - 1; 194 set_bit(i, misc_minors); 195 } 206 dev = MKDEV(MISC_MAJOR, misc->minor); 208 misc->this_device = device_create(misc_class, misc->parent, dev, 209 misc, "%s", misc->name); 210 if (IS_ERR(misc->this_device)) { 211 int i = DYNAMIC_MINORS - misc->minor - 1; 212 if (i = 0) 213 clear_bit(i, misc_minors); 214 err = PTR_ERR(misc->this_device); 216 } 222 list_add(&misc->list, &misc_list); 226 }
“
misc_register()
–187–> 如果指定的minor是动态分配,那么进入相关语句块。
–188–> 使dev用位图遍历API-find_first_zero_bit找到最小未用的设备号。
–193–> 得到分配好的次设备号。
–208–> (根据设备号)创建设备文件,使用的是misc_init中创建的misc_class,至此就可以实现misc设备文件的自动创建。就相当与我们在纯粹的cdev驱动中使用class_create()+device_create()创建设备文件。一个设备文件和一个设备号相联系,而misc的所有的设备号都和misc_init创建的cdev对象相联系,所以打开的任何一个misc设备文件首先回调的就是(chrdev_open()->)misc_open()。
–222–> 关键,将这个新分配的misc加入到misc链表中,用于管理所有的misc设备,便于misc_open()提取具体设备的fops。”
misc_open
构建的misc子系统,将设备添加到了该子系统中,接下来我们来看一下应用层程序是如何打开一个misc设备的。由于misc也是一种字符设备,所以其提供的接口也是位于/dev中。但是正如misc的定义,其中的设备五花八门却共用同一个主设备号,这就意味着最终被chrdev_open回调的misc_open一定要具备根据被打开的不同文件为file结构准备不同的操作方法这一能力,即在驱动中实现对子设备的识别,或者称之为”多态”。
112 static int misc_open(struct inode * inode, struct file * file) 113 { 114 int minor = iminor(inode); 115 struct miscdevice *c; 116 int err = -ENODEV; 117 const struct file_operations *new_fops = NULL; 121 list_for_each_entry(c, &misc_list, list) { 122 if (c->minor == minor) { 123 new_fops = fops_get(c->fops); 124 break; 125 } 126 } 144 replace_fops(file, new_fops); 145 if (file->f_op->open) { 146 file->private_data = c; 147 err = file->f_op->open(inode,file); 148 } 152 }
“
misc_open()
–121–>遍历misc设备链表,根据被打开的设备的次设备号找到设备对象。
–123–>存储这个设备对象的操作方法集unique_fops。
–144–>将misc设备具体的操作方法集unique_fops替换到filp中的f_op中,这个位置原来是misc的cdev对象的fops,filp带着这个unique_fops从open()返回,就实现了不同的设备对应不同的操作方法,即面向对象的”多态””
3+2+1多设备识别驱动模型
通过上述对misc机制的分析,我们不难总结出一个支持设备识别的3+2+1驱动模型(3个函数+2个数据结构+1个封装):
- 初始化整个驱动组的**xxx_init()**,通常用模块加载函数或总线的probe函数实现;
- 用于注册一个子驱动的**xxx_register()**,需要EXPORT到符号表;
- 能够根据传入的inode识别具体的设备并将其操作方法集放到filp中的**xxx_open()**。
+
- 用于存储每一个驱动对象的通用链表或数组+priv_data
- 用于存储子设备号的位图。
+
- 将所有的不同的设备用一个统一的结构进行封装
至此,我们就可以写一写这个3+2+1驱动模型的模板。
1个封装
struct multidevice{ struct list_head head; int minor; struct file_operations* fops; void *priv; //私有数据,供read/write等接口识别的信息,以及其他数据都放这里 };
2个数据结构
struct list_head multi_dev_list; unsigned int minors_map; //根据设备号数目的不同选数据类型or数组
3个函数
int major,baseminor = 0,max_dev = sizeof(minors_map)*8; #define DEV_NAME "multi_device" struct class *cls; xxx_open(struct inode *inode,struct file *file){ int minor = iminor(inode); struct multidevice *dp; const struct file_operations *new_fops = NULL; list_for_each_entry(dp, &multi_dev_list, head) { if (dp->minor == minor) { new_fops = fops_get(dp->fops); break; } } replace_fops(file, new_fops); if (file->f_op->open) { file->private_data = dp file->f_op->open(inode,file); } } xxx_init(void){ dev_t devno, INIT_LIST_HEAD(&multi_dev_list); init_map(&minors_map); struct cdev *multi_cdev = cdev_alloc(); cdev_init(multi_cdev, multi_fops); alloc_chrdev_region(&devno, baseminor, count,DEV_NAME); major = MAJOR(devno); cdev_add(multi_cdev , devno, count); cls = class_create(THIS_MODULE, DEV_NAME); } /*---------------下面是给待加驱动用的----------------------*/ xxx_register(struct *multidevice dev){ dev_t dev; if (dev->minor == MISC_DYNAMIC_MINOR) { int i = find_first_zero_bit(minors_map, DYNAMIC_MINORS); dev->minor = DYNAMIC_MINORS - i - 1; set_bit(i, minors_map); } dev_t pri_devno = MKDEV(major, dev->minor); device_create(multi_class, NULL, pri_devno, "%s", misc->name); list_add(dev->head, &multi_dev_list); } EXPORT_SYMBOL(xxx_register)
通过本文,我们了解了misc子系统和3+2+1设备识别驱动框架的原理和方法,它们可以用来实现一些特殊的设备驱动,如识别设备,虚拟设备,混合设备等。我们应该根据实际需求选择合适的框架,并遵循一些基本原则,如使用正确的注册和注销函数,使用正确的文件操作结构体,使用正确的设备树节点等。misc子系统和3+2+1设备识别驱动框架是Linux内核中两个有用而灵活的设备驱动框架,它们可以提升设备驱动的兼容性和可扩展性,也可以提升开发者的效率和质量。希望本文能够对你有所帮助和启发。
Atas ialah kandungan terperinci Dua rangka kerja pemacu peranti khas dalam kernel Linux: subsistem misc dan rangka kerja pemacu pengecaman peranti 3+2+1. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Alat AI Hot

Undresser.AI Undress
Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover
Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool
Gambar buka pakaian secara percuma

Clothoff.io
Penyingkiran pakaian AI

AI Hentai Generator
Menjana ai hentai secara percuma.

Artikel Panas

Alat panas

Notepad++7.3.1
Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina
Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1
Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6
Alat pembangunan web visual

SublimeText3 versi Mac
Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Topik panas



Langkah -langkah untuk memulakan Nginx di Linux: Periksa sama ada Nginx dipasang. Gunakan SistemCTL Mula Nginx untuk memulakan perkhidmatan Nginx. Gunakan SistemCTL Dayakan NGINX untuk membolehkan permulaan automatik Nginx pada permulaan sistem. Gunakan Status SistemCTL Nginx untuk mengesahkan bahawa permulaan berjaya. Lawati http: // localhost dalam pelayar web untuk melihat halaman selamat datang lalai.

Bagaimana untuk mengesahkan sama ada nginx dimulakan: 1. Gunakan baris arahan: status sistem sistem nginx (linux/unix), netstat -ano | Findstr 80 (Windows); 2. Periksa sama ada port 80 dibuka; 3. Semak mesej permulaan Nginx dalam log sistem; 4. Gunakan alat pihak ketiga, seperti Nagios, Zabbix, dan Icinga.

Pelayan tidak mempunyai kebenaran untuk mengakses sumber yang diminta, mengakibatkan ralat NGINX 403. Penyelesaian termasuk: Periksa keizinan fail. Semak konfigurasi .htaccess. Semak konfigurasi Nginx. Konfigurasikan keizinan Selinux. Semak peraturan firewall. Menyelesaikan masalah lain seperti masalah penyemak imbas, kegagalan pelayan, atau kesilapan lain yang mungkin.

Perbezaan utama antara CentOS dan Ubuntu adalah: asal (CentOS berasal dari Red Hat, untuk perusahaan; Ubuntu berasal dari Debian, untuk individu), pengurusan pakej (CentOS menggunakan yum, yang memberi tumpuan kepada kestabilan; Ubuntu menggunakan APT, untuk kekerapan yang tinggi) Pelbagai tutorial dan dokumen), kegunaan (CentOS berat sebelah ke arah pelayan, Ubuntu sesuai untuk pelayan dan desktop), perbezaan lain termasuk kesederhanaan pemasangan (CentOS adalah nipis)

Docker menggunakan ciri -ciri kernel Linux untuk menyediakan persekitaran berjalan yang cekap dan terpencil. Prinsip kerjanya adalah seperti berikut: 1. Cermin digunakan sebagai templat baca sahaja, yang mengandungi semua yang anda perlukan untuk menjalankan aplikasi; 2. Sistem Fail Kesatuan (Unionfs) menyusun pelbagai sistem fail, hanya menyimpan perbezaan, menjimatkan ruang dan mempercepatkan; 3. Daemon menguruskan cermin dan bekas, dan pelanggan menggunakannya untuk interaksi; 4. Ruang nama dan cgroups melaksanakan pengasingan kontena dan batasan sumber; 5. Pelbagai mod rangkaian menyokong interkoneksi kontena. Hanya dengan memahami konsep -konsep teras ini, anda boleh menggunakan Docker dengan lebih baik.

CentOS akan ditutup pada tahun 2024 kerana pengedaran hulu, RHEL 8, telah ditutup. Penutupan ini akan menjejaskan sistem CentOS 8, menghalangnya daripada terus menerima kemas kini. Pengguna harus merancang untuk penghijrahan, dan pilihan yang disyorkan termasuk CentOS Stream, Almalinux, dan Rocky Linux untuk memastikan sistem selamat dan stabil.

Langkah Pemasangan CentOS: Muat turun Imej ISO dan Burn Bootable Media; boot dan pilih sumber pemasangan; Pilih susun atur bahasa dan papan kekunci; Konfigurasikan rangkaian; memisahkan cakera keras; Tetapkan jam sistem; Buat pengguna root; pilih pakej perisian; Mulakan pemasangan; Mulakan semula dan boot dari cakera keras selepas pemasangan selesai.

Bagaimana cara menggunakan desktop Docker? Docktop Docktop adalah alat untuk menjalankan bekas Docker pada mesin tempatan. Langkah -langkah untuk digunakan termasuk: 1. Pasang desktop Docker; 2. Mulakan desktop Docker; 3. Buat imej Docker (menggunakan Dockerfile); 4. Membina imej Docker (menggunakan Docker Build); 5. Jalankan bekas Docker (menggunakan Docker Run).
