UNIX环境高级编程APUE练习3.2-不用fcntl实现dup2的功能
1 题面
编写与dup2功能相同的函数,要求不调用fcntl函数,并且要有正确的出错处理。
2 基本思路
不能用fcntl,能够返回一个文件描述符的只有open和dup。而open会创建一个新的文件表项,返回的fd指向新的文件表项,与dup2的表现不符。dup基本能满足要求,但是返回的是最小的可用fd,需要进一步操作满足要求。另外需要自己添加错误处理,以及处理oldfd与newfd相等的情况等。具体地,
- 当dup返回出错时,直接返回出错
- 当dup返回值等于newfd时,直接返回
- 当dup返回值小于newfd时,记录返回值,循环调用dup直到返回值等于newfd。关闭前面记录的所有fd,返回newfd
- 当dup返回值大于newfd时,关闭返回值的fd。如果oldfd等于newfd,直接返回newfd;如果不相等,关掉newfd,然后再dup(因为不是原子的,返回值需要再判断)
3 出错处理
- oldfd的出错处理可以直接交给dup
- newfd的出错处理,需要判断是否超出文件描述符范围(RLIMIT_NOFILE in getrlimit)
- 对于dup返回EMFILE的情况,newfd如果没超过进程可打开的最大文件数,则不影响
- 另外还有一个判断顺序问题,是先判断参数是否合法还是oldfd==newfd, 这个可以根据dup2函数实测来确定
4 测试用例
进程打开的文件数没满的情况下
- 都超出范围,相同(MAX+1,MAX+1)
- 未打开描述符,相同 (100, 100)
- newfd超出范围 (1, MAX+1)
- newfd正好没超出 (1, MAX)
- oldfd和newfd相同 (2, 2)
进程打开的文件数满的情况下
- newfd正好超出范围 (1, MAX+1)
- newfd正好没超出 (1, MAX)
- oldfd和newfd相同 (2, 2)
5 开始撸码实测
5.1 先验证dup2的判断顺序问题
- 测试源码
#include <stdio.h>
#include <unistd.h>
#include <sys/resource.h>
int main()
{
int r;
struct rlimit old_rlim={0};
getrlimit(RLIMIT_NOFILE, &old_rlim);
printf("NOFILE limits: soft=%lld; hard=%lld\n",
(long long) old_rlim.rlim_cur, (long long) old_rlim.rlim_max);
r = dup2(10000, 10000);
if(r == -1) {
perror("dup2(10000, 10000) fail: ");
}
else {
printf("dup2(10000, 10000) success return %d\n", r);
}
r = dup2(100, 100);
if(r == -1) {
perror("dup2(100, 100) fail: ");
}
else {
printf("dup2(100, 100) success return %d\n", r);
}
r = dup2(1, 10000);
if(r == -1) {
perror("dup2(1, 10000) fail: ");
}
else {
printf("dup2(1, 10000) success return %d\n", r);
}
r = dup2(1, 100);
if(r == -1) {
perror("dup2(1, 100) fail: ");
}
else {
printf("dup2(1, 100) success return %d\n", r);
}
return 0;
}
- MAC OSX下运行结果
^_^$ ./a.out
NOFILE limits: soft=7168; hard=9223372036854775807
dup2(10000, 10000) fail: : Bad file descriptor
dup2(100, 100) fail: : Bad file descriptor
dup2(1, 10000) fail: : Bad file descriptor
dup2(1, 100) success return 100
可见是参数出错判断是先于oldfd == newfd判断的
5.2 测试进程打开的最大文件数到上限时,dup2是否能成功
- 测试源码
#include <stdio.h>
#include <unistd.h>
#include <sys/resource.h>
int main()
{
int r;
int max_fd = 0;
struct rlimit old_rlim={0};
getrlimit(RLIMIT_NOFILE, &old_rlim);
printf("NOFILE limits: soft=%lld; hard=%lld\n",
(long long) old_rlim.rlim_cur, (long long) old_rlim.rlim_max);
while((r = dup(0))!= -1)
{
max_fd = r;
}
perror(NULL);
printf("max fd is %d\n", max_fd);
r = dup2(1, 10000);
if(r == -1) {
perror("dup2(1, 10000) fail: ");
}
else {
printf("dup2(1, 10000) success return %d\n", r);
}
r = dup2(1, 100);
if(r == -1) {
perror("dup2(1, 100) fail: ");
}
else {
printf("dup2(1, 100) success return %d\n", r);
}
return 0;
}
- MAC OSX下运行结果
^_^$ ./a.out
NOFILE limits: soft=7168; hard=9223372036854775807
Too many open files
max fd is 7167
dup2(1, 7168) fail: : Bad file descriptor
dup2(1, 7167) success return 7167
可见在进程打开文件数达到上限时,dup2替换已经打开的文件是可以的
5.3 实现dup2的功能
- 源码
#include <stdio.h>
#include <unistd.h>
#include <sys/resource.h>
#include <sys/errno.h>
/*dup实现dup2的功能*/
int dup2_(int oldfd, int newfd) {
int ret;
int stack[7168];
int count = 0;
struct rlimit old_rlim={0};
getrlimit(RLIMIT_NOFILE, &old_rlim);
if (newfd < 0 || newfd > old_rlim.rlim_cur - 1) {
errno = EBADF;
return -1;
}
while(1) {
ret = dup(oldfd);
if(ret == -1 && errno != EMFILE) {
break;
}
else if(ret == -1 && errno == EMFILE) {
if(oldfd == newfd) {
return newfd;
}
printf("close(newfd)\n");
close(newfd);
}
else {
if(oldfd == newfd) {
close(ret);
return newfd;
}
if(ret == newfd) {
break;
}
else if(ret < newfd) {
stack[count++] = ret;
}
else {
close(ret);
printf("close(newfd)\n");
close(newfd);
}
}
}
while(count) {
close(stack[--count]);
}
return ret;
}
int main()
{
int r, max_fd;
struct rlimit old_rlim={0};
getrlimit(RLIMIT_NOFILE, &old_rlim);
printf("NOFILE limits: soft=%lld; hard=%lld\n",
(long long) old_rlim.rlim_cur, (long long) old_rlim.rlim_max);
r = dup2_(7168, 7168);
if(r == -1) {
perror("dup2_(7168, 7168) fail: ");
}
else {
printf("dup2_(7168, 7168) success return %d\n", r);
}
r = dup2_(100, 100);
if(r == -1) {
perror("dup2_(100, 100) fail: ");
}
else {
printf("dup2_(100, 100) success return %d\n", r);
}
r = dup2_(1, 7168);
if(r == -1) {
perror("dup2_(1, 7168) fail: ");
}
else {
printf("dup2_(1, 7168) success return %d\n", r);
}
r = dup2_(1, 7167);
if(r == -1) {
perror("dup2_(1, 7167) fail: ");
}
else {
printf("dup2_(1, 7167) success return %d\n", r);
}
r = dup2_(2, 2);
if(r == -1) {
perror("dup2_(2, 2) fail: ");
}
else {
printf("dup2_(2, 2) success return %d\n", r);
}
while((r = dup(0))!= -1)
{
max_fd = r;
}
perror(NULL);
printf("max fd is %d\n", max_fd);
r = dup2_(1, 7168);
if(r == -1) {
perror("dup2_(1, 7168) fail: ");
}
else {
printf("dup2_(1, 7168) success return %d\n", r);
}
r = dup2_(1, 7167);
if(r == -1) {
perror("dup2_(1, 7167) fail: ");
}
else {
printf("dup2_(1, 7167) success return %d\n", r);
}
r = dup2_(2, 2);
if(r == -1) {
perror("dup2_(2, 2) fail: ");
}
else {
printf("dup2_(2, 2) success return %d\n", r);
}
return 0;
}
- MAC OSX下的运行结果
NOFILE limits: soft=7168; hard=9223372036854775807
dup2_(7168, 7168) fail: : Bad file descriptor
dup2_(100, 100) fail: : Bad file descriptor
dup2_(1, 7168) fail: : Bad file descriptor
dup2_(1, 7167) success return 7167
dup2_(2, 2) success return 2
Too many open files
max fd is 7167
dup2_(1, 7168) fail: : Bad file descriptor
close(newfd)
dup2_(1, 7167) success return 7167
close(newfd)
dup2_(1, 100) success return 100
dup2_(2, 2) success return 2
结果都符合预期
UNIX环境高级编程APUE练习3.2-不用fcntl实现dup2的功能的更多相关文章
- (十三) [终篇] 一起学 Unix 环境高级编程 (APUE) 之 网络 IPC:套接字
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (三) 一起学 Unix 环境高级编程 (APUE) 之 文件和目录
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (四) 一起学 Unix 环境高级编程(APUE) 之 系统数据文件和信息
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (五) 一起学 Unix 环境高级编程 (APUE) 之 进程环境
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (六) 一起学 Unix 环境高级编程 (APUE) 之 进程控制
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (七) 一起学 Unix 环境高级编程(APUE) 之 进程关系 和 守护进程
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (八) 一起学 Unix 环境高级编程 (APUE) 之 信号
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
随机推荐
- 什么是 Shell 脚本?
Shell 既是一种命令语言,又是一种程序设计语言.Shell 是指一种应用程序,这个应用程序提供了一个界面,用户通过这个界面访问操作系统内核的服务.Windows Explorer 是一个典型的图形 ...
- [Kick Start] 2021 Round B
题目:Kick Start 2021 Round-B . Increasing Substring 输出字符串中每个字符的最长 Increasing Substring 的长度,非常简单的动态规划问题 ...
- NoSql非关系型数据库之MongoDB应用(三):MongoDB在项目中的初步应用
业精于勤,荒于嬉:行成于思,毁于随. 我们可以结合相关的IDE做一个简单的增删改查了,实现MongoDB在项目中的初步应用. 前提是安装了MongoDB服务和MongoDB可视化工具,没有安装的可以点 ...
- Linux:Linux操作防火墙命令
首先查看Linux的防火墙是否关闭 firewall-cmd Linux上新用的防火墙软件,跟iptables差不多的工具. firewall-cmd --state # 显示防火墙状态 system ...
- 1.3.2、通过Cookie匹配
server: port: 8080 spring: application: name: gateway cloud: gateway: routes: - id: guo-system4 uri: ...
- Linux守护进程列表/守护进程
在linux或者unix操作系统中在系统引导的时候会开启很多服务,这些服务就叫做守护进程.为了增加灵活性,root可以选择系统开启的模式,这些模式叫做运行级别,每一种运行级别以一定的方式配置系统. ...
- 3shell命令替换
Shell 命令替换是指将命令的输出结果赋值给某个变量.比如,将使用ls命令查看到的某个目录中的内容保存到某个变量中,这就需要使用命令替换. Shell 中有两种方式可以完成命令替换,一种是反引号` ...
- webpack(11)配置文件分离为开发配置、生成配置和基础配置
前言 上篇我们已经配置好了本地开发服务器,但是配置的相对比较凌乱,一个文件中有些是开发时用到的配置,有些是生成时用到的配置,有些是开发和生成都要用到的配置,所以我们这里把环境分为3个环境 webpac ...
- TCP/IP 5层协议簇/协议栈
TCP/IP 5层协议簇/协议栈 数据/PDU 应用层 PC.防火墙 数据段/段Fragment 传输层 防火墙 报文/包/IP包packet 网络层 路由器 帧Frame 数据链路层 交换机.网卡 ...
- [网络流24题]最长k可重区间集[题解]
最长 \(k\) 可重区间集 题目大意 给定实心直线 \(L\) 上 \(n\) 个开区间组成的集合 \(I\) ,和一个正整数 \(k\) ,试设计一个算法,从开区间集合 \(I\) 中选取开区间集 ...