0

生の IP パケットの実装については、既にこの質問をしました。しかし、私は何の解決策も得られませんでした。

私のコード:

if((s = WSASocket(AF_INET, SOCK_RAW, IPPROTO_TCP, 0, 0, 0))==SOCKET_ERROR) // Socket 
    {
        printf("Creation of raw socket failed.");
        return 0;
    }
if(setsockopt(s, IPPROTO_IP, IP_HDRINCL, (char *)&optval, sizeof(optval))==SOCKET_ERROR)
    {
        printf("failed to set socket in raw mode.");
        return 0;
    }
if((sendto(s ,(char *) buf , sizeof(IPV4_HDR)+sizeof(TCP_HDR) + payload, 0,(SOCKADDR *)&dest, sizeof(dest)))==SOCKET_ERROR)
    {
    printf("Error sending Packet : %d",WSAGetLastError());
    break;
    }

エラー:

WSAGetLastError() は 10022 を返します。

説明:

setsockopt (Wsapiref_94aa.asp) 関数に無効な引数 (無効なレベルを指定した引数など) が指定されました。場合によっては、ソケットの現在の状態を参照することもあります。たとえば、リッスンしていないソケットで accept (Wsapiref_13aq.asp) を呼び出します。

解説:

しかし、正しいオプション値とサイズを設定しました。

私は何を間違っていますか?Windows XP (SP3) を使用しています。setsocketopt で、そのプログラムに対して IP_OPTIONS を試してみましたが、正常に動作し、IP パケットも送信します。しかし、すべての IP パケットのエーテルでは、宛先から ICMP パケットが生成されます。

どうすればこれを修正できますか?

ソースコード:

// raw tcp パケット クラフター

#include "stdio.h"
#include "winsock2.h"
#include "ws2tcpip.h"  //IP_HDRINCL is here
#include "conio.h"

typedef struct ip_hdr
{
    unsigned char  ip_header_len:4;  // 4-bit header length (in 32-bit words) normally=5 (Means 20 Bytes may be 24 also)
    unsigned char  ip_version   :4;  // 4-bit IPv4 version
    unsigned char  ip_tos;           // IP type of service
    unsigned short ip_total_length;  // Total length
    unsigned short ip_id;            // Unique identifier 

    unsigned char  ip_frag_offset   :5;        // Fragment offset field

    unsigned char  ip_more_fragment :1;
    unsigned char  ip_dont_fragment :1;
    unsigned char  ip_reserved_zero :1;

    unsigned char  ip_frag_offset1;    //fragment offset

    unsigned char  ip_ttl;           // Time to live
    unsigned char  ip_protocol;      // Protocol(TCP,UDP etc)
    unsigned short ip_checksum;      // IP checksum
    unsigned int   ip_srcaddr;       // Source address
    unsigned int   ip_destaddr;      // Source address
}   IPV4_HDR, *PIPV4_HDR, FAR * LPIPV4_HDR;

// TCP header
typedef struct tcp_header 
{ 
    unsigned short source_port;  // source port 
    unsigned short dest_port;    // destination port 
    unsigned int   sequence;     // sequence number - 32 bits 
    unsigned int   acknowledge;  // acknowledgement number - 32 bits 

    unsigned char  ns   :1;          //Nonce Sum Flag Added in RFC 3540.
    unsigned char  reserved_part1:3; //according to rfc
    unsigned char  data_offset:4;    /*The number of 32-bit words in the TCP header. 
                                       This indicates where the data begins. 
                                       The length of the TCP header is always a multiple 
                                       of 32 bits.*/

    unsigned char  fin  :1;      //Finish Flag
    unsigned char  syn  :1;      //Synchronise Flag
    unsigned char  rst  :1;      //Reset Flag
    unsigned char  psh  :1;      //Push Flag 
    unsigned char  ack  :1;      //Acknowledgement Flag 
    unsigned char  urg  :1;      //Urgent Flag

    unsigned char  ecn  :1;      //ECN-Echo Flag
    unsigned char  cwr  :1;      //Congestion Window Reduced Flag

    ////////////////////////////////

    unsigned short window;  // window 
    unsigned short checksum;  // checksum 
    unsigned short urgent_pointer;  // urgent pointer 
}   TCP_HDR , *PTCP_HDR , FAR * LPTCP_HDR , TCPHeader , TCP_HEADER;

int main()
{
    char host[100],buf[1000],*data=NULL,source_ip[20]; //buf is the complete packet
    SOCKET s;
    int k=1;

    IPV4_HDR *v4hdr=NULL;
    TCP_HDR  *tcphdr=NULL;

    int payload=512 ;
    int optval= 1;
    SOCKADDR_IN dest;
    hostent *server;

    //Initialise Winsock
    WSADATA wsock;
    printf("\nInitialising Winsock...");
    if (WSAStartup(MAKEWORD(2,2),&wsock) != 0)
    {
        fprintf(stderr,"WSAStartup() failed");
        exit(EXIT_FAILURE); 
    } 
    printf("Initialised successfully.");
    ////////////////////////////////////////////////

    //Create Raw TCP Packet
    printf("\nCreating Raw TCP Socket...");
    if((s = WSASocket(AF_INET, SOCK_RAW, IPPROTO_TCP, 0, 0, 0))==SOCKET_ERROR)
    {
        printf("Creation of raw socket failed.");
        return 0;
    }
    printf("Raw TCP Socket Created successfully.");
    ////////////////////////////////////////////////

    //Put Socket in RAW Mode.
    printf("\nSetting the socket in RAW mode...");
    if(setsockopt(s, IPPROTO_IP, IP_HDRINCL, (char *)&optval, sizeof(optval))==SOCKET_ERROR)
    {
        printf("failed to set socket in raw mode.");
        return 0;
    }
    printf("Successful.");
    ////////////////////////////////////////////////
    //Target Hostname
    printf("\nEnter hostname : ");
    gets(host);
    printf("\nResolving Hostname...");
    if((server=gethostbyname(host))==0)
    {
        printf("Unable to resolve.");
        return 0;
    }
    dest.sin_family = AF_INET;
    dest.sin_port   = htons(8888);  //your destination port
    memcpy(&dest.sin_addr.s_addr,server->h_addr,server->h_length);
    printf("Resolved.");
    /////////////////////////////////////////////////

    printf("\nEnter Source IP : ");
    gets(source_ip);


    v4hdr = (IPV4_HDR *)buf;  //lets point to the ip header portion
    v4hdr->ip_version=4;
    v4hdr->ip_header_len=5;
    v4hdr->ip_tos    = 0;
    v4hdr->ip_total_length = htons ( sizeof(IPV4_HDR) + sizeof(TCP_HDR) + payload );
    v4hdr->ip_id     = htons(2);
    v4hdr->ip_frag_offset = 0;
    v4hdr->ip_frag_offset1 = 0;
    v4hdr->ip_reserved_zero = 0;
    v4hdr->ip_dont_fragment = 1;
    v4hdr->ip_more_fragment = 0;
    v4hdr->ip_ttl    = 8;
    v4hdr->ip_protocol = IPPROTO_TCP;
    v4hdr->ip_srcaddr  = inet_addr(source_ip);
    v4hdr->ip_destaddr = inet_addr(inet_ntoa(dest.sin_addr));
    v4hdr->ip_checksum = 0;

    tcphdr = (TCP_HDR *)&buf[sizeof(IPV4_HDR)]; //get the pointer to the tcp header in the packet

    tcphdr->source_port = htons(1234);
    tcphdr->dest_port = htons(8888);

    tcphdr->cwr=0;
    tcphdr->ecn=1;
    tcphdr->urg=0;
    tcphdr->ack=0;
    tcphdr->psh=0;
    tcphdr->rst=1;
    tcphdr->syn=0;
    tcphdr->fin=0;
    tcphdr->ns=1;

    tcphdr->checksum = 0;


    // Initialize the TCP payload to some rubbish
    data = &buf[sizeof(IPV4_HDR) + sizeof(TCP_HDR)];
    memset(data, '^', payload);


    printf("\nSending packet...\n");

    while(!_kbhit())
    {
        printf("  %d  packets send\r",k++);
        if((sendto(s ,(char *) buf , sizeof(IPV4_HDR)+sizeof(TCP_HDR) + payload, 0,(SOCKADDR *)&dest, sizeof(dest)))==SOCKET_ERROR)
        {
        printf("Error sending Packet : %d",WSAGetLastError());
        break;
        }
    }
    return 0;
}
4

3 に答える 3

3

Windows では生の tcp ソケットでデータを送信できません。

ここから:

「Raw ソケットの制限

Windows 7、Windows Vista、Service Pack 2 (SP2) を適用した Windows XP、および Service Pack 3 (SP3) を適用した Windows XP では、raw ソケット経由でトラフィックを送信する機能がいくつかの方法で制限されています。

  • TCP データは raw ソケット経由で送信できません。

  • 無効な送信元アドレスを持つ UDP データグラムは、raw ソケット経由で送信できません。発信 UDP データグラムの IP ソース アドレスは、ネットワーク インターフェイスに存在する必要があります。存在しない場合、データグラムはドロップされます。この変更は、悪意のあるコードが分散サービス拒否攻撃を作成する機能を制限し、スプーフィングされたパケット (偽造された送信元 IP アドレスを持つ TCP/IP パケット) を送信する機能を制限するために行われました。

  • IPPROTO_TCP プロトコルの raw ソケットを使用した bind 関数の呼び出しは許可されていません。"
于 2011-06-07T22:20:36.667 に答える
1

さて、あなたには送る相手がいないようです。

bind()ソケットを作成してそのオプションを設定しましたが、着信接続 ( + accept()) またはconnect()他のパーティをリッスンする必要があります。

エラーの説明: Sometimes, it also refers to the current state of the sockets- あなたのケースだと思います。ソケットが接続状態ではないためsendto()、無効です。

于 2009-03-04T14:29:27.433 に答える
0

ところで、参考までに、bool か int かに関係なく、「optval」に関する議論があります。どうやら「int」がより良い選択ですが、bool の例をたくさん見てきました。

win32 で IP_HDRINCL を setsockopt 関数に設定する

「bool」を使用したことがあり、私のプログラムは Windows XP で正常に動作しました。現在、Win 7 では動作せず、エラー コード 10022 が表示されます。

于 2011-06-07T20:25:44.737 に答える