ERROR_INVALID_HANDLE при вызове ConnectNamedPipe

Я пытаюсь реализовать систему, в которой произвольное количество процессов будет общаться друг с другом через именованные каналы. Реализация на C++ в Windows. Однако я застрял, потому что процессы, похоже, не могут получать сообщения. Похожий вопрос не нашел, извиняюсь, если уже задавался. Во всяком случае, вот код, связанный с созданием дескриптора. Идея состоит в том, что у меня есть n (n = приблизительно 5 или 6) процессов, и у каждого процесса есть именованный канал для записи (pipesIn) и чтения (pipesOut).

pipesIn = std::vector<HANDLE *>();
    pipesOut = std::vector<HANDLE *>();
    this->name = name;
    wchar_t pipeName[20];
    for (int i = 0; i < total; i++) {
        if (i != name) {
            HANDLE hIn;
            swprintf(pipeName, L"\\\\.\\pipe\\Pipe(%d,%d)", name, i);
            do {
                hIn = CreateNamedPipe(pipeName,
                    PIPE_ACCESS_DUPLEX | PIPE_TYPE_BYTE | PIPE_READMODE_BYTE,   // FILE_FLAG_FIRST_PIPE_INSTANCE is not needed but forces CreateNamedPipe(..) to fail if the pipe already exists...
                    PIPE_WAIT,
                    1,
                    1024 * 16,
                    1024 * 16,
                    NMPWAIT_USE_DEFAULT_WAIT,
                    NULL);
            } while (hIn == INVALID_HANDLE_VALUE);
            pipesIn.push_back(&hIn);
        }
    }

    for (int i = 0; i < total; i++) {
        if (i != name) {
            HANDLE hOut;
            swprintf(pipeName, L"\\\\.\\pipe\\Pipe(%d,%d)", i, name);
            do {
                hOut = CreateFile(pipeName,
                    GENERIC_READ | GENERIC_WRITE,
                    0,
                    NULL,
                    OPEN_EXISTING,
                    0,
                    NULL);
            } while (hOut == INVALID_HANDLE_VALUE);
            pipesOut.push_back(&hOut);
        }
        else {
            pipesIn.push_back(NULL);
            pipesOut.push_back(NULL);
        }
    }

У меня проблема в моей функции приема. Когда я вызываю ConnectNamedPipe, результат равен 0, а GetLastError() возвращает ERROR_INVALID_HANDLE.

char response[20];
    DWORD dwRead;

    bool recieved = false;
    while (!recieved) {
        bool flag = false;
        for (int i = 0; i < pipesIn.size(); i++) {
            if (name == i) {
                continue;
            }
            if (ConnectNamedPipe(pipesIn.at(i), NULL) != FALSE) {
                while (ReadFile(*(pipesIn.at(i)), response, sizeof(response) - 1, &dwRead, NULL) != FALSE)
                {
                    response[dwRead] = '\0';
                    recieved = true;
                    break;
                }
            }
            else {
                printf("%d\n", GetLastError());
            }
        }
    }

Поскольку я новичок в C++ и использую каналы, для меня это не имеет особого смысла. Я пытался воспроизвести метод связи, приведенный в качестве примера здесь. Любые советы будут считаться полезными.


person Zlatan Sičanica    schedule 29.05.2016    source источник


Ответы (1)


Вы выстрелили себе в ногу, излишне усложняя ситуацию.

        pipesIn.push_back(&hIn);

Это твоя проблема. hIn — это локальная переменная стека, которая исчезает в конце области видимости, но вы берете адрес этого места в стеке и сохраняете его.

Решение для вас состоит в том, чтобы просто сделать ваши векторы std::vector<HANDLE>; HANDLE — это в первую очередь очень маленькие объекты, такого же размера, как указатель, поэтому вы не делаете себе никаких одолжений, беря их по адресу.

person kfsone    schedule 29.05.2016