Permalink
Cannot retrieve contributors at this time
Name already in use
A tag already exists with the provided branch name. Many Git commands accept both tag and branch names, so creating this branch may cause unexpected behavior. Are you sure you want to create this branch?
redsocks/base.c
Go to fileThis commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
541 lines (482 sloc)
13.8 KB
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| /* redsocks - transparent TCP-to-proxy redirector | |
| * Copyright (C) 2007-2018 Leonid Evdokimov <leon@darkk.net.ru> | |
| * | |
| * Licensed under the Apache License, Version 2.0 (the "License"); you may not | |
| * use this file except in compliance with the License. You may obtain a copy | |
| * of the License at | |
| * | |
| * http://www.apache.org/licenses/LICENSE-2.0 | |
| * | |
| * Unless required by applicable law or agreed to in writing, software | |
| * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | |
| * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | |
| * License for the specific language governing permissions and limitations | |
| * under the License. | |
| */ | |
| #include <sys/types.h> | |
| #include <sys/socket.h> | |
| #include <netinet/in.h> | |
| #include <netinet/tcp.h> | |
| #include <arpa/inet.h> | |
| #include <sys/stat.h> | |
| #include <sys/resource.h> | |
| #include <fcntl.h> | |
| #include <unistd.h> | |
| #include <string.h> | |
| #include <pwd.h> | |
| #include <grp.h> | |
| #include <stdlib.h> | |
| #include "config.h" | |
| #if defined USE_IPTABLES | |
| # include <limits.h> | |
| # include <linux/netfilter_ipv4.h> | |
| #endif | |
| #if defined USE_PF | |
| # include <net/if.h> | |
| # include <net/pfvar.h> | |
| # include <sys/ioctl.h> | |
| # include <errno.h> | |
| #endif | |
| #include "log.h" | |
| #include "main.h" | |
| #include "parser.h" | |
| #include "redsocks.h" | |
| typedef struct redirector_subsys_t { | |
| int (*init)(); | |
| void (*fini)(); | |
| int (*getdestaddr)(int fd, const struct sockaddr_in *client, const struct sockaddr_in *bindaddr, struct sockaddr_in *destaddr); | |
| const char *name; | |
| // some subsystems may store data here: | |
| int private; | |
| } redirector_subsys; | |
| typedef struct base_instance_t { | |
| int configured; | |
| char *chroot; | |
| char *user; | |
| char *group; | |
| char *redirector_name; | |
| redirector_subsys *redirector; | |
| char *log_name; | |
| bool log_debug; | |
| bool log_info; | |
| bool daemon; | |
| #if defined(TCP_KEEPIDLE) && defined(TCP_KEEPCNT) && defined(TCP_KEEPINTVL) | |
| uint16_t tcp_keepalive_time; | |
| uint16_t tcp_keepalive_probes; | |
| uint16_t tcp_keepalive_intvl; | |
| #endif | |
| uint32_t rlimit_nofile; | |
| uint32_t redsocks_conn_max; | |
| uint32_t connpres_idle_timeout; | |
| uint32_t max_accept_backoff_ms; | |
| } base_instance; | |
| static base_instance instance; | |
| #if defined __FreeBSD__ || defined USE_PF | |
| static int redir_open_private(const char *fname, int flags) | |
| { | |
| int fd = open(fname, flags); | |
| if (fd < 0) { | |
| log_errno(LOG_ERR, "open(%s)", fname); | |
| return -1; | |
| } | |
| instance.redirector->private = fd; | |
| return 0; | |
| } | |
| static void redir_close_private() | |
| { | |
| close(instance.redirector->private); | |
| instance.redirector->private = -1; | |
| } | |
| #endif | |
| #ifdef __FreeBSD__ | |
| static int redir_init_ipf() | |
| { | |
| #ifdef IPNAT_NAME | |
| const char *fname = IPNAT_NAME; | |
| #else | |
| const char *fname = IPL_NAME; | |
| #endif | |
| return redir_open_private(fname, O_RDONLY); | |
| } | |
| static int getdestaddr_ipf(int fd, const struct sockaddr_in *client, const struct sockaddr_in *bindaddr, struct sockaddr_in *destaddr) | |
| { | |
| int natfd = instance.redirector->private; | |
| struct natlookup natLookup; | |
| int x; | |
| #if defined(IPFILTER_VERSION) && (IPFILTER_VERSION >= 4000027) | |
| struct ipfobj obj; | |
| #else | |
| static int siocgnatl_cmd = SIOCGNATL & 0xff; | |
| #endif | |
| #if defined(IPFILTER_VERSION) && (IPFILTER_VERSION >= 4000027) | |
| obj.ipfo_rev = IPFILTER_VERSION; | |
| obj.ipfo_size = sizeof(natLookup); | |
| obj.ipfo_ptr = &natLookup; | |
| obj.ipfo_type = IPFOBJ_NATLOOKUP; | |
| obj.ipfo_offset = 0; | |
| #endif | |
| natLookup.nl_inport = bindaddr->sin_port; | |
| natLookup.nl_outport = client->sin_port; | |
| natLookup.nl_inip = bindaddr->sin_addr; | |
| natLookup.nl_outip = client->sin_addr; | |
| natLookup.nl_flags = IPN_TCP; | |
| #if defined(IPFILTER_VERSION) && (IPFILTER_VERSION >= 4000027) | |
| x = ioctl(natfd, SIOCGNATL, &obj); | |
| #else | |
| /* | |
| * IP-Filter changed the type for SIOCGNATL between | |
| * 3.3 and 3.4. It also changed the cmd value for | |
| * SIOCGNATL, so at least we can detect it. We could | |
| * put something in configure and use ifdefs here, but | |
| * this seems simpler. | |
| */ | |
| if (63 == siocgnatl_cmd) { | |
| struct natlookup *nlp = &natLookup; | |
| x = ioctl(natfd, SIOCGNATL, &nlp); | |
| } else { | |
| x = ioctl(natfd, SIOCGNATL, &natLookup); | |
| } | |
| #endif | |
| if (x < 0) { | |
| if (errno != ESRCH) | |
| log_errno(LOG_WARNING, "ioctl(SIOCGNATL)\n"); | |
| return -1; | |
| } else { | |
| destaddr->sin_family = AF_INET; | |
| destaddr->sin_port = natLookup.nl_realport; | |
| destaddr->sin_addr = natLookup.nl_realip; | |
| return 0; | |
| } | |
| } | |
| #endif | |
| #ifdef USE_PF | |
| static int redir_init_pf() | |
| { | |
| return redir_open_private("/dev/pf", O_RDWR); | |
| } | |
| static int getdestaddr_pf( | |
| int fd, const struct sockaddr_in *client, const struct sockaddr_in *bindaddr, | |
| struct sockaddr_in *destaddr) | |
| { | |
| int pffd = instance.redirector->private; | |
| struct pfioc_natlook nl; | |
| int saved_errno; | |
| char clientaddr_str[INET6_ADDRSTRLEN], bindaddr_str[INET6_ADDRSTRLEN]; | |
| memset(&nl, 0, sizeof(struct pfioc_natlook)); | |
| nl.saddr.v4.s_addr = client->sin_addr.s_addr; | |
| nl.sport = client->sin_port; | |
| nl.daddr.v4.s_addr = bindaddr->sin_addr.s_addr; | |
| nl.dport = bindaddr->sin_port; | |
| nl.af = AF_INET; | |
| nl.proto = IPPROTO_TCP; | |
| nl.direction = PF_OUT; | |
| if (ioctl(pffd, DIOCNATLOOK, &nl) != 0) { | |
| if (errno == ENOENT) { | |
| nl.direction = PF_IN; // required to redirect local packets | |
| if (ioctl(pffd, DIOCNATLOOK, &nl) != 0) { | |
| goto fail; | |
| } | |
| } | |
| else { | |
| goto fail; | |
| } | |
| } | |
| destaddr->sin_family = AF_INET; | |
| destaddr->sin_port = nl.rdport; | |
| destaddr->sin_addr = nl.rdaddr.v4; | |
| return 0; | |
| fail: | |
| saved_errno = errno; | |
| if (!inet_ntop(client->sin_family, &client->sin_addr, clientaddr_str, sizeof(clientaddr_str))) | |
| strncpy(clientaddr_str, "???", sizeof(clientaddr_str)); | |
| if (!inet_ntop(bindaddr->sin_family, &bindaddr->sin_addr, bindaddr_str, sizeof(bindaddr_str))) | |
| strncpy(bindaddr_str, "???", sizeof(bindaddr_str)); | |
| errno = saved_errno; | |
| log_errno(LOG_WARNING, "ioctl(DIOCNATLOOK {src=%s:%d, dst=%s:%d})", | |
| clientaddr_str, ntohs(nl.sport), bindaddr_str, ntohs(nl.dport)); | |
| return -1; | |
| } | |
| #endif | |
| #ifdef USE_IPTABLES | |
| static int getdestaddr_iptables(int fd, const struct sockaddr_in *client, const struct sockaddr_in *bindaddr, struct sockaddr_in *destaddr) | |
| { | |
| socklen_t socklen = sizeof(*destaddr); | |
| int error; | |
| error = getsockopt(fd, SOL_IP, SO_ORIGINAL_DST, destaddr, &socklen); | |
| if (error) { | |
| log_errno(LOG_WARNING, "getsockopt"); | |
| return -1; | |
| } | |
| return 0; | |
| } | |
| #endif | |
| static int getdestaddr_generic(int fd, const struct sockaddr_in *client, const struct sockaddr_in *bindaddr, struct sockaddr_in *destaddr) | |
| { | |
| socklen_t socklen = sizeof(*destaddr); | |
| int error; | |
| error = getsockname(fd, (struct sockaddr*)destaddr, &socklen); | |
| if (error) { | |
| log_errno(LOG_WARNING, "getsockopt"); | |
| return -1; | |
| } | |
| return 0; | |
| } | |
| int getdestaddr(int fd, const struct sockaddr_in *client, const struct sockaddr_in *bindaddr, struct sockaddr_in *destaddr) | |
| { | |
| return instance.redirector->getdestaddr(fd, client, bindaddr, destaddr); | |
| } | |
| int apply_tcp_keepalive(int fd) | |
| { | |
| struct { int level, option, value; } opt[] = { | |
| { SOL_SOCKET, SO_KEEPALIVE, 1 }, | |
| { IPPROTO_TCP, TCP_KEEPIDLE, instance.tcp_keepalive_time }, | |
| { IPPROTO_TCP, TCP_KEEPCNT, instance.tcp_keepalive_probes }, | |
| { IPPROTO_TCP, TCP_KEEPINTVL, instance.tcp_keepalive_intvl }, | |
| }; | |
| for (int i = 0; i < SIZEOF_ARRAY(opt); ++i) { | |
| if (opt[i].value) { | |
| int error = setsockopt(fd, opt[i].level, opt[i].option, &opt[i].value, sizeof(opt[i].value)); | |
| if (error) { | |
| log_errno(LOG_WARNING, "setsockopt(%d, %d, %d, &%d, %zu)", fd, opt[i].level, opt[i].option, opt[i].value, sizeof(opt[i].value)); | |
| return -1; | |
| } | |
| } | |
| } | |
| return 0; | |
| } | |
| uint32_t max_accept_backoff_ms() | |
| { | |
| return instance.max_accept_backoff_ms; | |
| } | |
| uint32_t redsocks_conn_max() | |
| { | |
| return instance.redsocks_conn_max; | |
| } | |
| uint32_t connpres_idle_timeout() | |
| { | |
| return instance.connpres_idle_timeout; | |
| } | |
| static redirector_subsys redirector_subsystems[] = | |
| { | |
| #ifdef __FreeBSD__ | |
| { .name = "ipf", .init = redir_init_ipf, .fini = redir_close_private, .getdestaddr = getdestaddr_ipf }, | |
| #endif | |
| #ifdef USE_PF | |
| { .name = "pf", .init = redir_init_pf, .fini = redir_close_private, .getdestaddr = getdestaddr_pf }, | |
| #endif | |
| #ifdef USE_IPTABLES | |
| { .name = "iptables", .getdestaddr = getdestaddr_iptables }, | |
| #endif | |
| { .name = "generic", .getdestaddr = getdestaddr_generic }, | |
| }; | |
| /*********************************************************************** | |
| * `base` config parsing | |
| */ | |
| static parser_entry base_entries[] = | |
| { | |
| { .key = "chroot", .type = pt_pchar, .addr = &instance.chroot }, | |
| { .key = "user", .type = pt_pchar, .addr = &instance.user }, | |
| { .key = "group", .type = pt_pchar, .addr = &instance.group }, | |
| { .key = "redirector", .type = pt_pchar, .addr = &instance.redirector_name }, | |
| { .key = "log", .type = pt_pchar, .addr = &instance.log_name }, | |
| { .key = "log_debug", .type = pt_bool, .addr = &instance.log_debug }, | |
| { .key = "log_info", .type = pt_bool, .addr = &instance.log_info }, | |
| { .key = "daemon", .type = pt_bool, .addr = &instance.daemon }, | |
| #if defined(TCP_KEEPIDLE) && defined(TCP_KEEPCNT) && defined(TCP_KEEPINTVL) | |
| { .key = "tcp_keepalive_time", .type = pt_uint16, .addr = &instance.tcp_keepalive_time }, | |
| { .key = "tcp_keepalive_probes", .type = pt_uint16, .addr = &instance.tcp_keepalive_probes }, | |
| { .key = "tcp_keepalive_intvl", .type = pt_uint16, .addr = &instance.tcp_keepalive_intvl }, | |
| #endif | |
| { .key = "rlimit_nofile", .type = pt_uint32, .addr = &instance.rlimit_nofile }, | |
| { .key = "redsocks_conn_max", .type = pt_uint32, .addr = &instance.redsocks_conn_max }, | |
| { .key = "connpres_idle_timeout", .type = pt_uint32, .addr = &instance.connpres_idle_timeout }, | |
| { .key = "max_accept_backoff", .type = pt_uint32, .addr = &instance.max_accept_backoff_ms }, | |
| { } | |
| }; | |
| static int base_onenter(parser_section *section) | |
| { | |
| if (instance.configured) { | |
| parser_error(section->context, "only one instance of base is valid"); | |
| return -1; | |
| } | |
| memset(&instance, 0, sizeof(instance)); | |
| instance.configured = 1; | |
| instance.max_accept_backoff_ms = 60000; | |
| instance.connpres_idle_timeout = 7440; | |
| return 0; | |
| } | |
| static int base_onexit(parser_section *section) | |
| { | |
| if (!instance.max_accept_backoff_ms) { | |
| parser_error(section->context, "`max_accept_backoff` must be positive, 0 ms is too low"); | |
| return -1; | |
| } | |
| if (instance.redirector_name) { | |
| redirector_subsys *ss; | |
| FOREACH(ss, redirector_subsystems) { | |
| if (!strcmp(ss->name, instance.redirector_name)) { | |
| instance.redirector = ss; | |
| instance.redirector->private = -1; | |
| break; | |
| } | |
| } | |
| if (!instance.redirector) { | |
| parser_error(section->context, "invalid `redirector` set <%s>", instance.redirector_name); | |
| return -1; | |
| } | |
| } | |
| else { | |
| parser_error(section->context, "no `redirector` set"); | |
| return -1; | |
| } | |
| return 0; | |
| } | |
| static parser_section base_conf_section = | |
| { | |
| .name = "base", | |
| .entries = base_entries, | |
| .onenter = base_onenter, | |
| .onexit = base_onexit | |
| }; | |
| /*********************************************************************** | |
| * `base` initialization | |
| */ | |
| static int base_fini(); | |
| static int base_init() | |
| { | |
| uid_t uid = -1; | |
| gid_t gid = -1; | |
| int devnull = -1; | |
| if (!instance.configured) { | |
| log_error(LOG_ERR, "there is no configured instance of `base`, check config file"); | |
| return -1; | |
| } | |
| if (instance.redirector->init && instance.redirector->init() < 0) | |
| return -1; | |
| if (instance.user) { | |
| struct passwd *pw = getpwnam(instance.user); | |
| if (pw == NULL) { | |
| log_errno(LOG_ERR, "getpwnam(%s)", instance.user); | |
| goto fail; | |
| } | |
| uid = pw->pw_uid; | |
| } | |
| if (instance.group) { | |
| struct group *gr = getgrnam(instance.group); | |
| if (gr == NULL) { | |
| log_errno(LOG_ERR, "getgrnam(%s)", instance.group); | |
| goto fail; | |
| } | |
| gid = gr->gr_gid; | |
| } | |
| if (log_preopen( | |
| instance.log_name ? instance.log_name : instance.daemon ? "syslog:daemon" : "stderr", | |
| instance.log_debug, | |
| instance.log_info | |
| ) < 0 ) { | |
| goto fail; | |
| } | |
| if (instance.rlimit_nofile) { | |
| struct rlimit rlmt; | |
| rlmt.rlim_cur = instance.rlimit_nofile; | |
| rlmt.rlim_max = instance.rlimit_nofile; | |
| if (setrlimit(RLIMIT_NOFILE, &rlmt) != 0) { | |
| log_errno(LOG_ERR, "setrlimit(RLIMIT_NOFILE, %u)", instance.rlimit_nofile); | |
| goto fail; | |
| } | |
| } else { | |
| struct rlimit rlmt; | |
| if (getrlimit(RLIMIT_NOFILE, &rlmt) != 0) { | |
| log_errno(LOG_ERR, "getrlimit(RLIMIT_NOFILE)"); | |
| goto fail; | |
| } | |
| instance.rlimit_nofile = rlmt.rlim_cur; | |
| } | |
| if (!instance.redsocks_conn_max) { | |
| instance.redsocks_conn_max = (instance.rlimit_nofile - instance.rlimit_nofile / 4) | |
| / (redsocks_has_splice_instance() ? 6 : 2); | |
| } | |
| if (instance.daemon) { | |
| devnull = open("/dev/null", O_RDWR); | |
| if (devnull == -1) { | |
| log_errno(LOG_ERR, "open(\"/dev/null\", O_RDWR"); | |
| goto fail; | |
| } | |
| } | |
| if (instance.chroot) { | |
| if (chroot(instance.chroot) < 0) { | |
| log_errno(LOG_ERR, "chroot(%s)", instance.chroot); | |
| goto fail; | |
| } | |
| } | |
| if (instance.daemon || instance.chroot) { | |
| if (chdir("/") < 0) { | |
| log_errno(LOG_ERR, "chdir(\"/\")"); | |
| goto fail; | |
| } | |
| } | |
| if (instance.group) { | |
| if (setgid(gid) < 0) { | |
| log_errno(LOG_ERR, "setgid(%i)", gid); | |
| goto fail; | |
| } | |
| } | |
| if (instance.user) { | |
| if (setuid(uid) < 0) { | |
| log_errno(LOG_ERR, "setuid(%i)", uid); | |
| goto fail; | |
| } | |
| } | |
| if (instance.daemon) { | |
| switch (fork()) { | |
| case -1: // error | |
| log_errno(LOG_ERR, "fork()"); | |
| goto fail; | |
| case 0: // child | |
| break; | |
| default: // parent, pid is returned | |
| exit(EXIT_SUCCESS); | |
| } | |
| } | |
| log_open(); // child has nothing to do with TTY | |
| if (instance.daemon) { | |
| if (setsid() < 0) { | |
| log_errno(LOG_ERR, "setsid()"); | |
| goto fail; | |
| } | |
| int fds[] = { STDIN_FILENO, STDOUT_FILENO, STDERR_FILENO }; | |
| int *pfd; | |
| FOREACH(pfd, fds) | |
| if (dup2(devnull, *pfd) < 0) { | |
| log_errno(LOG_ERR, "dup2(devnull, %i)", *pfd); | |
| goto fail; | |
| } | |
| close(devnull); | |
| } | |
| return 0; | |
| fail: | |
| if (devnull != -1) | |
| close(devnull); | |
| base_fini(); | |
| return -1; | |
| } | |
| static int base_fini() | |
| { | |
| if (instance.redirector->fini) | |
| instance.redirector->fini(); | |
| free(instance.chroot); | |
| free(instance.user); | |
| free(instance.group); | |
| free(instance.redirector_name); | |
| free(instance.log_name); | |
| memset(&instance, 0, sizeof(instance)); | |
| return 0; | |
| } | |
| app_subsys base_subsys = | |
| { | |
| .init = base_init, | |
| .fini = base_fini, | |
| .conf_section = &base_conf_section, | |
| }; | |
| /* vim:set tabstop=4 softtabstop=4 shiftwidth=4: */ | |
| /* vim:set foldmethod=marker foldlevel=32 foldmarker={,}: */ |