1

我正在研究一个伪终端库。代码以 C 代码实现,代码由基于 Web 的终端使用。只要我不使用 sudo 或登录,代码就可以工作。

这是我在 Mac 上运行服务器时遇到的错误:

sh-3.2$ sudo ls
Password:
[1]+  Stopped(SIGTTOU)
sh-3.2$

以上适用于Linux:

$ sudo ls
  readme.txt

但是,我在 Linux 上使用 sudo bash 得到以下信息:

$ sudo bash
bash: cannot set terminal process group (-1): Inappropriate ioctl for device
bash: no job control in this shell
]0;root@ubuntu: /tmproot@ubuntu:/tmp#

注意:以上工作,但我没有工作控制。

我可能忘了在终端上设置一些控制位,但谷歌在找到这个方面并没有多大帮助。另外,您是否知道任何可以非常详细地解释伪终端管理的好书。

我有 setid 调用,但我没有使用 openpty。我在打开 pty 时使用以下代码:

static int createPty(lua_State* L, char* ttyName, int* pty)
{
   *pty = getpt();
   if (*pty < 0 || grantpt(*pty) < 0 || unlockpt(*pty) < 0)
      return lDoErr(L,"Cannot open PTY: %s",strerror(errno));
   if(ptsname_r(*pty, ttyName, PTY_NAME_SIZE-1))
      return lDoErr(L,"ptsname_r: %s",strerror(errno));
   return 0;
}

我已经编辑了下面的代码,并且该代码有效。我的第一个版本不起作用的原因是我试图创建两个 PTY 频道。我希望能够区分 stdout 和 stderr,但 Linux 内核不允许多个 TIOCSCTTY 调用。

static int
childOpenTTY(const char* ttyName)
{
   struct termios termbuf;
   int fd=open(ttyName, O_RDWR);
   if(fd < 0)
      doClientError("open %s: %s",ttyName, strerror(errno));
   tcsetpgrp(fd, getpid());
   ioctl(fd,TIOCSCTTY,NULL);
   tcgetattr(fd, &termbuf);
   cfmakeraw(&termbuf); /* turn off NL to CR/NL mapping on output. */
   tcsetattr(fd, TCSANOW, &termbuf);
   return fd;
}

if( (ret = createPty(L, ttyName, &te->pty)) != 0)
   return ret;
if ((te->pid = zzbafork()) < 0)
   return lDoErr(L,"fork: %s",strerror(errno));
if(te->pid == 0)
{  /* Child process */
   static const char efmt[]={"Cannot set '%s' (dup2 err)"};
   int fd;
   if(setsid() < 0) /* make new process group */
      doClientError("setsid: %s",strerror(errno));
   fd=childOpenTTY(ttyName);
   if(dup2(fd, STDIN_FILENO) != STDIN_FILENO)
      doClientError(efmt,"stdin");
   if(dup2(fd, STDOUT_FILENO) != STDOUT_FILENO)
      doClientError(efmt,"stdout");
   if(dup2(fd, STDERR_FILENO) != STDERR_FILENO)
      doClientError(efmt,"stderr");
   if(fd != STDIN_FILENO && fd != STDOUT_FILENO && fd != STDERR_FILENO)
      close(fd);
   execve(cmd, (char**)cmdArgv, environ);
   /* execve should not return, unless error exec cmd */
   doClientError("Executing %s failed: %s",cmd,strerror(errno));
}
4

1 回答 1

0

由于此处没有显示实际代码,因此很难确定,但我怀疑您遇到了 POSIX 风格的“会话”管理。您需要执行一个setsid调用,然后打开 pty(从端),使其成为控制终端。和例程openptylogin_tty您完成低级垃圾工作;你在用那些吗?

于 2013-03-20T20:16:09.677 回答