2013-07-10 80 views
0

(請在Z卡,如果你想,它會減輕情緒)發送命令來管在C++

我很遠了這一新的項目我的安樂窩,我決定潛入,在至少與其中的一部分。

整個項目將是一個可以加載到TeamSpeak 3中的DLL,並允許人們(通過一小組命令)來控制Pianobar(一個Pandora命令行播放器)。

這裏的答案引導我足以讓Pianobar(一個控制檯應用程序)https://stackoverflow.com/a/17502224/1733365啓動並運行,我可以獲得它的STDOUT並顯示直到它顯示歌曲當前時間的位置,以及它接受用戶的位置輸入。整個過程鎖定在那一點,我猜測,因爲ReadFromPipe()命令認爲有更多的讀取,因爲該行不斷刷新。

爲了讓我能夠從外部線程調用它,我還採取了一個重寫WriteToPipe(void *)的初始WriteToPipe(void)的方法。 (聽特定命令的TeamSpeak 3服務器聊天的人。)

現在我的代碼是一個巨大的混亂,但我把它清理了一下,所以希望有人能幫助我理解。

真的這只是一個夏天的項目,我決定嘗試在我失學的時候,以及我第一次創建DLL的經驗。

Pianobar for Windows

遠低於從Creating a Child Process with Redirected Input and Output

#include "pianobar.h" 
//#include <windows.h> 
//#include <tchar.h> 
//#include <stdio.h> 
#include <strsafe.h> 
//#include <stdlib.h> 
//#include <sys/types.h> 
//#include <string.h> 

#define BUFSIZE 4096 

HANDLE g_hChildStd_IN_Rd = NULL; 
HANDLE g_hChildStd_IN_Wr = NULL; 
HANDLE g_hChildStd_OUT_Rd = NULL; 
HANDLE g_hChildStd_OUT_Wr = NULL; 

HANDLE g_hInputFile = NULL; 
PROCESS_INFORMATION piProcInfo; 
STARTUPINFO siStartInfo; 
SECURITY_ATTRIBUTES saAttr; 

void CreateChildProcess(void); 
void WriteToPipe(char *command); 
void ReadFromPipe(void); 
void ErrorExit(PTSTR); 

int pianobar (struct TS3Functions ts3Functions) { 
    int iFound = 0; 

    printf("\n->Start of parent execution.\n"); 

    // Set the bInheritHandle flag so pipe handles are inherited. 

    saAttr.nLength = sizeof(SECURITY_ATTRIBUTES); 
    saAttr.bInheritHandle = TRUE; 
    saAttr.lpSecurityDescriptor = NULL; 

    // Create a pipe for the child process's STDOUT. 

    if (! CreatePipe(&g_hChildStd_OUT_Rd, &g_hChildStd_OUT_Wr, &saAttr, 0)) 
     ErrorExit(TEXT("StdoutRd CreatePipe")); 

    // Ensure the read handle to the pipe for STDOUT is not inherited. 

    if (! SetHandleInformation(g_hChildStd_OUT_Rd, HANDLE_FLAG_INHERIT, 0)) 
     ErrorExit(TEXT("Stdout SetHandleInformation")); 

    // Create a pipe for the child process's STDIN. 

    if (! CreatePipe(&g_hChildStd_IN_Rd, &g_hChildStd_IN_Wr, &saAttr, 0)) 
     ErrorExit(TEXT("Stdin CreatePipe")); 

    // Ensure the write handle to the pipe for STDIN is not inherited. 

    if (! SetHandleInformation(g_hChildStd_IN_Wr, HANDLE_FLAG_INHERIT, 0)) 
     ErrorExit(TEXT("Stdin SetHandleInformation")); 

    // Create the child process. 

    CreateChildProcess(); 

    // Write to the pipe that is the standard input for a child process. 
    // Data is written to the pipe's buffers, so it is not necessary to wait 
    // until the child process is running before writing data. 

    // This should cause a help menu to be displayed on the next ReadFromPipe() 
    // However, ReadFromPipe() doesn't show help commands 
    //WriteToPipe("?\r\n"); 

    // Read from pipe that is the standard output for child process. 
    // Reading causes a lock. 
    //ReadFromPipe(); 


    printf("\n->End of parent execution.\n"); 
    printf("\n->Pianobar started.\n"); 
    iFound = 1; 
    return iFound; 
} 

void CloseChildProcess() { 
    //CloseHandle(piProcInfo.hProcess); 
    CloseHandle(piProcInfo.hThread); 
    TerminateProcess(piProcInfo.hProcess,0); 
} 

void CreateChildProcess() 
    // Create a child process that uses the previously created pipes for STDIN and STDOUT. 
{ 
    TCHAR szCmdline[]=TEXT("c:\\pianobar\\pianobar.exe"); 
    BOOL bSuccess = FALSE; 

    // Set up members of the PROCESS_INFORMATION structure. 

    ZeroMemory(&piProcInfo, sizeof(PROCESS_INFORMATION)); 

    // Set up members of the STARTUPINFO structure. 
    // This structure specifies the STDIN and STDOUT handles for redirection. 
    ZeroMemory(&siStartInfo, sizeof(STARTUPINFO)); 
    siStartInfo.cb = sizeof(STARTUPINFO); 
    siStartInfo.hStdError = g_hChildStd_OUT_Wr; 
    siStartInfo.hStdOutput = g_hChildStd_OUT_Wr; 
    siStartInfo.hStdInput = g_hChildStd_IN_Rd; 
    siStartInfo.dwFlags |= STARTF_USESTDHANDLES; 

    // Create the child process. 

    bSuccess = CreateProcess(NULL, 
     szCmdline,  // command line 
     NULL,   // process security attributes 
     NULL,   // primary thread security attributes 
     TRUE,   // handles are inherited 
     0,    // creation flags 
     NULL,   // use parent's environment 
     TEXT("c:\\pianobar\\"),   // use parent's current directory 
     &siStartInfo, // STARTUPINFO pointer 
     &piProcInfo); // receives PROCESS_INFORMATION 

    // If an error occurs, exit the application. 
    if (! bSuccess) 
     ErrorExit(TEXT("CreateProcess")); 
    else 
    { 
     // Close handles to the child process and its primary thread. 
     // Some applications might keep these handles to monitor the status 
     // of the child process, for example. 

     // I think I need these while I'm running... 
     //CloseHandle(piProcInfo.hProcess); 
     //CloseHandle(piProcInfo.hThread); 
    } 
} 

void WriteToPipe(char *command) 

    // Read from a file and write its contents to the pipe for the child's STDIN. 
    // Stop when there is no more data. 
{ 

    DWORD dwRead, dwWritten; 
    DWORD dw; 
    CHAR chBuf[BUFSIZE]; 
    BOOL bSuccess = FALSE; 
    LPTSTR lpTStr; 

    printf("\n-> In WriteToPipe()\n"); 
    bSuccess = WriteFile(g_hChildStd_IN_Wr, command, sizeof(command), &dwWritten, NULL); 
     if(bSuccess) { 
      printf("bSuccess was TRUE\n->Sent: "); 
      printf(command); 
     } else { 
      printf("bSuccess was FALSE\n"); 
     } 

     // Close the pipe handle so the child process stops reading. 
     // my 2nd call to WriteToPipe results in a "The handle is invalid" error 
     if (! CloseHandle(g_hChildStd_IN_Wr)) { 

     dw = GetLastError(); 
     FormatMessage(
      FORMAT_MESSAGE_ALLOCATE_BUFFER | 
      FORMAT_MESSAGE_FROM_SYSTEM | 
      FORMAT_MESSAGE_IGNORE_INSERTS, 
      NULL, 
      dw, 
      MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), 
      (LPTSTR) &lpTStr, 
      0, NULL); 
      printf(lpTStr); 
     } 
     if(command == "q\r\n") { 
      printf("Quit received.\n"); 
      // this should have killed the process if it was received correctly... 
      CloseChildProcess(); 
     } 
} 

void ReadFromPipe(void) 
    // Read output from the child process's pipe for STDOUT 
    // and write to the parent process's pipe for STDOUT. 
    // Stop when there is no more data. 
{ 
    DWORD dwRead, dwWritten; 
    CHAR chBuf[BUFSIZE]; 
    BOOL bSuccess = FALSE; 
    HANDLE hParentStdOut = GetStdHandle(STD_OUTPUT_HANDLE); 

    printf("\n-> In ReadFromPipe()\n"); 
    for (;;) 
    { 
     bSuccess = ReadFile(g_hChildStd_OUT_Rd, chBuf, BUFSIZE, &dwRead, NULL); 
     if(! bSuccess || dwRead == 0) break; 
     printf("In ReadFromPipe loop\n"); 
     bSuccess = WriteFile(hParentStdOut, chBuf, 
      dwRead, &dwWritten, NULL); 
     if (! bSuccess) { 
      // we never get to this, it just waits... 
      printf("Leaving loop\n"); 
      break; 
     } 
    } 
} 

void ErrorExit(PTSTR lpszFunction) 

    // Format a readable error message, display a message box, 
    // and exit from the application. 
{ 
    LPVOID lpMsgBuf; 
    LPVOID lpDisplayBuf; 
    DWORD dw = GetLastError(); 

    FormatMessage(
     FORMAT_MESSAGE_ALLOCATE_BUFFER | 
     FORMAT_MESSAGE_FROM_SYSTEM | 
     FORMAT_MESSAGE_IGNORE_INSERTS, 
     NULL, 
     dw, 
     MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), 
     (LPTSTR) &lpMsgBuf, 
     0, NULL); 

    lpDisplayBuf = (LPVOID)LocalAlloc(LMEM_ZEROINIT, 
     (lstrlen((LPCTSTR)lpMsgBuf)+lstrlen((LPCTSTR)lpszFunction)+40)*sizeof(TCHAR)); 
    StringCchPrintf((LPTSTR)lpDisplayBuf, 
     LocalSize(lpDisplayBuf)/sizeof(TCHAR), 
     TEXT("%s failed with error %d: %s"), 
     lpszFunction, dw, lpMsgBuf); 
    MessageBox(NULL, (LPCTSTR)lpDisplayBuf, TEXT("Error"), MB_OK); 

    LocalFree(lpMsgBuf); 
    LocalFree(lpDisplayBuf); 
    ExitProcess(1); 
} 

回答

1

我拍攝並沒有真正理解你的設置代碼,但對於這樣:

...也它接受用戶輸入。整個過程鎖定在 點,我猜測,因爲ReadFromPipe()命令認爲有更多的 閱讀,因爲該行不斷刷新。

這很可能。如果沒有什麼要從管道讀取,那麼你的進程阻塞,即卡住ReadFile()調用。如果只有在需要讀取內容時才需要讀取,則需要異步I/O或通知機制。我並不真正瞭解Windows,但看起來好像是IO的完成端口(IOCP)和異步回調函數。或許這些鏈接可以幫助:

What is the best epoll/kqueue/select equvalient on Windows?

IOCP and ReadFileEx usage

+0

大部分什麼在有相當類似微軟提供的代碼示例,但是,是的,它不是100%清楚。我希望有另一個應用程序啓動Pianobar控制檯應用程序,並且有一些監視聊天窗口(這發生在其他地方並且有詳細記錄),以便將命令發送回Pianobar控制檯應用程序。我想我現在可以跳過標準輸出,我似乎無法正確輸入以重新定向到推出的Pianobar應用程序的STDIN。 –

+0

另外,謝謝。我對這些東西的術語知識有限,這絕對阻礙了我尋找可能答案的能力。異步I/O向我展示了Boost,在我的研究中我讀過的某些地方是用來處理通過管道發送東西的,但我不知道爲什麼。 –