c - 使用管道连接到控制台程序时如何处理缓冲

标签 c winapi

我有一个 win32 程序(是的,我一直在使用 win32)启动一个子进程并通过匿名管道连接到它的标准输入/标准输出。大多数时候我不拥有子进程(因此,我无权访问代码)。我发现 child 的标准输出缓冲妨碍了。在我终止进程之前,我看不到 child 的任何输出;但是,我知道输入在收到时会被处理。

所以,我读过 http://support.microsoft.com/kb/190351/en-us ,它警告子进程使用 printf(等)关于缓冲的问题,微软的 brilliant 修复是用 fflush() 跟踪子进程中的所有 printfs(更简单的方法是只需使用 setbuf(stdout,0); 但是,这假定您可以访问子源代码。

显然,子进程在连接到父管道时不会刷新标准输出。有没有办法解决这个问题而不需要修改 child ?

我已经用一个简单的 child 验证了这一点:

    #include <stdio.h>
    #include <fcntl.h>

    char line[256];

    int
    main(int argc, char *argv[])
    {
            char *lp;

    //      setbuf(stdout,0);
            while(1) {
                    printf("Prompt:");
                    lp = gets(line);
                    printf("Got: <%s>\n",lp);
                    if (strcmp(lp,"yada") == 0) {
                            int fd = open("C:/tmp/yada123",O_BINARY|O_RDWR|O_CREAT);
                            if (fd > 0) {
                                    write (fd,"hi!\n",4);
                                    close(fd);
                            }
                    }
                    if (strcmp(lp,"exit") == 0) {
                            printf("bye!\n");
                            break;
                    }
            }
}

如果我取消注释顶部的 setbuf() 调用,那么交互是干净的;但是请注意,我通常没有修改子代码的选项。另外,我验证了另一个方向没有被缓冲(即使我没有看到输出);因为文件是在收到字符串“yada”时写入的。

最佳答案

前言:在问题的评论中概述了代码注入(inject)可能会解决问题。这篇文章展示了 dll 注入(inject)的一种可能实现,因此补充了评论。我将代码作为普通帖子发布,以便更好地展示多行代码。

以下代码可能是 dll 注入(inject)的良好起点。它使用一个全局钩子(Hook)。这是一个非常简单的方法,但它会将您的 dll 注入(inject)到所有正在运行的进程中(仅注入(inject)您有权访问的进程)。因此,您必须询问当前程序名称并调用 setbuf(stdout,0); 只有在您处于所需应用程序中的情况下:-) 也许你需要添加一些头文件(我只是把一些代码片段放在一起,没有测试它)。

void InstallHook(void)void RemoveHook(void) 函数必须声明为 dll 导出函数,因此您要么使用 *.def 文件,要么声明它们作为 __declspec(dllexport)。这可能取决于您的编译器和设置。

#include <windows.h>
#include <cstdlib>
#include <cstdio>

BOOL WINAPI DllMain( HINSTANCE hModule, DWORD  ul_reason_for_call, LPVOID lpReserved)
{
  UNREFERENCED_PARAMETER(lpReserved);

  switch (ul_reason_for_call)
  {
  case DLL_PROCESS_ATTACH:
  {
    wchar_t buffer [1024];
    // get the path and executable name of the current process
    GetModuleFileNameW( GetModuleHandle( NULL ), buffer, 1024 );
    //test if we are in the target app
    if ( _wcsicmp( buffer, "C:/some_dir/some_app.exe" ) == 0 )
    {
      setbuf(stdout,0);
    }
    break;
  }
  case DLL_PROCESS_DETACH:
    break;
  default:
    break;
  }

  return (true);
}

HHOOK gl_hHook = NULL;
LRESULT CALLBACK HookProc(int nCode, WPARAM wParam, LPARAM lParam)
{
  return (CallNextHookEx( gl_hHook, nCode, wParam, lParam));
}

// call this function before or after you have started the child process
// the hook gets installed and your dll is loaded into each running process
// and the DllMain functions gets called by each process
void InstallHook(void)
{
  gl_hHook = SetWindowsHookEx( WH_CBT, HookProc, gl_hThisInstance, 0 );
}

void RemoveHook(void)
{
  UnhookWindowsHookEx( gl_hHook );
}

关于c - 使用管道连接到控制台程序时如何处理缓冲,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22919166/

相关文章:

c++ - 在 NT 之前的系统上调用 NT 函数

c++ - output objdump -t的输出中的 ".hidden"是什么意思?

objective-c - 什么是 int foo[] = {1200,100} 以及如何在 Swift 中重新创建它?

windows - 如何获取 Microsoft Edge 和 ApplicationFrameHost.exe 中托管的其他窗口的键盘布局

c++ - CreateDC() 导致 glutInit() 失败?

c++ - 保持窗口始终在顶部——包括菜单 (win32)

c++ - 如何在两个 C++ MFC 插件之间进行通信?

c - 如何从文件中删除记录

c - 在 core_cm4.h 上,为什么会有类似 ((uint32_t)(int32_t)IRQn) 的转换?

对 recv() 感到困惑