Linux 文件锁 fcntl 函数详解

函数原型

#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 对文件描述符可以执行的操作也非常多,用的最多的是文件记录锁,也就是 F_SETLK 命令,此命令搭配 flock 结构体,对文件进行加解锁操作,例如执行加锁操作,如果不解锁,本进程或者其他进程再次使用 F_SETLK 命令访问同一文件则会告知目前此文件已经上锁,加锁进程退出(正常、异常)后会自行解锁,使用此特性可以实现避免程序多次运行、锁定文件防止其他进行访问等操作。

函数功能
复制一个现有的描述符(cmd = F_DUPFD).
获得/设置文件描述符标记(cmd = F_GETFD 或 F_SETFD).
获得/设置文件状态标记(cmd = F_GETFL 或 F_SETFL).
获得/设置异步I/O所有权(cmd = F_GETOWN 或 F_SETOWN).
获得/设置记录锁(cmd = F_GETLK , F_SETLK 或 F_SETLKW).

函数参数
fd 文件描述符

cmd 操作命令

lock 类型为结构体flock,设置记录锁的具体状态。
struct flcok 

    short int l_type;   //锁定的状态

    /* 以下的三个参数用于分段对文件加锁,若对整个文件加锁,则:l_whence = SEEK_SET,              l_start = 0, l_len = 0 */
    short int l_whence; //决定l_start位置
    off_t l_start;      //锁定区域的开头位置 
    off_t l_len;        //锁定区域的大小

    pid_t l_pid;        //锁定动作的进程
};

cmd 操作命令详解

cmd = F_DUPFD
F_DUPFD 返回值为一个如下描述的(文件)描述符:

  1. 最小的大于或等于arg的一个可用的描述符
  2. 与原始操作符一样的某对象的引用
  3. 如果对象是文件(file)的话,则返回一个新的描述符,这个描述符与arg共享相同的偏移量(offset)
  4. 相同的访问模式(读,写或读/写)
  5. 相同的文件状态标志(如:两个文件描述符共享相同的状态标志)
  6. 与新的文件描述符结合在一起的close-on-exec标志被设置成交叉式访问execve(2)的系统调用

实际上调用
dup(oldfd);
等效于:
fcntl(oldfd, F_DUPFD, 0);
而调用
dup2(oldfd, newfd);
等效于
close(oldfd); fcntl(oldfd, F_DUPFD, newfd);
 

cmd = F_GETFD 或 F_SETFD
F_GETFD 取得与文件描述符 fd 联合的 close-on-exec 标志,类似 FD_CLOEXEC。
如果返回值和 FD_CLOEXEC 进行与运算结果是 0 的话,文件保持交叉式访问 exec(),否则如果通过exec运行的话,文件将被关闭(arg 被忽略)。
F_SETFD 设置close-on-exec标志,该标志以参数 arg 的 FD_CLOEXEC 位决定,应当了解很多现存的涉及文件描述符标志的程序并不使用常数 FD_CLOEXEC,而是将此标志设置为0(系统默认,在exec时不关闭)或1(在exec时关闭)。
在修改文件描述符标志或文件状态标志时必须谨慎,先要取得现在的标志值,然后按照希望修改它,最后设置新标志值。不能只是执行F_SETFD或F_SETFL命令,这样会关闭以前设置的标志位。


cmd = F_GETFL 或 F_SETFL
F_GETFL 取得 fd 的文件状态标志,如同下面的描述一样(arg 被忽略),在说明 open 函数时,已说明了文件状态标志。不幸的是,三个存取方式标志 (O_RDONLY , O_WRONLY , 以及 O_RDWR) 并不各占1位,这三种标志的值各是 0、1、2,由于历史原因,这三种值互斥,即一个文件只能有这三种值之一,因此首先必须用屏蔽字O_ACCMODE相与取得存取方式位,然后将结果与这三种值相比较。

F_SETFL 设置给 arg 描述符状态标志,可以更改的几个标志是:O_APPEND、O_NONBLOCK、O_SYNC、O_ASYNC。而fcntl的文件状态标志总共有7个:O_RDONLY , O_WRONLY , O_RDWR , O_APPEND , O_NONBLOCK , O_SYNC和O_ASYNC。
可更改的几个标志如下面的描述:


cmd = F_GETOWN 或 F_SETOWN
F_GETOWN 取得当前正在接收 SIGIO 或者 SIGURG 信号的进程id 或进程组id,进程组id返回的是负值(arg被忽略)
F_SETOWN 设置将接收 SIGIO 和 SIGURG 信号的进程id或进程组id,进程组id 通过提供负值的 arg 来说明(arg 绝对值的一个进程组ID),否则 arg 将被认为是进程id。

cmd = F_GETLK 或 F_SETLK 或 F_SETLKW
获得/设置记录锁的功能,成功则返回0,若有错误则返回-1,错误原因存于errno。

F_GETLK 通过第三个参数 arg(一个指向 flock 的结构体)取得第一个阻塞 lock description 指向的锁。取得的信息将覆盖传到fcntl()的flock结构的信息。如果没有发现能够阻止本次锁 (flock) 生成的锁,这个结构将不被改变,除非锁的类型被设置成 F_UNLCK

F_SETLK 按照指向结构体 flock 的指针的第三个参数 arg 所描述的锁的信息设置或者清除一个文件的 segment 锁。F_SETLK 被用来实现共享(读) 锁 (F_RDLCK) 或独占(写) 锁 (F_WRLCK),同样可以去掉这两种锁 (F_UNLCK)。如果共享锁或独占锁不能被设置,fcntl() 将立即返回 EAGAIN。

F_SETLKW 除了共享锁或独占锁被其他的锁阻塞这种情况外,这个命令和 F_SETLK 是一样的。如果共享锁或独占锁被其他的锁阻塞,进程将等待直到这个请求能够完成。当 fcntl() 正在等待文件的某个区域的时候捕捉到一个信号,如果这个信号没有被指定 SA_RESTART, fcntl 将被中断

当一个共享锁被 set 到一个文件的某段的时候,其他的进程可以 set 共享锁到这个段或这个段的一部分。共享锁阻止任何其他进程 set 独占锁到这段保护区域的任何部分。如果文件描述符没有以读的访问方式打开的话,共享锁的设置请求会失败。

独占锁阻止任何其他的进程在这段保护区域任何位置设置共享锁或独占锁。如果文件描述符不是以写的访问方式打开的话,独占锁的请求会失败。

l_type 有三种状态:


l_whence 也有三种方式:


fcntl 文件锁有两种类型:建议性锁和强制性锁

建议性锁是这样规定的:每个使用上锁文件的进程都要检查是否有锁存在,当然还得尊重已有的锁。内核和系统总体上都坚持不使用建议性锁,它们依靠程序员遵守这个规定。
强制性锁是由内核执行的:当文件被上锁来进行写入操作时,在锁定该文件的进程释放该锁之前,内核会阻止任何对该文件的读或写访问,每次读或写访问都得检查锁是否存在。


系统默认 fcntl 都是建议性锁,强制性锁是非 POSIX 标准的。如果要使用强制性锁,要使整个系统可以使用强制性锁,那么得需要重新挂载文件系统,mount 使用参数 -0 mand 打开强制性锁,或者关闭已加锁文件的组执行权限并且打开该文件的 set-GID 权限位。
建议性锁只在 cooperating processes 之间才有用。对 cooperating process 的理解是最重要的,它指的是会影响其它进程的进程或被别的进程所影响的进程,举两个例子:

我们可以同时在两个窗口中运行同一个命令,对同一个文件进行操作,那么这两个进程就是 cooperating processes。
cat file | sort,那么 cat 和 sort 产生的进程就是使用了 pipe 的 cooperating processes。


使用 fcntl 文件锁进行 I/O 操作必须小心:进程在开始任何 I/O 操作前如何去处理锁,在对文件解锁前如何完成所有的操作,是必须考虑的。如果在设置锁之前打开文件,或者读取该锁之后关闭文件,另一个进程就可能在上锁/解锁操作和打开/关闭操作之间的几分之一秒内访问该文件。当一个进程对文件加锁后,无论它是否释放所加的锁,只要文件关闭,内核都会自动释放加在文件上的建议性锁(这也是建议性锁和强制性锁的最大区别),所以不要想设置建议性锁来达到永久不让别的进程访问文件的目的(强制性锁才可以);强制性锁则对所有进程起作用。

fcntl 使用三个参数 F_SETLK/F_SETLKW, F_UNLCK和F_GETLK 来分别要求、释放、测试 record locks。record locks 是对文件一部分而不是整个文件的锁,这种细致的控制使得进程更好地协作以共享文件资源。fcntl 能够用于读取锁和写入锁,read lock 也叫 shared lock(共享锁), 因为多个 cooperating process 能够在文件的同一部分建立读取锁;write lock 被称为 exclusive lock(排斥锁),因为任何时刻只能有一个 cooperating process 在文件的某部分上建立写入锁。如果 cooperating processes 对文件进行操作,那么它们可以同时对文件加 read lock,在一个 cooperating process 加 write lock 之前,必须释放别的 cooperating process 加在该文件的 read lock 和 wrtie lock,也就是说,对于文件只能有一个 write lock存在,read lock 和 wrtie lock 不能共存。


函数返回值
fcntl() 的返回值与命令有关,如果出错,所有命令都返回 -1,错误码保存在 error 中;

如果成功则返回某个其他值。下列三个命令有特定返回值:F_DUPFD、F_GETFD、F_GETFL 以及 F_GETOWN;


除此之外的命令成功返回 0,失败返回 -1。


 

例子:

使用F_GETFL获取fd的文件状态标志。

#include<fcntl.h>
#include<unistd.h>
#include<iostream>
#include<errno.h>
using namespace std;

int main(int argc,char* argv[])
{
  int fd, var;

  if (argc!=2)
  {
      perror("--");
      cout<<"请输入参数,即文件名!"<<endl;
  }

  if((var=fcntl(atoi(argv[1]), F_GETFL, 0))<0)
  {
     strerror(errno);
     cout<<"fcntl file error."<<endl;
  }

  switch(var & O_ACCMODE)
  {
   case O_RDONLY : cout<<"Read only.."<<endl;
                   break;
   case O_WRONLY : cout<<"Write only.."<<endl;
                   break;
   case O_RDWR   : cout<<"Read wirte.."<<endl;
                   break;
   default  : break;
  }

 if (val & O_APPEND)
    cout<<",append"<<endl;

 if (val & O_NONBLOCK)
    cout<<",noblocking"<<endl;

 cout<<"exit 0"<<endl;

 exit(0);
}


原文链接:

Linux 文件锁 fcntl 函数详解_f_setlk-CSDN博客

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值