Почему recvfrom () для сокета дейтаграммы домена UNIX возвращает неверный аргумент? - PullRequest
0 голосов
/ 17 января 2020

Я работаю над программой, которая использует доменные датаграммы UNIX для IP C. Половина времени работает нормально, а другая половина возвращает «Неверный аргумент». Я подтвердил, что это не должно быть связано с ошибкой повторного использования пути сокета, SO_REUSEADDR установлен на сокете, и ошибка возникает, даже если любой остаточный файл в пути сокета был удален. Почему это происходит? Вот код:

Сервер

/* receive data from the client and return a structure containing operation information and argument */
int handle_input(int server_socket, module *module_registry) {
        input client_input;     // operation and argument sent from client
        struct sockaddr *client_address;
        unsigned int cl_address_len;

        if(recvfrom(server_socket, &client_input, sizeof(input), 0, client_address, &cl_address_len) < 0) {
                fprintf(stderr, "failed to receive from datagram socket\n");
                perror("guru meditation");
                return 0;
        }

        /* parse input ... */
}

Клиент

int main(int argc, char **argv) {    
    /* parse input ... */

    /* initialize client socket and addressing information */
    if((client_socket = socket(AF_UNIX, SOCK_DGRAM, 0)) < 0) {
        fprintf(stderr, "failed to bind client socket\n");
        exit(1);
    }

    /* initialize server socket addressing information */
    memset(&server_addr, 0, sizeof(struct sockaddr_un));
    server_addr.sun_family = AF_UNIX;
    snprintf(server_addr.sun_path, sizeof server_addr.sun_path, "%s", SOCK_PATH);

    if(sendto(client_socket, &client_input, sizeof(input), 0, (struct sockaddr *)&server_addr, sizeof(struct sockaddr_un))
        fprintf(stderr, "failed to send message to daemon\n");
        exit(1);
    }

    exit(0);
}

1 Ответ

0 голосов
/ 17 января 2020

Половина времени работает нормально, но другая половина возвращает «Неверный аргумент»

Это потому, что cl_address_len нужно инициализировать с размером вашего адресного буфера, но он содержит неопределенное значение.

Вам необходимо внести следующие изменения:

struct sockaddr_un client_address;
socklen_t cl_address_len = sizeof client_address;
if(recvfrom(server_socket, &client_input, sizeof(input), 0, (struct sockaddr*)&client_address, &cl_address_len) < 0) {
...