summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authormiau <miaukatzemiau@priveasy.de>2011-12-01 14:13:35 (GMT)
committermiau <miaukatzemiau@priveasy.de>2011-12-01 14:13:35 (GMT)
commitb4b4c79cf6276e8d2cb851f4ea3d40102d11c5a4 (patch)
treea6d27d9602d0e9492bb69b9eb83a1dd08f0ab2b0
parente99b31fcedd2e4b453b3023ea5cd8662fc335c9b (diff)
change to IPv6
-rw-r--r--src/broadcaster.c56
-rw-r--r--src/common.h9
-rw-r--r--src/listener.c174
3 files changed, 100 insertions, 139 deletions
diff --git a/src/broadcaster.c b/src/broadcaster.c
index 9c02f79..c945b91 100644
--- a/src/broadcaster.c
+++ b/src/broadcaster.c
@@ -3,35 +3,22 @@
** this one can broadcast
*/
-#include <stdio.h>
-#include <stdlib.h>
-#include <unistd.h>
-#include <errno.h>
-#include <string.h>
-#include <sys/types.h>
-#include <sys/socket.h>
-#include <netinet/in.h>
-#include <arpa/inet.h>
-#include <netdb.h>
-
#include "common.h"
-int main(int argc, char *argv[])
+int main(int argc, char **argv)
{
- int sockfd;
- struct sockaddr_in their_addr; // connector's address information
- struct hostent *he;
- int numbytes;
+ int fd;
+ struct sockaddr_in6 addr;
int broadcast = 1;
- // assemble packet
+ /* assemble packet, parse cmd line */
if (argc < 3 || argc > 4) {
- fprintf(stderr,"usage: broadcaster hostname task [message]\n");
- exit(1);
+ fprintf(stderr, "usage: %s interface task [message]\n", argv[0]);
+ exit(1);
}
char buf[MAXBUFLEN],
*task = argv[2],
- *message = (argc == 4) ? argv[3] : NULL;
+ *message = (argc == 4) ? argv[3] : "";
int task_len = strlen(task),
total_len = task_len + 1 + strlen(message);
if (total_len > MAXBUFLEN) {
@@ -42,30 +29,23 @@ int main(int argc, char *argv[])
buf[task_len] = 0;
strcpy(buf + 1 + task_len, message);
- if ((he=gethostbyname(argv[1])) == NULL) { // get the host info
- perror("gethostbyname");
- exit(1);
- }
-
- if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) == -1) {
+ /* setup socket */
+ if ((fd = socket(AF_INET6, SOCK_DGRAM, 0)) == -1) {
perror("socket");
exit(1);
}
- // this call is what allows broadcast packets to be sent:
- if (setsockopt(sockfd, SOL_SOCKET, SO_BROADCAST, &broadcast,
- sizeof broadcast) == -1) {
- perror("setsockopt (SO_BROADCAST)");
- exit(1);
+ addr.sin6_family = AF_INET6;
+ addr.sin6_port = htons(SERVERPORT);
+ addr.sin6_flowinfo = 0;
+ inet_pton(AF_INET6, "ff02::1", &(addr.sin6_addr));
+ if ((addr.sin6_scope_id = if_nametoindex(argv[1])) == 0) {
+ fprintf(stderr, "interface not found\n");
+ exit(1);
}
- their_addr.sin_family = AF_INET; // host byte order
- their_addr.sin_port = htons(SERVERPORT); // short, network byte order
- their_addr.sin_addr = *((struct in_addr *)he->h_addr);
- memset(their_addr.sin_zero, '\0', sizeof their_addr.sin_zero);
-
- if ((numbytes=sendto(sockfd, buf, total_len, 0,
- (struct sockaddr *)&their_addr, sizeof their_addr)) == -1) {
+ /* send packet */
+ if (sendto(fd, buf, total_len, 0, (struct sockaddr *)&addr, sizeof addr) == -1) {
perror("sendto");
exit(1);
}
diff --git a/src/common.h b/src/common.h
index c64c764..0274856 100644
--- a/src/common.h
+++ b/src/common.h
@@ -1,6 +1,15 @@
#ifndef UDP_BROADCAST_COMMON
#define UDP_BROADCAST_COMMON
+#include <arpa/inet.h>
+#include <errno.h>
+#include <netinet/in.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <sys/socket.h>
+#include <sys/types.h>
+
#define SERVERPORT 4950 // the port users will be connecting to
#define SERVERPORT_S "4950"
#define MAXBUFLEN 100
diff --git a/src/listener.c b/src/listener.c
index c0d81ef..3c2ee64 100644
--- a/src/listener.c
+++ b/src/listener.c
@@ -2,118 +2,90 @@
** listener.c -- a datagram sockets "server" demo
*/
-#include <stdio.h>
-#include <stdlib.h>
-#include <unistd.h>
-#include <errno.h>
-#include <string.h>
-#include <sys/types.h>
-#include <sys/socket.h>
-#include <netinet/in.h>
-#include <arpa/inet.h>
-#include <netdb.h>
-
#include "common.h"
-// get sockaddr, IPv4 or IPv6:
-void *get_in_addr(struct sockaddr *sa)
-{
- if (sa->sa_family == AF_INET) {
- return &(((struct sockaddr_in*)sa)->sin_addr);
+int main(int argc, char **argv, char **envp) {
+ int fd;
+ int numbytes;
+ struct sockaddr_storage their_addr;
+ char buf[MAXBUFLEN];
+ socklen_t addr_len;
+
+ /* setup socket, parse cmd line */
+ {
+ struct sockaddr_in6 ba;
+ struct ipv6_mreq ga;
+ if (argc > 2) {
+ fprintf(stderr, "usage: %s [interface]\n", argv[0]);
+ exit(1);
}
-
- return &(((struct sockaddr_in6*)sa)->sin6_addr);
-}
-
-int main(int argc, char *argv[], char *envp[])
-{
- int sockfd;
- struct addrinfo hints, *servinfo, *p;
- int rv;
- int numbytes;
- struct sockaddr_storage their_addr;
- char buf[MAXBUFLEN];
- socklen_t addr_len;
- char s[INET6_ADDRSTRLEN];
-
- memset(&hints, 0, sizeof hints);
- hints.ai_family = AF_UNSPEC; // set to AF_INET to force IPv4
- hints.ai_socktype = SOCK_DGRAM;
- hints.ai_flags = AI_PASSIVE; // use my IP
-
- if ((rv = getaddrinfo(NULL, SERVERPORT_S, &hints, &servinfo)) != 0) {
- fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(rv));
- return 1;
+ if ((fd = socket(AF_INET6, SOCK_DGRAM, IPPROTO_UDP)) == -1) {
+ perror("listener: socket");
+ exit(1);
}
-
- // loop through all the results and bind to the first we can
- for(p = servinfo; p != NULL; p = p->ai_next) {
- if ((sockfd = socket(p->ai_family, p->ai_socktype,
- p->ai_protocol)) == -1) {
- perror("listener: socket");
- continue;
- }
-
- if (bind(sockfd, p->ai_addr, p->ai_addrlen) == -1) {
- close(sockfd);
- perror("listener: bind");
- continue;
- }
-
- break;
+ if (argc == 2) {
+ inet_pton(AF_INET6, "ff02::1", &(ba.sin6_addr));
+ if ((ba.sin6_scope_id = if_nametoindex(argv[1])) == 0) {
+ fprintf(stderr, "interface not found\n");
+ exit(1);
+ }
+ }else{
+ ba.sin6_addr = in6addr_any;
+ ba.sin6_scope_id = 0;
}
-
- if (p == NULL) {
- fprintf(stderr, "listener: failed to bind socket\n");
- return 2;
+ ba.sin6_family = AF_INET6;
+ ba.sin6_port = htons(SERVERPORT);
+ ba.sin6_flowinfo = 0;
+ if (bind(fd, (struct sockaddr*) &ba, sizeof ba) != 0) {
+ perror("listener: bind");
+ exit(1);
}
+ }
- freeaddrinfo(servinfo);
-
- // daemonize
+ /* daemonize */
#ifndef DEBUG
+ switch (fork()) {
+ case 0:
+ setsid();
+ umask(0);
+ break;
+ case -1:
+ perror("fork");
+ exit(1);
+ default:
+ exit(0);
+ }
+#endif
+
+ /* receive loop */
+ addr_len = sizeof their_addr;
+ while ((numbytes = recvfrom(fd, buf, MAXBUFLEN-1 , 0,
+ (struct sockaddr *) &their_addr, &addr_len)) != -1) {
+ /* decode packet & launch handler */
+ buf[numbytes] = '\0';
+ char *task = buf,
+ *cl_argv[3] = {task, (strlen(task)<numbytes) ? (buf+strlen(task)+1) : NULL, NULL };
+
+ if (task[0] == '/' || strstr(task, "..")) {
+ fprintf(stderr, "payload tried directory traversal\n");
+ continue;
+ }
+
switch (fork()) {
case 0:
- setsid();
- umask(0);
- break;
+ close(fd); // close socket handler, leave std io
+ execve(task, cl_argv, envp);
+ perror("exec");
+ exit(1);
case -1:
- perror("fork (serious!)");
- exit(-1);
+ perror("fork");
+ exit(1);
default:
- exit(0);
+ wait();
}
-#endif
-
- // receive loop
- addr_len = sizeof their_addr;
- while ((numbytes = recvfrom(sockfd, buf, MAXBUFLEN-1 , 0,
- (struct sockaddr *)&their_addr, &addr_len)) != -1) {
- // decode packet & launch handler
- buf[numbytes] = '\0';
- char *task = buf,
- *cl_argv[3] = {task, (strlen(task)<numbytes) ? (buf+strlen(task)+1) : NULL, NULL };
-
- if (task[0] == '/' || strstr(task, "..")) {
- fprintf(stderr, "payload tried directory traversal\n");
- continue;
- }
-
- switch (fork()) {
- case 0:
- close(sockfd); // close socket handler, leave std io
- execve(task, cl_argv, envp);
- perror("exec");
- exit(-1);
- case -1:
- perror("fork (serious!)");
- exit(-1);
- default:
- wait();
- }
- }
-
- perror("recvfrom");
-
- return -1;
+ }
+
+ perror("recvfrom");
+
+ return 1;
}
contact: Jan Huwald // Impressum