3

CのDNSサーバーからMXレコードを取得しようとしています。問題は、sendtoまたはrecvfromを呼び出すたびに、アクセス許可が拒否されたというエラーが発生することです。(IDE-Xcode 4、Mac OS X Lion)これまでCで何もしていませんが、割り当てにはこれが必要です。http://www.binarytides.com/blog/dns-query-code-in-c-with-linux-sockets/からの「インスピレーション」 これまでの私のコード:

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; 
} 
4

2 に答える 2

8

このinet_addr関数は、ホスト名ではなく、文字列形式のIPアドレスを取ります。"ns1.sil.at"IPアドレスに置き換えてみてください。"213.129.232.1"

于 2012-01-16T13:31:34.667 に答える
0

@ joni-salonenの答えに加えて、ソケットをバインドする必要があります。関数呼び出し
を使用します。ソケットのローカル側を識別する、が 必要です。ファミリ、ポート0、およびアドレスを使用して初期化できます。bind
struct sockaddr_inAF_INETINADDR_ANY

于 2012-01-16T13:59:18.773 に答える