begin process at 2012 05 30 11:12:32
  Trouver un code source :
 
dans
 
Accueil > Forum > 

Archive C/C++

 > 

Archives

 > 

Réseau / Internet

 > 

RAW SOCKET - IP - ICMP - Ping


Derniers messages déposésPoser une question dans le forum ou lancer une discussion

RAW SOCKET - IP - ICMP - Ping

dimanche 14 septembre 2003 à 12:26:44 | RAW SOCKET - IP - ICMP - Ping

dark1933

Salut à tous,

Voici le code d'un programme qui envoye un ping (ici à Google), mais ne reçoit aucune réponse.

La socket utilisée est du type :
socket(AF_INET,SOCK_RAW,IPPROTO_RAW).

Le même programme marche très bien avec une socket du type :
socket(AF_INET,SOCK_RAW,IPPROTO_ICMP)


Il semblerait donc qu'il y ait un problème dans l'entête IP (peut-être 1 pb de checksum, Network Bytes Order ou firewall).

Si quelqu'un pouvait m'apporter une explication je lui serais très reconnaissant.



=====================================================

#include <stdio.h>
#include <windows.h>


// ------- Structures ip et icmp ----------------------------
typedef struct _IP_HEADER {
unsigned char VS_LET;
unsigned char TOS;
unsigned short Size;
unsigned short ID;
unsigned Flags :3;
unsigned Offset :13;
unsigned char TTL;
unsigned char Proto;
unsigned short Checksum;
unsigned int IP_S;
unsigned int IP_D;
char IP_DATA[1];
} IPHEADER;

typedef struct _ICMP_HEADER{
unsigned char Type;
unsigned char Code;
unsigned short Checksum;
unsigned short ID;
unsigned short SEQ;
char ICMP_DATA[1];
} ICMPHEADER;


// ------- Déclarations ----------------------------
ICMPHEADER * ICMP_HEADER;
IPHEADER * IP_HEADER;

char ip_buf[100],icmp_buf[100],rcv_buf[100];

int Socket_IP;
struct sockaddr_in ip_sock_info;
int lsock=sizeof(struct sockaddr_in);

void START_WS(void); // ---- WSACleanup()
void CLEAR_WS(void); // ---- WSAStartup()
// ----- Calcule le checksum, Nb_S : nombre de Short
unsigned short CALCUL_CHECKSUM(unsigned short * ptr, int Nb_S);



// ------- Main ----------------------------
int main(){

CLEAR_WS();
START_WS();

(char*)ICMP_HEADER=&icmp_buf[0];
(char*)IP_HEADER=&ip_buf[0];

if((Socket_IP=socket(AF_INET,SOCK_RAW,IPPROTO_RAW))==-1)
{printf("[Erreur] Création Socket_IP");getchar(); return;}

ip_sock_info.sin_family=AF_INET;
ip_sock_info.sin_addr.s_addr=inet_addr("216.239.59.99");

// ------ Initialisation de la partie ICMP
ICMP_HEADER->Type=8;
ICMP_HEADER->Code=0;
ICMP_HEADER->ID=1;
ICMP_HEADER->SEQ=1;
ICMP_HEADER->Checksum=0;
memcpy(&ICMP_HEADER->ICMP_DATA,"12345678",8);
ICMP_HEADER->Checksum=htons(CALCUL_CHECKSUM((unsigned short *)ICMP_HEADER,8));

// ------ Initialisation de la partie IP
IP_HEADER->VS_LET=69; // --- (ip_version << 4) | ip_len;
IP_HEADER->TOS=0;
IP_HEADER->Size=htons(36);
IP_HEADER->ID=htons(16);
IP_HEADER->Flags=2;
IP_HEADER->Offset=0;
IP_HEADER->TTL=100;
IP_HEADER->Proto=1;
IP_HEADER->Checksum=0;
IP_HEADER->IP_S=inet_addr("82.65.1.50"); // --- mon ip lors des essais
IP_HEADER->IP_D=inet_addr("216.239.59.99");
IP_HEADER->Checksum=htons(CALCUL_CHECKSUM((unsigned short *)IP_HEADER,10));

memcpy(&IP_HEADER->IP_DATA,&ICMP_HEADER,16);

// --- Envoit de la requète
if(sendto(Socket_IP,(char*)IP_HEADER_X,36,0,(struct sockaddr *)&ip_sock_info,lsock)==-1)
{ printf("[Erreur] sendto socket_ip"); getchar(); return;}
else printf("Sendto socket_ip[ok]\n");

// --- Attente de la réponse
do{
if(recvfrom(Socket_IP,rcv_buf,100,0,NULL,NULL)==-1)
{printf("[Erreur] receivefrom\n"); getchar(); }
else printf("recvfrom [ok]\n");
memcpy(IP_HEADER,&rcv_buf[0],36); // --- on récupère le datagramme ip en entier
memcpy(ICMP_HEADER,&rcv_buf[20],16); // --- on récupère la partie icmp
}
while(ICMP_HEADER->Type!=0);

printf("Session ICMP terminée\n");
getchar();
}
dimanche 14 septembre 2003 à 13:05:32 | Re : RAW SOCKET - IP - ICMP - Ping

aardman

Membre Club
Salut,
Puisque une requete ping se fait pas le protocole ICMP, il est donc normal que créér un socket avec le protocole ICMP, non ?

-------------------------------
Réponse au message :
-------------------------------

> Salut à tous,
>
> Voici le code d'un programme qui envoye un ping (ici à Google), mais ne reçoit aucune réponse.
>
> La socket utilisée est du type :
> socket(AF_INET,SOCK_RAW,IPPROTO_RAW).
>
> Le même programme marche très bien avec une socket du type :
> socket(AF_INET,SOCK_RAW,IPPROTO_ICMP)
>
>
> Il semblerait donc qu'il y ait un problème dans l'entête IP (peut-être 1 pb de checksum, Network Bytes Order ou firewall).
>
> Si quelqu'un pouvait m'apporter une explication je lui serais très reconnaissant.
>
>
>
> =====================================================
>
> #include <stdio.h>
> #include <windows.h>
>
>
> // ------- Structures ip et icmp ----------------------------
> typedef struct _IP_HEADER {
> unsigned char VS_LET;
> unsigned char TOS;
> unsigned short Size;
> unsigned short ID;
> unsigned Flags :3;
> unsigned Offset :13;
> unsigned char TTL;
> unsigned char Proto;
> unsigned short Checksum;
> unsigned int IP_S;
> unsigned int IP_D;
> char IP_DATA[1];
> } IPHEADER;
>
> typedef struct _ICMP_HEADER{
> unsigned char Type;
> unsigned char Code;
> unsigned short Checksum;
> unsigned short ID;
> unsigned short SEQ;
> char ICMP_DATA[1];
> } ICMPHEADER;
>
>
> // ------- Déclarations ----------------------------
> ICMPHEADER * ICMP_HEADER;
> IPHEADER * IP_HEADER;
>
> char ip_buf[100],icmp_buf[100],rcv_buf[100];
>
> int Socket_IP;
> struct sockaddr_in ip_sock_info;
> int lsock=sizeof(struct sockaddr_in);
>
> void START_WS(void); // ---- WSACleanup()
> void CLEAR_WS(void); // ---- WSAStartup()
> // ----- Calcule le checksum, Nb_S : nombre de Short
> unsigned short CALCUL_CHECKSUM(unsigned short * ptr, int Nb_S);
>
>
>
> // ------- Main ----------------------------
> int main(){
>
> CLEAR_WS();
> START_WS();
>
> (char*)ICMP_HEADER=&icmp_buf[0];
> (char*)IP_HEADER=&ip_buf[0];
>
> if((Socket_IP=socket(AF_INET,SOCK_RAW,IPPROTO_RAW))==-1)
> {printf("[Erreur] Création Socket_IP");getchar(); return;}
>
> ip_sock_info.sin_family=AF_INET;
> ip_sock_info.sin_addr.s_addr=inet_addr("216.239.59.99");
>
> // ------ Initialisation de la partie ICMP
> ICMP_HEADER->Type=8;
> ICMP_HEADER->Code=0;
> ICMP_HEADER->ID=1;
> ICMP_HEADER->SEQ=1;
> ICMP_HEADER->Checksum=0;
> memcpy(&ICMP_HEADER->ICMP_DATA,"12345678",8);
> ICMP_HEADER->Checksum=htons(CALCUL_CHECKSUM((unsigned short *)ICMP_HEADER,8));
>
> // ------ Initialisation de la partie IP
> IP_HEADER->VS_LET=69; // --- (ip_version << 4) | ip_len;
> IP_HEADER->TOS=0;
> IP_HEADER->Size=htons(36);
> IP_HEADER->ID=htons(16);
> IP_HEADER->Flags=2;
> IP_HEADER->Offset=0;
> IP_HEADER->TTL=100;
> IP_HEADER->Proto=1;
> IP_HEADER->Checksum=0;
> IP_HEADER->IP_S=inet_addr("82.65.1.50"); // --- mon ip lors des essais
> IP_HEADER->IP_D=inet_addr("216.239.59.99");
> IP_HEADER->Checksum=htons(CALCUL_CHECKSUM((unsigned short *)IP_HEADER,10));
>
> memcpy(&IP_HEADER->IP_DATA,&ICMP_HEADER,16);
>
> // --- Envoit de la requète
> if(sendto(Socket_IP,(char*)IP_HEADER_X,36,0,(struct sockaddr *)&ip_sock_info,lsock)==-1)
> { printf("[Erreur] sendto socket_ip"); getchar(); return;}
> else printf("Sendto socket_ip[ok]\n");
>
> // --- Attente de la réponse
> do{
> if(recvfrom(Socket_IP,rcv_buf,100,0,NULL,NULL)==-1)
> {printf("[Erreur] receivefrom\n"); getchar(); }
> else printf("recvfrom [ok]\n");
> memcpy(IP_HEADER,&rcv_buf[0],36); // --- on récupère le datagramme ip en entier
> memcpy(ICMP_HEADER,&rcv_buf[20],16); // --- on récupère la partie icmp
> }
> while(ICMP_HEADER->Type!=0);
>
> printf("Session ICMP terminée\n");
> getchar();
> }
dimanche 14 septembre 2003 à 13:14:13 | Re : RAW SOCKET - IP - ICMP - Ping

dark1933

Un ping se fait à l'aide d'une requète ICMP encapsulée dans un datagramme IP.

La socket socket(AF_INET,SOCK_RAW,IPPROTO_RAW) permet de manipuler directement l'entête IP

alors que la socket socket(F_INET,SOCK_RAW,IPPROTO_ICMP) permet seulement de manipuler la partie ICMP autrement dit la partie Data du datagramme IP.

Non?


dimanche 14 septembre 2003 à 13:53:08 | Re : RAW SOCKET - IP - ICMP - Ping

aardman

Membre Club
Salut,
Ben justement, pas besoin de toucher a l'entete IP puisque c'est l'entete ICMP t'interesse.






-------------------------------
Réponse au message :
-------------------------------

> Un ping se fait à l'aide d'une requète ICMP encapsulée dans un datagramme IP.
>
> La socket socket(AF_INET,SOCK_RAW,IPPROTO_RAW) permet de manipuler directement l'entête IP
>
> alors que la socket socket(F_INET,SOCK_RAW,IPPROTO_ICMP) permet seulement de manipuler la partie ICMP autrement dit la partie Data du datagramme IP.
>
> Non?
>
>
>
dimanche 14 septembre 2003 à 14:04:29 | Re : RAW SOCKET - IP - ICMP - Ping

dark1933

Effectivement, mais à titre pédagogique se serait intéressant de savoir forger ses propres packets.

dimanche 14 septembre 2003 à 14:14:28 | Re : RAW SOCKET - IP - ICMP - Ping

aardman

Membre Club
Salut,
setsockopt(...) avec IP_HDRINCL.



-------------------------------
Réponse au message :
-------------------------------

> Effectivement, mais à titre pédagogique se serait intéressant de savoir forger ses propres packets.
>
>
dimanche 14 septembre 2003 à 15:11:36 | Re : RAW SOCKET - IP - ICMP - Ping

dark1933

Salut,

Je développe sous Dev C++ et ai un problème de compilation à cause de IP_HDRINCL qui n'est déclaré nulle part

et le header file <winsock2.h> fournit par Dev C++ ne contient pas la définission de IP_HDRINCL.

J'ai trouvé dans une source la valeur 2 pour IP_HDRINCL mais lors de l'exécution sendto() renvoye une erreur.

Voici les appels des fonctions concernées et leurs paramètres :

Socket_IP=socket(AF_INET,SOCK_RAW,IPPROTO_RAW);
setsockopt(Socket_IP, IPPROTO_IP, 2, (char *) &optval, sizeof(optval));


Merci d'avance.


dimanche 14 septembre 2003 à 16:06:14 | Re : RAW SOCKET - IP - ICMP - Ping

aardman

Membre Club
Salut,
Si optval est a 1 c'est ok.
Le probleme vient surement de l'initialisation du header ip, qui est assez long.



-------------------------------
Réponse au message :
-------------------------------

> Salut,
>
> Je développe sous Dev C++ et ai un problème de compilation à cause de IP_HDRINCL qui n'est déclaré nulle part
>
> et le header file <winsock2.h> fournit par Dev C++ ne contient pas la définission de IP_HDRINCL.
>
> J'ai trouvé dans une source la valeur 2 pour IP_HDRINCL mais lors de l'exécution sendto() renvoye une erreur.
>
> Voici les appels des fonctions concernées et leurs paramètres :
>
> Socket_IP=socket(AF_INET,SOCK_RAW,IPPROTO_RAW);
> setsockopt(Socket_IP, IPPROTO_IP, 2, (char *) &optval, sizeof(optval));
>
>
> Merci d'avance.
>
>
>
mercredi 8 octobre 2003 à 00:18:50 | Re : RAW SOCKET - IP - ICMP - Ping

darkstorm


Salut,

Il n'y a pas de réponse tout simplement parce que tu ne calcules PAS le checksum de ton paquet ICMP, le noyau calcule automatiquement celui ip mais le ICMP s'obtient avec cette fonction :

function CheckSum(Var Buffer; Size : integer) : Word;
type
TWordArray = Array[0..1] of word;
var
ChkSum : LongWord;
i : Integer;

begin
ChkSum := 0;
i := 0;
While Size > 1 do begin
ChkSum := ChkSum + TWordArray(Buffer)[i];
inc(i);
Size := Size - SizeOf(Word);
end;

if Size=1 then ChkSum := ChkSum + Byte(TWordArray(Buffer)[i]);

ChkSum := (ChkSum and $FFFF) + (ChkSum shr 16);


Result := Word(not(ChkSum));
end;

Alors cette fonction est en delphi et je suis désolé parce que je connais très mal le C je viens de m'y mettre. alors le checksum s'obtient en aditionnant les WORD (2 bytes) du paquet ICMP + les données (32 bytes de base), ensuite on le transforme en WORD et on ajoute la retenue , et on inverse le tout (not) et on a notre checksum, arrête de me regarder avec ces yeux je te fais un cas pratique ;-)

0033 ADDD 1344 AAAA 4444 4342 235D 3423
6162 6364 6566 6768 6970 7172 7273 7475

Alors cette partie est la partie ICMP + données (totalement inventée)

alors le chk c'est

la somme des DWORD

33 + ADDD + 1344 ..... + 7475 = 59E62

On transforme en word :

59E62 = 9E62

On ajoute la retenue

9E62 + 5 = 9E67

On l'inverse
not(9E67)=6198
TADA !!!!

alors en t'inspirant de ma fonction delphi ben j'espère que tu t'en sortes ;-), le mieux est d'avoir un sniffer et cela te permettra d'analyser la trame pour savoir ce qui est faux.
Le meilleur sniffer gratuit est Ethereal et en plus Open-Source >:-P

A+ bon travail



..................................
mardi 22 juin 2004 à 14:56:43 | Re : RAW SOCKET - IP - ICMP - Ping

valoue

Salut,
Je viens encore t'embeter ...

Je suis toujours sur mon client serveur, et je veux envoyer depuis le client un ping au serveur.
Pour la formulation du ping, no pb !
Lorsque le serveur recoit la requete, dois-je moi même traiter la reception et formuler la reponse, ou existe t-il un moyen de faire repondre le système automatiquement ?

Si je dois le faire moi même, je ne sais pas koi mettre dans la trame.

Peux tu me donner quelques explications stp ?
merci aardman

Valoue

1 2

Cette discussion est classée dans : icmp, ip, socket, header, unsigned


Répondre à ce message

Sujets en rapport avec ce message

RAW SOCKET - IP - Sendto() - Erreur 'WSAEADDRNOTAVAIL' [ par dark1933 ] Salut à tous,J'essaye désespérément d'envoyer un Ping en forgeant le datagramme IP adéquat.J'obtiens l'erreur "WSAEADDRNOTAVAIL" au moment de l'appel socket RAW ICMP - fonctionnement et implémentation... [ par smirnoff69ice ] Salut, Je suis sur un projet et je dois simuler l'envoi de paquets ICMP sur le réseau. J'ai déjà fait un petit programme de test (repris en partie dep pb lors de l'envoie de donnée avec les sockets [ par poporiding ] bonjour tout le monde. Je me galère vraiment sur la création d'un programme qui envoie des socket en mode RAW. j'ai un ensemble d'erreur, capturé winsock raw socket ping icmp recv [ par 4aBestWord ] bonjour,j'ai repris un code source en c pour faire un un ping icmp en utilisant les raw sockets sous windows (XP). L'envoie du ping fontionne bien par Thread socket icmp [ par ghanmi04_05 ] Bonjour j'ai un programme qui lance des threads de contrôle des interfaces réseaux disponible sur une machine quelconque(windows). Si je lance une fo bonjour j'ai un problème dans un code ping implémenter avec le c++, que je n'arrive pas à résoudre. s'il vous plait aider moi. Merci d'avance. [ par talelham ] bonjour j'ai un problème dans un code ping implémenter avec le c++, que je n'arrive pas à résoudre. s'il vous plait aider moi. Merci d'avance. voilà programmation socket [ par hanaw ] Comment écrire un programme simple communication client serveur en TCP/IP en C qui retourne au serveur un message contenant l'adresse IP du client. Problème avec ntohs et htons sous VS 2010 [ par marroucha ] Bonjour tout le monde, je suis entrain de développer une application qui récupère et affiche les trames réseau mais je confronte un pb de compilation Manipulation de Tableaux, structures en C++ [ par katerson ] Bonjour,Je travaille actuellement sur un projet (en stage) dans lequel je dois envoyer deux trames: l'une d'entre elles faisant 88 et l'autre 161 octe


Nos sponsors


Sondage...

CalendriCode

Mai 2012
LMMJVSD
 123456
78910111213
14151617181920
21222324252627
28293031   

Consulter la suite du CalendriCode

A découvrir



 
Développement réalisé par Nicolas SOREL (Nix) avec l'aide de : Cyril DURAND et Emmanuel (EBArtSoft), Merci à Vincent pour ses précieux conseils.
CodeS-SourceS.com© Toute reproduction même partielle est interdite sauf accord écrit du Webmaster
CodeS-SourceS.com© est une marque déposée tous droits réservés

Google Coop CodeS-SourceS Google Coop CodeS-SourceS
Temps d'éxécution de la page : 0,374 sec (3)

Nous contacter | Annoncer sur CodeS-SourceS | Mentions légales