当前位置:首页 > 嵌入式 > 嵌入式教程
[导读]QNX 4.25设备驱动程序的编写

摘要:介绍实时操作系统QNX4.25下编写设备驱动程序的大体框架、底层细节以及诸多注意点。针对使用较为普遍的PCI设备作为较为详细的描述。

    关键词:驱动程序 QNX 实时操作系统 PCI

引言

QNX是一个多任务、多用户、分布式、可嵌入式符合POSIX标准的微内核的主流实时操作系统,广泛用于实时性能、开发灵活性、网络灵活性要求较高的场合,如电信系统、医疗仪器、航空航天、工业自动化、交通运输、POS机、信息家电等。

QNX是一个适合软件/硬件定制的实时操作系统。如果你曾经试图在传统的UNIX或Windows平台下开发设备驱动程序,那么,QNX下开发驱动程序一定会让你受宠若惊。由于QNX的微内核结构,QNX下的系统进程和用户所写的进程没有什么不同,甚至没有私有的隐藏起来的以至用户不能使用的界面。正是这种结构给QNX带来了无与伦比的可扩展性,使得在QNX下写驱动程序如同写其它程序一般方便。设备驱动程序能够获取普通程序所能获得的任务服务。在QNX中增加一个新的驱动程序不会影响操作系统其它程序的任何部分,QNX环境所需的唯一改变是实现地启动新的驱动程序。

当然,我们会遇到形形色色的硬件设备,某些驱动程序可能将以特殊方式控制设备的存在和配置。本文只想集中讨论QNX下如何进入、控制设备级的通用硬件,对所有驱动程序来讲这是一个共性问题。其中,将对使用较多的PCI设备作较为详细的叙述。以下是硬件驱动程序的编写。

1 探测硬件

首先,需要判断设备是否存在,然后查询该设备的配置(例如,设备基地址、中断号等)。对于某类设备,一般会有一大相应的标准机制来判断其配置。每块设备的基地址、中断号等是编程必须的资源,例如,常用的ISA及PCI硬件设备。对于ISA设备,一般由板上手工跳线设定,不言自明;对于常用的PCI设备,这些资源会由系统自动分配,特别是添减设备,可能会发生变化。因此,在驱动程序中能够动态查找这些资源显得比较重要。对于诸如A/D、D/A、定时卡、I/O板卡这类设备,对照硬件手册编写一些简单的驱动程序并不困难。如果有DOS下驱动程序的C源码,移值应该更容易一些。

为了实现对PCI总线设备的控制和管理,必须访问PCI设备的配置空间。配置空间是一容量为256字节并具特定纪录结构的地址空间。该地址空间的结构如图1所示。NQX4.25pp sys/pci.h中对应的结构体定义。

    每个PCI设备具有唯一的厂商标识(vendor id)和设备标识(device id),这些信息由硬件手册提供或系统启动时可以看到。下面一段代码展示了于一个给定的PCI设备如何调用QNX相关的函数、侦测设备的存在以及系统分配的资源。其中,标识(index)用来支持和区分具有同样厂商标识和设备标识的几块同样的设备。Index从0开始,如果指定为1,将标识第二块同型号的设备。

本例中,YOUR_PCI_DEVICE_ID、YOUR_PCI)CENDOR)OD值是研华的PCL-1713采集卡,可以根据所使用的硬件填以合适的值。

以根据所使用的硬件填以合适的值。

#include<stdlib.h>

#include<stddef.h>

#include<stdio.h>

#include<fcntl.h>

#include<sys/mman.h>

#include<sys/osinfo.h>

#include<sys/pci.h>

#include<i86.h>

#define YOUR_PCI_DEVICE_ID0x1713 //根据具体设备提供对应的厂商标识及设备标识

#define YOUR_PCI_VENDOR_ID 0x13fe

int main(void){

unsigned busnum,devfuncnum; //总线号(PC仅有一条)及设备功能号

long address;

long io_base; //I/O基地址

unsigned char irq; //中断号

int pci_index=0 //标识为零标识第一块此种型号设备

if(_CA_PCI_Fin

d_Device(YOUR_PCI_DEVICE_ID,

YOUR_PCI_VENDOR_ID,pci_index,&busnum,&devfuncnum)!=PCI_SUCCESS){

printf("Can not find device");

exit(EXIT_FAILURE);

}

//侦测设备中断

if(_CA_PCI_Read_Config_Byte(busnum,devfuncnum,offsetof(struct_pci_config_regs,Interrupt_Line),

1,&irq)!=PCI_SUCCESS){

printf("Error reading interrupt");

exit(EXIT_FAILURE);

}

//侦测设备I/O基地址

if_CA_PCI_Read_Config_DWord(busnum,devfuncnum,offsetof(struct_pci_config_regs,Base_[2]),

1,(char *)&address)!=PCI_SUCCESS){

printf("Error reading address");

exit(EXIT_FAILURE);

}

io_base=PCI_IO_ADDR(adress);

printf("IO address:%x",io_base);

printf("IRQ:"%x",irq);

exit(EXIT_SUCCESS);

}

注意:各种设备的Base_Address_Regs[x],x可能不尽相同,需要查看具体的硬件手册决定。

2 进入硬件

一旦获得了系统分配给某个硬件设备的资源信息,就可以同这个设备进行通信了。至于如何做取决于需要访问的硬件资源。

2.1 I/O资源

一个进程试图进行I/O操作,必须具有正确的权限等级。你必须是超及用户(root),在编译的时候加上适当参数T1,以确何该进程拥有访问I/O口的权限。若忽视这一点,该运行进程将获得一个口的权限。若忽视这一点,该运行进行将获得一个SIGSEGV信号,表示一个非法的内存引用,并结束进程运行。

现在就可以利用inp()、inpd()、inpw(),outp(),inpd(),inpw(0等函数,对I/O基地址(I/O base address)加上寄存器偏移量(offset)处的I/O进行操作了。例如:

outpw(baseaddress+offset_reg,0xdeadbeef);

此外,对于一些设备,其I/O口是固定、众所皆知的,例如,一块VGA兼容的设备,并无上述所谓基地址。通过0x3c0、0x3d4、0x3d5,可以直接进入这些VGA的控制器。例如:

outp(0x3d4,0x11);

outp(0x3d5,inp(0x3d5)& ~0x80);

2.2 存储映射资源

某些设备,可以通过一般的内存操作进入寄存器,这就需要获得内存基地址(memory base address)。为了能够获进入此类设备的寄存器,需要将其映射到驱动程序虚拟地址空间。QNX下的技术资料/etc/readme/technotes/shmem.txt描述了如何创建一个共享内存对象,然后将这个内存对象的一段内存映射到PCI卡中,以便能够进入这个PCI设备。(接着上面的代码)可以利用mmap():

char *mem_base;

if(PCI_IS_MEM(address)){ //判断内存基地址

int fd;

char *page_ptr;

fd=shm_open("Physical",O_RDWR,0777);//创建一个共享内存对象

if(fd= =-1){

perror("Error shm_open:");

exit(EXIT_FAILURE);

}

page_ptr=mmap(0,4096,PROT_READ|PROT_WRITE,

MAP_SHARED,fd,PCI_MEM_ADDR(address)&~0xfff);//将内存基地址映射

if(page_ptr= =(char *)

perror("Error mmap:");

exit(EXIT_FAILURE);

}

mem_base=page_ptr+(PCI_MEM_ADDR(address)&0xfff);

close(fd);

}

printf("MEM" address:%lx",PCI_MEM_ADDR(address));

if(PCI_IS_MEM(address))

printf("mapped at : %lx",mem_base);

现在可以使用指针mem_base来进入设备寄存器了。例如:

mem_base[SHUTDOWN_REGISTER]=0x0xdeadbeef;

2.3 中断资源

超级用户(root)可以调用qnx_hint_attach()将一个中断处理程序绑定到一个设备上。中断处理程序作为一个远程调用(far),在进程空间(Localdescriptor Table set)运行。该函数最后一个参数设置数据段。寄存器SS为一个特别的内核栈,这不同于数据段(DS)。因此,需要在中断处理程序及其调用的函数中关断栈检查。大部分系统库中的函数在编译的时候都关断了栈检查,然而,对于需要使用大量内存的函数可能并非如此。后者即是那些在中断处理程序中不可调用的函数,如printf()、open()。通过QNX具体函数在线资源的Safety→Interrupt handler项进行判断该函数是否可以调用。如果函数中包括任何自动(auto)变量,强烈建议将中断函数放在自身文件中,然后利用参数-zu选项编译之。这样能够告知编译器,使得SS!=DS。

任何被中断处理程序修改的变量需要指定为volatile关键字。中断处理程序的返回值必须为0;或某个有效的代码号(proxy pid),以此来触发一个代码从而发送一则消息。

下面总结一个中断处理程序编写时的注意点:

①只能和自己的硬件对话(如,清除设备的中断状态位),千万不要对8259中断控制器编程!

②使中断处理程序尽可能的短小。如果有很多的工作需要做,必须触发一个代理,并且它唤醒一个进程完成这些工作,以保证其它进程及低优先级的中断正常运行,提高系统的实时响应能力。

③中断处理程序不能调用含有内核调用的例程。

④中断处理程序必须是一个远程(far)调用函数。

⑤中断处理程序必须在自己的模块中。

⑥无论程序中其它模块是如何编译的,包含中断处理程序的模块必须是利用-zu和-s选项编译。(利用cc-zu-Wc-s)这些选项能够保证SS!=DS,并且关断栈检查。当然,也可使用:

#pragma off(check_stack);

pid_t far handler_xxx(){

return(proxy_xxx);

}

#pragma on(check_stack);

在试图编写执行一个中断处理程序前,务必仔细阅读在线文档。现在,可以参照硬件手册自由地对您的设备寄存器进行操作了。

结语

在HT-7U极向场电源控制系统中,我们在QNX4.25下开发了多种设备的驱动程序。这些程序工作稳定、性能优异、工作量小且易于控制。此外,QSSL公司的新版本QNX6.x下开发驱动更为方便,其原理同QNX4.25相似或者是对应的。

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭