diff src/common.c @ 41:54d28a81ca99

Small updates in preparation for stateful protocols
author Ivo Smits <Ivo@UCIS.nl>
date Thu, 16 May 2013 01:15:01 +0200
parents 47a34fe75c57
children e896392f7e03
line wrap: on
line diff
--- a/src/common.c	Mon May 06 23:58:25 2013 +0200
+++ b/src/common.c	Thu May 16 01:15:01 2013 +0200
@@ -62,6 +62,7 @@
 	int (*decode)(struct qtsession* sess, char* enc, char* raw, int len);
 	int (*init)(struct qtsession* sess);
 	int protocol_data_size;
+	void (*idle)(struct qtsession* sess);
 };
 struct qtsession {
 	struct qtproto protocol;
@@ -71,6 +72,8 @@
 	int remote_float;
 	struct sockaddr_in remote_addr;
 	int use_pi;
+	int poll_timeout;
+	void (*sendnetworkpacket)(struct qtsession* sess, char* msg, int len);
 };
 
 #ifdef COMBINED_BINARY
@@ -101,7 +104,7 @@
 	fprintf(stderr, "More information: http://wiki.ucis.nl/QuickTun\n");
 }
 
-int init_udp(struct qtsession* session) {
+static int init_udp(struct qtsession* session) {
 	char* envval;
 	fprintf(stderr, "Initializing UDP socket...\n");
 	int sfd = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP);
@@ -146,7 +149,7 @@
 	return sfd;
 }
 
-int init_tuntap(struct qtsession* session) {
+static int init_tuntap(struct qtsession* session) {
 	char* envval;
 	fprintf(stderr, "Initializing tun/tap device...\n");
 	int ttfd; //Tap device file descriptor
@@ -204,7 +207,7 @@
 	}
 }
 
-int drop_privileges() {
+static int drop_privileges() {
 	char* envval;
 	if (envval = getconf("SETUID")) {
 		if (setgroups(0, NULL) == -1) return errorexitp("setgroups");
@@ -216,14 +219,25 @@
 	chdir("/");
 }
 
+static void qtsendnetworkpacket(struct qtsession* session, char* msg, int len) {
+	if (session->remote_float == 0) {
+		len = write(session->fd_socket, msg, len);
+	} else if (session->remote_float == 2) {
+		len = sendto(session->fd_socket, msg, len, 0, (struct sockaddr*)&session->remote_addr, sizeof(struct sockaddr_in));
+	}
+}
+
 int qtrun(struct qtproto* p) {
 	if (getconf("DEBUG")) debug = 1;
 	struct qtsession session;
+	session.poll_timeout = -1;
 	session.protocol = *p;
 
 	if (init_udp(&session) < 0) return -1;
 	int sfd = session.fd_socket;
 
+	session.sendnetworkpacket = qtsendnetworkpacket;
+
 	if (init_tuntap(&session) < 0) return -1;
 	int ttfd = session.fd_dev;
 
@@ -253,21 +267,19 @@
 	char* buffer_enc = buffer_enc_a;
 
 	while (1) {
-		int len = poll(fds, 2, -1);
+		int len = poll(fds, 2, session.poll_timeout);
 		if (len < 0) return errorexitp("poll error");
 		else if (fds[0].revents & (POLLERR | POLLHUP | POLLNVAL)) return errorexit("poll error on tap device");
 		else if (fds[1].revents & (POLLHUP | POLLNVAL)) return errorexit("poll error on udp socket");
+		if (len == 0 && p->idle) p->idle(&session);
 		if (fds[0].revents & POLLIN) {
 			len = read(ttfd, buffer_raw + p->offset_raw, p->buffersize_raw + pi_length);
 			if (len < pi_length) errorexit("read packet smaller than header from tun device");
 			if (session.remote_float == 0 || session.remote_float == 2) {
 				len = p->encode(&session, buffer_raw + pi_length, buffer_enc, len - pi_length);
 				if (len < 0) return len;
-				if (session.remote_float == 0) {
-					len = write(sfd, buffer_enc + p->offset_enc, len);
-				} else {
-					len = sendto(sfd, buffer_enc + p->offset_enc, len, 0, (struct sockaddr*)&session.remote_addr, sizeof(session.remote_addr));
-				}
+				if (len == 0) continue; //encoding is not yet possible
+				qtsendnetworkpacket(&session, buffer_enc + p->offset_enc, len);
 			}
 		}
 		if (fds[1].revents & POLLERR) {
@@ -287,18 +299,17 @@
 				long long out;
 				len = sizeof(out);
 				getsockopt(sfd, SOL_SOCKET, SO_ERROR, &out, &len);
-				fprintf(stderr, "Received end of file on udp socket (error %d)\n", out);
+				fprintf(stderr, "Received end of file on udp socket (error %lld)\n", out);
 			} else {
 				len = p->decode(&session, buffer_enc, buffer_raw + pi_length, len);
-				if (len < 0) return len;
-				if (len != 0 && session.remote_float != 0 && (session.remote_addr.sin_addr.s_addr != recvaddr.sin_addr.s_addr || session.remote_addr.sin_port != recvaddr.sin_port)) {
+				if (len < 0) continue;
+				if (session.remote_float != 0 && (session.remote_addr.sin_addr.s_addr != recvaddr.sin_addr.s_addr || session.remote_addr.sin_port != recvaddr.sin_port)) {
 					fprintf(stderr, "Remote endpoint has changed to %08X:%d\n", ntohl(recvaddr.sin_addr.s_addr), ntohs(recvaddr.sin_port));
 					session.remote_addr = recvaddr;
 					session.remote_float = 2;
 				}
-				if (session.use_pi == 2) {
-					int ipver = 0;
-					if (len >= 1) ipver = (buffer_raw[p->offset_raw + pi_length] >> 4) & 0xf;
+				if (len > 0 && session.use_pi == 2) {
+					int ipver = (buffer_raw[p->offset_raw + pi_length] >> 4) & 0xf;
 					int pihdr = 0;
 #if defined linux
 					if (ipver == 4) pihdr = 0x0000 | (0x0008 << 16); //little endian: flags and protocol are swapped
@@ -309,14 +320,14 @@
 #endif
 					*(int*)(buffer_raw + p->offset_raw) = ipver;
 				}
-				write(ttfd, buffer_raw + p->offset_raw, len + pi_length);
+				if (len > 0) write(ttfd, buffer_raw + p->offset_raw, len + pi_length);
 			}
 		}
 	}
 	return 0;
 }
 
-char* getconfcmdargs(const char* name) {
+static char* getconfcmdargs(const char* name) {
 	int i;
 	for (i = 1; i < gargc - 2; i++) {
 		if (strcmp(gargv[i], "-c")) continue;