/* * w9xpopen.c * * Serves as an intermediate stub Win32 console application to * avoid a hanging pipe when redirecting 16-bit console based * programs (including MS-DOS console based programs and batch * files) on Window 95 and Windows 98. * * This program is to be launched with redirected standard * handles. It will launch the command line specified 16-bit * console based application in the same console, forwarding * its own redirected standard handles to the 16-bit child. * AKA solution to the problem described in KB: Q150956. */ #define WIN32_LEAN_AND_MEAN #include <windows.h> #include <stdio.h> #include <stdlib.h> /* for malloc and its friends */ const char *usage = "This program is used by Python's os.popen function\n" "to work around a limitation in Windows 95/98. It is\n" "not designed to be used as a stand-alone program."; int main(int argc, char *argv[]) { BOOL bRet; STARTUPINFO si; PROCESS_INFORMATION pi; DWORD exit_code=0; size_t cmdlen = 0; int i; char *cmdline, *cmdlinefill; if (argc < 2) { if (GetFileType(GetStdHandle(STD_INPUT_HANDLE))==FILE_TYPE_CHAR) /* Attached to a console, and therefore not executed by Python Display a message box for the inquisitive user */ MessageBox(NULL, usage, argv[0], MB_OK); else { /* Eeek - executed by Python, but args are screwed! Write an error message to stdout so there is at least some clue for the end user when it appears in their output. A message box would be hidden and blocks the app. */ fprintf(stdout, "Internal popen error - no args specified\n%s\n", usage); } return 1; } /* Build up the command-line from the args. Args with a space are quoted, existing quotes are escaped. To keep things simple calculating the buffer size, we assume every character is a quote - ie, we allocate double what we need in the worst case. As this is only double the command line passed to us, there is a good chance this is reasonably small, so the total allocation will almost always be < 512 bytes. */ for (i=1;i<argc;i++) cmdlen += strlen(argv[i])*2 + 3; /* one space, maybe 2 quotes */ cmdline = cmdlinefill = (char *)malloc(cmdlen+1); if (cmdline == NULL) return -1; for (i=1;i<argc;i++) { const char *arglook; int bQuote = strchr(argv[i], ' ') != NULL; if (bQuote) *cmdlinefill++ = '"'; /* escape quotes */ for (arglook=argv[i];*arglook;arglook++) { if (*arglook=='"') *cmdlinefill++ = '\\'; *cmdlinefill++ = *arglook; } if (bQuote) *cmdlinefill++ = '"'; *cmdlinefill++ = ' '; } *cmdlinefill = '\0'; /* Make child process use this app's standard files. */ ZeroMemory(&si, sizeof si); si.cb = sizeof si; si.dwFlags = STARTF_USESTDHANDLES; si.hStdInput = GetStdHandle(STD_INPUT_HANDLE); si.hStdOutput = GetStdHandle(STD_OUTPUT_HANDLE); si.hStdError = GetStdHandle(STD_ERROR_HANDLE); bRet = CreateProcess( NULL, cmdline, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi ); free(cmdline); if (bRet) { if (WaitForSingleObject(pi.hProcess, INFINITE) != WAIT_FAILED) { GetExitCodeProcess(pi.hProcess, &exit_code); } CloseHandle(pi.hProcess); CloseHandle(pi.hThread); return exit_code; } return 1; }