Программирование сокетов IPv6 с собственным C - PullRequest
2 голосов
/ 02 ноября 2011

Я работаю над IPv6 и мне нужно создать пакет IPv6 с нуля и поместить его в буфер.К сожалению, у меня нет большого опыта работы с C. Из руководства я успешно проделал то же самое с IPv4, определив

struct ipheader {
 unsigned char      iph_ihl:5, /* Little-endian */
                iph_ver:4;
 unsigned char      iph_tos;
 unsigned short int iph_len;
 unsigned short int iph_ident;
 unsigned char      iph_flags;
 unsigned short int iph_offset;
 unsigned char      iph_ttl;
 unsigned char      iph_protocol;
 unsigned short int iph_chksum;
 unsigned int       iph_sourceip;
 unsigned int       iph_destip;
};

/* Structure of a TCP header */
struct tcpheader {
 unsigned short int tcph_srcport;
 unsigned short int tcph_destport;
 unsigned int       tcph_seqnum;
 unsigned int       tcph_acknum;
 unsigned char      tcph_reserved:4, tcph_offset:4;
 // unsigned char tcph_flags;
  unsigned int
       tcp_res1:4,       /*little-endian*/
       tcph_hlen:4,      /*length of tcp header in 32-bit words*/
       tcph_fin:1,       /*Finish flag "fin"*/
       tcph_syn:1,       /*Synchronize sequence numbers to start a connection*/
       tcph_rst:1,       /*Reset flag */
       tcph_psh:1,       /*Push, sends data to the application*/
       tcph_ack:1,       /*acknowledge*/
       tcph_urg:1,       /*urgent pointer*/
       tcph_res2:2;
 unsigned short int tcph_win;
 unsigned short int tcph_chksum;
 unsigned short int tcph_urgptr;
};

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

// IP structure
   ip->iph_ihl = 5;
   ip->iph_ver = 6;
   ip->iph_tos = 16;
   ip->iph_len = sizeof (struct ipheader) + sizeof (struct tcpheader);
   ip->iph_ident = htons(54321);
   ip->iph_offset = 0;
   ip->iph_ttl = 64;
   ip->iph_protocol = 6; // TCP
   ip->iph_chksum = 0; // Done by kernel

   // Source IP, modify as needed, spoofed, we accept through command line argument
   ip->iph_sourceip = inet_addr("1922.168.1.128");
   // Destination IP, modify as needed, but here we accept through command line argument
   ip->iph_destip = inet_addr(1922.168.1.1);

   // The TCP structure. The source port, spoofed, we accept through the command line
   tcp->tcph_srcport = htons(atoi("1024"));
   // The destination port, we accept through command line
   tcp->tcph_destport = htons(atoi("4201"));
   tcp->tcph_seqnum = htons(1);
   tcp->tcph_acknum = 0;
   tcp->tcph_offset = 5;
   tcp->tcph_syn = 1;
   tcp->tcph_ack = 0;
   tcp->tcph_win = htons(32767);
   tcp->tcph_chksum = 0; // Done by kernel
   tcp->tcph_urgptr = 0;
   // IP checksum calculation
   ip->iph_chksum = csum((unsigned short *) buffer, (sizeof (struct ipheader) + sizeof (struct tcpheader)));

Однако для IPv6 я не нашел аналогичного способа.Я уже нашел эту структуру из IETF,

struct ip6_hdr {

   union {

      struct ip6_hdrctl {
         uint32_t ip6_un1_flow; /* 4 bits version, 8 bits TC, 20 bits
                                      flow-ID */
         uint16_t ip6_un1_plen; /* payload length */
         uint8_t ip6_un1_nxt; /* next header */
         uint8_t ip6_un1_hlim; /* hop limit */
      } ip6_un1;

      uint8_t ip6_un2_vfc; /* 4 bits version, top 4 bits
                                      tclass */
   } ip6_ctlun;
   struct in6_addr ip6_src; /* source address */
   struct in6_addr ip6_dst; /* destination address */
};

Но я не знал, как заполнять информацию, например, как отправлять TCP / SYN с 2001: 220: 806: 22:aacc: ff: fe00: 1 порт 1024–2001: 220: 806: 21 :: 4 порт 1025?

Может ли кто-нибудь мне помочь или есть какие-либо ссылки?

Спасибо вам большое.

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

Кто-нибудь может помочь?

#define PCKT_LEN 2000

int main(void) {
   unsigned char buffer[PCKT_LEN];
   int s;
   struct sockaddr_in6 din;
   struct ipv6_header *ip = (struct ipv6_header *) buffer;
   struct tcpheader *tcp = (struct tcpheader *) (buffer + sizeof (struct ipv6_header));

   memset(buffer, 0, PCKT_LEN);
   din.sin6_family = AF_INET6;
   din.sin6_port = htons(0);
   inet_pton(AF_INET6, "::1", &(din.sin6_addr)); // For routing 

   ip->version = 6;
   ip->traffic_class = 0;
   ip->flow_label = 0;
   ip->length = 40;
   ip->next_header = 6;
   ip->hop_limit = 64;
   inet_pton(AF_INET6, "::1", &(ip->dst)); // IPv6 
   inet_pton(AF_INET6, "::1", &(ip->src)); // IPv6

   tcp->tcph_srcport = htons(atoi("11111"));
   tcp->tcph_destport = htons(atoi("13"));
   tcp->tcph_seqnum = htons(0);
   tcp->tcph_acknum = 0;
   tcp->tcph_offset = 5;
   tcp->tcph_syn = 1;
   tcp->tcph_ack = 0;
   tcp->tcph_win = htons(32752);
   tcp->tcph_chksum = 0; // Done by kernel
   tcp->tcph_urgptr = 0;

   s = socket(PF_INET6, SOCK_RAW, IPPROTO_RAW);
   if (s < 0) {
      perror("socket()");
      return 1;
   }
   unsigned short int packet_len = sizeof (struct ipv6_header) + sizeof (struct tcpheader);
   if (sendto(s, buffer, packet_len, 0, (struct sockaddr*) &din, sizeof (din)) == -1) {
      perror("sendto()");
      close(s);
      return 1;
   }
   close(s);
   return 0;
}

Ответы [ 2 ]

2 голосов
/ 02 ноября 2011

Может быть эта статья может помочь вам начать?

Edit: Используя приведенную выше статью в википедии, я создал эту структуру (не зная, что означают некоторые поля):

struct ipv6_header
{
    unsigned int
        version : 4,
        traffic_class : 8,
        flow_label : 20;
    uint16_t length;
    uint8_t  next_header;
    uint8_t  hop_limit;
    struct in6_addr src;
    struct in6_addr dst;
};

Это ничем не отличается от того, как была создана структура заголовка для IPv4 в вашем примере. Просто создайте struct, содержащий поля, в правильном порядке и в нужном размере, и заполните его правильными значениями.

Просто сделайте то же самое для заголовков TCP.

0 голосов
/ 31 октября 2017

К сожалению, RFC ipv6 не предоставляют тот же интерфейс необработанных сокетов, который вы получаете с ipv4. Из того, что я видел для создания пакетов ipv6, вы должны пойти на уровень глубже и использовать сокет AF_PACKET для отправки кадра Ethernet, включая ваш пакет ipv6.

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