8

データを受信せずにデータを送信するためだけに SOCK_RAW タイプのソケットを作成した場合、カーネルがネットワーク パケットを受信し続け、そのデータグラムを (アプリケーションの) バッファーにコピーするときに問題はありますか? 言い換えれば、一部のバッファがいっぱいになった後、何が起こるのでしょうか? エラーまたは無視?

カーネルがデータグラムのコピーをアプリケーションに配信しないようにする方法がわかりません。

参照http://sock-raw.org/papers/sock_raw 0x4 raw_input

IP レイヤーは、新しい着信 IP データグラムを処理した後、ip_local_deliver_finish() カーネル関数を呼び出します。この関数は、IP ヘッダーのプロトコル フィールドを調べることによって、登録済みのトランスポート プロトコル ハンドラーを呼び出す役割を果たします (上記を思い出してください)。ただし、データグラムをハンドラーに配信する前に、アプリケーションが同じプロトコル番号で raw ソケットを作成したかどうかを毎回チェックします。そのようなアプリケーションが 1 つ以上ある場合、データグラムのコピーを作成し、それらのアプリケーションにも配信します。

4

2 に答える 2

4

ソケットの受信部分をシャットダウンするには、shutdown(2)を使用できます。シャットダウンのマニュアルページを参照してください

編集:シャットダウンは、接続された(つまりTCP)ソケットでのみ機能することがわかりました。Raw ソケットには、次の 2 つの可能性があります。

  • ( recvを使用して) 一時バッファーにデータを受信し、それらを (おそらく別のスレッドで) 破棄します。
  • よく覚えているのですが、ソケット バッファーがいっぱいになると、受信データは自動的に破棄されます (バッファー内のデータは変更されません)。そのため、ソケット受信バッファー サイズを 0 に設定できます (後で必要に応じて大きくします)。

受信バッファ サイズを 0 に設定する方法は次のとおりです。

int opt = 0;
setsockopt(sock_fd, SOL_SOCKET, SO_RCVBUF, &opt, sizeof(opt));

テスト

/**
 * @file raw_print_pkt.c
 * @brief 
 * @author Airead Fan <fgh1987168@gmail.com>
 * @date 2012/08/22 12:35:22
 */

#include <stdio.h> 
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/ioctl.h>
#include <netinet/ip.h>
#include <netinet/tcp.h>

int main(int argc, char *argv[])
{
    int s;
    ssize_t rn;                 /* receive number */
    struct sockaddr_in saddr;
    char packet[4096];
    int count;

    if ((s = socket(AF_INET, SOCK_RAW, IPPROTO_TCP)) < 0) {
        perror("error:");
        exit(EXIT_FAILURE);
    }

    memset(packet, 0, sizeof(packet));
    socklen_t *len = (socklen_t *)sizeof(saddr);
    int fromlen = sizeof(saddr);
    int opt = 0;

    count = 0;
    while(1) {
        if ((rn = recvfrom(s, (char *)&packet, sizeof(packet), 0,
                           (struct sockaddr *)&saddr, &fromlen)) < 0)
            perror("packet receive error:");
        if (rn == 0) {
            printf("the peer has performed an orderly shutdown\n");
            break;
        }

        printf("[%d] rn = %lu \n", count++, rn);

        if (count == 16) {
            if (setsockopt(s, SOL_SOCKET, SO_RCVBUF, &opt, sizeof(opt)) < 0) {
                perror("setsocketopt failed");
            } else {
                fprintf(stdout, "setsocketopt successful\n");
            }
            // int shutdown(int sockfd, int how);
            /* if (shutdown(s, SHUT_RD) < 0) {
             *     perror("shutdown failed");
             * } */
        }
    }

    return 0;
}

テスト 2 (同じ内容):

int main(int argc, char *argv[])
{
int s;
ssize_t rn;                 /* receive number */
char packet[4096];
int count;

if ((s = socket(AF_INET, SOCK_RAW, IPPROTO_TCP)) < 0) {
    perror("error:");
    exit(EXIT_FAILURE);
}

memset(packet, 0, sizeof(packet));
int opt = 0;
count = 0;

//Set recv buffer size
if (setsockopt(s, SOL_SOCKET, SO_RCVBUF, &opt, sizeof(opt)) < 0) {
    perror("setsocketopt failed");
} else {
    fprintf(stdout, "setsocketopt successful\n");
}

//10 seconds countdown
int i = 10;
while(i > 0)
{
    printf("\r%d              ", i);
    fflush(stdout);
    i--;
    sleep(1);
}
printf("\n");
while(1) {
    if ((rn = recv(s, (char *)&packet, sizeof(packet), 0)) <= 0)
        perror("packet receive error:");

    printf("[%d] rn = %lu \n", count++, rn);
    }
return 0;
}

テスト 2 を続行する方法は次のとおりです。

まず、バッファ サイズを 4096 (ネットワークのトラフィックが多い場合はそれ以上) に設定します。コンパイルして起動します。データの受信を開始する前の 10 秒間に、大量のデータをソケットに送信します。10 秒後、プログラムはカウントダウン中に送信したすべてを受信します。

その後、バッファ サイズを 0 に設定します。前と同様に続行します。10 秒が経過すると、プログラムはカウントダウン中に送信したデータを受信しなくなります。しかし、recvfromにある間にデータを送信すると、正常に読み取られます。

于 2012-08-22T15:21:17.077 に答える
3

私はあなたが何を望んでいるのかよくわかりません!いくつかのパケットを注入したいだけなら、それは簡単です:

#include<netinet/tcp.h> /* TCP header */
#include<netinet/ip.h>  /* IP header */

/* Checksum compute function */
/* source : http://www.winpcap.org/pipermail/winpcap-users/2007-July/001984.html */
unsigned short checksum(unsigned short *buffer, int size)
{
    unsigned long cksum=0;
    while(size >1)
    {
        cksum+=*buffer++;
        size -=sizeof(unsigned short);
    }
    if(size)
        cksum += *(UCHAR*)buffer;

    cksum = (cksum >> 16) + (cksum & 0xffff);
    cksum += (cksum >>16);
    return (unsigned short)(~cksum);
}

int main (int argc, char **argv)
{
    char packet_buffer[BUFFER_SIZE];
    struct sockaddr_in sin;     
    struct iphdr *ip_header;    /* IP header */
    struct tcphdr *tcp_header;  /* TCP header */
    int flag = 1;

    /* Creating RAW socket */
    int raw_socket = socket (PF_INET, SOCK_RAW, IPPROTO_TCP);

    ip_header = (struct iphdr *) packet_buffer;

    tcp_header = (struct tcphdr *) (packet_buffer + sizeof (struct ip));

    sin.sin_family = AF_INET;
    sin.sin_port = htons(PORT_NUMBER);
    sin.sin_addr.s_addr = inet_addr (IP_ADDRESS);

    /* Zeroing the bbuffer */ 
    memset (packet_buffer, 0, BUFFER_SIZE);

    /* Construct your IP Header */
    ip_header->ihl = 5;
    ip_header->version = 4;
    ip_header->tos = 0;
    ip_header->tot_len = sizeof (struct ip) + sizeof (struct tcphdr);
    ip_header->id = htonl(CHOOSE_PACKET_ID);
    ip_header->frag_off = 0;
    ip_header->ttl = 255;
    ip_header->protocol = 6;    /* TCP. Change to 17 if you want UDP */
    ip_header->check = 0;
    ip_header->saddr = inet_addr (SOURCE_IP_ADDRESS_TO_SPOOF);
    ip_header->daddr = sin.sin_addr.s_addr;

    /* Construct your TCP Header */
    tcp_header->source = htons (SOURCE);
    tcp_header->dest = htons(DEST);
    tcp_header->seq = random();
    tcp_header->ack_seq = 0;
    tcp_header->doff = 0;
    tcp_header->syn = 1; 
    tcp_header->window = htonl(65535);
    tcp_header->check = 0;
    tcp_header->urg_ptr = 0;

    /* IP Checksum */
    ip_header->check = checksum((unsigned short *) packet_buffer, ip_header->tot_len >> 1);

    if (setsockopt(raw_socket, IPPROTO_IP, IP_HDRINCL, &flag, sizeof(flag)) < 0)
    {
        /* ERROR handling */
    }

    while (1)
    {
        /* Send the packet */
    if (sendto(raw_socket, packet_buffer, ip_header->tot_len, 0,  (struct sockaddr *) &sin, sizeof (sin)) < 0)
    {
        /* ERROR handling */
    }
    /* The rest of your need */
 }

 return 0;
}
于 2012-08-22T11:29:11.800 に答える