diff options
author | miau <miaukatzemiau@priveasy.de> | 2011-12-01 14:13:35 (GMT) |
---|---|---|
committer | miau <miaukatzemiau@priveasy.de> | 2011-12-01 14:13:35 (GMT) |
commit | b4b4c79cf6276e8d2cb851f4ea3d40102d11c5a4 (patch) | |
tree | a6d27d9602d0e9492bb69b9eb83a1dd08f0ab2b0 | |
parent | e99b31fcedd2e4b453b3023ea5cd8662fc335c9b (diff) |
change to IPv6
-rw-r--r-- | src/broadcaster.c | 56 | ||||
-rw-r--r-- | src/common.h | 9 | ||||
-rw-r--r-- | src/listener.c | 174 |
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; } |