Pregunta

Estoy intentando obtener un registro MX de un servidor DNS en C.Mi problema es que cada vez que llamo a sendto o recvfrom aparece un error de Permiso denegado.(IDE - Xcode 4, Mac OS X Lion) Realmente no he hecho nada en C hasta este punto, pero necesito esto para una tarea."Inspiración" de http://www.binarytides.com/blog/dns-query-code-in-c-with-linux-sockets/Mi código hasta ahora:

struct QUESTION
{
unsigned short qtype;
unsigned short qclass;
};

typedef struct
{
unsigned char *name;
struct QUESTION *ques;
} QUERY;

typedef struct
{
unsigned short type;
unsigned short _class;
unsigned int ttl;
unsigned short data_len;
} R_DATA;



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



int sockfd, i;
unsigned char buf[65536],*qname;
struct QUESTION *qinfo = NULL;

struct sockaddr_in dest;

struct dnsheader *dns = NULL;

sockfd = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP);


dest.sin_family=AF_INET;
dest.sin_port=htons(53);
dest.sin_addr.s_addr= inet_addr("ns1.sil.at");


srand((unsigned)time(NULL)); 
int16_t rand_number = (rand()%1000+1);
printf("generated random number : %i \n", rand_number);


dns = (struct dnsheader *)&buf;
dns->flags = STANDARD_QUERY_FLAGS;
dns->question_count = 1;
dns->transaction_id = rand_number;
dns->answer_rr_count = 0;
dns->additional_rr_count = 0;
dns->nameserver_rr_count = 0;


qname =(unsigned char*)&buf[sizeof(struct dnsheader)];


qinfo =(struct QUESTION*)&buf[sizeof(struct dnsheader) + (strlen((const char*)qname) + 1)]; //fill it

qinfo->qtype = htons( 15 ); // MX = 15
qinfo->qclass = htons(1); 

printf("\nSending Packet...");
if( sendto(sockfd,(char*)buf,sizeof(struct dnsheader) + (strlen((const char*)qname)+1) + sizeof(struct QUESTION),0,(struct sockaddr*)&dest,sizeof(dest)) < 0){
    perror("\nsendto failed");
}
printf("\nDone\n");


i = sizeof dest;
printf("\nReceiving answer...");
if(recvfrom (sockfd,(char*)buf , 65536 , 0 , (struct sockaddr*)&dest , (socklen_t*)&i ) < 0){
    perror("recvfrom failed");
}
printf("Done");

dns = (struct dnsheader*) buf;

print_dns_answers(buf, sizeof((char*)buf));




close(sockfd); 


return 0; 
} 
¿Fue útil?

Solución

La función inet_addr toma una dirección IP en forma de cadena, no un nombre de host.Intente reemplazar "ns1.sil.at" con su dirección IP, "213.129.232.1"

Otros consejos

Además de la respuesta de @ joni-salonen, debes vincular el socket.
Usar una bind Llamada de función.
necesita un struct sockaddr_in, que identificaría el lado local del socket.Puedes inicializarlo con tu familia. AF_INET, puerto 0 y dirección INADDR_ANY.

Licenciado bajo: CC-BY-SA con atribución
No afiliado a StackOverflow
scroll top