566 lines
19 KiB
C++
566 lines
19 KiB
C++
// Copyright (c) 2011-2012 Ryan Prichard
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to
|
|
// deal in the Software without restriction, including without limitation the
|
|
// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
// sell copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
// IN THE SOFTWARE.
|
|
|
|
#include "Agent.h"
|
|
#include "Win32Console.h"
|
|
#include "ConsoleInput.h"
|
|
#include "Terminal.h"
|
|
#include "NamedPipe.h"
|
|
#include "AgentAssert.h"
|
|
#include "../shared/DebugClient.h"
|
|
#include "../shared/AgentMsg.h"
|
|
#include "../shared/Buffer.h"
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdint.h>
|
|
#include <windows.h>
|
|
#include <vector>
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
const int SC_CONSOLE_MARK = 0xFFF2;
|
|
const int SC_CONSOLE_SELECT_ALL = 0xFFF5;
|
|
const int SYNC_MARKER_LEN = 16;
|
|
|
|
static BOOL WINAPI consoleCtrlHandler(DWORD dwCtrlType)
|
|
{
|
|
if (dwCtrlType == CTRL_C_EVENT) {
|
|
// Do nothing and claim to have handled the event.
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static std::string wstringToUtf8String(const std::wstring &input)
|
|
{
|
|
int mblen = WideCharToMultiByte(CP_UTF8, 0,
|
|
input.c_str(), input.size() + 1,
|
|
NULL, 0, NULL, NULL);
|
|
if (mblen <= 0) {
|
|
return std::string();
|
|
}
|
|
std::vector<char> tmp(mblen);
|
|
int mblen2 = WideCharToMultiByte(CP_UTF8, 0,
|
|
input.c_str(), input.size() + 1,
|
|
tmp.data(), tmp.size(),
|
|
NULL, NULL);
|
|
ASSERT(mblen2 == mblen);
|
|
return tmp.data();
|
|
}
|
|
|
|
Agent::Agent(LPCWSTR controlPipeName,
|
|
LPCWSTR dataPipeName,
|
|
int initialCols,
|
|
int initialRows) :
|
|
m_closingDataSocket(false),
|
|
m_terminal(NULL),
|
|
m_childProcess(NULL),
|
|
m_childExitCode(-1),
|
|
m_syncCounter(0)
|
|
{
|
|
trace("Agent starting...");
|
|
|
|
m_bufferData = new CHAR_INFO[BUFFER_LINE_COUNT][MAX_CONSOLE_WIDTH];
|
|
|
|
m_console = new Win32Console;
|
|
m_console->setSmallFont();
|
|
m_console->reposition(
|
|
Coord(initialCols, BUFFER_LINE_COUNT),
|
|
SmallRect(0, 0, initialCols, initialRows));
|
|
m_console->setCursorPosition(Coord(0, 0));
|
|
m_console->setTitle(m_currentTitle);
|
|
|
|
m_controlSocket = makeSocket(controlPipeName);
|
|
m_dataSocket = makeSocket(dataPipeName);
|
|
m_terminal = new Terminal(m_dataSocket);
|
|
m_consoleInput = new ConsoleInput(m_console, this);
|
|
|
|
resetConsoleTracking(false);
|
|
|
|
// Setup Ctrl-C handling. First restore default handling of Ctrl-C. This
|
|
// attribute is inherited by child processes. Then register a custom
|
|
// Ctrl-C handler that does nothing. The handler will be called when the
|
|
// agent calls GenerateConsoleCtrlEvent.
|
|
SetConsoleCtrlHandler(NULL, FALSE);
|
|
SetConsoleCtrlHandler(consoleCtrlHandler, TRUE);
|
|
|
|
setPollInterval(25);
|
|
}
|
|
|
|
Agent::~Agent()
|
|
{
|
|
trace("Agent exiting...");
|
|
m_console->postCloseMessage();
|
|
if (m_childProcess != NULL)
|
|
CloseHandle(m_childProcess);
|
|
delete [] m_bufferData;
|
|
delete m_console;
|
|
delete m_terminal;
|
|
delete m_consoleInput;
|
|
}
|
|
|
|
// Write a "Device Status Report" command to the terminal. The terminal will
|
|
// reply with a row+col escape sequence. Presumably, the DSR reply will not
|
|
// split a keypress escape sequence, so it should be safe to assume that the
|
|
// bytes before it are complete keypresses.
|
|
void Agent::sendDsr()
|
|
{
|
|
m_dataSocket->write("\x1B[6n");
|
|
}
|
|
|
|
NamedPipe *Agent::makeSocket(LPCWSTR pipeName)
|
|
{
|
|
NamedPipe *pipe = createNamedPipe();
|
|
if (!pipe->connectToServer(pipeName)) {
|
|
trace("error: could not connect to %ls", pipeName);
|
|
::exit(1);
|
|
}
|
|
pipe->setReadBufferSize(64 * 1024);
|
|
return pipe;
|
|
}
|
|
|
|
void Agent::resetConsoleTracking(bool sendClear)
|
|
{
|
|
memset(m_bufferData, 0, sizeof(CHAR_INFO) * BUFFER_LINE_COUNT * MAX_CONSOLE_WIDTH);
|
|
m_syncRow = -1;
|
|
m_scrapedLineCount = m_console->windowRect().top();
|
|
m_scrolledCount = 0;
|
|
m_maxBufferedLine = -1;
|
|
m_dirtyWindowTop = -1;
|
|
m_dirtyLineCount = 0;
|
|
m_terminal->reset(sendClear, m_scrapedLineCount);
|
|
}
|
|
|
|
void Agent::onPipeIo(NamedPipe *namedPipe)
|
|
{
|
|
if (namedPipe == m_controlSocket)
|
|
pollControlSocket();
|
|
else if (namedPipe == m_dataSocket)
|
|
pollDataSocket();
|
|
}
|
|
|
|
void Agent::pollControlSocket()
|
|
{
|
|
if (m_controlSocket->isClosed()) {
|
|
trace("Agent shutting down");
|
|
shutdown();
|
|
return;
|
|
}
|
|
|
|
while (true) {
|
|
int32_t packetSize;
|
|
int size = m_controlSocket->peek((char*)&packetSize, sizeof(int32_t));
|
|
if (size < (int)sizeof(int32_t))
|
|
break;
|
|
int totalSize = sizeof(int32_t) + packetSize;
|
|
if (m_controlSocket->bytesAvailable() < totalSize) {
|
|
if (m_controlSocket->readBufferSize() < totalSize)
|
|
m_controlSocket->setReadBufferSize(totalSize);
|
|
break;
|
|
}
|
|
std::string packetData = m_controlSocket->read(totalSize);
|
|
ASSERT((int)packetData.size() == totalSize);
|
|
ReadBuffer buffer(packetData);
|
|
buffer.getInt(); // Discard the size.
|
|
handlePacket(buffer);
|
|
}
|
|
}
|
|
|
|
void Agent::handlePacket(ReadBuffer &packet)
|
|
{
|
|
int type = packet.getInt();
|
|
int32_t result = -1;
|
|
switch (type) {
|
|
case AgentMsg::Ping:
|
|
result = 0;
|
|
break;
|
|
case AgentMsg::StartProcess:
|
|
result = handleStartProcessPacket(packet);
|
|
break;
|
|
case AgentMsg::SetSize:
|
|
result = handleSetSizePacket(packet);
|
|
break;
|
|
case AgentMsg::GetExitCode:
|
|
ASSERT(packet.eof());
|
|
result = m_childExitCode;
|
|
break;
|
|
case AgentMsg::GetProcessId:
|
|
ASSERT(packet.eof());
|
|
if (m_childProcess == NULL)
|
|
result = -1;
|
|
else
|
|
result = GetProcessId(m_childProcess);
|
|
break;
|
|
case AgentMsg::SetConsoleMode:
|
|
m_terminal->setConsoleMode(packet.getInt());
|
|
result = 0;
|
|
break;
|
|
default:
|
|
trace("Unrecognized message, id:%d", type);
|
|
}
|
|
m_controlSocket->write((char*)&result, sizeof(result));
|
|
}
|
|
|
|
int Agent::handleStartProcessPacket(ReadBuffer &packet)
|
|
{
|
|
BOOL success;
|
|
ASSERT(m_childProcess == NULL);
|
|
|
|
std::wstring program = packet.getWString();
|
|
std::wstring cmdline = packet.getWString();
|
|
std::wstring cwd = packet.getWString();
|
|
std::wstring env = packet.getWString();
|
|
std::wstring desktop = packet.getWString();
|
|
ASSERT(packet.eof());
|
|
|
|
LPCWSTR programArg = program.empty() ? NULL : program.c_str();
|
|
std::vector<wchar_t> cmdlineCopy;
|
|
LPWSTR cmdlineArg = NULL;
|
|
if (!cmdline.empty()) {
|
|
cmdlineCopy.resize(cmdline.size() + 1);
|
|
cmdline.copy(&cmdlineCopy[0], cmdline.size());
|
|
cmdlineCopy[cmdline.size()] = L'\0';
|
|
cmdlineArg = &cmdlineCopy[0];
|
|
}
|
|
LPCWSTR cwdArg = cwd.empty() ? NULL : cwd.c_str();
|
|
LPCWSTR envArg = env.empty() ? NULL : env.data();
|
|
|
|
STARTUPINFO sui;
|
|
PROCESS_INFORMATION pi;
|
|
memset(&sui, 0, sizeof(sui));
|
|
memset(&pi, 0, sizeof(pi));
|
|
sui.cb = sizeof(STARTUPINFO);
|
|
sui.lpDesktop = desktop.empty() ? NULL : (LPWSTR)desktop.c_str();
|
|
|
|
success = CreateProcess(programArg, cmdlineArg, NULL, NULL,
|
|
/*bInheritHandles=*/FALSE,
|
|
/*dwCreationFlags=*/CREATE_UNICODE_ENVIRONMENT |
|
|
/*CREATE_NEW_PROCESS_GROUP*/0,
|
|
(LPVOID)envArg, cwdArg, &sui, &pi);
|
|
int ret = success ? 0 : GetLastError();
|
|
|
|
trace("CreateProcess: %s %d",
|
|
(success ? "success" : "fail"),
|
|
(int)pi.dwProcessId);
|
|
|
|
if (success) {
|
|
CloseHandle(pi.hThread);
|
|
m_childProcess = pi.hProcess;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int Agent::handleSetSizePacket(ReadBuffer &packet)
|
|
{
|
|
int cols = packet.getInt();
|
|
int rows = packet.getInt();
|
|
ASSERT(packet.eof());
|
|
resizeWindow(cols, rows);
|
|
return 0;
|
|
}
|
|
|
|
void Agent::pollDataSocket()
|
|
{
|
|
m_consoleInput->writeInput(m_dataSocket->readAll());
|
|
|
|
// If the child process had exited, then close the data socket if we've
|
|
// finished sending all of the collected output.
|
|
if (m_closingDataSocket &&
|
|
!m_dataSocket->isClosed() &&
|
|
m_dataSocket->bytesToSend() == 0) {
|
|
trace("Closing data pipe after data is sent");
|
|
m_dataSocket->closePipe();
|
|
}
|
|
}
|
|
|
|
void Agent::onPollTimeout()
|
|
{
|
|
// Give the ConsoleInput object a chance to flush input from an incomplete
|
|
// escape sequence (e.g. pressing ESC).
|
|
m_consoleInput->flushIncompleteEscapeCode();
|
|
|
|
// Check if the child process has exited.
|
|
if (WaitForSingleObject(m_childProcess, 0) == WAIT_OBJECT_0) {
|
|
DWORD exitCode;
|
|
if (GetExitCodeProcess(m_childProcess, &exitCode))
|
|
m_childExitCode = exitCode;
|
|
CloseHandle(m_childProcess);
|
|
m_childProcess = NULL;
|
|
|
|
// Close the data socket to signal to the client that the child
|
|
// process has exited. If there's any data left to send, send it
|
|
// before closing the socket.
|
|
m_closingDataSocket = true;
|
|
}
|
|
|
|
// Scrape for output *after* the above exit-check to ensure that we collect
|
|
// the child process's final output.
|
|
if (!m_dataSocket->isClosed())
|
|
scrapeOutput();
|
|
|
|
if (m_closingDataSocket &&
|
|
!m_dataSocket->isClosed() &&
|
|
m_dataSocket->bytesToSend() == 0) {
|
|
trace("Closing data pipe after child exit");
|
|
m_dataSocket->closePipe();
|
|
}
|
|
}
|
|
|
|
// Detect window movement. If the window moves down (presumably as a
|
|
// result of scrolling), then assume that all screen buffer lines down to
|
|
// the bottom of the window are dirty.
|
|
void Agent::markEntireWindowDirty()
|
|
{
|
|
SmallRect windowRect = m_console->windowRect();
|
|
m_dirtyLineCount = std::max(m_dirtyLineCount,
|
|
windowRect.top() + windowRect.height());
|
|
}
|
|
|
|
// Scan the screen buffer and advance the dirty line count when we find
|
|
// non-empty lines.
|
|
void Agent::scanForDirtyLines()
|
|
{
|
|
const SmallRect windowRect = m_console->windowRect();
|
|
CHAR_INFO prevChar;
|
|
if (m_dirtyLineCount >= 1) {
|
|
m_console->read(SmallRect(windowRect.width() - 1,
|
|
m_dirtyLineCount - 1,
|
|
1, 1),
|
|
&prevChar);
|
|
} else {
|
|
m_console->read(SmallRect(0, 0, 1, 1), &prevChar);
|
|
}
|
|
int attr = prevChar.Attributes;
|
|
|
|
for (int line = m_dirtyLineCount;
|
|
line < windowRect.top() + windowRect.height();
|
|
++line) {
|
|
CHAR_INFO lineData[MAX_CONSOLE_WIDTH]; // TODO: bufoverflow
|
|
SmallRect lineRect(0, line, windowRect.width(), 1);
|
|
m_console->read(lineRect, lineData);
|
|
for (int col = 0; col < windowRect.width(); ++col) {
|
|
int newAttr = lineData[col].Attributes;
|
|
if (lineData[col].Char.UnicodeChar != L' ' || attr != newAttr)
|
|
m_dirtyLineCount = line + 1;
|
|
newAttr = attr;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Agent::resizeWindow(int cols, int rows)
|
|
{
|
|
freezeConsole();
|
|
|
|
Coord bufferSize = m_console->bufferSize();
|
|
SmallRect windowRect = m_console->windowRect();
|
|
Coord newBufferSize(cols, bufferSize.Y);
|
|
SmallRect newWindowRect;
|
|
|
|
// This resize behavior appears to match what happens when I resize the
|
|
// console window by hand.
|
|
if (windowRect.top() + windowRect.height() == bufferSize.Y ||
|
|
windowRect.top() + rows >= bufferSize.Y) {
|
|
// Lock the bottom of the new window to the bottom of the buffer if either
|
|
// - the window was already at the bottom of the buffer, OR
|
|
// - there isn't enough room.
|
|
newWindowRect = SmallRect(0, newBufferSize.Y - rows, cols, rows);
|
|
} else {
|
|
// Keep the top of the window where it is.
|
|
newWindowRect = SmallRect(0, windowRect.top(), cols, rows);
|
|
}
|
|
|
|
if (m_dirtyWindowTop != -1 && m_dirtyWindowTop < windowRect.top())
|
|
markEntireWindowDirty();
|
|
m_dirtyWindowTop = newWindowRect.top();
|
|
|
|
m_console->reposition(newBufferSize, newWindowRect);
|
|
unfreezeConsole();
|
|
}
|
|
|
|
void Agent::scrapeOutput()
|
|
{
|
|
freezeConsole();
|
|
|
|
std::wstring newTitle = m_console->title();
|
|
if (newTitle != m_currentTitle) {
|
|
std::string command = std::string("\x1b]0;") +
|
|
wstringToUtf8String(newTitle) + "\x07";
|
|
m_dataSocket->write(command.c_str());
|
|
m_currentTitle = newTitle;
|
|
}
|
|
|
|
const Coord cursor = m_console->cursorPosition();
|
|
const SmallRect windowRect = m_console->windowRect();
|
|
|
|
if (m_syncRow != -1) {
|
|
// If a synchronizing marker was placed into the history, look for it
|
|
// and adjust the scroll count.
|
|
int markerRow = findSyncMarker();
|
|
if (markerRow == -1) {
|
|
// Something has happened. Reset the terminal.
|
|
trace("Sync marker has disappeared -- resetting the terminal"
|
|
" (m_syncCounter=%d)",
|
|
m_syncCounter);
|
|
resetConsoleTracking();
|
|
} else if (markerRow != m_syncRow) {
|
|
ASSERT(markerRow < m_syncRow);
|
|
m_scrolledCount += (m_syncRow - markerRow);
|
|
m_syncRow = markerRow;
|
|
// If the buffer has scrolled, then the entire window is dirty.
|
|
markEntireWindowDirty();
|
|
}
|
|
}
|
|
|
|
// Update the dirty line count:
|
|
// - If the window has moved, the entire window is dirty.
|
|
// - Everything up to the cursor is dirty.
|
|
// - All lines above the window are dirty.
|
|
// - Any non-blank lines are dirty.
|
|
if (m_dirtyWindowTop != -1) {
|
|
if (windowRect.top() > m_dirtyWindowTop) {
|
|
// The window has moved down, presumably as a result of scrolling.
|
|
markEntireWindowDirty();
|
|
} else if (windowRect.top() < m_dirtyWindowTop) {
|
|
// The window has moved upward. This is generally not expected to
|
|
// happen, but the CMD/PowerShell CLS command will move the window
|
|
// to the top as part of clearing everything else in the console.
|
|
trace("Window moved upward -- resetting the terminal"
|
|
" (m_syncCounter=%d)",
|
|
m_syncCounter);
|
|
resetConsoleTracking();
|
|
}
|
|
}
|
|
m_dirtyWindowTop = windowRect.top();
|
|
m_dirtyLineCount = std::max(m_dirtyLineCount, cursor.Y + 1);
|
|
m_dirtyLineCount = std::max(m_dirtyLineCount, (int)windowRect.top());
|
|
scanForDirtyLines();
|
|
|
|
// Note that it's possible for all the lines on the current window to
|
|
// be non-dirty.
|
|
|
|
int firstLine = std::min(m_scrapedLineCount,
|
|
windowRect.top() + m_scrolledCount);
|
|
int stopLine = std::min(m_dirtyLineCount,
|
|
windowRect.top() + windowRect.height()) +
|
|
m_scrolledCount;
|
|
|
|
bool sawModifiedLine = false;
|
|
|
|
for (int line = firstLine; line < stopLine; ++line) {
|
|
CHAR_INFO curLine[MAX_CONSOLE_WIDTH]; // TODO: bufoverflow
|
|
const int w = windowRect.width();
|
|
m_console->read(SmallRect(0, line - m_scrolledCount, w, 1), curLine);
|
|
|
|
// TODO: The memcpy can overflow the m_bufferData buffer.
|
|
CHAR_INFO (&bufLine)[MAX_CONSOLE_WIDTH] =
|
|
m_bufferData[line % BUFFER_LINE_COUNT];
|
|
if (sawModifiedLine ||
|
|
line > m_maxBufferedLine ||
|
|
memcmp(curLine, bufLine, sizeof(CHAR_INFO) * w) != 0) {
|
|
//trace("sent line %d", line);
|
|
m_terminal->sendLine(line, curLine, windowRect.width());
|
|
memset(bufLine, 0, sizeof(bufLine));
|
|
memcpy(bufLine, curLine, sizeof(CHAR_INFO) * w);
|
|
for (int col = w; col < MAX_CONSOLE_WIDTH; ++col) {
|
|
bufLine[col].Attributes = curLine[w - 1].Attributes;
|
|
bufLine[col].Char.UnicodeChar = L' ';
|
|
}
|
|
m_maxBufferedLine = std::max(m_maxBufferedLine, line);
|
|
sawModifiedLine = true;
|
|
}
|
|
}
|
|
|
|
m_scrapedLineCount = windowRect.top() + m_scrolledCount;
|
|
|
|
// Creating a new sync row requires clearing part of the console buffer, so
|
|
// avoid doing it if there's already a sync row that's good enough.
|
|
// TODO: replace hard-coded constants
|
|
const int newSyncRow = static_cast<int>(windowRect.top()) - 200;
|
|
if (newSyncRow >= 1 && newSyncRow >= m_syncRow + 200) {
|
|
createSyncMarker(newSyncRow);
|
|
}
|
|
|
|
m_terminal->finishOutput(std::pair<int, int>(cursor.X,
|
|
cursor.Y + m_scrolledCount));
|
|
|
|
unfreezeConsole();
|
|
}
|
|
|
|
void Agent::freezeConsole()
|
|
{
|
|
SendMessage(m_console->hwnd(), WM_SYSCOMMAND, SC_CONSOLE_SELECT_ALL, 0);
|
|
}
|
|
|
|
void Agent::unfreezeConsole()
|
|
{
|
|
SendMessage(m_console->hwnd(), WM_CHAR, 27, 0x00010001);
|
|
}
|
|
|
|
void Agent::syncMarkerText(CHAR_INFO *output)
|
|
{
|
|
char str[SYNC_MARKER_LEN + 1];// TODO: use a random string
|
|
sprintf(str, "S*Y*N*C*%08x", m_syncCounter);
|
|
memset(output, 0, sizeof(CHAR_INFO) * SYNC_MARKER_LEN);
|
|
for (int i = 0; i < SYNC_MARKER_LEN; ++i) {
|
|
output[i].Char.UnicodeChar = str[i];
|
|
output[i].Attributes = 7;
|
|
}
|
|
}
|
|
|
|
int Agent::findSyncMarker()
|
|
{
|
|
ASSERT(m_syncRow >= 0);
|
|
CHAR_INFO marker[SYNC_MARKER_LEN];
|
|
CHAR_INFO column[BUFFER_LINE_COUNT];
|
|
syncMarkerText(marker);
|
|
SmallRect rect(0, 0, 1, m_syncRow + SYNC_MARKER_LEN);
|
|
m_console->read(rect, column);
|
|
int i;
|
|
for (i = m_syncRow; i >= 0; --i) {
|
|
int j;
|
|
for (j = 0; j < SYNC_MARKER_LEN; ++j) {
|
|
if (column[i + j].Char.UnicodeChar != marker[j].Char.UnicodeChar)
|
|
break;
|
|
}
|
|
if (j == SYNC_MARKER_LEN)
|
|
return i;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
void Agent::createSyncMarker(int row)
|
|
{
|
|
ASSERT(row >= 1);
|
|
|
|
// Clear the lines around the marker to ensure that Windows 10's rewrapping
|
|
// does not affect the marker.
|
|
m_console->clearLines(row - 1, SYNC_MARKER_LEN + 1);
|
|
|
|
// Write a new marker.
|
|
m_syncCounter++;
|
|
CHAR_INFO marker[SYNC_MARKER_LEN];
|
|
syncMarkerText(marker);
|
|
m_syncRow = row;
|
|
SmallRect markerRect(0, m_syncRow, 1, SYNC_MARKER_LEN);
|
|
m_console->write(markerRect, marker);
|
|
}
|