gpt4 book ai didi

c - 如何使用共享内存段重定向命令输出?

转载 作者:行者123 更新时间:2023-11-30 16:37:08 25 4
gpt4 key购买 nike

我正在尝试使用我的共享内存重定向我的命令。

总结一下:我的程序运行如下:我运行 ./server 等待信息,当我运行 ./client 时写入。客户端通过命名管道(mkfifo)将信息发送到服务器。该信息是:要执行的命令类型、共享内存段的数据和名称(在客户端中使用 shm_open ftruncate shm_unlink 创建,最后 char * addr = mmap () ....)。如果我在没有重定向标准输出的情况下启动程序,命令的结果将显示在执行服务器的终端中......尽管如此,我希望它出现在客户端中!这是服务器中的代码段:

    if ((fd = shm_open(req->shm_name, O_RDWR, S_IRUSR | S_IWUSR)) == -1) {
perror("shm_open");
exit(EXIT_FAILURE);
}

char *dir = malloc(sizeof(char) * 256);
int tube[2];
if (pipe(tube) == -1) {
perror("tube");
exit(EXIT_FAILURE);
}
switch (fork()) {
case -1:
perror("fork");
exit(EXIT_FAILURE);
case 0:
getcwd(dir, 256);
strcat(dir, "/info_proc");
char *pid = malloc(10 * sizeof(char));
sprintf(pid, "%d", req->data);
if (close(tube[0]) == -1) {
perror("close");
exit(EXIT_FAILURE);
}

if (dup2(tube[1], STDOUT_FILENO) == -1) {
perror("dup2");
exit(EXIT_FAILURE);
}

/*if (dup2(fd, STDIN_FILENO) == -1) {
perror("dup2");
exit(EXIT_FAILURE);
}*/

execl(dir, "info_proc", pid, NULL);
perror("execl");
exit(EXIT_FAILURE);
default:
wait(NULL);
/*if (dup2(fd, tube[0]) == -1) {
perror("dup2");
exit(EXIT_FAILURE);
}*/

if (close(tube[1]) == -1) {
perror("close");
exit(EXIT_FAILURE);
}

char *received = malloc(sizeof(char) * BUFFER_SIZE);
while (read(tube[0], &received, sizeof(received)));
//addr = received;

/*fd = tube[0];
if (read(tube[0], &addr, BUFFER_SIZE) == -1) {
perror("write");
exit(EXIT_FAILURE);
}*/

break;
}

我该怎么做?

最佳答案

虽然您应该能够通过 fmemopen()映射到共享内存段,但您需要一个文件描述符来执行通过 dup2() 或其同级重定向。您可以通过 fileno() 函数提取大多数流的文件描述符,但我发现这不适用于内存流,大概是因为此类流与底层打开文件描述不对应内核。

因此,如果您想捕获执行命令的标准输出(也许还有标准错误?),那么您需要将其重定向到管道,并让另一个进程为管道的另一端提供复制服务将数据写入共享内存段。

这是一种对我来说有意义的方法:

  • 客户端创建共享内存段,然后
  • 客户端打开 fifo 并向其写入请求
  • 服务器读取请求并派生一个子进程来处理它
  • 子进程打开指定的共享内存段并取消链接
  • child 创建一个管道,然后
  • 子进程 fork 自己的子进程(孙子进程),在其中运行客户端的命令
  • 孙子将其标准输出重定向到管道的写入端
  • 子进程从管道读取输出并将其写入共享内存段,直到到达末尾

注释:

  • 虽然您无法使用fmemopen()直接重定向到共享内存段,但它仍然可以帮助您
  • 您可以使用fstat()来获取服务器需要知道的共享内存段的大小
  • 使用popen()而不是管道、 fork 等可以简化孙子的创建和管理。
  • 您需要一种方法来告诉客户端输出已准备就绪,并且顺便同步对共享内存段的访问。例如,客户端可能会创建一个命名信号量供服务器用于此目的。

关于c - 如何使用共享内存段重定向命令输出?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48104735/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com