12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193 |
- /***************************************************************************
- * _ _ ____ _
- * Project ___| | | | _ \| |
- * / __| | | | |_) | |
- * | (__| |_| | _ <| |___
- * \___|\___/|_| \_\_____|
- *
- * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
- *
- * This software is licensed as described in the file COPYING, which
- * you should have received as part of this distribution. The terms
- * are also available at https://curl.se/docs/copyright.html.
- *
- * You may opt to use, copy, modify, merge, publish, distribute and/or sell
- * copies of the Software, and permit persons to whom the Software is
- * furnished to do so, under the terms of the COPYING file.
- *
- * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
- * KIND, either express or implied.
- *
- * SPDX-License-Identifier: curl
- *
- ***************************************************************************/
- #include "server_setup.h"
- #include <stdlib.h>
- /* Function
- *
- * Accepts a TCP connection on a custom port (IPv4 or IPv6). Connects to a
- * given addr + port backend (that is NOT extracted form the client's
- * request). The backend server default to connect to can be set with
- * --backend and --backendport.
- *
- * Read commands from FILE (set with --config). The commands control how to
- * act and is reset to defaults each client TCP connect.
- *
- * Config file keywords:
- *
- * "version [number: 5]" - requires the communication to use this version.
- * "nmethods_min [number: 1]" - the minimum numberf NMETHODS the client must
- * state
- * "nmethods_max [number: 3]" - the minimum numberf NMETHODS the client must
- * state
- * "user [string]" - the user name that must match (if method is 2)
- * "password [string]" - the password that must match (if method is 2)
- * "backend [IPv4]" - numerical IPv4 address of backend to connect to
- * "backendport [number:0]" - TCP port of backend to connect to. 0 means use
- the client's specified port number.
- * "method [number: 0]" - connect method to respond with:
- * 0 - no auth
- * 1 - GSSAPI (not supported)
- * 2 - user + password
- * "response [number]" - the decimal number to respond to a connect
- * SOCKS5: 0 is OK, SOCKS4: 90 is ok
- *
- */
- /* based on sockfilt.c */
- #include <signal.h>
- #ifdef HAVE_NETINET_IN_H
- #include <netinet/in.h>
- #endif
- #ifdef HAVE_NETINET_IN6_H
- #include <netinet/in6.h>
- #endif
- #ifdef HAVE_ARPA_INET_H
- #include <arpa/inet.h>
- #endif
- #ifdef HAVE_NETDB_H
- #include <netdb.h>
- #endif
- #include "curlx.h" /* from the private lib dir */
- #include "getpart.h"
- #include "inet_pton.h"
- #include "util.h"
- #include "server_sockaddr.h"
- #include "warnless.h"
- /* include memdebug.h last */
- #include "memdebug.h"
- #ifdef USE_WINSOCK
- #undef EINTR
- #define EINTR 4 /* errno.h value */
- #undef EAGAIN
- #define EAGAIN 11 /* errno.h value */
- #undef ENOMEM
- #define ENOMEM 12 /* errno.h value */
- #undef EINVAL
- #define EINVAL 22 /* errno.h value */
- #endif
- #define DEFAULT_PORT 8905
- #ifndef DEFAULT_LOGFILE
- #define DEFAULT_LOGFILE "log/socksd.log"
- #endif
- #ifndef DEFAULT_REQFILE
- #define DEFAULT_REQFILE "log/socksd-request.log"
- #endif
- #ifndef DEFAULT_CONFIG
- #define DEFAULT_CONFIG "socksd.config"
- #endif
- static const char *backendaddr = "127.0.0.1";
- static unsigned short backendport = 0; /* default is use client's */
- struct configurable {
- unsigned char version; /* initial version byte in the request must match
- this */
- unsigned char nmethods_min; /* minimum number of nmethods to expect */
- unsigned char nmethods_max; /* maximum number of nmethods to expect */
- unsigned char responseversion;
- unsigned char responsemethod;
- unsigned char reqcmd;
- unsigned char connectrep;
- unsigned short port; /* backend port */
- char addr[32]; /* backend IPv4 numerical */
- char user[256];
- char password[256];
- };
- #define CONFIG_VERSION 5
- #define CONFIG_NMETHODS_MIN 1 /* unauth, gssapi, auth */
- #define CONFIG_NMETHODS_MAX 3
- #define CONFIG_RESPONSEVERSION CONFIG_VERSION
- #define CONFIG_RESPONSEMETHOD 0 /* no auth */
- #define CONFIG_REQCMD 1 /* CONNECT */
- #define CONFIG_PORT backendport
- #define CONFIG_ADDR backendaddr
- #define CONFIG_CONNECTREP 0
- static struct configurable config;
- const char *serverlogfile = DEFAULT_LOGFILE;
- static const char *reqlogfile = DEFAULT_REQFILE;
- static const char *configfile = DEFAULT_CONFIG;
- static const char *socket_type = "IPv4";
- static unsigned short port = DEFAULT_PORT;
- static void resetdefaults(void)
- {
- logmsg("Reset to defaults");
- config.version = CONFIG_VERSION;
- config.nmethods_min = CONFIG_NMETHODS_MIN;
- config.nmethods_max = CONFIG_NMETHODS_MAX;
- config.responseversion = CONFIG_RESPONSEVERSION;
- config.responsemethod = CONFIG_RESPONSEMETHOD;
- config.reqcmd = CONFIG_REQCMD;
- config.connectrep = CONFIG_CONNECTREP;
- config.port = CONFIG_PORT;
- strcpy(config.addr, CONFIG_ADDR);
- strcpy(config.user, "user");
- strcpy(config.password, "password");
- }
- static unsigned char byteval(char *value)
- {
- unsigned long num = strtoul(value, NULL, 10);
- return num & 0xff;
- }
- static unsigned short shortval(char *value)
- {
- unsigned long num = strtoul(value, NULL, 10);
- return num & 0xffff;
- }
- static enum {
- socket_domain_inet = AF_INET
- #ifdef USE_IPV6
- , socket_domain_inet6 = AF_INET6
- #endif
- #ifdef USE_UNIX_SOCKETS
- , socket_domain_unix = AF_UNIX
- #endif
- } socket_domain = AF_INET;
- static void getconfig(void)
- {
- FILE *fp = fopen(configfile, FOPEN_READTEXT);
- resetdefaults();
- if(fp) {
- char buffer[512];
- logmsg("parse config file");
- while(fgets(buffer, sizeof(buffer), fp)) {
- char key[32];
- char value[260];
- if(2 == sscanf(buffer, "%31s %259s", key, value)) {
- if(!strcmp(key, "version")) {
- config.version = byteval(value);
- logmsg("version [%d] set", config.version);
- }
- else if(!strcmp(key, "nmethods_min")) {
- config.nmethods_min = byteval(value);
- logmsg("nmethods_min [%d] set", config.nmethods_min);
- }
- else if(!strcmp(key, "nmethods_max")) {
- config.nmethods_max = byteval(value);
- logmsg("nmethods_max [%d] set", config.nmethods_max);
- }
- else if(!strcmp(key, "backend")) {
- strcpy(config.addr, value);
- logmsg("backend [%s] set", config.addr);
- }
- else if(!strcmp(key, "backendport")) {
- config.port = shortval(value);
- logmsg("backendport [%d] set", config.port);
- }
- else if(!strcmp(key, "user")) {
- strcpy(config.user, value);
- logmsg("user [%s] set", config.user);
- }
- else if(!strcmp(key, "password")) {
- strcpy(config.password, value);
- logmsg("password [%s] set", config.password);
- }
- /* Methods:
- o X'00' NO AUTHENTICATION REQUIRED
- o X'01' GSSAPI
- o X'02' USERNAME/PASSWORD
- */
- else if(!strcmp(key, "method")) {
- config.responsemethod = byteval(value);
- logmsg("method [%d] set", config.responsemethod);
- }
- else if(!strcmp(key, "response")) {
- config.connectrep = byteval(value);
- logmsg("response [%d] set", config.connectrep);
- }
- }
- }
- fclose(fp);
- }
- }
- static void loghex(unsigned char *buffer, ssize_t len)
- {
- char data[1200];
- ssize_t i;
- unsigned char *ptr = buffer;
- char *optr = data;
- ssize_t width = 0;
- int left = sizeof(data);
- for(i = 0; i < len && (left >= 0); i++) {
- msnprintf(optr, left, "%02x", ptr[i]);
- width += 2;
- optr += 2;
- left -= 2;
- }
- if(width)
- logmsg("'%s'", data);
- }
- /* RFC 1928, SOCKS5 byte index */
- #define SOCKS5_VERSION 0
- #define SOCKS5_NMETHODS 1 /* number of methods that is listed */
- /* in the request: */
- #define SOCKS5_REQCMD 1
- #define SOCKS5_RESERVED 2
- #define SOCKS5_ATYP 3
- #define SOCKS5_DSTADDR 4
- /* connect response */
- #define SOCKS5_REP 1
- #define SOCKS5_BNDADDR 4
- /* auth request */
- #define SOCKS5_ULEN 1
- #define SOCKS5_UNAME 2
- #define SOCKS4_CD 1
- #define SOCKS4_DSTPORT 2
- /* connect to a given IPv4 address, not the one asked for */
- static curl_socket_t socksconnect(unsigned short connectport,
- const char *connectaddr)
- {
- int rc;
- srvr_sockaddr_union_t me;
- curl_socket_t sock = socket(AF_INET, SOCK_STREAM, 0);
- if(sock == CURL_SOCKET_BAD)
- return CURL_SOCKET_BAD;
- memset(&me.sa4, 0, sizeof(me.sa4));
- me.sa4.sin_family = AF_INET;
- me.sa4.sin_port = htons(connectport);
- me.sa4.sin_addr.s_addr = INADDR_ANY;
- Curl_inet_pton(AF_INET, connectaddr, &me.sa4.sin_addr);
- rc = connect(sock, &me.sa, sizeof(me.sa4));
- if(rc) {
- int error = SOCKERRNO;
- logmsg("Error connecting to %s:%hu: (%d) %s",
- connectaddr, connectport, error, sstrerror(error));
- return CURL_SOCKET_BAD;
- }
- logmsg("Connected fine to %s:%d", connectaddr, connectport);
- return sock;
- }
- static curl_socket_t socks4(curl_socket_t fd,
- unsigned char *buffer,
- ssize_t rc)
- {
- unsigned char response[256 + 16];
- curl_socket_t connfd;
- unsigned char cd;
- unsigned short s4port;
- if(buffer[SOCKS4_CD] != 1) {
- logmsg("SOCKS4 CD is not 1: %d", buffer[SOCKS4_CD]);
- return CURL_SOCKET_BAD;
- }
- if(rc < 9) {
- logmsg("SOCKS4 connect message too short: %zd", rc);
- return CURL_SOCKET_BAD;
- }
- if(!config.port)
- s4port = (unsigned short)((buffer[SOCKS4_DSTPORT] << 8) |
- (buffer[SOCKS4_DSTPORT + 1]));
- else
- s4port = config.port;
- connfd = socksconnect(s4port, config.addr);
- if(connfd == CURL_SOCKET_BAD) {
- /* failed */
- cd = 91;
- }
- else {
- /* success */
- cd = 90;
- }
- response[0] = 0; /* reply version 0 */
- response[1] = cd; /* result */
- /* copy port and address from connect request */
- memcpy(&response[2], &buffer[SOCKS4_DSTPORT], 6);
- rc = (send)(fd, (char *)response, 8, 0);
- if(rc != 8) {
- logmsg("Sending SOCKS4 response failed!");
- return CURL_SOCKET_BAD;
- }
- logmsg("Sent %zd bytes", rc);
- loghex(response, rc);
- if(cd == 90)
- /* now do the transfer */
- return connfd;
- if(connfd != CURL_SOCKET_BAD)
- sclose(connfd);
- return CURL_SOCKET_BAD;
- }
- static curl_socket_t sockit(curl_socket_t fd)
- {
- unsigned char buffer[2*256 + 16];
- unsigned char response[2*256 + 16];
- ssize_t rc;
- unsigned char len;
- unsigned char type;
- unsigned char rep = 0;
- unsigned char *address;
- unsigned short socksport;
- curl_socket_t connfd = CURL_SOCKET_BAD;
- unsigned short s5port;
- getconfig();
- rc = recv(fd, (char *)buffer, sizeof(buffer), 0);
- if(rc <= 0) {
- logmsg("SOCKS identifier message missing, recv returned %zd", rc);
- return CURL_SOCKET_BAD;
- }
- logmsg("READ %zd bytes", rc);
- loghex(buffer, rc);
- if(buffer[SOCKS5_VERSION] == 4)
- return socks4(fd, buffer, rc);
- if(rc < 3) {
- logmsg("SOCKS5 identifier message too short: %zd", rc);
- return CURL_SOCKET_BAD;
- }
- if(buffer[SOCKS5_VERSION] != config.version) {
- logmsg("VERSION byte not %d", config.version);
- return CURL_SOCKET_BAD;
- }
- if((buffer[SOCKS5_NMETHODS] < config.nmethods_min) ||
- (buffer[SOCKS5_NMETHODS] > config.nmethods_max)) {
- logmsg("NMETHODS byte not within %d - %d ",
- config.nmethods_min, config.nmethods_max);
- return CURL_SOCKET_BAD;
- }
- /* after NMETHODS follows that many bytes listing the methods the client
- says it supports */
- if(rc != (buffer[SOCKS5_NMETHODS] + 2)) {
- logmsg("Expected %d bytes, got %zd", buffer[SOCKS5_NMETHODS] + 2, rc);
- return CURL_SOCKET_BAD;
- }
- logmsg("Incoming request deemed fine!");
- /* respond with two bytes: VERSION + METHOD */
- response[0] = config.responseversion;
- response[1] = config.responsemethod;
- rc = (send)(fd, (char *)response, 2, 0);
- if(rc != 2) {
- logmsg("Sending response failed!");
- return CURL_SOCKET_BAD;
- }
- logmsg("Sent %zd bytes", rc);
- loghex(response, rc);
- /* expect the request or auth */
- rc = recv(fd, (char *)buffer, sizeof(buffer), 0);
- if(rc <= 0) {
- logmsg("SOCKS5 request or auth message missing, recv returned %zd", rc);
- return CURL_SOCKET_BAD;
- }
- logmsg("READ %zd bytes", rc);
- loghex(buffer, rc);
- if(config.responsemethod == 2) {
- /* RFC 1929 authentication
- +----+------+----------+------+----------+
- |VER | ULEN | UNAME | PLEN | PASSWD |
- +----+------+----------+------+----------+
- | 1 | 1 | 1 to 255 | 1 | 1 to 255 |
- +----+------+----------+------+----------+
- */
- unsigned char ulen;
- unsigned char plen;
- bool login = TRUE;
- if(rc < 5) {
- logmsg("Too short auth input: %zd", rc);
- return CURL_SOCKET_BAD;
- }
- if(buffer[SOCKS5_VERSION] != 1) {
- logmsg("Auth VERSION byte not 1, got %d", buffer[SOCKS5_VERSION]);
- return CURL_SOCKET_BAD;
- }
- ulen = buffer[SOCKS5_ULEN];
- if(rc < 4 + ulen) {
- logmsg("Too short packet for username: %zd", rc);
- return CURL_SOCKET_BAD;
- }
- plen = buffer[SOCKS5_ULEN + ulen + 1];
- if(rc < 3 + ulen + plen) {
- logmsg("Too short packet for ulen %d plen %d: %zd", ulen, plen, rc);
- return CURL_SOCKET_BAD;
- }
- if((ulen != strlen(config.user)) ||
- (plen != strlen(config.password)) ||
- memcmp(&buffer[SOCKS5_UNAME], config.user, ulen) ||
- memcmp(&buffer[SOCKS5_UNAME + ulen + 1], config.password, plen)) {
- /* no match! */
- logmsg("mismatched credentials!");
- login = FALSE;
- }
- response[0] = 1;
- response[1] = login ? 0 : 1;
- rc = (send)(fd, (char *)response, 2, 0);
- if(rc != 2) {
- logmsg("Sending auth response failed!");
- return CURL_SOCKET_BAD;
- }
- logmsg("Sent %zd bytes", rc);
- loghex(response, rc);
- if(!login)
- return CURL_SOCKET_BAD;
- /* expect the request */
- rc = recv(fd, (char *)buffer, sizeof(buffer), 0);
- if(rc <= 0) {
- logmsg("SOCKS5 request message missing, recv returned %zd", rc);
- return CURL_SOCKET_BAD;
- }
- logmsg("READ %zd bytes", rc);
- loghex(buffer, rc);
- }
- if(rc < 6) {
- logmsg("Too short for request: %zd", rc);
- return CURL_SOCKET_BAD;
- }
- if(buffer[SOCKS5_VERSION] != config.version) {
- logmsg("Request VERSION byte not %d", config.version);
- return CURL_SOCKET_BAD;
- }
- /* 1 == CONNECT */
- if(buffer[SOCKS5_REQCMD] != config.reqcmd) {
- logmsg("Request COMMAND byte not %d", config.reqcmd);
- return CURL_SOCKET_BAD;
- }
- /* reserved, should be zero */
- if(buffer[SOCKS5_RESERVED]) {
- logmsg("Request COMMAND byte not %d", config.reqcmd);
- return CURL_SOCKET_BAD;
- }
- /* ATYP:
- o IP V4 address: X'01'
- o DOMAINNAME: X'03'
- o IP V6 address: X'04'
- */
- type = buffer[SOCKS5_ATYP];
- address = &buffer[SOCKS5_DSTADDR];
- switch(type) {
- case 1:
- /* 4 bytes IPv4 address */
- len = 4;
- break;
- case 3:
- /* The first octet of the address field contains the number of octets of
- name that follow */
- len = buffer[SOCKS5_DSTADDR];
- len++;
- break;
- case 4:
- /* 16 bytes IPv6 address */
- len = 16;
- break;
- default:
- logmsg("Unknown ATYP %d", type);
- return CURL_SOCKET_BAD;
- }
- if(rc < (4 + len + 2)) {
- logmsg("Request too short: %zd, expected %d", rc, 4 + len + 2);
- return CURL_SOCKET_BAD;
- }
- logmsg("Received ATYP %d", type);
- {
- FILE *dump;
- dump = fopen(reqlogfile, "ab");
- if(dump) {
- int i;
- fprintf(dump, "atyp %u =>", type);
- switch(type) {
- case 1:
- /* 4 bytes IPv4 address */
- fprintf(dump, " %u.%u.%u.%u\n",
- address[0], address[1], address[2], address[3]);
- break;
- case 3:
- /* The first octet of the address field contains the number of octets
- of name that follow */
- fprintf(dump, " %.*s\n", len-1, &address[1]);
- break;
- case 4:
- /* 16 bytes IPv6 address */
- for(i = 0; i < 16; i++) {
- fprintf(dump, " %02x", address[i]);
- }
- fprintf(dump, "\n");
- break;
- }
- fclose(dump);
- }
- }
- if(!config.port) {
- unsigned char *portp = &buffer[SOCKS5_DSTADDR + len];
- s5port = (unsigned short)((portp[0] << 8) | (portp[1]));
- }
- else
- s5port = config.port;
- if(!config.connectrep)
- connfd = socksconnect(s5port, config.addr);
- if(connfd == CURL_SOCKET_BAD) {
- /* failed */
- rep = 1;
- }
- else {
- rep = config.connectrep;
- }
- /* */
- response[SOCKS5_VERSION] = config.responseversion;
- /*
- o REP Reply field:
- o X'00' succeeded
- o X'01' general SOCKS server failure
- o X'02' connection not allowed by ruleset
- o X'03' Network unreachable
- o X'04' Host unreachable
- o X'05' Connection refused
- o X'06' TTL expired
- o X'07' Command not supported
- o X'08' Address type not supported
- o X'09' to X'FF' unassigned
- */
- response[SOCKS5_REP] = rep;
- response[SOCKS5_RESERVED] = 0; /* must be zero */
- response[SOCKS5_ATYP] = type; /* address type */
- /* mirror back the original addr + port */
- /* address or hostname */
- memcpy(&response[SOCKS5_BNDADDR], address, len);
- /* port number */
- memcpy(&response[SOCKS5_BNDADDR + len],
- &buffer[SOCKS5_DSTADDR + len], sizeof(socksport));
- rc = (send)(fd, (char *)response, (SEND_TYPE_ARG3)(len + 6), 0);
- if(rc != (len + 6)) {
- logmsg("Sending connect response failed!");
- return CURL_SOCKET_BAD;
- }
- logmsg("Sent %zd bytes", rc);
- loghex(response, rc);
- if(!rep)
- return connfd;
- if(connfd != CURL_SOCKET_BAD)
- sclose(connfd);
- return CURL_SOCKET_BAD;
- }
- struct perclient {
- size_t fromremote;
- size_t fromclient;
- curl_socket_t remotefd;
- curl_socket_t clientfd;
- bool used;
- };
- /* return non-zero when transfer is done */
- static int tunnel(struct perclient *cp, fd_set *fds)
- {
- ssize_t nread;
- ssize_t nwrite;
- char buffer[512];
- if(FD_ISSET(cp->clientfd, fds)) {
- /* read from client, send to remote */
- nread = recv(cp->clientfd, buffer, sizeof(buffer), 0);
- if(nread > 0) {
- nwrite = send(cp->remotefd, (char *)buffer,
- (SEND_TYPE_ARG3)nread, 0);
- if(nwrite != nread)
- return 1;
- cp->fromclient += nwrite;
- }
- else
- return 1;
- }
- if(FD_ISSET(cp->remotefd, fds)) {
- /* read from remote, send to client */
- nread = recv(cp->remotefd, buffer, sizeof(buffer), 0);
- if(nread > 0) {
- nwrite = send(cp->clientfd, (char *)buffer,
- (SEND_TYPE_ARG3)nread, 0);
- if(nwrite != nread)
- return 1;
- cp->fromremote += nwrite;
- }
- else
- return 1;
- }
- return 0;
- }
- /*
- sockfdp is a pointer to an established stream or CURL_SOCKET_BAD
- if sockfd is CURL_SOCKET_BAD, listendfd is a listening socket we must
- accept()
- */
- static bool incoming(curl_socket_t listenfd)
- {
- fd_set fds_read;
- fd_set fds_write;
- fd_set fds_err;
- int clients = 0; /* connected clients */
- struct perclient c[2];
- memset(c, 0, sizeof(c));
- if(got_exit_signal) {
- logmsg("signalled to die, exiting...");
- return FALSE;
- }
- #ifdef HAVE_GETPPID
- /* As a last resort, quit if socks5 process becomes orphan. */
- if(getppid() <= 1) {
- logmsg("process becomes orphan, exiting");
- return FALSE;
- }
- #endif
- do {
- int i;
- ssize_t rc;
- int error = 0;
- curl_socket_t sockfd = listenfd;
- int maxfd = (int)sockfd;
- FD_ZERO(&fds_read);
- FD_ZERO(&fds_write);
- FD_ZERO(&fds_err);
- /* there's always a socket to wait for */
- #if defined(__DJGPP__)
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Warith-conversion"
- #endif
- FD_SET(sockfd, &fds_read);
- #if defined(__DJGPP__)
- #pragma GCC diagnostic pop
- #endif
- for(i = 0; i < 2; i++) {
- if(c[i].used) {
- curl_socket_t fd = c[i].clientfd;
- #if defined(__DJGPP__)
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Warith-conversion"
- #endif
- FD_SET(fd, &fds_read);
- #if defined(__DJGPP__)
- #pragma GCC diagnostic pop
- #endif
- if((int)fd > maxfd)
- maxfd = (int)fd;
- fd = c[i].remotefd;
- #if defined(__DJGPP__)
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Warith-conversion"
- #endif
- FD_SET(fd, &fds_read);
- #if defined(__DJGPP__)
- #pragma GCC diagnostic pop
- #endif
- if((int)fd > maxfd)
- maxfd = (int)fd;
- }
- }
- do {
- /* select() blocking behavior call on blocking descriptors please */
- rc = select(maxfd + 1, &fds_read, &fds_write, &fds_err, NULL);
- if(got_exit_signal) {
- logmsg("signalled to die, exiting...");
- return FALSE;
- }
- } while((rc == -1) && ((error = errno) == EINTR));
- if(rc < 0) {
- logmsg("select() failed with error: (%d) %s",
- error, strerror(error));
- return FALSE;
- }
- if((clients < 2) && FD_ISSET(sockfd, &fds_read)) {
- curl_socket_t newfd = accept(sockfd, NULL, NULL);
- if(CURL_SOCKET_BAD == newfd) {
- error = SOCKERRNO;
- logmsg("accept(%" FMT_SOCKET_T ", NULL, NULL) "
- "failed with error: (%d) %s",
- sockfd, error, sstrerror(error));
- }
- else {
- curl_socket_t remotefd;
- logmsg("====> Client connect, fd %" FMT_SOCKET_T ". "
- "Read config from %s", newfd, configfile);
- remotefd = sockit(newfd); /* SOCKS until done */
- if(remotefd == CURL_SOCKET_BAD) {
- logmsg("====> Client disconnect");
- sclose(newfd);
- }
- else {
- struct perclient *cp = &c[0];
- logmsg("====> Tunnel transfer");
- if(c[0].used)
- cp = &c[1];
- cp->fromremote = 0;
- cp->fromclient = 0;
- cp->clientfd = newfd;
- cp->remotefd = remotefd;
- cp->used = TRUE;
- clients++;
- }
- }
- }
- for(i = 0; i < 2; i++) {
- struct perclient *cp = &c[i];
- if(cp->used) {
- if(tunnel(cp, &fds_read)) {
- logmsg("SOCKS transfer completed. Bytes: < %zu > %zu",
- cp->fromremote, cp->fromclient);
- sclose(cp->clientfd);
- sclose(cp->remotefd);
- cp->used = FALSE;
- clients--;
- }
- }
- }
- } while(clients);
- return TRUE;
- }
- static curl_socket_t sockdaemon(curl_socket_t sock,
- unsigned short *listenport
- #ifdef USE_UNIX_SOCKETS
- , const char *unix_socket
- #endif
- )
- {
- /* passive daemon style */
- srvr_sockaddr_union_t listener;
- int flag;
- int rc;
- int totdelay = 0;
- int maxretr = 10;
- int delay = 20;
- int attempt = 0;
- int error = 0;
- do {
- attempt++;
- flag = 1;
- rc = setsockopt(sock, SOL_SOCKET, SO_REUSEADDR,
- (void *)&flag, sizeof(flag));
- if(rc) {
- error = SOCKERRNO;
- logmsg("setsockopt(SO_REUSEADDR) failed with error: (%d) %s",
- error, sstrerror(error));
- if(maxretr) {
- rc = wait_ms(delay);
- if(rc) {
- /* should not happen */
- error = errno;
- logmsg("wait_ms() failed with error: (%d) %s",
- error, strerror(error));
- sclose(sock);
- return CURL_SOCKET_BAD;
- }
- if(got_exit_signal) {
- logmsg("signalled to die, exiting...");
- sclose(sock);
- return CURL_SOCKET_BAD;
- }
- totdelay += delay;
- delay *= 2; /* double the sleep for next attempt */
- }
- }
- } while(rc && maxretr--);
- if(rc) {
- logmsg("setsockopt(SO_REUSEADDR) failed %d times in %d ms. Error: (%d) %s",
- attempt, totdelay, error, strerror(error));
- logmsg("Continuing anyway...");
- }
- /* When the specified listener port is zero, it is actually a
- request to let the system choose a non-zero available port. */
- switch(socket_domain) {
- case AF_INET:
- memset(&listener.sa4, 0, sizeof(listener.sa4));
- listener.sa4.sin_family = AF_INET;
- listener.sa4.sin_addr.s_addr = INADDR_ANY;
- listener.sa4.sin_port = htons(*listenport);
- rc = bind(sock, &listener.sa, sizeof(listener.sa4));
- break;
- #ifdef USE_IPV6
- case AF_INET6:
- memset(&listener.sa6, 0, sizeof(listener.sa6));
- listener.sa6.sin6_family = AF_INET6;
- listener.sa6.sin6_addr = in6addr_any;
- listener.sa6.sin6_port = htons(*listenport);
- rc = bind(sock, &listener.sa, sizeof(listener.sa6));
- break;
- #endif /* USE_IPV6 */
- #ifdef USE_UNIX_SOCKETS
- case AF_UNIX:
- rc = bind_unix_socket(sock, unix_socket, &listener.sau);
- #endif
- }
- if(rc) {
- error = SOCKERRNO;
- #ifdef USE_UNIX_SOCKETS
- if(socket_domain == AF_UNIX)
- logmsg("Error binding socket on path %s: (%d) %s",
- unix_socket, error, sstrerror(error));
- else
- #endif
- logmsg("Error binding socket on port %hu: (%d) %s",
- *listenport, error, sstrerror(error));
- sclose(sock);
- return CURL_SOCKET_BAD;
- }
- if(!*listenport
- #ifdef USE_UNIX_SOCKETS
- && !unix_socket
- #endif
- ) {
- /* The system was supposed to choose a port number, figure out which
- port we actually got and update the listener port value with it. */
- curl_socklen_t la_size;
- srvr_sockaddr_union_t localaddr;
- #ifdef USE_IPV6
- if(socket_domain == AF_INET6)
- la_size = sizeof(localaddr.sa6);
- else
- #endif
- la_size = sizeof(localaddr.sa4);
- memset(&localaddr.sa, 0, (size_t)la_size);
- if(getsockname(sock, &localaddr.sa, &la_size) < 0) {
- error = SOCKERRNO;
- logmsg("getsockname() failed with error: (%d) %s",
- error, sstrerror(error));
- sclose(sock);
- return CURL_SOCKET_BAD;
- }
- switch(localaddr.sa.sa_family) {
- case AF_INET:
- *listenport = ntohs(localaddr.sa4.sin_port);
- break;
- #ifdef USE_IPV6
- case AF_INET6:
- *listenport = ntohs(localaddr.sa6.sin6_port);
- break;
- #endif
- default:
- break;
- }
- if(!*listenport) {
- /* Real failure, listener port shall not be zero beyond this point. */
- logmsg("Apparently getsockname() succeeded, with listener port zero.");
- logmsg("A valid reason for this failure is a binary built without");
- logmsg("proper network library linkage. This might not be the only");
- logmsg("reason, but double check it before anything else.");
- sclose(sock);
- return CURL_SOCKET_BAD;
- }
- }
- /* start accepting connections */
- rc = listen(sock, 5);
- if(0 != rc) {
- error = SOCKERRNO;
- logmsg("listen(%" FMT_SOCKET_T ", 5) failed with error: (%d) %s",
- sock, error, sstrerror(error));
- sclose(sock);
- return CURL_SOCKET_BAD;
- }
- return sock;
- }
- int main(int argc, char *argv[])
- {
- curl_socket_t sock = CURL_SOCKET_BAD;
- curl_socket_t msgsock = CURL_SOCKET_BAD;
- int wrotepidfile = 0;
- int wroteportfile = 0;
- const char *pidname = ".socksd.pid";
- const char *portname = NULL; /* none by default */
- bool juggle_again;
- int error;
- int arg = 1;
- #ifdef USE_UNIX_SOCKETS
- const char *unix_socket = NULL;
- bool unlink_socket = false;
- #endif
- while(argc > arg) {
- if(!strcmp("--version", argv[arg])) {
- printf("socksd IPv4%s\n",
- #ifdef USE_IPV6
- "/IPv6"
- #else
- ""
- #endif
- );
- return 0;
- }
- else if(!strcmp("--pidfile", argv[arg])) {
- arg++;
- if(argc > arg)
- pidname = argv[arg++];
- }
- else if(!strcmp("--portfile", argv[arg])) {
- arg++;
- if(argc > arg)
- portname = argv[arg++];
- }
- else if(!strcmp("--config", argv[arg])) {
- arg++;
- if(argc > arg)
- configfile = argv[arg++];
- }
- else if(!strcmp("--backend", argv[arg])) {
- arg++;
- if(argc > arg)
- backendaddr = argv[arg++];
- }
- else if(!strcmp("--backendport", argv[arg])) {
- arg++;
- if(argc > arg)
- backendport = (unsigned short)atoi(argv[arg++]);
- }
- else if(!strcmp("--logfile", argv[arg])) {
- arg++;
- if(argc > arg)
- serverlogfile = argv[arg++];
- }
- else if(!strcmp("--reqfile", argv[arg])) {
- arg++;
- if(argc > arg)
- reqlogfile = argv[arg++];
- }
- else if(!strcmp("--ipv6", argv[arg])) {
- #ifdef USE_IPV6
- socket_domain = AF_INET6;
- socket_type = "IPv6";
- #endif
- arg++;
- }
- else if(!strcmp("--ipv4", argv[arg])) {
- /* for completeness, we support this option as well */
- #ifdef USE_IPV6
- socket_type = "IPv4";
- #endif
- arg++;
- }
- else if(!strcmp("--unix-socket", argv[arg])) {
- arg++;
- if(argc > arg) {
- #ifdef USE_UNIX_SOCKETS
- struct sockaddr_un sau;
- unix_socket = argv[arg];
- if(strlen(unix_socket) >= sizeof(sau.sun_path)) {
- fprintf(stderr,
- "socksd: socket path must be shorter than %zu chars: %s\n",
- sizeof(sau.sun_path), unix_socket);
- return 0;
- }
- socket_domain = AF_UNIX;
- socket_type = "unix";
- #endif
- arg++;
- }
- }
- else if(!strcmp("--port", argv[arg])) {
- arg++;
- if(argc > arg) {
- char *endptr;
- unsigned long ulnum = strtoul(argv[arg], &endptr, 10);
- port = curlx_ultous(ulnum);
- arg++;
- }
- }
- else {
- puts("Usage: socksd [option]\n"
- " --backend [ipv4 addr]\n"
- " --backendport [TCP port]\n"
- " --config [file]\n"
- " --version\n"
- " --logfile [file]\n"
- " --pidfile [file]\n"
- " --portfile [file]\n"
- " --reqfile [file]\n"
- " --ipv4\n"
- " --ipv6\n"
- " --unix-socket [file]\n"
- " --bindonly\n"
- " --port [port]\n");
- return 0;
- }
- }
- #ifdef _WIN32
- win32_init();
- atexit(win32_cleanup);
- setmode(fileno(stdin), O_BINARY);
- setmode(fileno(stdout), O_BINARY);
- setmode(fileno(stderr), O_BINARY);
- #endif
- install_signal_handlers(false);
- sock = socket(socket_domain, SOCK_STREAM, 0);
- if(CURL_SOCKET_BAD == sock) {
- error = SOCKERRNO;
- logmsg("Error creating socket: (%d) %s",
- error, sstrerror(error));
- goto socks5_cleanup;
- }
- {
- /* passive daemon style */
- sock = sockdaemon(sock, &port
- #ifdef USE_UNIX_SOCKETS
- , unix_socket
- #endif
- );
- if(CURL_SOCKET_BAD == sock) {
- goto socks5_cleanup;
- }
- #ifdef USE_UNIX_SOCKETS
- unlink_socket = true;
- #endif
- msgsock = CURL_SOCKET_BAD; /* no stream socket yet */
- }
- logmsg("Running %s version", socket_type);
- #ifdef USE_UNIX_SOCKETS
- if(socket_domain == AF_UNIX)
- logmsg("Listening on Unix socket %s", unix_socket);
- else
- #endif
- logmsg("Listening on port %hu", port);
- wrotepidfile = write_pidfile(pidname);
- if(!wrotepidfile) {
- goto socks5_cleanup;
- }
- if(portname) {
- wroteportfile = write_portfile(portname, port);
- if(!wroteportfile) {
- goto socks5_cleanup;
- }
- }
- do {
- juggle_again = incoming(sock);
- } while(juggle_again);
- socks5_cleanup:
- if((msgsock != sock) && (msgsock != CURL_SOCKET_BAD))
- sclose(msgsock);
- if(sock != CURL_SOCKET_BAD)
- sclose(sock);
- #ifdef USE_UNIX_SOCKETS
- if(unlink_socket && socket_domain == AF_UNIX) {
- error = unlink(unix_socket);
- logmsg("unlink(%s) = %d (%s)", unix_socket, error, strerror(error));
- }
- #endif
- if(wrotepidfile)
- unlink(pidname);
- if(wroteportfile)
- unlink(portname);
- restore_signal_handlers(false);
- if(got_exit_signal) {
- logmsg("============> socksd exits with signal (%d)", exit_signal);
- /*
- * To properly set the return status of the process we
- * must raise the same signal SIGINT or SIGTERM that we
- * caught and let the old handler take care of it.
- */
- raise(exit_signal);
- }
- logmsg("============> socksd quits");
- return 0;
- }
|