fcntl系统调用可以用来对已打开的文件描述符进行各种控制操作以改变已打开文件的的各种属性

函数原型:
 
  1. #include<unistd.h>
  2. #include<fcntl.h>
  3. int fcntl(int fd, int cmd);
  4. int fcntl(int fd, int cmd, long arg);
  5. int fcntl(int fd, int cmd ,struct flock* lock);
#include<unistd.h>
#include<fcntl.h>
int fcntl(int fd, int cmd);
int fcntl(int fd, int cmd, long arg);
int fcntl(int fd, int cmd ,struct flock* lock);

fcntl函数功能依据cmd的值的不同而不同。参数对应功能如下:

(1)F_DUPFD

与dup函数功能一样,复制由fd指向的文件描述符,调用成功后返回新的文件描述符,与旧的文件描述符共同指向同一个文件。

(2)F_GETFD

读取文件描述符close-on-exec标志

(3)F_SETFD

将文件描述符close-on-exec标志设置为第三个参数arg的最后一位

(4)F_GETFL

获取文件打开方式的标志,标志值含义与open调用一致

(5)F_SETF

设置文件打开方式为arg指定方式

文件记录锁是fcntl函数的主要功能。

记录锁:实现只锁文件的某个部分,并且可以灵活的选择是阻塞方式还是立刻返回方式

当fcntl用于管理文件记录锁的操作时,第三个参数指向一个struct flock *lock的结构体

  1. struct flock
  2. {
  3. short_l_type;    /*锁的类型*/
  4. short_l_whence;  /*偏移量的起始位置:SEEK_SET,SEEK_CUR,SEEK_END*/
  5. off_t_l_start;     /*加锁的起始偏移*/
  6. off_t_l_len;    /*上锁字节*/
  7. pid_t_l_pid;   /*锁的属主进程ID */
  8. };
struct flock
{
short_l_type; /*锁的类型*/
short_l_whence; /*偏移量的起始位置:SEEK_SET,SEEK_CUR,SEEK_END*/
off_t_l_start; /*加锁的起始偏移*/
off_t_l_len; /*上锁字节*/
pid_t_l_pid; /*锁的属主进程ID */
};

short_l_type用来指定设置共享锁(F_RDLCK,读锁)还是互斥锁(F_WDLCK,写锁).

当short_l_type的值为F_UNLCK时,传入函数中将解锁。

每个进程可以在该字节区域上设置不同的读锁。

但给定的字节上只能设置一把写锁,并且写锁存在就不能再设其他任何锁,且该写锁只能被一个进程单独使用。

这是多个进程的情况。

单个进程时,文件的一个区域上只能有一把锁,若该区域已经存在一个锁,再在该区域设置锁时,新锁会覆盖掉旧的锁,无论是写锁还时读锁。

l_whence,l_start,l_len三个变量来确定给文件上锁的区域。

l_whence确定文件内部的位置指针从哪开始,l_star确定从l_whence开始的位置的偏移量,两个变量一起确定了文件内的位置指针先所指的位置,即开始上锁的位置,然后l_len的字节数就确定了上锁的区域。

特殊的,当l_len的值为0时,则表示锁的区域从起点开始直至最大的可能位置,就是从l_whence和l_start两个变量确定的开始位置开始上锁,将开始以后的所有区域都上锁。

为了锁整个文件,我们会把l_whence,l_start,l_len都设为0。

(6)F_SETLK

此时fcntl函数用来设置或释放锁。当short_l_type为F_RDLCK为读锁,F_WDLCK为写锁,F_UNLCK为解锁。

如果锁被其他进程占用,则返回-1;

这种情况设的锁遇到锁被其他进程占用时,会立刻停止进程。

(7)F_SETLKW

此时也是给文件上锁,不同于F_SETLK的是,该上锁是阻塞方式。当希望设置的锁因为其他锁而被阻止设置时,该命令会等待相冲突的锁被释放。

(8)F_GETLK

第3个参数lock指向一个希望设置的锁的属性结构,如果锁能被设置,该命令并不真的设置锁,而是只修改lock的l_type为F_UNLCK,然后返回该结构体。如果存在一个或多个锁与希望设置的锁相互冲突,则fcntl返回其中的一个锁的flock结构。

====================================================================================================================================================

linux下串口的阻塞和非阻塞操作

[cpp] 

有两个可以进行控制串口阻塞性(同时控制read和write):一个是在打开串口的时候,open函数是否带O_NDELAY;第二个是可以在打开串口之后通过fcntl()函数进行控制。

阻塞的定义:

对于read,block指当串口输入缓冲区没有数据的时候,read函数将会阻塞在这里,移植到串口输入缓冲区中有数据可读取,read读到了需要的字节数之后,返回值为读到的字节数;

对于write,block指当串口输出缓冲区满,或剩下的空间小于将要写入的字节数,则write将阻塞,一直到串口输出缓冲区中剩下的空间大于等于将要写入的字节数,执行写入操作,返回写入的字节数。

非阻塞的定义:

对于read,no block指当串口输入缓冲区没有数据的时候,read函数立即返回,返回值为0。

对于write,no block指当串口输出缓冲区满,或剩下的空间小于将要写入的字节数,则write将进行写操作,写入当前串口输出缓冲区剩下空间允许的字节数,然后返回写入的字节数。

  1. static int set_opt(int fd, int nSpeed, int nBits, char nEvent, int nStop)
  2. {
  3. struct termios newtio;
  4. struct termios oldtio;
  5. if(tcgetattr(fd,&oldtio) != 0)
  6. {
  7. perror("SetupSerial 1");
  8. return -1;
  9. }
  10. bzero(&newtio,sizeof(newtio));
  11. newtio.c_cflag |= CLOCAL |CREAD;
  12. newtio.c_cflag &= ~CSIZE;
  13. /***********数据位选择****************/
  14. switch(nBits)
  15. {
  16. case 7:
  17. newtio.c_cflag |= CS7;
  18. break;
  19. case 8:
  20. newtio.c_cflag |= CS8;
  21. break;
  22. }
  23. /***********校验位选择****************/
  24. switch(nEvent)
  25. {
  26. case 'O':
  27. newtio.c_cflag |= PARENB;
  28. newtio.c_cflag |= PARODD;
  29. newtio.c_iflag |= (INPCK | ISTRIP);
  30. break;
  31. case 'E':
  32. newtio.c_iflag |= (INPCK |ISTRIP);
  33. newtio.c_cflag |= PARENB;
  34. newtio.c_cflag &= ~PARODD;
  35. break;
  36. case 'N':
  37. newtio.c_cflag &= ~PARENB;
  38. break;
  39. }
  40. /***********波特率选择****************/
  41. switch(nSpeed)
  42. {
  43. case 2400:
  44. cfsetispeed(&newtio,B2400);
  45. cfsetospeed(&newtio,B2400);
  46. break;
  47. case 4800:
  48. cfsetispeed(&newtio,B4800);
  49. cfsetospeed(&newtio,B4800);
  50. break;
  51. case 9600:
  52. cfsetispeed(&newtio,B9600);
  53. cfsetospeed(&newtio,B9600);
  54. break;
  55. case 57600:
  56. cfsetispeed(&newtio,B57600);
  57. cfsetospeed(&newtio,B57600);
  58. break;
  59. case 115200:
  60. cfsetispeed(&newtio,B115200);
  61. cfsetospeed(&newtio,B115200);
  62. break;
  63. case 460800:
  64. cfsetispeed(&newtio,B460800);
  65. cfsetospeed(&newtio,B460800);
  66. break;
  67. default:
  68. cfsetispeed(&newtio,B9600);
  69. cfsetospeed(&newtio,B9600);
  70. break;
  71. }
  72. /***********停止位选择****************/
  73. if(nStop == 1){
  74. newtio.c_cflag &= ~CSTOPB;
  75. }
  76. else if(nStop ==2){
  77. newtio.c_cflag |= CSTOPB;
  78. }
  79. newtio.c_cc[VTIME] = 1;
  80. newtio.c_cc[VMIN] = FRAME_MAXSIZE;   //阻塞条件下有效
  81. tcflush(fd,TCIFLUSH);
  82. if((tcsetattr(fd,TCSANOW,&newtio)) != 0)
  83. {
  84. perror("com set error");
  85. return -1;
  86. }
  87. printf("set done!\n");
  88. return 0;
  89. }
  1. static int open_port(int fd,int comport)
  2. {
  3. /***********打开串口1****************/
  4. if(comport == 1)
  5. {
  6. fd = open("/dev/ttyAT1",O_RDWR|O_NOCTTY|O_NDELAY);
  7. if(fd == -1){
  8. perror("Can't Open Serial Port");
  9. return -1;
  10. }
  11. }
  12. /***********打开串口2****************/
  13. else if(comport == 2)
  14. {
  15. fd = open("/dev/ttyAT2",O_RDWR|O_NOCTTY|O_NDELAY);
  16. if(fd == -1){
  17. perror("Can't Open Serial Port");
  18. return -1;
  19. }
  20. }
  21. /***********打开串口3****************/
  22. else if(comport == 3)
  23. {
  24. fd = open("/dev/ttyAT3",O_RDWR|O_NOCTTY|O_NDELAY);
  25. if(fd == -1){
  26. perror("Can't Open Serial Port");
  27. return -1;
  28. }
  29. }
  30. if(comport == 1)
  31. {
  32. if(fcntl(fd,F_SETFL,FNDELAY) < 0)//非阻塞,覆盖前面open的属性
  33. {
  34. printf("fcntl failed\n");
  35. }
  36. else{
  37. printf("fcntl=%d\n",fcntl(fd,F_SETFL,FNDELAY));
  38. }
  39. }
  40. else
  41. {
  42. if(fcntl(fd,F_SETFL,0) < 0){   //阻塞,即使前面在open串口设备时设置的是非阻塞的,这里设为阻塞后,以此为准
  43. printf("fcntl failed\n");
  44. }
  45. else{
  46. printf("fcntl=%d\n",fcntl(fd,F_SETFL,0));
  47. }
  48. }
  49. if(isatty(STDIN_FILENO) == 0){
  50. printf("standard input is not a terminal device\n");
  51. }
  52. else{
  53. printf("isatty sucess!\n");
  54. }
  55. printf("fd-open=%d\n",fd);
  56. return fd;
  57. }

所以,linux的串口的阻塞性通过fcntl()函数进行设置即可。

  1. 阻塞:fcntl(fd,F_SETFL,0)
  1. 非阻塞:fcntl(fd,F_SETFL,FNDELAY

fcntl函数的用法总结的更多相关文章

  1. linxu fcntl 函数用法 【转】

    功能描述:根据文件描述词来操作文件的特性. 文件控制函数         fcntl -- file control 头文件: #include <fcntl.h>;          i ...

  2. Linux下 fcntl 函数用法说明

    功能描述:根据文件描述词来操作文件的特性. 文件控制函数         fcntl -- file control LIBRARY         Standard C Library (libc, ...

  3. fcntl函数用法详解

    功能描述:根据文件描述词来操作文件的特性. #include <unistd.h> #include <fcntl.h>  int fcntl(int fd, int cmd) ...

  4. fcntl函数用法——操纵文件描述符状态

    fcntl函数:操纵文件描述符,改变已经打开的文件的属性int fcntl(int fd, int cmd, ... //arg  );cmd选项:一.复制文件描述符:F_DUPFD二.更改设置文件描 ...

  5. fcntl函数用法——设置文件锁

    fcntl函数.锁定文件,设置文件锁.设置获取文件锁:F_GETLK .F_SETLK  .F_SETLKW文件锁结构,设置好用于fcntl函数的第三个参数.struct flock{    shor ...

  6. Linux系统编程(3)——文件与IO之fcntl函数

    linux文件I/O用:open.read.write.lseek以及close函数实现了文件的打开.读写等基本操作.fcntl函数可以根据文件描述词来操作文件. 用法: int fcntl(int ...

  7. Daemon函数的用法

    Daemon函数的用法 说明: 让一个程序后台运行. 原型: #include <unistd.h> int daemon(int nochdir, int noclose); #incl ...

  8. 有关日期的函数操作用法总结,to_date(),trunc(),add_months();

    相关知识链接: Oracle trunc()函数的用法 oracle add_months函数 Oracle日期格式转换,tochar(),todate() №2:取得当前日期是一个星期中的第几天,注 ...

  9. Oracle to_date()函数的用法

    Oracle to_date()函数的用法 to_date()是Oracle数据库函数的代表函数之一,下文对Oracle to_date()函数的几种用法作了详细的介绍说明,供您参考学习. 在Orac ...

随机推荐

  1. Business Cards UVALive - 4384(画图看图。。)

    只能由三种情况 都横着放  都竖着放  横和竖交错放 那就去判断好了... 具体看代码 #include <iostream> #include <cstdio> #inclu ...

  2. 学习Spring Boot:(四)应用日志

    前言 应用日志是一个系统非常重要的一部分,后来不管是开发还是线上,日志都起到至关重要的作用.这次使用的是 Logback 日志框架. 正文 Spring Boot在所有内部日志中使用Commons L ...

  3. Weblogic CVE-2018-3191远程代码命令执行漏洞复现

      0x00 简介 北京时间10月17日,Oracle官方发布的10月关键补充更新CPU(重要补丁更新)中修复了一个高危的WebLogic远程代码执行漏洞(CVE-2018-3191).该漏洞允许未经 ...

  4. Maven问题合集

    背景:总结maven使用过程中的相关问题,方便以后查询! 1 问题汇总 今天在导入github上下载的maven项目时候,出现了一堆错误,一时间不知道该如何下手. 根据错误提示,发现是一些依赖的jar ...

  5. CentOS 7.0源码包搭建LNMP方法分享(实际环境下)

    CentOS 7.0编译安装Nginx1.6.0+MySQL5.6.19+PHP5.5.14 一.配置防火墙,开启80端口.3306端口 CentOS 7.0默认使用的是firewall作为防火墙,这 ...

  6. 【题解】新型城市化 HAOI2017 网络流 二分图最大匹配 强连通分量

    Prelude 好,HAOI2017终于会做一道题了! 传送到洛谷:→_→ 传送到LOJ:←_← 本篇博客链接:(●'◡'●) Solution 首先要读懂题. 考场上我是这样想的QAQ. 我们把每个 ...

  7. PDF文本内容批量提取到Excel

    QQ:231469242,版权所有 sklearn实战-乳腺癌细胞数据挖掘 https://study.163.com/course/introduction.htm?courseId=1005269 ...

  8. Linux下U盘、SD卡挂载与卸载

    1.手动挂载/卸载U盘.SD卡 对于ARM Linux来说,第一次使用U盘或SD时,U盘这个文件目录是不能直接进入的,我们需要对其进行挂载,然后再接下来的使用中就可以直接进行使用了.通过再网上查资料, ...

  9. bzoj千题计划150:bzoj2738: 矩阵乘法

    http://www.lydsy.com/JudgeOnline/problem.php?id=2738 整体二分 二维树状数组累积 #include<cstdio> #include&l ...

  10. 在Emacs中画思维导图

    是的,你没有看错.其实,不只画思维导图,画结构图.流程图等,都可以.那怎么办呢?就是借助 Graphviz . Graphviz 可以方便地表达概念之间的联系,因此用它画思维导图是可行的,再加上它是个 ...