Socket: permiso denegado en C
-
28-10-2019 - |
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;
}
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
.