/* * This file is part of Red Panda C++ * Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com) * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <https://www.gnu.org/licenses/>. */ #include <string> using std::string; #include <stdio.h> #include <windows.h> #include <psapi.h> #include <processthreadsapi.h> #include <conio.h> #ifndef WINBOOL #define WINBOOL BOOL #endif #ifndef ENABLE_VIRTUAL_TERMINAL_PROCESSING #define ENABLE_VIRTUAL_TERMINAL_PROCESSING 0x0004 #endif #ifndef ENABLE_PROCESSED_OUTPUT #define ENABLE_PROCESSED_OUTPUT 0x0001 #endif #define MAX_COMMAND_LENGTH 32768 #define MAX_ERROR_LENGTH 2048 enum RunProgramFlag { RPF_PAUSE_CONSOLE = 0x0001, RPF_REDIRECT_INPUT = 0x0002, RPF_ENABLE_VIRTUAL_TERMINAL_PROCESSING = 0x0004 }; HANDLE hJob; LONGLONG GetClockTick() { LARGE_INTEGER dummy; QueryPerformanceCounter(&dummy); return dummy.QuadPart; } LONGLONG GetClockFrequency() { LARGE_INTEGER dummy; QueryPerformanceFrequency(&dummy); return dummy.QuadPart; } string GetErrorMessage() { string result(MAX_ERROR_LENGTH,0); FormatMessageA( FORMAT_MESSAGE_FROM_SYSTEM|FORMAT_MESSAGE_IGNORE_INSERTS, NULL,GetLastError(),MAKELANGID(LANG_NEUTRAL,SUBLANG_DEFAULT),&result[0],result.size(),NULL); // Clear newlines at end of string for(int i = result.length()-1;i >= 0;i--) { if(isspace(result[i])) { result[i] = 0; } else { break; } } return result; } void PauseExit(int exitcode, bool reInp) { HANDLE hInp=NULL; if (reInp) { SECURITY_ATTRIBUTES sa; sa.nLength = sizeof(sa); sa.lpSecurityDescriptor = NULL; sa.bInheritHandle = TRUE; HANDLE hInp = CreateFileA("CONIN$", GENERIC_WRITE | GENERIC_READ, FILE_SHARE_READ , &sa, OPEN_EXISTING, /*FILE_ATTRIBUTE_NORMAL*/0, NULL); //si.hStdInput = hInp; SetStdHandle(STD_INPUT_HANDLE,hInp); freopen("CONIN$","r",stdin); } FlushConsoleInputBuffer(GetStdHandle(STD_INPUT_HANDLE)); fflush(stdin); printf("\n"); printf("Press ANY key to exit..."); getch(); if (reInp) { CloseHandle(hInp); } exit(exitcode); } string GetCommand(int argc,char** argv,bool &reInp,bool &pauseAfterExit, bool &enableVisualTerminalSeq) { string result; int flags = atoi(argv[1]); reInp = flags & RPF_REDIRECT_INPUT; pauseAfterExit = flags & RPF_PAUSE_CONSOLE; enableVisualTerminalSeq = flags & RPF_ENABLE_VIRTUAL_TERMINAL_PROCESSING; for(int i = 3;i < argc;i++) { // Quote the argument in case the path name contains spaces result += string("\"") + string(argv[i]) + string("\""); // Add a space except for the last argument if(i != (argc-1)) { result += string(" "); } } if(result.length() > MAX_COMMAND_LENGTH) { printf("\n--------------------------------"); printf("\nError: Length of command line string is over %d characters\n",MAX_COMMAND_LENGTH); PauseExit(EXIT_FAILURE,reInp); } return result; } DWORD ExecuteCommand(string& command,bool reInp, LONGLONG &peakMemory, LONGLONG &execTime) { STARTUPINFOA si; PROCESS_INFORMATION pi; memset(&si,0,sizeof(si)); si.cb = sizeof(si); memset(&pi,0,sizeof(pi)); DWORD dwCreationFlags = CREATE_BREAKAWAY_FROM_JOB; if(!CreateProcessA(NULL, (LPSTR)command.c_str(), NULL, NULL, true, dwCreationFlags, NULL, NULL, &si, &pi)) { printf("\n--------------------------------"); printf("\nFailed to execute \"%s\":",command.c_str()); printf("\nError %lu: %s\n",GetLastError(),GetErrorMessage().c_str()); PauseExit(EXIT_FAILURE,reInp); } WINBOOL bSuccess = AssignProcessToJobObject( hJob, pi.hProcess ); if ( bSuccess == FALSE ) { printf( "AssignProcessToJobObject failed: error %lu\n", GetLastError() ); return 0; } WaitForSingleObject(pi.hProcess, INFINITE); // Wait for it to finish peakMemory = 0; PROCESS_MEMORY_COUNTERS counter; counter.cb = sizeof(counter); if (GetProcessMemoryInfo(pi.hProcess,&counter, sizeof(counter))){ peakMemory = counter.PeakWorkingSetSize/1024; } FILETIME creationTime; FILETIME exitTime; FILETIME kernelTime; FILETIME userTime; execTime=0; if (GetProcessTimes(pi.hProcess,&creationTime,&exitTime,&kernelTime,&userTime)) { execTime=((LONGLONG)kernelTime.dwHighDateTime<<32) +((LONGLONG)userTime.dwHighDateTime<<32) +(kernelTime.dwLowDateTime)+(userTime.dwLowDateTime); } DWORD result = 0; GetExitCodeProcess(pi.hProcess, &result); return result; } void EnableVtSequence() { DWORD mode; HANDLE hConsole = GetStdHandle(STD_OUTPUT_HANDLE); if (GetConsoleMode(hConsole, &mode)) SetConsoleMode(hConsole, mode | ENABLE_VIRTUAL_TERMINAL_PROCESSING | ENABLE_PROCESSED_OUTPUT); hConsole = GetStdHandle(STD_ERROR_HANDLE); if (GetConsoleMode(hConsole, &mode)) SetConsoleMode(hConsole, mode | ENABLE_VIRTUAL_TERMINAL_PROCESSING | ENABLE_PROCESSED_OUTPUT); } int main(int argc, char** argv) { const char *sharedMemoryId; // First make sure we aren't going to read nonexistent arrays if(argc < 4) { printf("\n--------------------------------"); printf("\nUsage: consolepauser.exe <0|1> <shared_memory_id> <filename> <parameters>\n"); printf("\n 1 means the STDIN is redirected by Red Panda C++; 0 means not\n"); PauseExit(EXIT_SUCCESS,false); } // Make us look like the paused program SetConsoleTitleA(argv[3]); sharedMemoryId = argv[2]; SECURITY_ATTRIBUTES sa; sa.nLength = sizeof(sa); sa.lpSecurityDescriptor = NULL; sa.bInheritHandle = FALSE; hJob= CreateJobObject( &sa, NULL ); if ( hJob == NULL ) { printf( "CreateJobObject failed: error %lu\n", GetLastError() ); return 0; } JOBOBJECT_EXTENDED_LIMIT_INFORMATION info; memset(&info,0,sizeof(JOBOBJECT_EXTENDED_LIMIT_INFORMATION)); info.BasicLimitInformation.LimitFlags = JOB_OBJECT_LIMIT_KILL_ON_JOB_CLOSE; WINBOOL bSuccess = SetInformationJobObject( hJob, JobObjectExtendedLimitInformation, &info, sizeof( info ) ); if ( bSuccess == FALSE ) { printf( "SetInformationJobObject failed: error %lu\n", GetLastError() ); return 0; } bool reInp; bool pauseAfterExit; bool enableVisualTerminalSeq; // Then build the to-run application command string command = GetCommand(argc,argv,reInp, pauseAfterExit, enableVisualTerminalSeq); HANDLE hOutput = NULL; if (reInp) { SECURITY_ATTRIBUTES sa; sa.nLength = sizeof(sa); sa.lpSecurityDescriptor = NULL; sa.bInheritHandle = TRUE; hOutput = CreateFileA("CONOUT$", GENERIC_WRITE | GENERIC_READ, FILE_SHARE_WRITE , &sa, OPEN_EXISTING, /*FILE_ATTRIBUTE_NORMAL*/0, NULL); SetStdHandle(STD_OUTPUT_HANDLE, hOutput); SetStdHandle(STD_ERROR_HANDLE, hOutput); freopen("CONOUT$","w+",stdout); freopen("CONOUT$","w+",stderr); } else { FlushConsoleInputBuffer(GetStdHandle(STD_INPUT_HANDLE)); } if (enableVisualTerminalSeq) { EnableVtSequence(); } HANDLE hSharedMemory=INVALID_HANDLE_VALUE; int BUF_SIZE=1024; char* pBuf=nullptr; hSharedMemory = OpenFileMappingA( FILE_MAP_ALL_ACCESS, FALSE, sharedMemoryId ); if (hSharedMemory != NULL) { pBuf = (char*) MapViewOfFile(hSharedMemory, // handle to map object FILE_MAP_ALL_ACCESS, // read/write permission 0, 0, BUF_SIZE); // } else { // printf("can't open shared memory!\n"); } // Save starting timestamp LONGLONG starttime = GetClockTick(); LONGLONG peakMemory=0; LONGLONG execTime=0; // Then execute said command DWORD returnvalue = ExecuteCommand(command,reInp,peakMemory,execTime); // Get ending timestamp LONGLONG endtime = GetClockTick(); double seconds = (endtime - starttime) / (double)GetClockFrequency(); double execSeconds = (double)execTime/10000; if (pBuf) { strcpy(pBuf,"FINISHED"); UnmapViewOfFile(pBuf); } if (hSharedMemory != NULL && hSharedMemory!=INVALID_HANDLE_VALUE) { CloseHandle(hSharedMemory); } // Done? Print return value of executed program printf("\n--------------------------------"); printf("\nProcess exited after %.4g seconds with return value %lu (%.4g ms cpu time, %lld KB mem used).\n",seconds,returnvalue, execSeconds, peakMemory); if (pauseAfterExit) PauseExit(returnvalue,reInp); return 0; }