#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);
}
удачи 🙂