-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathudp.c
143 lines (122 loc) · 3.72 KB
/
udp.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
/*
* udpserver.c - A simple UDP echo server
* usage: udpserver <port>
*/
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include "dnsparse.h"
#define BUFSIZE 1024
/*
* error - wrapper for perror
*/
void error(char *msg) {
perror(msg);
// exit(1);
}
int main(int argc, char **argv) {
int sockfd; /* socket */
int portno; /* port to listen on */
unsigned int clientlen; /* byte size of client's address */
struct sockaddr_in serveraddr; /* server's addr */
struct sockaddr_in clientaddr; /* client addr */
struct hostent *hostp; /* client host info */
char buf[BUFSIZE]; /* message buf */
char *hostaddrp; /* dotted decimal host addr string */
int optval; /* flag value for setsockopt */
int n; /* message byte size */
int sendsockfd; /* socket */
struct sockaddr_in googleaddr;
unsigned int googlelen; /* byte size of client's address */
/*
* check command line arguments
*/
if (argc != 2) {
fprintf(stderr, "usage: %s <port>\n", argv[0]);
// exit(1);
}
// portno = atoi(argv[1]);
portno = 53;
/*
* socket: create the parent socket
*/
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (sockfd < 0)
error("ERROR opening socket");
sendsockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (sendsockfd < 0)
error("ERROR opening socket");
bzero((char *) &googleaddr, sizeof(googleaddr));
googleaddr.sin_family = AF_INET;
inet_aton("114.114.114.114", &googleaddr.sin_addr); // store IP in antelope
googleaddr.sin_port = htons(53);
/* setsockopt: Handy debugging trick that lets
* us rerun the server immediately after we kill it;
* otherwise we have to wait about 20 secs.
* Eliminates "ERROR on binding: Address already in use" error.
*/
optval = 1;
setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR,
(const void *)&optval , sizeof(int));
/*
* build the server's Internet address
*/
bzero((char *) &serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET;
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);
serveraddr.sin_port = htons((unsigned short)portno);
/*
* bind: associate the parent socket with a port
*/
if (bind(sockfd, (struct sockaddr *) &serveraddr,
sizeof(serveraddr)) < 0)
error("ERROR on binding");
/*
* main loop: wait for a datagram, then echo it
*/
clientlen = sizeof(clientaddr);
while (1) {
/*
* recvfrom: receive a UDP datagram from a client
*/
bzero(buf, BUFSIZE);
n = recvfrom(sockfd, buf, BUFSIZE, 0,
(struct sockaddr *) &clientaddr, &clientlen);
if (n < 0)
error("ERROR in recvfrom");
hostaddrp = inet_ntoa(clientaddr.sin_addr);
if (hostaddrp == NULL)
error("ERROR on inet_ntoa\n");
printf("server received datagram from %s (%s)\n",
"", hostaddrp);
printf("server received %lu/%d bytes: %s\n", strlen(buf), n, buf);
// print_hex(buf, n);
parse_dns(buf, n);
/* send the message to the server */
googlelen = sizeof(googleaddr);
n = sendto(sendsockfd, buf, n, 0, (struct sockaddr *)&googleaddr, googlelen);
if (n < 0)
error("ERROR in sendto");
bzero(buf, BUFSIZE);
/* print the server's reply */
n = recvfrom(sendsockfd, buf, BUFSIZE, 0, (struct sockaddr *)&googleaddr, &googlelen);
if (n < 0)
error("ERROR in recvfrom");
printf("Echo from server: %d %s\n", n, buf);
// print_hex(buf, n);
parse_dns(buf, n);
/*
* sendto: echo the input back to the client
*/
n = sendto(sockfd, buf, n, 0,
(struct sockaddr *) &clientaddr, clientlen);
if (n < 0)
error("ERROR in sendto");
}
}