V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
Distributions
Ubuntu
Fedora
CentOS
中文资源站
网易开源镜像站
davinci
V2EX  ›  Linux

自制 shell,进程前后台切换问题。

  •  
  •   davinci · 2017-04-24 17:57:33 +08:00 · 2358 次点击
    这是一个创建于 2553 天前的主题,其中的信息可能已经有所发展或是发生改变。

    shell 进程fork子进程后,位于前台的 shell 进程setpgid使子进程的process group id=子进程的pid后,

    tcsetpgrp使子进程成为前台进程。

    可是当子进程 execv 结束后,成为前台进程的是系统的 bash 进程而不是我的 shell 进程。

    我想要的效果是当位于前台的子进程终止后,之前转入后台的 shell 进程再度成为前台进程,而不是系统的 bash 成为前台进程。如何解决?部分代码如下

    
        pid_t pid = fork();
        if (pid < 0) return -1;
        if (pid == 0) {
          execv(argv[0], argv);
          exit(-1);
        }
        else {
          setpgid(pid, pid);
          tcsetpgrp(0, pid);
          waitpid(pid, NULL, 0);
          //。。。。做其他的事后面的代码略
        }
    
    
    7 条回复    2017-04-24 18:47:50 +08:00
    xiaq
        1
    xiaq  
       2017-04-24 18:22:00 +08:00 via iPad
    waitpid 完之后 tcsetpgrp 把自己弄到前台
    davinci
        2
    davinci  
    OP
       2017-04-24 18:24:57 +08:00
    @xiaq 试验过了这样不行。因为 tcsetpgrp 必须由属于前台进程组的进程执行才能生效, waitpid 后,前台进程是 bash 而不是 shell 。
    xiaq
        3
    xiaq  
       2017-04-24 18:26:37 +08:00 via iPad   ❤️ 1
    哦……你得先 ignore SIGTTOU
    xiaq
        4
    xiaq  
       2017-04-24 18:27:51 +08:00 via iPad
    打个广告: https://github.com/elves/elvish/blob/master/eval/eval.go#L168

    这个行为在 tcsetpgrp 的 man page 里是没有写的,很坑
    davinci
        5
    davinci  
    OP
       2017-04-24 18:42:54 +08:00 via iPhone
    @xiaq 忽略后前台进程依然是 bash 还是没办法在 shell 中输入新命令
    davinci
        6
    davinci  
    OP
       2017-04-24 18:45:18 +08:00 via iPhone
    @xiaq 我漏了一步我再试试
    davinci
        7
    davinci  
    OP
       2017-04-24 18:47:50 +08:00 via iPhone
    @xiaq 可以了 谢谢
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   我们的愿景   ·   实用小工具   ·   2535 人在线   最高记录 6543   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 34ms · UTC 15:47 · PVG 23:47 · LAX 08:47 · JFK 11:47
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.