伪终端是指对于一个应用程序而言,他看上去像一个终端,但事实上它并不是一个真正的终端。

  • 进程打开伪终端设备,然后fork。子进程建立一个新的会话,打开一个相应的伪终端从设备。复制输入、输出和标准错误文件描述符,调用exec,子进程从设备编程伪终端。
  • 伪终端能像终端一样,但是无意义的函数调用如改变波特率、发送中断符、设置奇偶校验将被忽略。
  • 伪终端可以做输入和输出。

posix_opent函数提供了一种可移植的方法来打开下一个可用伪终端主设备

#include <stdlib.h>
#include <fcntl.h> int posix_openpt(int flags);
返回值:成功返回下一个可用的PTY主设备文件描述符,出错-
flags:设备操作标记,可以是0或者以下两项的之一,O_RDWR允许对设备同时进行读写操作,此标记通常需要指定
O_NOCTTY不将设备作为进程的控制终端

在伪终端从设备可用之前,它的权限必须设置,以便应用程序可以访问它。

改变指定master对应从设备的属主与访问权限

#define _XOPEN_SOURCE
#include <stdlib.h> int grantpt(int fd);
int unlockpt(int fd);
返回值:成功0,出错-1
fd:文件描述符

ptsname返回PTY从设备的名字

#define _XOPEN_SOURCE
#include <stdlib.h> char *ptsname(int fd); #define _GNU_SOURCE
#include <stdlib.h>
int ptsname_r(int fd, char *buf, size_t buflen);
fd:文件描述符

apue写的函数,打开下一个可用的PTY主设备。调用者必须分配一个数组来存放主设备或从设备名字。

#include "apue.h"

int ptym_open(char *pts_name, int pts_namesz);
返回值:成功返回PTY主设备文件描述符,出错-1
int ptys_open(char *pts_name);
返回值:成功返回PTY设备文件描述符,出错-1
pts_name:打开设备的名字
pts_namesz:缓冲区字节长度

实现函数:

#include "apue.h"
#include <errno.h>
#include <fcntl.h>
#if defined(SOLARIS)
#include <stropts.h>
#endif int ptym_open(char *pts_name, int pts_namesz)
{
char *ptr;
int fdm, err; if((fdm = posix_openpt(O_RDWR)) < )
return(-);
if(grantpt(fdm) < )
goto errout;
if(unlock(fdm) < )
goto errout;
if((ptr = ptsname(fmd)) == NULL)
goto errout; strncpy(pts_name, ptr, pts_namesz);
pts_name[pts_namesz - ] = '\0';
return(fdm);
errout:
err = errno;
close(fdm);
errno = err;
return(-);
} int ptys_open(char *pts_name)
{
int fds;
#if defined(SOLARIS)
int err, setup;
#endif if((fds = open(pts_name, O_RDWR)) < )
goto errout; if(setup == ) {
if(ioctl(fds, I_PUSH, "ptem") < )
goto errout;
if(ioctl(fds, I_PUSH, "ldterm") < )
goto errout;
if(ioctl(fds, I_PUSH, "ttcompat") < ) {
errout:
err = errno;
close(fds);
errno = err;
return(-);
}
}
#endif
return(fds);
}

pty_open

用fork调用打开主设备和从设备,创建作为会话首进程的子进程并使其具有控制终端。

#include "apue.h"
#include <termios.h> pid_t pty_fork(int *ptrfdm, char *slave_name, int slave_namesz,
const struct termios *slave_termios,
const struct winsize *slave_winsize);

pty_fork函数

 #include "apue.h"
#include <termios.h> pid_t pty_fork(int *ptrfdm, char *slave_name, int slave_namesz,
const struct termios *slave_termios,
const struct winsize *slave_winsize)
{
int fdm, fds;
pid_t pid;
char pts_name[]; if((fdm = ptym_open(pts_name, sizeof(pts_name))) < )
err_sys("can't open master pty: %s, error %d", pts_name, fdm); if(slave_name != NULL) {
strncpy(slave_name, pts_name, slave_namesz);
slave_name[slave_namesz - ] = '\0';
} if((pid = fork()) < )
return(-);
else if(pid == ) { /* childe */
if(setsid() < )
err_sys("setsid error"); if((fds = ptys_open(pts_name)) < )
err_sys("can't open slave pty");
close(fdm);
#if defined(BSD)
if(ioctl(fds, TIOCSCTTY, (char *)) < )
err_sys("TIOCSCTTY error");
#endif
if(slave_termios != NULL) {
if(tcsetattr(fds, TCSANOW, slave_termios) < )
err_sys("tcsetattr error on slave pty");
}
if(slave_winsize != NULL) {
if(ioctl(fds, TIOCSWINSZ, slave_winsize) < )
err_sys("TIOCSWINSZ error on slave pty");
} if(dup2(fds, STDIN_FILENO) != STDIN_FILENO)
err_sys("dup2 error to stdin");
if(dup2(fds, STDOUT_FILENO) != STDOUT_FILENO)
err_sys("dup2 error to stdout");
if(dup2(fds, STDERR_FILENO) != STDERR_FILENO)
err_sys("dup2 error to stderr");
if(fds != STDIN_FILENO && fds != STDOUT_FILENO &&
fds != STDERR_FILENO)
close(fds);
return();
} else {
*ptrfdm = fdm;
return(pid);
}
}

pty_fork

apue 第19章 伪终端的更多相关文章

  1. [apue] 书中关于伪终端的一个纰漏

    在看 apue 第 19 章伪终端第 6 节使用 pty 程序时,发现“检查长时间运行程序的输出”这一部分内容的实际运行结果,与书上所说有出入. 于是展开一番研究,最终发现是书上讲的有问题,现在摘出来 ...

  2. apue 第18章 终端I/O

    终端I/O有两种不同的工作模式: (1)规范模式:输入以行单位进行处理,每个读请求也最多返回一行. (2)非规范模式:输入字符不装配成行. 终端设备是由通常位于内核中的终端驱动程序控制的.每个终端设备 ...

  3. apue第四章学习总结

    apue第四章学习总结 4.1.若以stat函数去替换lstat函数,会发生: 原来的目录路径: $:~/workspace/apue2/include$ ls -l apue.h abc lrwxr ...

  4. linux的终端,网络虚拟终端,伪终端(转)

      blog.csdn.net/todd911/article/details/8025540 Linux上许多网络服务应用,如l2tp.pptp.telnet,都用到了伪终端.有朋友在问这方面的概念 ...

  5. linux的终端,网络虚拟终端,伪终端(转)

    转自http://www.xuebuyuan.com/877887.html 2013年09月07日 ⁄ 综合 ⁄ 共 4047字 ⁄ 字号 小 中 大 ⁄ 评论关闭 Linux上许多网络服务应用,如 ...

  6. Unix环境高级编程(二十)伪终端

    1.综述 伪终端对于一个应用程序而言,看上去像一个终端,但事实上伪终端并不是一个真正的终端.从内核角度看,伪终端看起来像一个双向管道,而事实上Solaris的伪终端就是用STREAMS构建的.伪终端总 ...

  7. 第19章 集合框架(3)-Map接口

    第19章 集合框架(3)-Map接口 1.Map接口概述 Map是一种映射关系,那么什么是映射关系呢? 映射的数学解释 设A,B是两个非空集合,如果存在一个法则,使得对A中的每一个元素a,按法则f,在 ...

  8. 第19章 queue队列容器

    /* 第19章 queue队列容器 19.1 queue技术原理 19.2 queue应用基础 19.3 本章小结 */ // 第19章 queue队列容器 // 19.1 queue技术原理 // ...

  9. 《TCP/IP详解卷1:协议》第19章 TCP的交互数据流-读书笔记

    章节回顾: <TCP/IP详解卷1:协议>第1章 概述-读书笔记 <TCP/IP详解卷1:协议>第2章 链路层-读书笔记 <TCP/IP详解卷1:协议>第3章 IP ...

随机推荐

  1. IO操作之ObjectInputStream与ObjectOutputStream

    之前写过DataInputStream和DataOutputStream,使用这两个类可以对java基本数据类型进行序列化和反序列化. 本篇再来两个新东西:ObjectInputStream,Obje ...

  2. Java线程通信-生产者消费者问题

    线程通信示例——生产者消费者问题 这类问题描述了一种情况,假设仓库中只能存放一件产品,生产者将生产出来的产品放入仓库,消费者将仓库中的产品取走消费.假设仓库中没有产品,则生产者可以将 产品放入仓库,有 ...

  3. SpringMvc的过滤器。

    一:过滤器的原理: 过滤器放在web资源之前,可以在请求抵达它所应用的web资源(可以是一个Servlet.一个Jsp页面,甚至是一个HTML页面)之前截获进入的请求,并且在它返回到客户之前截获输出请 ...

  4. fiddler使用笔记1

    转载地址:写得很不错的fildder教程   http://kb.cnblogs.com/page/130367/ Fiddler的基本介绍 Fiddler的官方网站:  www.fiddler2.c ...

  5. js中的投掷筛子的小游戏

    代码 <!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8&qu ...

  6. 建站手册-网站构建:万维网联盟(World Wide Web Consortium)

    ylbtech-建站手册-网站构建:万维网联盟(World Wide Web Consortium) 1.返回顶部 1. http://www.w3school.com.cn/site/site_w3 ...

  7. 测开之路四十八:Django之重定向与cookie

    基础配置与上一篇一致 404错误 定义一个error页面 <!DOCTYPE html><html lang="en"><head> <m ...

  8. oo_project_1

    Project 1题目要求分析: 实现多项式的加减运算,主要问题是解决输入格式的判断问题. 输入实例: {(3,0), (2,2), (12,3)} + {(3,1), (-5,3)} – {(-19 ...

  9. Html5 学习笔记 【PC固定布局】 实战3 热门旅游展示区

    最终效果图: html 代码: <!DOCTYPE html> <html lang="zh-cn"> <head> <meta char ...

  10. HttpClient 之 4.x.x版本以上的发送Https请求

    https请求比http更安全 是在http的基础上加了SSL数据加密协议. http的连接很简单,是无状态的:HTTPS协议是由SSL+HTTP协议构建的可进行加密传输.身份认证的网络协议,比htt ...