10

Linuxループバックインターフェイスをリッスンしているサーバーとrawソケットを介して通信しようとしていますが、サーバーが単一のパケットを取得していないようです。送信するパケットはWiresharkに表示されます。

ループバックのrawソケットは可能ですか?(なぜそれが必要なのか尋ねないでください。ここで説明するには複雑すぎます)

編集:これは私たちがそれを開く方法です

_I_RawSocket = socket( PF_PACKET, SOCK_RAW, htons(ETH_P_ALL)))

memset( &ifr, 0, sizeof( ifr ) );
strcpy( ifr.ifr_ifrn.ifrn_name, _InterfaceName);

ioctl( _I_RawSocket, SIOCGIFINDEX, &ifr )

memset( &sll, 0, sizeof( sll ) );
sll.sll_family   = AF_PACKET;
sll.sll_ifindex  = ifr.ifr_ifindex;
sll.sll_protocol = htons( ETH_P_ALL );

bind( _I_RawSocket, (struct sockaddr *) &sll, sizeof( sll ))

サーバーはlighttpdであり、ローカルホストの通常のソケットを介して到達可能です。netstat --rawは空のテーブルを出力しますが、通常のethデバイスには2つの機能するrawソケットがあると確信しています。

4

2 に答える 2

3

raw ソケットは、bind() と connect() で特に不安定に動作しますが、問題がそれらにあることを確認できません。より簡単なアプローチに従うことをお勧めします。

送信者

#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>

#define DEST "127.0.0.1"

int main(int argc, char **argv)
{

 int s;
 struct sockaddr_in dst_addr;
 char packet[50];

 struct iphdr *ip = (struct iphdr *)packet;  

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

 dst_addr.sin_family = AF_INET;
 dst_addr.sin_port = 0; /* not needed in SOCK_RAW */
 inet_pton(AF_INET, DEST, (struct in_addr *)&dst_addr.sin_addr.s_addr);
 memset(dst_addr.sin_zero, 0, sizeof(dst_addr.sin_zero));

 memset(packet, 'A', sizeof(packet));   /* payload will be all As */

 ip->ihl = 5;
 ip->version = 4;
 ip->tos = 0;
 ip->tot_len = htons(40);
 ip->frag_off = 0;  /* NF */
 ip->ttl = 64;
 ip->protocol = IPPROTO_RAW; /* this has to be IPPROTO_RAW */
 ip->check = 0;
 ip->saddr = dst_addr.sin_addr.s_addr;
 ip->daddr = dst_addr.sin_addr.s_addr;

 while(42) {
  sleep(5);
  if (sendto(s, packet, sizeof(packet), 0, 
   (struct sockaddr *)&dst_addr, (socklen_t)sizeof(dst_addr)) < 0)
   perror("uh oh:");
 }
 return(0);
}

レシーバー

#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>


int main(int argc, char **argv)
{
 int s;
 struct sockaddr_in src_addr;
 char packet[50];

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

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

 while(42) {
  if (recvfrom(s, &packet, sizeof(packet), 0,
   (struct sockaddr *)&src_addr, &fromlen) < 0)
   perror("uh oh:");

  int i = sizeof(struct iphdr); /* print the payload */
  for(; i < sizeof(packet); i++) {
   printf("%c", packet[i]);
  }
  printf("\n");
 }
 return(0);
}

これらがあなたが望むように正確に動作することを願っています. man 7 rawなぜこれが機能するのか、さらに重要な man 7 packetことに拡張したい場合は、詳細を読んでください。また、IPPROTO_RAW は IP_HDRINCL ソケット オプションを意味することに注意してください。これが、ip ヘッダーを自分で構築している理由です。ただし、IP チェックサムと全長はカーネルによって計算され、埋められます。

編集: さらに、有効なデータを lighttpd などのアプリケーションに送信するための raw ソケットが必要な場合は、protocol引数を一致させsocket()、IP ヘッダー フィールドに有効な値を提供する必要があります。適切なイーサネット ヘッダーは必須ではありません。唯一の重要なフィールドは、カーネル スタックによって埋められます。

于 2010-07-26T16:04:26.887 に答える
0

if_index に必ずバインドしてください

if (ioctl(sock, SIOCGIFINDEX, &stEthReq ) < 0 )
{
    printf( "failed to get IF index!" );
    return -1;
}
memset(&client_addr, 0, sizeof(client_addr));
client_addr.sll_family   = AF_PACKET;
client_addr.sll_ifindex  = stEthReq.ifr_ifru.ifru_ivalue;
client_addr.sll_protocol = VOS_HTONS(usEthType);
ret = bind(sock,(struct sockaddr *)(&client_addr), sizeof(client_addr));
于 2012-07-04T07:24:58.840 に答える