一 创建管道函数:
#include <sys/types.h> #include <sys/stat.h> int mkfifo(const char * pathname, mode_t mode) |
函数说明 参数pathname 指向欲打开的文件路径字符串。下列是参数flags 所能使用的旗标:
S_IRWXU00700 权限,代表该文件所有者具有可读、可写及可执行的权限。
S_IRUSR 或S_IREAD,00400权限,代表该文件所有者具有可读取的权限。
S_IWUSR 或S_IWRITE,00200 权限,代表该文件所有者具有可写入的权限。
S_IXUSR 或S_IEXEC,00100 权限,代表该文件所有者具有可执行的权限。
S_IRWXG 00070权限,代表该文件用户组具有可读、可写及可执行的权限。
S_IRGRP 00040 权限,代表该文件用户组具有可读的权限。
S_IWGRP 00020权限,代表该文件用户组具有可写入的权限。
S_IXGRP 00010 权限,代表该文件用户组具有可执行的权限。
S_IRWXO 00007权限,代表其他用户具有可读、可写及可执行的权限。
S_IROTH 00004 权限,代表其他用户具有可读的权限
S_IWOTH 00002权限,代表其他用户具有可写入的权限。
S_IXOTH 00001 权限,代表其他用户具有可执行的权限。
返回值 若所有欲核查的权限都通过了检查则返回0 值,表示成功,只要有一个权限被禁止则返回-1。
错误代码 EEXIST 参数pathname 所指的文件已存在,却使用了O_CREAT和O_EXCL旗标。
EACCESS 参数pathname所指的文件不符合所要求测试的权限。
EROFS 欲测试写入权限的文件存在于只读文件系统内。
EFAULT 参数pathname指针超出可存取内存空间。
EINVAL 参数mode 不正确。
ENAMETOOLONG 参数pathname太长。
ENOTDIR 参数pathname不是目录。
ENOMEM 核心内存不足。
ELOOP 参数pathname有过多符号连接问题。
EIO I/O 存取错误。
pathname参数是要创建的FIFO文件的名称, mode是给FIFO文件设定的权限。FIFO文件与其它文件一样,可以用remove和unlink进行删除。如果mkfifo的第一参数是一个已经存在的路径名时,会返回EEXIST错误,所以一般典型的调用代码首先会检查是否返回该错误,如果确实返回该错误,那么只要调用打开FIFO函数就可以了。一般文件的I/O函数都可以用于FIFO,如close、read、
write等。管道仅需要创建而不需要打开,因为使用它们的进程通过继承获得了管道的文件描述符。但命名管道则需要打开,因为使用它们的进程可以没有任何关系。对命名管道的打开通常使用文件打开
二 FIFO打开操作注意
1. 打开读的FIFO
如果已经存在打开写的FIFO,则打开成功立刻返回.
不存在写的FIFO被打开, open(“FIFO”, O_RDONLY)则阻塞直到有数据写入
open(“FIFO”, O_RDONLY|O_NONBLOCK)则立刻成功返回
注:如果FIFO中有数据,则设置了阻塞标志的读操作不会因为FIFO中的字节数小于请求读的字节数而阻塞,此时,读操作会返回FIFO中现有的数据量。
2. 打开写的FIFO
如果已经存在打开读的FIFO,则打开成功立刻返回.
不存在读的FIFO被打开,open(“FIFO”, O_WDONLY)则阻塞直到有为读而打开的FIFO
open(“FIFO”, O_WDONLY|O_NONBLOCK)则返回错误ENXIO.
对于设置了阻塞标志的写操作:
对于没有设置阻塞标志的写操作:
Write pipe demo:
#include <sys/types.h> #include <sys/stat.h> #include <errno.h> #include <fcntl.h> #include <stdio.h> #include <memory.h> #include <string.h> #define FIFO_TEST "/tmp/fifo_test"
void main(int argc,char** argv) { int fd; char write_buf[1024*5] = "first"; int ret = 0;
mkfifo(FIFO_TEST, 0x777); fd = open( FIFO_TEST, O_WRONLY ); if(fd == -1 ) { if(errno==ENXIO) { printf("open error; no reading process/n"); goto error_end; } } ret = write( fd, write_buf, strlen(write_buf)); if( -1 == ret ) { if( EAGAIN == errno ) {
printf("write to fifo error; try later/n"); }
goto error_end; } else { printf("real write num is %d/n", ret); }
for( ret = 0; ret < sizeof(write_buf); ret++) write_buf[ret] = ret%10+'0'; ret = write( fd, write_buf, sizeof(write_buf) ); // more than 4096 bytes writing everytime is test for the non-atomic write if( ret == -1) { if( EAGAIN == errno ) { printf("try later/n"); } } error_end: return;
} |
|
Read pipe demo:
#include <sys/types.h> #include <sys/stat.h> #include <errno.h> #include <fcntl.h> #include <stdio.h> #include <memory.h>
#define FIFO_TEST "/tmp/fifo_test" #define FIFO_READ_SIZE 1024
void main(int argc,char** argv) { char read_buf[FIFO_READ_SIZE] ={0} ; int fd; int r_byte; int ret;
mkfifo( FIFO_TEST, 0x777); fd=open(FIFO_TEST,O_RDONLY|O_NONBLOCK, 0); if( -1 == fd ) { printf("open for read error/n"); unlink(FIFO_TEST); return; //exit(0); } while(1) {
memset(read_buf, 0, sizeof(read_buf) ); ret = read( fd, read_buf, sizeof(read_buf) ); if(-1 == ret ) { if(errno==EAGAIN) { printf("test situation:FIFO that opened as WDONLY do reading operation first /n"); }
break;
} read_buf[sizeof(read_buf)-1] = 0; printf("real read bytes %d string=%s/n", ret, read_buf); sleep(1); } unlink(FIFO_TEST); } |