NOTE: 以下内容摘自《UNIX网络编程》(卷2),做下记录,方便以后查阅;
#include<fcntl.h>
int fcntl(int fd, int cmd, ... /* strct flock *arg */);
/* 若成功取决于cmd,若出错则返回-1; */
struct flock
{
short l_type; /* F_RDLCK, F_WRLCK, F_UNLCK */
short l_whence; /* SEEK_SET, SEEK_CUR, SEEK_END */
off_t l_start; /* relative starting offset in bytes */
off_t l_len; /* #bytes; 0 meas until end-of-file */
pid_t l_pid; /* PID returned by F_GETLK */
...; /* other members */
}
cmd参数的取值如下:
1. F_SETLK: 获取(l_type成员为F_RDLCK/F_WRLCK)或释放(l_type成员为F_UNLCK)由arg指向flock结构所描述的锁。如果无法将该锁授予调用进程,则函数立即返回一个EACCESS/EAGAIN的错误;
2. F_SETLKW: F_SETLK的阻塞版本。(W -> wait)
3. F_GETLK: 检查由arg指向的锁以确认是否有某个已存在的锁会妨碍将新锁授予调用进程。如果当前没有这样的锁存在,由arg指向的flock结构的l_type成员就被设置为F_UNLCK。
记录锁的使用
#include <fcntl.h>
void my_lock(int fd)
{
struct flock lock;
lock.l_type = F_WRLCK;
lock.l_whence = SEEK_SET;
lock.l_start = 0;
lock.l_len = 0;
fcntl(fd, F_SETLKW, &lock);
}
void my_unlock(int fd)
{
struct flock lock;
lock.l_type = F_UNLCK;
lock.l_whence = SEEK_SET;
lock.l_start = 0;
lock.l_len = 0;
fcntl(fd, F_SETLK, &lock);
}
劝告性上锁(advisory locking)
Posix记录上锁称为劝告性上锁。其含义是内核维护着已由各个进程上没的所有文件的正确信息,但是它不能防止一个进程写已由另一个进程读锁定的某个文件,同样也不能防止读一个已写锁定的某个文件。一个进程能够无视一个劝告性锁写一个读锁定文件,或者读一上写锁定文件,前提是该进程有读写该文件的足够权限。(那这个锁有什么用?)
强制性上锁 (mandatory locking)
使用强制性锁后,内核检查每一个read/write请求,以验证其操作不会干扰由某个进程持有的锁。
为对某个特定文件进行强制性上没必须满足:
1. 组成员执行权限必须关闭;
2. SGID权限必须打开;
fcntl记录上锁提供了对一个文件 的劝告性或强制性上锁功能,而我们是通过该文件打开着的描述符来访问它的。这些锁用于不同进程间的上锁,而不是同一进程内不同线程间的上锁。另外,使用fcntl记录上锁时,等待着的读岀者优先还是写入者优先是没有保证的。
NOTE: 锁不能通过fork由子进程继承;