Как определить размер для нескольких вспомогательных сообщений в linux управляющих сообщениях - PullRequest
3 голосов
/ 07 января 2020

Я пытаюсь отправить несколько файловых дескрипторов через сокет unix одновременно. Это не проблема для одного сокета. Хотя, когда я пытаюсь добавить еще один с CMSG_NXTHDR, я получаю нулевой указатель, который указывает, что мой буфер был слишком коротким. Для одного дескриптора файла я вычислил размер буфера с CMSG_SPACE и предположил, что мне просто нужно умножить это. Хотя этого, кажется, недостаточно. Я написал короткую тестовую программу, чтобы проверить это:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <sys/types.h>

int main(int argc, char *argv[])
{
    struct msghdr msg;

    if(argc != 2){
        return 1;
    }

    union {
        struct cmsghdr cmsghdr;
        char control[CMSG_SPACE(sizeof(int)) * atoi(argv[1])];
    } cmsgu;
    struct cmsghdr *cmsg;

    msg.msg_name = 0;
    msg.msg_namelen = 0;
    msg.msg_iov = 0;
    msg.msg_iovlen = 0;

    msg.msg_control = cmsgu.control;
    msg.msg_controllen = sizeof(cmsgu.control);

    cmsg = CMSG_FIRSTHDR(&msg);
    cmsg->cmsg_len = CMSG_LEN(sizeof(int));
    cmsg->cmsg_level = SOL_SOCKET;
    cmsg->cmsg_type = SCM_RIGHTS;
    *((int *)CMSG_DATA(cmsg)) = -1;

    cmsg = CMSG_NXTHDR(&msg, cmsg);
    fprintf(stderr, "%p\n", cmsg);
    return 0;
}

Когда я вызываю это с 1, он выводит нулевой указатель, который ожидается. Хотя я ожидал бы, что, если он вызывается с 2 CMSG_NXTHDR, вернет действительный указатель. Первое рабочее значение 5 (так что дополнительные 120 байтов). Я думал, что CMSG_SPACE позаботится об этом. Есть ли способ go для расчета необходимого пространства? Или есть какой-нибудь более простой способ отправить несколько файловых дескрипторов в одном сообщении?

1 Ответ

3 голосов
/ 08 января 2020

См. этот вопрос и ответ. Ваш код должен работать, если для memset () cmsgu установлено значение 0 с.

Однако (и для ответа на второй вопрос), если вы хотите передать несколько файловых дескрипторов, вы можете использовать один cmsghdr, содержащий массив целых Изменяя ваш пример,

    int n = atoi(argv[1]);
    int myfds[n]; // use this later

    union {
        struct cmsghdr cmsghdr; // for alignment
        char control[CMSG_SPACE(sizeof(int) * n)]; // space for one message
    } cmsgu;
    struct cmsghdr *cmsg;

    memset(&cmsgu, 0, sizeof(cmsgu)); // as noted

    msg.msg_name = 0;
    msg.msg_namelen = 0;
    msg.msg_iov = 0;
    msg.msg_iovlen = 0;

    msg.msg_control = cmsgu.control;
    msg.msg_controllen = sizeof(cmsgu.control);

    cmsg = CMSG_FIRSTHDR(&msg);
    cmsg->cmsg_len = CMSG_LEN(sizeof(int) * n);
    cmsg->cmsg_level = SOL_SOCKET;
    cmsg->cmsg_type = SCM_RIGHTS;

    int *fdptr = (int *) CMSG_DATA(cmsg);
    memcpy(fdptr, myfds, sizeof(int) * n);

Это в основном то же самое, что и пример на странице руководства cmsg (3).

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...