仅在使用消息模式时,GetLastError()在调用PeekNamedPipe后返回ERROR_BROKEN_PIPE

时间:2011-11-14 20:20:18

标签: c++ winapi named-pipes

我无法弄清楚为什么会发生这种情况......我有一个命名管道服务器和客户端应用程序。两者都处于读/写模式并在彼此之间传输数据。服务器有两个线程,一个从管道读取,另一个写入管道。当客户端正在编写一堆消息时,服务器的读取线程将退出,因为它对PeekNamedPipe的调用返回false。 GetLastError()的返回值是ERROR_BROKEN_PIPE。服务器中的任何线程都没有关闭管道,客户端仍在写入管道,所以我不明白为什么管道“坏了”。

如果我将服务器更改为BYTE模式,那么一切都可以正常运行。我真的很想使用消息模式,所以我的“消息”不能组合在一起。

如果我将客户端更改为BYTE模式,将服务器更改为消息模式,则可以正常工作。

调用CreateNamedPipe

hPipe = CreateNamedPipe(
    pszPipeName,
    PIPE_ACCESS_DUPLEX | FILE_FLAG_OVERLAPPED,
    PIPE_WAIT | PIPE_READMODE_MESSAGE | PIPE_TYPE_MESSAGE,
    PIPE_UNLIMITED_INSTANCES,
    dwOutBufferSize,
    dwInBufferSize,
    0,
    NULL);

致电PeekNamedPipe

while( RunningState == DDCMP_STATE_RUNNING )
{
    if( !PeekNamedPipe(hPipe,NULL,NULL,NULL,&dwBytesAvailable,NULL) || !dwBytesAvailable )
        if( GetLastError() == ERROR_BROKEN_PIPE || GetLastError() == ERROR_PIPE_NOT_CONNECTED || GetLastError() == ERROR_INVALID_HANDLE )
            break;
        else
        {
            Sleep( 100 );
            continue;
        }

    //call to ReadFile(hPipe,...) with dwBytesAvailable as size, and then processing of data
}

1 个答案:

答案 0 :(得分:2)

您不仅在GetLastError()失败时调用PeekNamedPipe(),而且在成功时返回0字节。在后一种情况下,您不应该调用GetLastError(),因为该值没有意义。使用它更像是这样:

while( RunningState == DDCMP_STATE_RUNNING ) 
{ 
    if( !PeekNamedPipe(hPipe, NULL, NULL, NULL, &dwBytesAvailable, NULL) )
    {
        DWORD dwError = GetLastError();

        if( (dwError == ERROR_BROKEN_PIPE) ||
            (dwError == ERROR_PIPE_NOT_CONNECTED) ||
            (dwError == ERROR_INVALID_HANDLE) )
        { 
            break;
        }

        dwBytesAvailable = 0;
    }

    if( !dwBytesAvailable )
    {
        Sleep(100);
        continue;
    }

    //call to ReadFile(hPipe,...) with dwBytesAvailable as size, and then processing of data 
}