1 题面

编写与dup2功能相同的函数,要求不调用fcntl函数,并且要有正确的出错处理。

2 基本思路

不能用fcntl,能够返回一个文件描述符的只有open和dup。而open会创建一个新的文件表项,返回的fd指向新的文件表项,与dup2的表现不符。dup基本能满足要求,但是返回的是最小的可用fd,需要进一步操作满足要求。另外需要自己添加错误处理,以及处理oldfd与newfd相等的情况等。具体地,

  1. 当dup返回出错时,直接返回出错
  2. 当dup返回值等于newfd时,直接返回
  3. 当dup返回值小于newfd时,记录返回值,循环调用dup直到返回值等于newfd。关闭前面记录的所有fd,返回newfd
  4. 当dup返回值大于newfd时,关闭返回值的fd。如果oldfd等于newfd,直接返回newfd;如果不相等,关掉newfd,然后再dup(因为不是原子的,返回值需要再判断)

3 出错处理

  1. oldfd的出错处理可以直接交给dup
  2. newfd的出错处理,需要判断是否超出文件描述符范围(RLIMIT_NOFILE in getrlimit)
  3. 对于dup返回EMFILE的情况,newfd如果没超过进程可打开的最大文件数,则不影响
  4. 另外还有一个判断顺序问题,是先判断参数是否合法还是oldfd==newfd, 这个可以根据dup2函数实测来确定

4 测试用例

进程打开的文件数没满的情况下

  1. 都超出范围,相同(MAX+1,MAX+1)
  2. 未打开描述符,相同 (100, 100)
  3. newfd超出范围 (1, MAX+1)
  4. newfd正好没超出 (1, MAX)
  5. oldfd和newfd相同 (2, 2)

进程打开的文件数满的情况下

  1. newfd正好超出范围 (1, MAX+1)
  2. newfd正好没超出 (1, MAX)
  3. 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的功能的更多相关文章

  1. (十三) [终篇] 一起学 Unix 环境高级编程 (APUE) 之 网络 IPC:套接字

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  2. (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  3. (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  4. (三) 一起学 Unix 环境高级编程 (APUE) 之 文件和目录

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  5. (四) 一起学 Unix 环境高级编程(APUE) 之 系统数据文件和信息

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  6. (五) 一起学 Unix 环境高级编程 (APUE) 之 进程环境

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  7. (六) 一起学 Unix 环境高级编程 (APUE) 之 进程控制

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  8. (七) 一起学 Unix 环境高级编程(APUE) 之 进程关系 和 守护进程

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  9. (八) 一起学 Unix 环境高级编程 (APUE) 之 信号

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

随机推荐

  1. [HNOI2006]公路修建问题题解

    题目 题目描述 OI island是一个非常漂亮的岛屿,自开发以来,到这儿来旅游的人很多.然而,由于该岛屿刚刚开发不久,所以那里的交通情况还是很糟糕.所以,OIER Association组织成立了, ...

  2. ES2021 新特性!

    大家好,我是前端队长Daotin,想要获取更多前端精彩内容,关注我(全网同名),解锁前端成长新姿势. 以下正文: 2021 年 6 月 22 日,第 121 届 Ecma 国际(Ecma Intern ...

  3. 在使用XStream时没有processAnnotations方法

    https://stackoverflow.com/questions/28770909/xstream-processannotations 我遇到这个问题的原因是xstream.jar的版本问题 ...

  4. 1.3.2、通过Cookie匹配

    server: port: 8080 spring: application: name: gateway cloud: gateway: routes: - id: guo-system4 uri: ...

  5. ptm经验总结

  6. Python----MongoDB数据库

    什么是MongoDB ? MongoDB 是由C++语言编写的,是一个基于分布式文件存储的开源数据库系统. 在高负载的情况下,添加更多的节点,可以保证服务器性能. MongoDB 旨在为WEB应用提供 ...

  7. 配置Mac 终端高亮

    mac下所有vim的配色方案的样式. 下面讲解如何设置这些好看的配色 首先:在终端输入 vim ~/.bash_profile 查看是否有上面提到的某些配色,所有配色均是以.vim结束的,果有的话,再 ...

  8. adb bat

    @REM 生成随机数@echo off@REM 设置延迟变量setlocal enabledelayedexpansionset min=9set max=21set /a mod=!max!-!mi ...

  9. python3安装pp过程

    并行计算的目的是将所有的核心都运行起来以提高代码的执行速度,在python中由于存在全局解释器锁(GIL)如果使用默认的python多线程进行并行计算可能会发现代码的执行速度并不会加快,甚至会比使用但 ...

  10. PYTHON IDLE同时运行两个PY,互不影响

    1.打开指定文件夹 2.右击a1.py,用IDLE打开(idle1) 3.右击a2.py,用IDLE打开(idle2) 4.则分别按F5运行,则两个互不影响. 5.再用IDLE1打开新程序,用IDLE ...