Как включить разгрузку в необработанном сокете

#c #linux #networking #ethernet #raw-sockets

#c #linux #сеть #ethernet #необработанные сокеты

Вопрос:

Я знаю, что этот вопрос, должно быть, задавался раньше, но я не могу найти для него хорошего решения.

Я пишу программу на C в Ubuntu 12.04, которая использует необработанные сокеты для отправки TCP-пакетов. Вычисление контрольной суммы занимает много времени, поэтому я попытался выгрузить ее на свой сетевой адаптер.

 struct sockaddr intfAddr
rawSock = socket(AF_INET,SOCK_PACKET,htons(3)); //ETH_P_ALL = 3
if (rawSock == -1) { perror("failed to create raw socket");  exit(-1); }
memset(amp;intfAddr, 0, sizeof (struct sockaddr));
intfAddr.sa_family = AF_INET;
strcpy(intfAddrs.sa_data,  "eth0"); 
system("ifconfig eth0 promisc);
//sprintf(cmd, "ethtool -K %s rx off", intfName);   system(cmd);
if (bind(rawSock, (struct sockaddr *)amp;intfAddr, sizeof(intfAddrs[intf])) == -1) {
    perror("failed to bind"); exit(-1);
}   
 

Я отправил TCP-пакеты с одного компьютера и использовал wireshark для захвата пакетов на другом компьютере, подключенном к отправляющему компьютеру. Wireshark может перехватывать все пакеты, но, к сожалению, контрольная сумма TCP неверна — сетевой адаптер отправляющего КОМПЬЮТЕРА не выполнил вычисление контрольной суммы. Если я установлю пакеты с контрольной суммой TCP, равной 0, wireshark увидит пакеты с контрольной суммой, равной 0.

Это сбивает с толку, поскольку команда Linux «ethtool -k eth0» на отправляющем КОМПЬЮТЕРЕ показывает, что загрузка tx включена.

 $ ethtool -k eth0
Offload parameters for eth0:
rx-checksumming: on
tx-checksumming: on
scatter-gather: off
tcp-segmentation-offload: off
udp-fragmentation-offload: off
generic-segmentation-offload: off
generic-receive-offload: on
large-receive-offload: off
rx-vlan-offload: on
tx-vlan-offload: on
ntuple-filters: off
receive-hashing: off
 

Есть идеи? Спасибо!

Ответ №1:

Почему вы используете ETH_P_ALL?, попробуйте использовать htons(8) для ETH_P_IP, и еще одна проблема в вашем коде заключается в том, что вы используете семейство пакетов, PF используется для отправки / получения данных в драйвере устройства (уровень OSI 2), если вы собираетесь использовать семейство пакетов вместе сс помощью RAW_SOCK вам необходимо самостоятельно генерировать каждый отдельный заголовок, поэтому попробуйте создать экземпляр сокета следующим образом :

 ; IP header would be provided by the kernel
int s = socket (AF_INET, SOCK_RAW, IPPROTO_TCP);
; if you wanna use the Packet Family, do the following
sd = socket(PF_INET, SOCK_RAW, IPPROTO_RAW);
 

В этом случае вы должны сами указать заголовки, а для контрольной суммы выполните повторное исследование псевдоголовка tcp и используйте эту функцию для его вычисления

     unsigned short csum(unsigned short *buf, int nwords)
{       //
        unsigned long sum;
        for(sum=0; nwords>0; nwords--)
                sum  = *buf  ;
        sum = (sum >> 16)   (sum amp;0xffff);
        sum  = (sum >> 16);
        return (unsigned short)(~sum);
}
 

удачи 🙂