linux c c++开发重要函数应用积累0.docx_第1页
linux c c++开发重要函数应用积累0.docx_第2页
linux c c++开发重要函数应用积累0.docx_第3页
linux c c++开发重要函数应用积累0.docx_第4页
linux c c++开发重要函数应用积累0.docx_第5页
已阅读5页,还剩7页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

tcdrain() 函数标准C语言tcdrain(fd) 等待直到所有写入 fd 引用的对象的输出都被传输fcntl() 函数#include 定义函数 int fcntl(int fd , int cmd); int fcntl(int fd,int cmd,long arg); int fcntl(int fd,int cmd,struct flock * lock); fcntl()用来操作文件描述符的一些特性。参数fd代表欲设置的文件描述词。参数cmd代表欲操作的指令。 有以下几种情况: F_DUPFD用来查找大于或等于参数arg的最小且仍未使用的文件描述词,并且复制参数fd的文件描述词。执行成功则返回新复制的文件描述词。请参考dup2()。F_GETFD取得close-on-exec旗标。若此旗标的FD_CLOEXEC位为0,代表在调用exec()相关函数时文件将不会关闭。 F_SETFD 设置close-on-exec 旗标。该旗标以参数arg 的FD_CLOEXEC位决定。 F_GETFL 取得文件描述词状态旗标,此旗标为open()的参数flags。 F_SETFL 设置文件描述词状态旗标,参数arg为新旗标,但只允许O_APPEND、O_NONBLOCK和O_ASYNC位的改变,其他位的改变将不受影响。 F_GETLK 取得文件锁定的状态。 F_SETLK 设置文件锁定的状态。此时flcok 结构的l_type 值必须是F_RDLCK、F_WRLCK或F_UNLCK。如果无法建立锁定,则返回-1,错误代码为EACCES 或EAGAIN。 F_SETLKW 与F_SETLK 作用相同,但是无法建立锁定时,此调用会一直等到锁定动作成功为止。若在等待锁定的过程中被信号中断时,会立即返回-1,错误代码为EINTR。参数lock指针为flock 结构指针,定义如下 struct flcok short int l_type; short int l_whence; off_t l_start; off_t l_len; pid_t l_pid; ; l_type 有三种状态: F_RDLCK 建立一个供读取用的锁定 F_WRLCK 建立一个供写入用的锁定 F_UNLCK 删除之前建立的锁定 l_whence 也有三种方式: SEEK_SET 以文件开头为锁定的起始位置。 SEEK_CUR 以目前文件读写位置为锁定的起始位置 SEEK_END 以文件结尾为锁定的起始位置。 返回值 成功则返回0,若有错误则返回-1,错误原因存于errno.ioctl()函数 一、ioctl是设备驱动程序中对设备的I/O通道进行管理的函数。所谓对I/O通道进行管理,就是对设备的一些特性进行控制,例如串口的传输波特率、马达的转速等等。它的调用形式如下: int ioctl(int fd, ind cmd, ); 其中fd就是用户程序打开设备时使用open函数返回的文件标示符,cmd就是用户程序对设备的控制命令,至于后面的省略号,那是一些补充参数,一般最多一个,有或没有是和cmd的意义相关的。 ioctl函数是文件结构中的一个属性分量,就是说如果你的驱动程序提供了对ioctl的支持,用户就可以在用户程序中使用ioctl函数控制设备的I/O通道。 二、 ioctl的必要性 如果不用ioctl的话,也可以实现对设备I/O通道的控制,但那就是蛮拧了。例如,我们可以在驱动程序中实现write的时候检查一下是否有特殊约定的数据流通过,如果有的话,那么后面就跟着控制命令(一般在socket编程中常常这样做)。但是如果这样做的话,会导致代码分工不明,程序结构混乱,程序员自己也会头昏眼花的。 所以,我们就使用ioctl来实现控制的功能。要记住,用户程序所作的只是通过命令码告诉驱动程序它想做什么,至于怎么解释这些命令和怎么实现这些命令,这都是驱动程序要做的事情。 三、 ioctl如何实现 这是一个很麻烦的问题,我是能省则省。要说清楚它,没有四五千字是不行的,所以我这里是不可能把它说得非常清楚了,不过如果有读者对用户程序怎么和驱动程序联系起来感兴趣的话,可以看我前一阵子写的write的奥秘。读者只要把write换成ioctl,就知道用户程序的ioctl是怎么和驱动程序中的ioctl实现联系在一起的了。 我这里说一个大概思路,因为我觉得Linux设备驱动程序这本书已经说的非常清楚了,但是得花一些时间来看。 在驱动程序中实现的ioctl函数体内,实际上是有一个switchcase结构,每一个case对应一个命令码,做出一些相应的操作。怎么实现这些操作,这是每一个程序员自己的事情,因为设备都是特定的,这里也没法说。关键在于怎么样组织命令码,因为在ioctl中命令码是唯一联系用户程序命令和驱动程序支持的途径。命令码的组织是有一些讲究的,因为我们一定要做到命令和设备是一一对应的,这样才不会将正确的命令发给错误的设备,或者是把错误的命令发给正确的设备,或者是把错误的命令发给错误的设备。这些错误都会导致不可预料的事情发生,而当程序员发现了这些奇怪的事情的时候,再来调试程序查找错误,那将是非常困难的事情。 所以在Linux核心中是这样定义一个命令码的: _ | 设备类型 | 序列号 | 方向 |数据尺寸| |-|-|-|-| | 8 bit |8 bit |2 bit |814 bit| |-|-|-|-| 这样一来,一个命令就变成了一个整数形式的命令码。但是命令码非常的不直观,所以Linux Kernel中提供了一些宏,这些宏可根据便于理解的字符串生成命令码,或者是从命令码得到一些用户可以理解的字符串以标明这个命令对应的设备类型、设备序列号、数据传送方向和数据传输尺寸。 这些宏我就不在这里解释了,具体的形式请读者察看Linux核心源代码中的文件里给出了这些宏完整的定义。这里我只多说一个地方,那就是幻数。幻数是一个字母,数据长度也是8,所以就用一个特定的字母来标明设备类型,这和用一个数字是一样的,只是更加利于记忆和理解。就是这样,再没有更复杂的了。 更多的说了也没有,读者还是看一看源代码吧,推荐各位阅读Linux 设备驱动程序所带源代码中的short一例,因为它比较短小,功能比较简单,可以看明白ioctl的功能和细节。 四、cmd参数如何得出 这里确实要说一说,cmd参数在用户程序端由一些宏根据设备类型、序列号、传送方向、数据尺寸等生成,这个整数通过系统调用传递到内核中的驱动程序,再由驱动程序使用解码宏从这个整数中得到设备的类型、序列号、传送方向、数据尺寸等信息,然后通过 switchcase结构进行相应的操作。 要透彻理解,只能是通过阅读源代码,我这篇文章实际上只是一个引子。Cmd参数的组织还是比较复杂的,我认为要搞熟它还是得花不少时间的,但是这是值得的,驱动程序中最难的是对中断的理解。 五、小结 ioctl其实没有什么很难的东西需要理解,关键是理解cmd命令码是怎么在用户程序里生成并在驱动程序里解析的,程序员最主要的工作量在switchcase结构中,因为对设备的I/O控制都是通过这一部分的代码实现的。 flock()函数当用户想使用函数flock()来锁定一个文件时,可以使用两种不同的锁定类型:一个是共享锁定,另一个是互斥锁定。如果用户采用共享锁定,若干个进程可以对某个文件拥有一个共同的锁定;如果用户申请了一个互斥锁定,其他进程都无法对该文件进行锁定。因此,根据它们的原理,用户可以对执行读取操作的系统采用一个共享锁定,而对执行写入操作的系统采用一个互斥锁定。(摘自Linux编程宝典) 那么我就有疑问了: 1.既然读取操作的时候是不会对文件内容进行改变的,各个进程是不会相互冲突的,那么为什么要锁定?锁定的意义何在? 2.读取操作的时候使用共享锁定,也就是说,其它进程也可以锁定,那么是不是进行写操作的进程也可以锁定也可以写入呢?共享锁定没有什么作用? 1. 若干进程在读,他们不会互相影响,但是,如果有进程写则会影响它们,所以这个共享锁是防止有进程对文件写 2. 第二个问题和第一个问题关联,共享锁和独占锁互斥。只有当一个程序试图施加它自己的锁时,锁才会起作用;而没有尝试对文件上锁的程序仍然能够访问该文件。因此,锁只能在协同工作的程序间起作用。”(摘自GNU/Linux编程指南)3. 据程序试验结果:当我对一个文件进行互斥锁定以后,然后进行共享锁定时,锁定没有成功这是因为互斥锁定是排斥其他锁定的;当我对一个文件进行共享锁定以后,然后进行互斥锁定,锁定成功!难道互斥锁定不排斥之前的共享锁定?请解答。4. 我错在先关闭了文件描述符,当然共享锁定也就关闭了,随之互斥锁定必然也就能够成功,我错了,不好意思。但是,据我试验:如果一个进程对文件进行了互斥锁定,另外一个进程没有对文件进行尝试锁定操作而是直接对文件进行写操作,结果是可以写入!? 5. 那么也就是说,如果进程没有考虑到锁定的问题时,还是会破坏我锁定想保持不变的文件?那么这个锁定的意义是不是大打折扣了? 6. 呵呵!这样的锁叫建议锁和建议一样 可听可不听 好像还有强制锁 不过我没用过 呵呵 APUE上有介绍。如果你和别的程序混用无法控制他们的话,建议你采用QMAIL一样的原子操作法,不需要锁。UART的CTS与RTS 在RS232中本来CTS与RTS有明确的意义,但自从贺氏(HAYES) 推出了聪明猫(SmartModem)后就有点混淆了。在RS232中RTS与CTS是用来半双工模式下的方向切换;HAYES Modem中的RTS,CTS是用来进 行硬件流控的。通常UART的RTC、CTS的含义指后者,即用来做硬流控的。硬流控的RTS、CTS:RTS(Require To Send,发送请求)为输出信号,用于指示本设备准备好可接收;CTS(Clear To Send,发送清除)为输入信号,有效时停止发送。假定A、B两设备通信,A设备的RTS连接B设备的CTS;A设备的CTS连接B设备 的RTS。 前一路信号控制B设备的发送,后一路信号控制A设备的发送。对B设备的发送(A设备接收)来说,如果A设备接收缓冲快满的时发出RTS信号(意思 通知B设备停止发送),B设备通过CTS检测到该信号,停止发送;一段时间后A设备接收缓冲有了空余,发出RTS信号,指示B设备开始发送数据。A设备发(B设备接收) 类似。上述功能也能在数据流中插入Xoff(特殊字符)和Xon(另一个特殊字符)信号来实现。A设备一旦接收到B设备发送过来的Xoff,立刻停止发 送;反之,如接收到B设备发送过来的Xon,则恢复发送数据给B设备。同理,B设备也类似,从而实现收发双方的速度匹配。半双工的方向切换:RS232中使用DTR(Date Terminal Ready,数据终端准备)与DSR(Data Set Ready ,数据设备准备好)进行主流控,类似上述的RTS与CTS。对半双工的通信的DTE(Date Terminal Equipment,数据终端设备)与DCE(Data circuit Equipment )来说,默认的方向是DTE接收,DCE发送。如果DTE要发送数据,必须发出RTS信号,请求发送数据。DCE收到后如果 空闲则发出CTS回 应RTS信 号,表示响应请求,这样通信方向就变为DTE-TCE,同时RTS与CTS信号必须一直保持。从这里可以看出,CTS,TRS虽然也有点流控的意思(如CTS没有发出,DTE也不能发送数据),但主要是用来进行方向切换的。如果UART只有RX、TX两个信号,要流控的话只能是软流控;如果有RX,TX,CTS,RTS四个信号,则多半是支持硬流控的UART;如果有 RX,TX,CTS,RTS,DTR,DSR 六个信号的话,RS232标准的可能性比较大。顺便提一下:DCD( Data Carrier Detect, 数据载波检测):DCE向DTE指示,线路上检测到载波。RI(Ring Indicator,振铃指示):DCE向DTE指示,有呼叫接入。linux 基础复习(7)串口应用开发据通信的基本方式可分为并行通信与串行通信两种。 并行通信是指利用多条数据传输线将一个资料的各位同时传送。它的特点是传输速度快,适用于短距离通信,但要求传输速度较高的应用场合。 串行通信是指利用一条传输线将资料一位位地顺序传送。特点是通信线路简单,利用简单的线缆就可实现通信,降低成本,适用于远距离通信,但传输速度慢的应用场合。串口设置详解本节主要讲解设置串口的主要方法。如前所述,设置串口中最基本的包括波特率设置,校验位和停止位设置。串口的设置主要是设置struct termios结构体的各成员值,如下所示:includestruct termio unsigned short c_iflag; /* 输入模式标志 */ unsigned short c_oflag; /* 输出模式标志 */ unsigned short c_cflag; /* 控制模式标志*/ unsigned short c_lflag; /*本地模式标志 */ unsigned char c_line; /* line discipline */ unsigned char c_ccNCC; /* control characters */;在这个结构中最为重要的是c_cflag,通过对它的赋值,用户可以设置波特率、字符大小、数据位、停止位、奇偶校验位和硬件流控等。另外c_iflag 和c_cc 也是比较常用的标志。在此主要对这3 个成员进行详细说明。 c_cflag支持的常量名称CBAUD 波特率的位掩码B0 0波特率(放弃DTR)B1800 1800波特率B2400 2400波特率B4800 4800波特率B9600 9600波特率B19200 19200波特率B38400 38400波特率B57600 57600波特率B115200 115200波特率EXTA 外部时钟率EXTB 外部时钟率CSIZE 数据位的位掩码CS5 5个数据位CS6 6个数据位CS7 7个数据位CS8 8个数据位CSTOPB 2个停止位(不设则是1个停止位)CREAD 接收使能PARENB 校验位使能PARODD 使用奇校验而不使用偶校验HUPCL 最后关闭时挂线(放弃DTR)CLOCAL 本地连接(不改变端口所有者)LOBLK 块作业控制输出CNET_CTSRTS硬件流控制使能c_iflag支持的常量名称INPCK 奇偶校验使能IGNPAR 忽略奇偶校验错误PARMRK 奇偶校验错误掩码ISTRIP 除去奇偶校验位IXON 启动出口硬件流控IXOFF 启动入口软件流控IXANY 允许字符重新启动流控IGNBRK 忽略中断情况BRKINT 当发生中断时发送SIGINT信号INLCR 将NL映射到CRIGNCR 忽略CRICRNL 将CR映射到NLIUCLC 将高位情况映射到低位情况IMAXBEL 当输入太长时回复ECHOc_cc 支持的常量名称VINTR 中断控制,对应键为CTRL+CVQUIT 退出操作,对应键为CRTL+ZVERASE 删除操作,对应键为Backspace(BS)VKILL 删除行,对应键为CTRL+UVEOF 位于文件结尾,对应键为CTRL+DVEOL 位于行尾,对应键为Carriage return(CR)VEOL2 位于第二行尾,对应键为Line feed(LF)VMIN 指定了最少读取的字符数VTIME 指定了读取每个字符的等待时间串口控制函数Tcgetattr 取属性(termios结构)Tcsetattr 设置属性(termios结构)cfgetispeed 得到输入速度Cfgetospeed 得到输出速度Cfsetispeed 设置输入速度Cfsetospeed 设置输出速度Tcdrain 等待所有输出都被传输tcflow 挂起传输或接收tcflush 刷清未决输入和/或输出Tcsendbreak 送BREAK字符tcgetpgrp 得到前台进程组IDtcsetpgrp 设置前台进程组ID完整的串口配置模板,实用!把常用的选项在函数里面列出,可大大方便用户的调试使用int set_opt(int fd,int nSpeed, int nBits, char nEvent, int nStop) struct termios newtio,oldtio; /*保存测试现有串口参数设置,在这里如果串口号等出错,会有相关的出错信息*/ if ( tcgetattr( fd,&oldtio) != 0) perror(SetupSerial 1); return -1; bzero( &newtio, sizeof( newtio ) ); /*步骤一,设置字符大小*/ newtio.c_cflag |= CLOCAL | CREAD; newtio.c_cflag &= CSIZE; /*设置停止位*/ switch( nBits ) case 7: newtio.c_cflag |= CS7; break; case 8: newtio.c_cflag |= CS8; break; /*设置奇偶校验位*/ switch( nEvent ) case O: /奇数 newtio.c_cflag |= PARENB; newtio.c_cflag |= PARODD; newtio.c_iflag |= (INPCK | ISTRIP); break; case E: /偶数 newtio.c_iflag |= (INPCK | ISTRIP); newtio.c_cflag |= PARENB; newtio.c_cflag &= PARODD; break; case N: /无奇偶校验位 newtio.c_cflag &= PARENB; break; /*设置波特率*/ switch( nSpeed ) case 2400: cfsetispeed(&newtio, B2400); cfsetospeed(&newtio, B2400); break; case 4800: cfsetispeed(&newtio, B4800); cfsetospeed(&newtio, B4800); break; case 9600: cfsetispeed(&newtio, B9600); cfsetospeed(&newtio, B9600); break; case 115200: cfsetispeed(&newtio, B115200); cfsetospeed(&newtio, B115200); break; case 460800: cfsetispeed(&newtio, B460800); cfsetospeed(&newtio, B460800); break; default: cfsetispeed(&newtio, B9600); cfsetospeed(&newtio, B9600); break; /*设置停止位*/ if( nStop = 1 ) newtio.c_cflag &= CSTOPB; else if ( nStop = 2 ) newtio.c_cflag |= CSTOPB; /*设置等待时间和最小接收字符*/ newtio.c_ccVTIME = 0; newtio.c_ccVMIN = 0; /*处理未接收字符*/ tcflush(fd,TCIFLUSH); /*激活新配置*/ if(tcsetattr(fd,TCSANOW,&newtio)!=0) perror(com set error); return -1; printf(set done!n); return 0; 串口使用详解在配置完串口的相关属性后,就可对串口进行打开,读写操作了。其使用方式与文件操作一样,区别在于串口是一个终端设备。打开串口fd = open( /dev/ttyS0, O_RDWR|O_NOCTTY|O_NDELAY); Open函数中除普通参数外,另有两个参数O_NOCTTY和O_NDELAY。 O_NOCTTY: 通知linix系统,这个程序不会成为这个端口的控制终端。 O_NDELAY: 通知linux系统不关心DCD信号线所处的状态(端口的另一端是否激活或者停止)。然后,恢复串口的状态为阻塞状态,用于等待串口数据的读入。用fcntl函数: fcntl(fd, F_SETFL, 0); 接着,测试打开的文件描述府是否引用一个终端设备,以进一步确认串口是否正确打开。 isatty(STDIN_FILENO);串口的读写与普通文件一样,使用read,write函数。 read(fd,buff,8); write(fd,buff,8);实例#include stdio.h#include string.h#include sys/types.h#include errno.h#include sys/stat.h#include fcntl.h#include unistd.h#include termios.h#include stdlib.hint set_opt(int fd,int nSpeed, int nBits, char nEvent, int nStop) struct termios newtio,oldtio; if ( tcgetattr( fd,&oldtio) != 0) perror(SetupSerial 1); return -1; bzero( &newtio, sizeof( newtio ) ); newtio.c_cflag |= CLOCAL | CREAD; newtio.c_cflag &= CSIZE; switch( nBits ) case 7: newtio.c_cflag |= CS7; break; case 8: newtio.c_cflag |= CS8; break; switch( nEvent ) case O: newtio.c_cflag |= PARENB; newtio.c_cflag |= PARODD; newtio.c_iflag |= (INPCK | ISTRIP); break; case E: newtio.c_iflag |= (INPCK | ISTRIP); newtio.c_cflag |= PARENB; newtio.c_cflag &= PARODD; break; case N: newtio.c_cflag &= PARENB; break; switch( nSpeed ) case 2400: cfsetispeed(&newtio, B2400); cfsetospeed(&newtio, B2400); break; case 4800: cfsetispeed(&newtio, B4800); cfsetospeed(&newtio, B4800); break; case 9600: cfsetispeed(&newtio, B9600); cfsetospeed(&newtio, B9600); break; case 115200: cfsetispeed(&newtio, B115200); cfsetospeed(&newtio, B115200); break; default: cfsetispeed(&newtio, B9600); cfsetospeed(&newtio, B9600); break; if( nStop = 1 ) newtio.c_cflag &= CSTOPB; else if ( nStop = 2 ) newtio.c_cflag |= CSTOPB; newtio.c_ccVTIME = 0; newtio.c_ccVMIN = 0; tcflush(fd,TCIFLUSH); if(tcsetattr(fd,TCSANOW,&newtio)!=0) perror(

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论