Linux: Может Recvmsg использоваться для получения IP_TOS каждого входящего пакета

Может одно использование recvmsg () для получения поля IP_TOS каждого входящего пакета, или делает это, просто показывают значение IP_TOS, которое установлено для конкретного сокета. В противном случае делает любой знает о решении получить значения IP_TOS каждого входящие пакеты. Я использую приложение UDP и поэтому не добираюсь для просмотра поля IP_TOS на прикладном уровне, как имеет место с TCP.Спасибо.

При добавлении кода, который я написал до сих пор, упакуйте его, помогает:

struct msghdr msg; 
struct iovec iov[1];  
memset(&msg, '\0', sizeof(msg));
msg.msg_iov = iov;
msg.msg_iovlen = 1;
iov[0].iov_base = (char *) &pkt;
iov[0].iov_len = sizeof(pkt);

struct cmsghdr cmsgcmsg[1];  
msg.msg_control = cmsgcmsg;
msg.msg_controllen = sizeof(struct cmsghdr);


nRet = recvmsg(udpSocket, &msg, 0);

if (nRet > 0) {
    struct cmsghdr *cmsg;
    for (cmsg = CMSG_FIRSTHDR(&msg); cmsg != NULL; cmsg = CMSG_NXTHDR(&msg,cmsg)) {

    if ((cmsg->cmsg_level == IPPROTO_IP) && (cmsg->cmsg_type == IP_TOS) &&
    (cmsg->cmsg_len) ){
              int tos = *(uint8_t *)CMSG_DATA(cmsg);
              int isecn =  ((tos & INET_ECN_MASK) == INET_ECN_CE);
            printf("the tos = %i , is ecn = %d \n", tos, isecn); 

     }
    }
8
задан Mayutan 21 May 2010 в 11:48
поделиться

2 ответа

Обычно вы получаете доступ к полю ToS через getsockopt () / setsockopt (), но, похоже, очень зависит от реализации . Вы можете посмотреть do_ip_setsockopt () в исходных кодах ядра, в дереве ядра Linux по адресу linux / net / ipv4 / ip_sockglue.c

Ваш лучший друг для навигации по исходному тексту - там .

0
ответ дан 5 December 2019 в 17:35
поделиться

Мне, наконец, удалось решить проблему, и я добавляю сюда код, чтобы другие могли его использовать. Я надеюсь, что это поможет другим. Это для IP_TTL:

Установите UDPSocket для получения значений IP_TTL:

int ttl = 60;
if(setsockopt(udpSocket, IPPROTO_IP, IP_RECVTTL, &ttl,sizeof(ttl))<0) 
{
    printf("cannot set recvttl\n");
} 
else
{
    printf("socket set to recvttl\n");
}

И получите значения IP_TTL из каждого пакета следующим образом (следующая программа может получить сообщение с данными через iov [0], фрагмент кода указан ниже):

struct msghdr msg; 
struct iovec iov[1];  
memset(&msg, '\0', sizeof(msg));
msg.msg_iov = iov;
msg.msg_iovlen = 1;
iov[0].iov_base = (char *) &pkt;
iov[0].iov_len = sizeof(pkt);

int *ttlptr=NULL;
int received_ttl = 0;

int cmsg_size = sizeof(struct cmsghdr)+sizeof(received_ttl); // NOTE: Size of header + size of data
char buf[CMSG_SPACE(sizeof(received_ttl))];
msg.msg_control = buf; // Assign buffer space for control header + header data/value
msg.msg_controllen = sizeof(buf); //just initializing it

nRet = recvmsg(udpSocket, &msg, 0);



if (nRet > 0) {
    struct cmsghdr *cmsg;
    for (cmsg = CMSG_FIRSTHDR(&msg); cmsg != NULL; cmsg = CMSG_NXTHDR(&msg,cmsg)) {

          if ((cmsg->cmsg_level == IPPROTO_IP) && (cmsg->cmsg_type == IP_TTL) &&
          (cmsg->cmsg_len) ){
                ttlptr = (int *) CMSG_DATA(cmsg);
                received_ttl = *ttlptr;
                printf("received_ttl = %i and %d \n", ttlptr, received_ttl); 
               break;
           }
    }
}

Сообщение с данными может быть отправлено и получено следующим образом:

Сторона отправителя:

struct DATA_to_SEND pkt;
struct msghdr msg; 
struct iovec iov[1];  
memset(&msg, '\0', sizeof(msg));
msg.msg_iov = iov;
msg.msg_iovlen = 1;
iov[0].iov_base = (char *) &pkt;
iov[0].iov_len = sizeof(pkt);
nRet = sendmsg(udpSocket, &msg,0); 

Сторона получателя (предположение, что DATA_To_SEND имеет параметр с именем «seq»):

struct DATA_to_SEND pkt;
seqNum = ((struct DATA_to_SEND *) iov[0].iov_base)->seq;

Следующее - для IP_TOS . Настройте сокет на получение IP_TOS:

unsigned char set = 0x03;
if(setsockopt(udpSocket, IPPROTO_IP, IP_RECVTOS, &set,sizeof(set))<0) 
{
    printf("cannot set recvtos\n");
} 
else
{
        printf("socket set to recvtos\n");

и получите значение IP_TOS из каждого заголовка пакета следующим образом:

 struct PC_Pkt pkt;
 int *ecnptr;
 unsigned char received_ecn;

 struct msghdr msg; 
 struct iovec iov[1];  
 memset(&msg, '\0', sizeof(msg));
 msg.msg_iov = iov; 
 msg.msg_iovlen = 1;
 iov[0].iov_base = (char *) &pkt;
 iov[0].iov_len = sizeof(pkt);

 int cmsg_size = sizeof(struct cmsghdr)+sizeof(received_ecn);
 char buf[CMSG_SPACE(sizeof(received_ecn))];
 msg.msg_control = buf;
 msg.msg_controllen = sizeof(buf); 

 nRet = recvmsg(udpSocket, &msg, 0);

 if (nRet > 0) {
struct cmsghdr *cmsg;
for (cmsg = CMSG_FIRSTHDR(&msg); cmsg != NULL;
cmsg = CMSG_NXTHDR(&msg,cmsg)) {
         if ((cmsg->cmsg_level == IPPROTO_IP) && 
         (cmsg->cmsg_type == IP_TOS) && (cmsg->cmsg_len) ){
                ecnptr = (int *) CMSG_DATA(cmsg);
        received_ecn = *ecnptr;
        int isecn =  ((received_ecn & INET_ECN_MASK) == INET_ECN_CE);

                printf("received_ecn = %i and %d, is ECN CE marked = %d \n", ecnptr, received_ecn, isecn); 

                 break;
    }
     }
}
9
ответ дан 5 December 2019 в 17:35
поделиться
Другие вопросы по тегам:

Похожие вопросы: