summaryrefslogtreecommitdiffstats
path: root/src/windows/leash/out2con.cpp
blob: f7a1d35a95836454940bef0fdcf91b2d0b63f390 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
#include "out2con.h"

#include <windows.h>
#include <stdio.h>
#include <io.h>

class ConsoleEcho
{
public:
    ConsoleEcho();
    ~ConsoleEcho();

private:
    DWORD ThreadLoop();

    static DWORD WINAPI ThreadFunc(void* param);

    FILE m_originalStdout;
    int m_stdoutfd;
    int m_pipefd;
    HANDLE m_hReadPipe, m_hWritePipe;
    HANDLE m_hThread;

    static const int BUFSIZE=512;
};


ConsoleEcho *
CreateConsoleEcho()
{
    return new ConsoleEcho;
}

void
DestroyConsoleEcho(ConsoleEcho *echo)
{
    delete echo;
}


DWORD WINAPI ConsoleEcho::ThreadFunc(void* param)
{
    return ((ConsoleEcho*)(param))->ThreadLoop();
}


DWORD ConsoleEcho::ThreadLoop()
{
    DWORD dwRead, dwWritten;
    CHAR chBuf[BUFSIZE];
    BOOL bSuccess = FALSE;
    // Note that the following does not work when running in the msvc2010
    // debugger with redirected output; you still get the redirected file
    // handle, not the console:
    //HANDLE hConsoleStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
    // This seems to be more reliable:
    HANDLE hConsoleStdOut = CreateFile("CONOUT$",
                                       GENERIC_WRITE,
                                       FILE_SHARE_WRITE,
                                       NULL, OPEN_EXISTING, 0, 0);
    for (;;) {
        // read from redirected stdout
        bSuccess = ReadFile(m_hReadPipe, chBuf, BUFSIZE, &dwRead, NULL);
        if (!bSuccess || (dwRead == 0))
            break;

        // write to console
        WriteFile(hConsoleStdOut, chBuf, dwRead, &dwWritten, NULL);
        // also write to original stdout
        if (m_stdoutfd>=0) {
            _write(m_stdoutfd, chBuf, dwRead);
            // _commit() causes assert if m_stdoutfd is a device (e.g., console or NUL).
            if (!_isatty(m_stdoutfd))
                _commit(m_stdoutfd);
        }
    }
    CloseHandle(hConsoleStdOut);
    return 0;
}

ConsoleEcho::ConsoleEcho()
{
    // setup console
    AllocConsole();
    // create pipe
    CreatePipe(&m_hReadPipe, &m_hWritePipe, NULL, 0);
    // save original stdout to preserve commandline-specified redirection
    m_stdoutfd = _fileno(stdout);
    // and copy the whole damn FILE structure so we can restore it
    // when we're done.  I don't know any other way to restore the
    // crazy windows gui default '-2' filedesc stdout.
    m_originalStdout = *stdout;
    // hook up the write end of our pipe to stdout
    m_pipefd = _open_osfhandle((intptr_t)m_hWritePipe, 0);
    // take our os file handle and allocate a crt FILE for it
    FILE* fp = _fdopen(m_pipefd, "w");
    // copy to stdout
    *stdout = *fp;
    // now slam the allocated FILE's _flag to zero to mark it as free without
    // actually closing the os file handle and pipe
    fp->_flag = 0;

    // disable buffering
    setvbuf(stdout, NULL, _IONBF, 0);

    // Create a thread to process our pipe, forwarding output
    // to both the console and the original stdout
    m_hThread = CreateThread(NULL, 0, &ThreadFunc, this, 0, NULL);
}

ConsoleEcho::~ConsoleEcho()
{
    // fclose() unfortunately immediately invalidates the read pipe before the
    // pipe thread has a chance to flush it, so don't do that.
    //fclose(stdout);

    // instead, just slam the original stdout
    *stdout = m_originalStdout;
    //printf("Safe to printf now and no longer echoed to console.\n");
    // Close write pipe
    _close(m_pipefd);
    // and wait here for pipe thread to exit
    WaitForSingleObject(m_hThread, 1000);
    // now close read pipe as well
    CloseHandle(m_hReadPipe);
}