要在上面的示例中增加 fork
的部分以实现一个简单的 echo 伪终端,你可以按照以下步骤进行修改。我们将创建一个子进程,该子进程会读取从主伪终端传入的数据并回显(echo)到该终端。
下面是修改后的完整代码:
#define _GNU_SOURCE // 添加这一行以启用GNU扩展
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <pty.h> // 包含伪终端相关函数
int main() {
int master_fd; // 主设备文件描述符
char slave_name[100]; // 从设备名称
// 创建一个新的伪终端
master_fd = posix_openpt(O_RDWR);
if (master_fd == -1) {
perror("posix_openpt");
exit(EXIT_FAILURE);
}
// 授权访问该从设备
if (grantpt(master_fd) == -1) {
perror("grantpt");
close(master_fd);
exit(EXIT_FAILURE);
}
// 解锁该从设备
if (unlockpt(master_fd) == -1) {
perror("unlockpt");
close(master_fd);
exit(EXIT_FAILURE);
}
// 获取从设备的名字
snprintf(slave_name, sizeof(slave_name), "%s", ptsname(master_fd));
printf("Master FD: %d\n", master_fd);
printf("Slave PTY: %s\n", slave_name);
pid_t pid = fork();
if (pid == -1) {
perror("fork");
close(master_fd);
exit(EXIT_FAILURE);
} else if (pid == 0) {
// 子进程部分
char buf[256];
// 重定向标准输入和输出到从设备
dup2(master_fd, STDIN_FILENO); // 将标准输入重定向到主伪终端
dup2(master_fd, STDOUT_FILENO); // 将标准输出重定向到主伪终端
while (1) {
ssize_t bytesRead = read(STDIN_FILENO, buf, sizeof(buf));
if (bytesRead <= 0) break; // 如果没有读取到数据,退出循环
write(STDOUT_FILENO, buf, bytesRead); // 回显数据
fflush(stdout); // 刷新输出缓冲区以确保数据被发送出去
}
close(master_fd);
exit(0);
} else {
// 父进程部分: 在父进程中可以直接与主设备交互,例如接收用户输入
char input[256];
while (fgets(input, sizeof(input), stdin)) {
write(master_fd, input, strlen(input)); // 将用户输入写入主伪终端
}
close(master_fd);
wait(NULL);
}
return 0;
}
编译命令:
使用以下命令进行编译:
gcc -D_GNU_SOURCE pty_echo.c -o pty_echo
如何运行:
编译成功后,运行程序 ./pty_echo
。你将在控制台看到 Master FD
和 Slave PTY
的信息。
- 输入文本后按 Enter 键,将会把文本通过主伪终端传递给子进程,子进程会将其回显回来。
- 在实际的使用过程中,你可以在一个终端中运行程序,然后打开另一个窗口连接到获得的从设备(如
/dev/pts/X
)进行测试。
注意事项:
- 在 Linux 系统中,要对特定的文件描述符执行读写操作。
- 确保你有适当的权限来访问和创建伪终端。
- 如果想在同一 terminal 中实现更复杂的交互,可以考虑使用更高层次的库或框架,如
ncurses
等。
内容由零声教学AI助手提供,问题来源于学员提问