Подтвердить что ты не робот

Как reset сокет вернуться в режим блокировки (после того, как я установил его в неблокирующий режим)?

Я прочитал это относительно установки сокета в неблокирующий режим.

http://www.gnu.org/software/libc/manual/html_mono/libc.html#File-Status-Flags

Вот что я сделал:

static void setnonblocking(int sock)
{
    int opts;

    opts = fcntl(sock,F_GETFL);
    if (opts < 0) {
        perror("fcntl(F_GETFL)");
        exit(EXIT_FAILURE);
    }
    opts = (opts | O_NONBLOCK);
    if (fcntl(sock,F_SETFL,opts) < 0) {
        perror("fcntl(F_SETFL)");
        exit(EXIT_FAILURE);
    }
    return;
}

Как установить сокет обратно в режим блокировки? Я не вижу флаг O_BLOCK?

Спасибо.

4b9b3361

Ответ 1

Вы пытались очистить флаг O_NONBLOCK?

opts = opts & (~O_NONBLOCK)

Ответ 2

Вот более кроссплатформенное решение:

bool set_blocking_mode(int socket, bool is_blocking)
{
    bool ret = true;

#ifdef WIN32
    /// @note windows sockets are created in blocking mode by default
    // currently on windows, there is no easy way to obtain the socket current blocking mode since WSAIsBlocking was deprecated
    u_long non_blocking = is_blocking ? 0 : 1;
    ret = NO_ERROR == ioctlsocket(socket, FIONBIO, &non_blocking);
#else
    const int flags = fcntl(socket, F_GETFL, 0);
    if ((flags & O_NONBLOCK) && !is_blocking) { info("set_blocking_mode(): socket was already in non-blocking mode"); return ret; }
    if (!(flags & O_NONBLOCK) && is_blocking) { info("set_blocking_mode(): socket was already in blocking mode"); return ret; }
    ret = 0 == fcntl(socket, F_SETFL, is_blocking ? flags ^ O_NONBLOCK : flags | O_NONBLOCK));
#endif

    return ret;
}

Ответ 3

Альтернативный способ очистки флага:

opts ^= O_NONBLOCK;

Это приведет к переключению неблокирующего флага, то есть отключить неблокирование, если оно включено.