winpty/Shared/AgentClient.cc
Ryan Prichard c74f82abdb Pass an initial window size to the agent on startup.
* The TestNet protocol starts with a terminal resize escape sequence, so
   the server discards all input until it sees the sequence, and it delays
   starting the agent.  Eventually, the goal is to implement an SSH server,
   and the SSH message that creates a PTY also provides the terminal size.
2011-11-24 03:03:25 -08:00

169 lines
5.2 KiB
C++

#define WINVER 0x501
#include "AgentClient.h"
#include <QLocalServer>
#include <QLocalSocket>
#include <QCoreApplication>
#include <QtDebug>
#include <windows.h>
#include "DebugClient.h"
#include "AgentMsg.h"
// TODO: Note that this counter makes AgentClient non-thread-safe.
int AgentClient::m_counter = 0;
AgentClient::AgentClient(int initialCols, int initialRows, QObject *parent) :
QObject(parent)
{
// Start a named pipe server.
QLocalServer *socketServer = new QLocalServer(this);
QString serverName =
"ConsoleAgent-" +
QString::number(QCoreApplication::applicationPid()) + "-" +
QString::number(++m_counter);
socketServer->listen(serverName);
// TODO: Improve this code. If we're in the release subdirectory,
// find the release Agent. Look in the same directory first.
QString agentProgram = QCoreApplication::applicationDirPath() + "\\..\\..\\Agent-build-desktop\\debug\\Agent.exe";
QString agentCmdLine =
QString("\"%1\" %2 %3 %4").arg(agentProgram,
socketServer->fullServerName())
.arg(initialCols).arg(initialRows);
Trace("Starting Agent: [%s]", agentCmdLine.toStdString().c_str());
// Start the agent.
BOOL success;
STARTUPINFO sui;
memset(&sui, 0, sizeof(sui));
sui.cb = sizeof(sui);
sui.dwFlags = STARTF_USESHOWWINDOW;
sui.wShowWindow = SW_SHOW; // TODO: put this back --- SW_HIDE;
PROCESS_INFORMATION pi;
memset(&pi, 0, sizeof(pi));
m_agentProcess = new PROCESS_INFORMATION;
QVector<wchar_t> cmdline(agentCmdLine.size() + 1);
agentCmdLine.toWCharArray(cmdline.data());
cmdline[agentCmdLine.size()] = L'\0';
success = CreateProcess(
(LPCWSTR)agentProgram.utf16(),
cmdline.data(),
NULL, NULL,
/*bInheritHandles=*/FALSE,
/*dwCreationFlags=*/CREATE_NEW_CONSOLE,
NULL, NULL,
&sui, m_agentProcess);
if (!success)
qFatal("Could not start agent subprocess.");
qDebug("New child process: PID %d", (int)m_agentProcess->dwProcessId);
if (!socketServer->waitForNewConnection(30000))
qFatal("Child process did not connect to parent pipe server.");
m_socket = socketServer->nextPendingConnection();
Q_ASSERT(m_socket != NULL);
// TODO: security -- Do we need to verify that this pipe connection was
// made by the right client? i.e. The Agent.exe process we just started?
socketServer->close();
}
void AgentClient::writeMsg(const AgentMsg &msg)
{
m_socket->write((char*)&msg, sizeof(msg));
}
int AgentClient::agentPid()
{
return m_agentProcess->dwProcessId;
}
void AgentClient::startShell()
{
FreeConsole();
AttachConsole(agentPid());
HANDLE conout1, conout2, conin;
{
// TODO: Should the permissions be more restrictive?
// TODO: Is this code necessary or even desirable? If I
// don't change these handles, then are the old values
// invalid? If so, what happens if I create a console
// subprocess?
// The handle must be inheritable. See comment below.
SECURITY_ATTRIBUTES sa;
memset(&sa, 0, sizeof(sa));
sa.nLength = sizeof(sa);
sa.bInheritHandle = TRUE;
conout1 = CreateFile(L"CONOUT$",
GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE,
&sa,
OPEN_EXISTING,
0, NULL);
conout2 = CreateFile(L"CONOUT$",
GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE,
&sa,
OPEN_EXISTING,
0, NULL);
conin = CreateFile(L"CONIN$",
GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE,
&sa,
OPEN_EXISTING,
0, NULL);
Q_ASSERT(conin != NULL);
Q_ASSERT(conout1 != NULL);
Q_ASSERT(conout2 != NULL);
/*
BOOL success;
success = SetStdHandle(STD_OUTPUT_HANDLE, conout1);
Q_ASSERT(success);
success = SetStdHandle(STD_ERROR_HANDLE, conout2);
Q_ASSERT(success);
success = SetStdHandle(STD_INPUT_HANDLE, conin);
Q_ASSERT(success);
*/
}
{
wchar_t program[80] = L"c:\\windows\\system32\\cmd.exe";
wchar_t cmdline[80];
wcscpy(cmdline, program);
STARTUPINFO sui;
memset(&sui, 0, sizeof(sui));
sui.cb = sizeof(sui);
sui.dwFlags = STARTF_USESTDHANDLES;
sui.hStdInput = conin;
sui.hStdOutput = conout1;
sui.hStdError = conout2;
PROCESS_INFORMATION pi;
memset(&pi, 0, sizeof(pi));
BOOL success = CreateProcess(
program,
cmdline,
NULL,
NULL,
FALSE,
0,
NULL,
NULL,
&sui,
&pi);
if (!success) {
qDebug("Could not start shell");
}
}
CloseHandle(conout1);
CloseHandle(conout2);
CloseHandle(conin);
FreeConsole();
}
QLocalSocket *AgentClient::getSocket()
{
return m_socket;
}