2009-01-03 23:27:21 +00:00
|
|
|
/* Copyright (c) 2006-2009 Bjorn Andersson <flex@kryo.se>, Erik Ekman <yarrick@kryo.se>
|
2008-07-12 12:26:41 +00:00
|
|
|
* Copyright (c) 2007 Albert Lee <trisk@acm.jhu.edu>.
|
2007-02-04 15:22:55 +00:00
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <time.h>
|
2008-07-12 12:26:41 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/param.h>
|
2007-03-01 21:19:01 +00:00
|
|
|
#include <sys/stat.h>
|
2007-02-04 15:22:55 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
2008-07-12 12:26:41 +00:00
|
|
|
#include <fcntl.h>
|
2009-08-15 21:52:49 +00:00
|
|
|
#include <errno.h>
|
2009-01-24 22:19:11 +00:00
|
|
|
|
|
|
|
#ifdef WINDOWS32
|
2009-02-22 16:38:43 +00:00
|
|
|
#include <winsock2.h>
|
|
|
|
#include <conio.h>
|
2009-01-24 22:19:11 +00:00
|
|
|
#else
|
|
|
|
#include <arpa/nameser.h>
|
|
|
|
#ifdef DARWIN
|
2009-09-19 08:32:57 +00:00
|
|
|
#define BIND_8_COMPAT
|
|
|
|
#include <arpa/nameser_compat.h>
|
2009-01-24 22:19:11 +00:00
|
|
|
#endif
|
2007-07-12 15:48:05 +00:00
|
|
|
#include <termios.h>
|
2009-01-24 22:19:11 +00:00
|
|
|
#include <err.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <netinet/in.h>
|
2009-08-15 21:52:49 +00:00
|
|
|
#include <syslog.h>
|
2009-01-24 22:19:11 +00:00
|
|
|
#endif
|
2007-02-04 15:22:55 +00:00
|
|
|
|
2009-06-24 17:28:13 +00:00
|
|
|
#ifdef HAVE_SETCON
|
|
|
|
# include <selinux/selinux.h>
|
|
|
|
#endif
|
|
|
|
|
2007-02-04 15:22:55 +00:00
|
|
|
#include "common.h"
|
|
|
|
|
2009-06-14 11:14:40 +00:00
|
|
|
/* The raw header used when not using DNS protocol */
|
|
|
|
const unsigned char raw_header[RAW_HDR_LEN] = { 0x10, 0xd1, 0x9e, 0x00 };
|
|
|
|
|
2008-07-12 12:26:41 +00:00
|
|
|
/* daemon(3) exists only in 4.4BSD or later, and in GNU libc */
|
2011-04-23 18:55:59 +00:00
|
|
|
#if !defined(ANDROID) && !defined(WINDOWS32) && !(defined(BSD) && (BSD >= 199306)) && !defined(__GLIBC__)
|
2008-07-12 12:26:41 +00:00
|
|
|
static int daemon(int nochdir, int noclose)
|
|
|
|
{
|
|
|
|
int fd, i;
|
|
|
|
|
|
|
|
switch (fork()) {
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
case -1:
|
|
|
|
return -1;
|
|
|
|
default:
|
|
|
|
_exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!nochdir) {
|
|
|
|
chdir("/");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (setsid() < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!noclose) {
|
2008-08-06 18:59:22 +00:00
|
|
|
if ((fd = open("/dev/null", O_RDWR)) >= 0) {
|
2008-07-12 12:26:41 +00:00
|
|
|
for (i = 0; i < 3; i++) {
|
|
|
|
dup2(fd, i);
|
|
|
|
}
|
|
|
|
if (fd > 2) {
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-08-07 16:53:59 +00:00
|
|
|
#if defined(__BEOS__) && !defined(__HAIKU__)
|
|
|
|
int setgroups(int count, int *groups)
|
|
|
|
{
|
|
|
|
/* errno = ENOSYS; */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-01-24 15:50:54 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
check_superuser(void (*usage_fn)(void))
|
|
|
|
{
|
2009-01-24 22:19:11 +00:00
|
|
|
#ifndef WINDOWS32
|
2009-01-24 15:50:54 +00:00
|
|
|
if (geteuid() != 0) {
|
|
|
|
warnx("Run as root and you'll be happy.\n");
|
|
|
|
usage_fn();
|
|
|
|
/* NOTREACHED */
|
|
|
|
}
|
2009-01-24 22:19:11 +00:00
|
|
|
#endif
|
2009-01-24 15:50:54 +00:00
|
|
|
}
|
|
|
|
|
2007-02-04 15:22:55 +00:00
|
|
|
int
|
|
|
|
open_dns(int localport, in_addr_t listen_ip)
|
|
|
|
{
|
|
|
|
struct sockaddr_in addr;
|
2009-01-24 22:19:11 +00:00
|
|
|
int flag = 1;
|
2007-02-04 15:22:55 +00:00
|
|
|
int fd;
|
|
|
|
|
|
|
|
memset(&addr, 0, sizeof(addr));
|
|
|
|
addr.sin_family = AF_INET;
|
|
|
|
addr.sin_port = htons(localport);
|
|
|
|
/* listen_ip already in network byte order from inet_addr, or 0 */
|
|
|
|
addr.sin_addr.s_addr = listen_ip;
|
|
|
|
|
2009-01-25 16:42:28 +00:00
|
|
|
if ((fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0) {
|
2009-02-28 14:35:03 +00:00
|
|
|
fprintf(stderr, "got fd %d\n", fd);
|
2007-02-04 15:22:55 +00:00
|
|
|
err(1, "socket");
|
2009-01-24 22:19:11 +00:00
|
|
|
}
|
2007-02-04 15:22:55 +00:00
|
|
|
|
|
|
|
flag = 1;
|
|
|
|
#ifdef SO_REUSEPORT
|
2009-01-25 16:42:28 +00:00
|
|
|
setsockopt(fd, SOL_SOCKET, SO_REUSEPORT, (const void*) &flag, sizeof(flag));
|
2007-02-04 15:22:55 +00:00
|
|
|
#endif
|
2009-01-25 16:42:28 +00:00
|
|
|
setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, (const void*) &flag, sizeof(flag));
|
2007-02-04 15:22:55 +00:00
|
|
|
|
2009-01-24 22:19:11 +00:00
|
|
|
#ifndef WINDOWS32
|
2008-08-07 22:12:10 +00:00
|
|
|
/* To get destination address from each UDP datagram, see iodined.c:read_dns() */
|
2009-01-25 16:42:28 +00:00
|
|
|
setsockopt(fd, IPPROTO_IP, DSTADDR_SOCKOPT, (const void*) &flag, sizeof(flag));
|
2009-01-24 22:19:11 +00:00
|
|
|
#endif
|
2008-08-07 22:12:10 +00:00
|
|
|
|
2009-03-07 00:21:06 +00:00
|
|
|
#ifdef IP_OPT_DONT_FRAG
|
2009-03-07 00:01:00 +00:00
|
|
|
/* Set dont-fragment ip header flag */
|
|
|
|
flag = DONT_FRAG_VALUE;
|
|
|
|
setsockopt(fd, IPPROTO_IP, IP_OPT_DONT_FRAG, (const void*) &flag, sizeof(flag));
|
2009-03-07 00:21:06 +00:00
|
|
|
#endif
|
2009-03-07 00:01:00 +00:00
|
|
|
|
2007-02-04 15:22:55 +00:00
|
|
|
if(bind(fd, (struct sockaddr*)&addr, sizeof(addr)) < 0)
|
|
|
|
err(1, "bind");
|
|
|
|
|
2009-02-28 14:35:03 +00:00
|
|
|
fprintf(stderr, "Opened UDP socket\n");
|
2007-02-04 15:22:55 +00:00
|
|
|
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
close_dns(int fd)
|
|
|
|
{
|
|
|
|
close(fd);
|
|
|
|
}
|
2007-02-04 20:37:36 +00:00
|
|
|
|
2007-03-01 21:14:51 +00:00
|
|
|
void
|
|
|
|
do_chroot(char *newroot)
|
|
|
|
{
|
2009-01-24 22:19:11 +00:00
|
|
|
#if !(defined(WINDOWS32) || defined(__BEOS__) || defined(__HAIKU__))
|
2007-07-12 13:36:24 +00:00
|
|
|
if (chroot(newroot) != 0 || chdir("/") != 0)
|
|
|
|
err(1, "%s", newroot);
|
2007-03-01 21:14:51 +00:00
|
|
|
|
2007-07-12 13:36:24 +00:00
|
|
|
seteuid(geteuid());
|
|
|
|
setuid(getuid());
|
2008-08-07 16:53:59 +00:00
|
|
|
#else
|
|
|
|
warnx("chroot not available");
|
|
|
|
#endif
|
2007-03-01 21:14:51 +00:00
|
|
|
}
|
2007-03-01 21:19:01 +00:00
|
|
|
|
2009-06-24 17:28:13 +00:00
|
|
|
void
|
|
|
|
do_setcon(char *context)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_SETCON
|
|
|
|
if (-1 == setcon(context))
|
|
|
|
err(1, "%s", context);
|
|
|
|
#else
|
|
|
|
warnx("No SELinux support built in");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2009-08-15 21:52:49 +00:00
|
|
|
void
|
|
|
|
do_pidfile(char *pidfile)
|
|
|
|
{
|
|
|
|
#ifndef WINDOWS32
|
|
|
|
FILE *file;
|
|
|
|
|
|
|
|
if ((file = fopen(pidfile, "w")) == NULL) {
|
|
|
|
syslog(LOG_ERR, "Cannot write pidfile to %s, exiting", pidfile);
|
|
|
|
err(1, "do_pidfile: Can not write pidfile to %s", pidfile);
|
|
|
|
} else {
|
|
|
|
fprintf(file, "%d\n", (int)getpid());
|
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
fprintf(stderr, "Windows version does not support pid file\n");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2007-03-01 21:19:01 +00:00
|
|
|
void
|
|
|
|
do_detach()
|
|
|
|
{
|
2009-01-24 22:19:11 +00:00
|
|
|
#ifndef WINDOWS32
|
2009-02-28 14:35:03 +00:00
|
|
|
fprintf(stderr, "Detaching from terminal...\n");
|
2007-03-01 21:19:01 +00:00
|
|
|
daemon(0, 0);
|
|
|
|
umask(0);
|
|
|
|
alarm(0);
|
2009-01-24 22:19:11 +00:00
|
|
|
#else
|
2009-02-28 14:35:03 +00:00
|
|
|
fprintf(stderr, "Windows version does not support detaching\n");
|
2009-01-24 22:19:11 +00:00
|
|
|
#endif
|
2007-03-01 21:19:01 +00:00
|
|
|
}
|
2007-07-12 15:48:05 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
read_password(char *buf, size_t len)
|
|
|
|
{
|
2010-11-25 17:48:50 +00:00
|
|
|
char pwd[80] = {0};
|
2009-01-24 22:19:11 +00:00
|
|
|
#ifndef WINDOWS32
|
2007-07-12 15:48:05 +00:00
|
|
|
struct termios old;
|
|
|
|
struct termios tp;
|
|
|
|
|
|
|
|
tcgetattr(0, &tp);
|
|
|
|
old = tp;
|
|
|
|
|
|
|
|
tp.c_lflag &= (~ECHO);
|
|
|
|
tcsetattr(0, TCSANOW, &tp);
|
2009-02-22 16:38:43 +00:00
|
|
|
#else
|
|
|
|
int i;
|
2009-01-24 22:19:11 +00:00
|
|
|
#endif
|
2007-07-12 15:48:05 +00:00
|
|
|
|
2009-02-28 14:35:03 +00:00
|
|
|
fprintf(stderr, "Enter password: ");
|
|
|
|
fflush(stderr);
|
2009-02-22 16:38:43 +00:00
|
|
|
#ifndef WINDOWS32
|
2010-11-25 17:48:50 +00:00
|
|
|
fscanf(stdin, "%79[^\n]", pwd);
|
2009-02-22 16:38:43 +00:00
|
|
|
#else
|
|
|
|
for (i = 0; i < sizeof(pwd); i++) {
|
|
|
|
pwd[i] = getch();
|
|
|
|
if (pwd[i] == '\r' || pwd[i] == '\n') {
|
|
|
|
pwd[i] = 0;
|
|
|
|
break;
|
|
|
|
} else if (pwd[i] == '\b') {
|
|
|
|
i--; /* Remove the \b char */
|
|
|
|
if (i >=0) i--; /* If not first char, remove one more */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2009-02-28 14:35:03 +00:00
|
|
|
fprintf(stderr, "\n");
|
2007-07-12 15:48:05 +00:00
|
|
|
|
2009-01-24 22:19:11 +00:00
|
|
|
#ifndef WINDOWS32
|
2007-07-12 15:48:05 +00:00
|
|
|
tcsetattr(0, TCSANOW, &old);
|
2009-01-24 22:19:11 +00:00
|
|
|
#endif
|
2007-07-12 15:48:05 +00:00
|
|
|
|
|
|
|
strncpy(buf, pwd, len);
|
|
|
|
buf[len-1] = '\0';
|
|
|
|
}
|
2008-07-12 11:41:01 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
check_topdomain(char *str)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if(str[0] == '.') /* special case */
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
for( i = 0; i < strlen(str); i++) {
|
|
|
|
if( isalpha(str[i]) || isdigit(str[i]) || str[i] == '-' || str[i] == '.' )
|
|
|
|
continue;
|
|
|
|
else
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2009-01-24 22:19:11 +00:00
|
|
|
|
2011-04-23 18:55:59 +00:00
|
|
|
#if defined(WINDOWS32) || defined(ANDROID)
|
|
|
|
#ifndef ANDROID
|
2009-01-24 22:19:11 +00:00
|
|
|
int
|
|
|
|
inet_aton(const char *cp, struct in_addr *inp)
|
|
|
|
{
|
|
|
|
inp->s_addr = inet_addr(cp);
|
|
|
|
return inp->s_addr != INADDR_ANY;
|
|
|
|
}
|
2011-04-23 18:55:59 +00:00
|
|
|
#endif
|
2009-01-25 19:34:33 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
warn(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list list;
|
|
|
|
|
|
|
|
va_start(list, fmt);
|
|
|
|
if (fmt) fprintf(stderr, fmt, list);
|
2011-04-23 18:55:59 +00:00
|
|
|
#ifndef ANDROID
|
2009-01-25 20:39:44 +00:00
|
|
|
if (errno == 0) {
|
|
|
|
fprintf(stderr, ": WSA error %d\n", WSAGetLastError());
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, ": %s\n", strerror(errno));
|
|
|
|
}
|
2011-04-23 18:55:59 +00:00
|
|
|
#endif
|
2009-01-25 19:34:33 +00:00
|
|
|
va_end(list);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
warnx(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list list;
|
|
|
|
|
|
|
|
va_start(list, fmt);
|
|
|
|
if (fmt) fprintf(stderr, fmt, list);
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
va_end(list);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
err(int eval, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list list;
|
|
|
|
|
|
|
|
va_start(list, fmt);
|
|
|
|
warn(fmt, list);
|
|
|
|
va_end(list);
|
|
|
|
exit(eval);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
errx(int eval, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list list;
|
|
|
|
|
|
|
|
va_start(list, fmt);
|
|
|
|
warnx(fmt, list);
|
|
|
|
va_end(list);
|
|
|
|
exit(eval);
|
|
|
|
}
|
2009-01-24 22:19:11 +00:00
|
|
|
#endif
|
|
|
|
|
2009-09-20 21:10:38 +00:00
|
|
|
|
|
|
|
int recent_seqno(int ourseqno, int gotseqno)
|
|
|
|
/* Return 1 if we've seen gotseqno recently (current or up to 3 back).
|
|
|
|
Return 0 if gotseqno is new (or very old).
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 4; i++, ourseqno--) {
|
|
|
|
if (ourseqno < 0)
|
|
|
|
ourseqno = 7;
|
|
|
|
if (gotseqno == ourseqno)
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|