• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /***************************************************************************
2  *                                  _   _ ____  _
3  *  Project                     ___| | | |  _ \| |
4  *                             / __| | | | |_) | |
5  *                            | (__| |_| |  _ <| |___
6  *                             \___|\___/|_| \_\_____|
7  *
8  * Copyright (C) 1998 - 2021, Daniel Stenberg, <daniel@haxx.se>, et al.
9  *
10  * This software is licensed as described in the file COPYING, which
11  * you should have received as part of this distribution. The terms
12  * are also available at https://curl.se/docs/copyright.html.
13  *
14  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15  * copies of the Software, and permit persons to whom the Software is
16  * furnished to do so, under the terms of the COPYING file.
17  *
18  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19  * KIND, either express or implied.
20  *
21  ***************************************************************************/
22 #include "server_setup.h"
23 #include <stdlib.h>
24 
25 /* Function
26  *
27  * Accepts a TCP connection on a custom port (IPv4 or IPv6). Connects to a
28  * given addr + port backend (that is NOT extracted form the client's
29  * request). The backend server default to connect to can be set with
30  * --backend and --backendport.
31  *
32  * Read commands from FILE (set with --config). The commands control how to
33  * act and is reset to defaults each client TCP connect.
34  *
35  * Config file keywords:
36  *
37  * "version [number: 5]" - requires the communication to use this version.
38  * "nmethods_min [number: 1]" - the minimum numberf NMETHODS the client must
39  *                              state
40  * "nmethods_max [number: 3]" - the minimum numberf NMETHODS the client must
41  *                              state
42  * "user [string]" - the user name that must match (if method is 2)
43  * "password [string]" - the password that must match (if method is 2)
44  * "backend [IPv4]" - numerical IPv4 address of backend to connect to
45  * "backendport [number:0]" - TCP port of backend to connect to. 0 means use
46                               the client's specified port number.
47  * "method [number: 0]" - connect method to respond with:
48  *                        0 - no auth
49  *                        1 - GSSAPI (not supported)
50  *                        2 - user + password
51  * "response [number]" - the decimal number to respond to a connect
52  *                       SOCKS5: 0 is OK, SOCKS4: 90 is ok
53  *
54  */
55 
56 /* based on sockfilt.c */
57 
58 #ifdef HAVE_SIGNAL_H
59 #include <signal.h>
60 #endif
61 #ifdef HAVE_NETINET_IN_H
62 #include <netinet/in.h>
63 #endif
64 #ifdef HAVE_NETINET_IN6_H
65 #include <netinet/in6.h>
66 #endif
67 #ifdef HAVE_ARPA_INET_H
68 #include <arpa/inet.h>
69 #endif
70 #ifdef HAVE_NETDB_H
71 #include <netdb.h>
72 #endif
73 
74 #define ENABLE_CURLX_PRINTF
75 /* make the curlx header define all printf() functions to use the curlx_*
76    versions instead */
77 #include "curlx.h" /* from the private lib dir */
78 #include "getpart.h"
79 #include "inet_pton.h"
80 #include "util.h"
81 #include "server_sockaddr.h"
82 #include "warnless.h"
83 
84 /* include memdebug.h last */
85 #include "memdebug.h"
86 
87 #ifdef USE_WINSOCK
88 #undef  EINTR
89 #define EINTR    4 /* errno.h value */
90 #undef  EAGAIN
91 #define EAGAIN  11 /* errno.h value */
92 #undef  ENOMEM
93 #define ENOMEM  12 /* errno.h value */
94 #undef  EINVAL
95 #define EINVAL  22 /* errno.h value */
96 #endif
97 
98 #define DEFAULT_PORT 8905
99 
100 #ifndef DEFAULT_LOGFILE
101 #define DEFAULT_LOGFILE "log/socksd.log"
102 #endif
103 
104 #ifndef DEFAULT_CONFIG
105 #define DEFAULT_CONFIG "socksd.config"
106 #endif
107 
108 static const char *backendaddr = "127.0.0.1";
109 static unsigned short backendport = 0; /* default is use client's */
110 
111 struct configurable {
112   unsigned char version; /* initial version byte in the request must match
113                             this */
114   unsigned char nmethods_min; /* minimum number of nmethods to expect */
115   unsigned char nmethods_max; /* maximum number of nmethods to expect */
116   unsigned char responseversion;
117   unsigned char responsemethod;
118   unsigned char reqcmd;
119   unsigned char connectrep;
120   unsigned short port; /* backend port */
121   char addr[32]; /* backend IPv4 numerical */
122   char user[256];
123   char password[256];
124 };
125 
126 #define CONFIG_VERSION 5
127 #define CONFIG_NMETHODS_MIN 1 /* unauth, gssapi, auth */
128 #define CONFIG_NMETHODS_MAX 3
129 #define CONFIG_RESPONSEVERSION CONFIG_VERSION
130 #define CONFIG_RESPONSEMETHOD 0 /* no auth */
131 #define CONFIG_REQCMD 1 /* CONNECT */
132 #define CONFIG_PORT backendport
133 #define CONFIG_ADDR backendaddr
134 #define CONFIG_CONNECTREP 0
135 
136 static struct configurable config;
137 
138 const char *serverlogfile = DEFAULT_LOGFILE;
139 static const char *configfile = DEFAULT_CONFIG;
140 
141 #ifdef ENABLE_IPV6
142 static bool use_ipv6 = FALSE;
143 #endif
144 static const char *ipv_inuse = "IPv4";
145 static unsigned short port = DEFAULT_PORT;
146 
resetdefaults(void)147 static void resetdefaults(void)
148 {
149   logmsg("Reset to defaults");
150   config.version = CONFIG_VERSION;
151   config.nmethods_min = CONFIG_NMETHODS_MIN;
152   config.nmethods_max = CONFIG_NMETHODS_MAX;
153   config.responseversion = CONFIG_RESPONSEVERSION;
154   config.responsemethod = CONFIG_RESPONSEMETHOD;
155   config.reqcmd = CONFIG_REQCMD;
156   config.connectrep = CONFIG_CONNECTREP;
157   config.port = CONFIG_PORT;
158   strcpy(config.addr, CONFIG_ADDR);
159   strcpy(config.user, "user");
160   strcpy(config.password, "password");
161 }
162 
byteval(char * value)163 static unsigned char byteval(char *value)
164 {
165   unsigned long num = strtoul(value, NULL, 10);
166   return num & 0xff;
167 }
168 
shortval(char * value)169 static unsigned short shortval(char *value)
170 {
171   unsigned long num = strtoul(value, NULL, 10);
172   return num & 0xffff;
173 }
174 
getconfig(void)175 static void getconfig(void)
176 {
177   FILE *fp = fopen(configfile, FOPEN_READTEXT);
178   resetdefaults();
179   if(fp) {
180     char buffer[512];
181     logmsg("parse config file");
182     while(fgets(buffer, sizeof(buffer), fp)) {
183       char key[32];
184       char value[32];
185       if(2 == sscanf(buffer, "%31s %31s", key, value)) {
186         if(!strcmp(key, "version")) {
187           config.version = byteval(value);
188           logmsg("version [%d] set", config.version);
189         }
190         else if(!strcmp(key, "nmethods_min")) {
191           config.nmethods_min = byteval(value);
192           logmsg("nmethods_min [%d] set", config.nmethods_min);
193         }
194         else if(!strcmp(key, "nmethods_max")) {
195           config.nmethods_max = byteval(value);
196           logmsg("nmethods_max [%d] set", config.nmethods_max);
197         }
198         else if(!strcmp(key, "backend")) {
199           strcpy(config.addr, value);
200           logmsg("backend [%s] set", config.addr);
201         }
202         else if(!strcmp(key, "backendport")) {
203           config.port = shortval(value);
204           logmsg("backendport [%d] set", config.port);
205         }
206         else if(!strcmp(key, "user")) {
207           strcpy(config.user, value);
208           logmsg("user [%s] set", config.user);
209         }
210         else if(!strcmp(key, "password")) {
211           strcpy(config.password, value);
212           logmsg("password [%s] set", config.password);
213         }
214         /* Methods:
215            o  X'00' NO AUTHENTICATION REQUIRED
216            o  X'01' GSSAPI
217            o  X'02' USERNAME/PASSWORD
218         */
219         else if(!strcmp(key, "method")) {
220           config.responsemethod = byteval(value);
221           logmsg("method [%d] set", config.responsemethod);
222         }
223         else if(!strcmp(key, "response")) {
224           config.connectrep = byteval(value);
225           logmsg("response [%d] set", config.connectrep);
226         }
227       }
228     }
229     fclose(fp);
230   }
231 }
232 
loghex(unsigned char * buffer,ssize_t len)233 static void loghex(unsigned char *buffer, ssize_t len)
234 {
235   char data[1200];
236   ssize_t i;
237   unsigned char *ptr = buffer;
238   char *optr = data;
239   ssize_t width = 0;
240   int left = sizeof(data);
241 
242   for(i = 0; i<len && (left >= 0); i++) {
243     msnprintf(optr, left, "%02x", ptr[i]);
244     width += 2;
245     optr += 2;
246     left -= 2;
247   }
248   if(width)
249     logmsg("'%s'", data);
250 }
251 
252 /* RFC 1928, SOCKS5 byte index */
253 #define SOCKS5_VERSION 0
254 #define SOCKS5_NMETHODS 1 /* number of methods that is listed */
255 
256 /* in the request: */
257 #define SOCKS5_REQCMD 1
258 #define SOCKS5_RESERVED 2
259 #define SOCKS5_ATYP 3
260 #define SOCKS5_DSTADDR 4
261 
262 /* connect response */
263 #define SOCKS5_REP 1
264 #define SOCKS5_BNDADDR 4
265 
266 /* auth request */
267 #define SOCKS5_ULEN 1
268 #define SOCKS5_UNAME 2
269 
270 #define SOCKS4_CD 1
271 #define SOCKS4_DSTPORT 2
272 
273 /* connect to a given IPv4 address, not the one asked for */
socksconnect(unsigned short connectport,const char * connectaddr)274 static curl_socket_t socksconnect(unsigned short connectport,
275                                   const char *connectaddr)
276 {
277   int rc;
278   srvr_sockaddr_union_t me;
279   curl_socket_t sock = socket(AF_INET, SOCK_STREAM, 0);
280   if(sock == CURL_SOCKET_BAD)
281     return CURL_SOCKET_BAD;
282   memset(&me.sa4, 0, sizeof(me.sa4));
283   me.sa4.sin_family = AF_INET;
284   me.sa4.sin_port = htons(connectport);
285   me.sa4.sin_addr.s_addr = INADDR_ANY;
286   Curl_inet_pton(AF_INET, connectaddr, &me.sa4.sin_addr);
287 
288   rc = connect(sock, &me.sa, sizeof(me.sa4));
289 
290   if(rc) {
291     int error = SOCKERRNO;
292     logmsg("Error connecting to %s:%hu: (%d) %s",
293            connectaddr, connectport, error, strerror(error));
294     return CURL_SOCKET_BAD;
295   }
296   logmsg("Connected fine to %s:%d", connectaddr, connectport);
297   return sock;
298 }
299 
socks4(curl_socket_t fd,unsigned char * buffer,ssize_t rc)300 static curl_socket_t socks4(curl_socket_t fd,
301                             unsigned char *buffer,
302                             ssize_t rc)
303 {
304   unsigned char response[256 + 16];
305   curl_socket_t connfd;
306   unsigned char cd;
307   unsigned short s4port;
308 
309   if(buffer[SOCKS4_CD] != 1) {
310     logmsg("SOCKS4 CD is not 1: %d", buffer[SOCKS4_CD]);
311     return CURL_SOCKET_BAD;
312   }
313   if(rc < 9) {
314     logmsg("SOCKS4 connect message too short: %d", rc);
315     return CURL_SOCKET_BAD;
316   }
317   if(!config.port)
318     s4port = (unsigned short)((buffer[SOCKS4_DSTPORT]<<8) |
319                               (buffer[SOCKS4_DSTPORT + 1]));
320   else
321     s4port = config.port;
322 
323   connfd = socksconnect(s4port, config.addr);
324   if(connfd == CURL_SOCKET_BAD) {
325     /* failed */
326     cd = 91;
327   }
328   else {
329     /* success */
330     cd = 90;
331   }
332   response[0] = 0; /* reply version 0 */
333   response[1] = cd; /* result */
334   /* copy port and address from connect request */
335   memcpy(&response[2], &buffer[SOCKS4_DSTPORT], 6);
336   rc = (send)(fd, (char *)response, 8, 0);
337   if(rc != 8) {
338     logmsg("Sending SOCKS4 response failed!");
339     return CURL_SOCKET_BAD;
340   }
341   logmsg("Sent %d bytes", rc);
342   loghex(response, rc);
343 
344   if(cd == 90)
345     /* now do the transfer */
346     return connfd;
347 
348   if(connfd != CURL_SOCKET_BAD)
349     sclose(connfd);
350 
351   return CURL_SOCKET_BAD;
352 }
353 
sockit(curl_socket_t fd)354 static curl_socket_t sockit(curl_socket_t fd)
355 {
356   unsigned char buffer[256 + 16];
357   unsigned char response[256 + 16];
358   ssize_t rc;
359   unsigned char len;
360   unsigned char type;
361   unsigned char rep = 0;
362   unsigned char *address;
363   unsigned short socksport;
364   curl_socket_t connfd = CURL_SOCKET_BAD;
365   unsigned short s5port;
366 
367   getconfig();
368 
369   rc = recv(fd, (char *)buffer, sizeof(buffer), 0);
370 
371   logmsg("READ %d bytes", rc);
372   loghex(buffer, rc);
373 
374   if(buffer[SOCKS5_VERSION] == 4)
375     return socks4(fd, buffer, rc);
376 
377   if(buffer[SOCKS5_VERSION] != config.version) {
378     logmsg("VERSION byte not %d", config.version);
379     return CURL_SOCKET_BAD;
380   }
381   if((buffer[SOCKS5_NMETHODS] < config.nmethods_min) ||
382      (buffer[SOCKS5_NMETHODS] > config.nmethods_max)) {
383     logmsg("NMETHODS byte not within %d - %d ",
384            config.nmethods_min, config.nmethods_max);
385     return CURL_SOCKET_BAD;
386   }
387   /* after NMETHODS follows that many bytes listing the methods the client
388      says it supports */
389   if(rc != (buffer[SOCKS5_NMETHODS] + 2)) {
390     logmsg("Expected %d bytes, got %d", buffer[SOCKS5_NMETHODS] + 2, rc);
391     return CURL_SOCKET_BAD;
392   }
393   logmsg("Incoming request deemed fine!");
394 
395   /* respond with two bytes: VERSION + METHOD */
396   response[0] = config.responseversion;
397   response[1] = config.responsemethod;
398   rc = (send)(fd, (char *)response, 2, 0);
399   if(rc != 2) {
400     logmsg("Sending response failed!");
401     return CURL_SOCKET_BAD;
402   }
403   logmsg("Sent %d bytes", rc);
404   loghex(response, rc);
405 
406   /* expect the request or auth */
407   rc = recv(fd, (char *)buffer, sizeof(buffer), 0);
408 
409   logmsg("READ %d bytes", rc);
410   loghex(buffer, rc);
411 
412   if(config.responsemethod == 2) {
413     /* RFC 1929 authentication
414        +----+------+----------+------+----------+
415        |VER | ULEN |  UNAME   | PLEN |  PASSWD  |
416        +----+------+----------+------+----------+
417        | 1  |  1   | 1 to 255 |  1   | 1 to 255 |
418        +----+------+----------+------+----------+
419     */
420     unsigned char ulen;
421     unsigned char plen;
422     bool login = TRUE;
423     if(rc < 5) {
424       logmsg("Too short auth input: %d", rc);
425       return CURL_SOCKET_BAD;
426     }
427     if(buffer[SOCKS5_VERSION] != 1) {
428       logmsg("Auth VERSION byte not 1, got %d", buffer[SOCKS5_VERSION]);
429       return CURL_SOCKET_BAD;
430     }
431     ulen = buffer[SOCKS5_ULEN];
432     if(rc < 4 + ulen) {
433       logmsg("Too short packet for username: %d", rc);
434       return CURL_SOCKET_BAD;
435     }
436     plen = buffer[SOCKS5_ULEN + ulen + 1];
437     if(rc < 3 + ulen + plen) {
438       logmsg("Too short packet for ulen %d plen %d: %d", ulen, plen, rc);
439       return CURL_SOCKET_BAD;
440     }
441     if((ulen != strlen(config.user)) ||
442        (plen != strlen(config.password)) ||
443        memcmp(&buffer[SOCKS5_UNAME], config.user, ulen) ||
444        memcmp(&buffer[SOCKS5_UNAME + ulen + 1], config.password, plen)) {
445       /* no match! */
446       logmsg("mismatched credentials!");
447       login = FALSE;
448     }
449     response[0] = 1;
450     response[1] = login ? 0 : 1;
451     rc = (send)(fd, (char *)response, 2, 0);
452     if(rc != 2) {
453       logmsg("Sending auth response failed!");
454       return CURL_SOCKET_BAD;
455     }
456     logmsg("Sent %d bytes", rc);
457     loghex(response, rc);
458     if(!login)
459       return CURL_SOCKET_BAD;
460 
461     /* expect the request */
462     rc = recv(fd, (char *)buffer, sizeof(buffer), 0);
463 
464     logmsg("READ %d bytes", rc);
465     loghex(buffer, rc);
466   }
467   if(rc < 6) {
468     logmsg("Too short for request: %d", rc);
469     return CURL_SOCKET_BAD;
470   }
471 
472   if(buffer[SOCKS5_VERSION] != config.version) {
473     logmsg("Request VERSION byte not %d", config.version);
474     return CURL_SOCKET_BAD;
475   }
476   /* 1 == CONNECT */
477   if(buffer[SOCKS5_REQCMD] != config.reqcmd) {
478     logmsg("Request COMMAND byte not %d", config.reqcmd);
479     return CURL_SOCKET_BAD;
480   }
481   /* reserved, should be zero */
482   if(buffer[SOCKS5_RESERVED]) {
483     logmsg("Request COMMAND byte not %d", config.reqcmd);
484     return CURL_SOCKET_BAD;
485   }
486   /* ATYP:
487      o  IP V4 address: X'01'
488      o  DOMAINNAME: X'03'
489      o  IP V6 address: X'04'
490   */
491   type = buffer[SOCKS5_ATYP];
492   address = &buffer[SOCKS5_DSTADDR];
493   switch(type) {
494   case 1:
495     /* 4 bytes IPv4 address */
496     len = 4;
497     break;
498   case 3:
499     /* The first octet of the address field contains the number of octets of
500        name that follow */
501     len = buffer[SOCKS5_DSTADDR];
502     len++;
503     break;
504   case 4:
505     /* 16 bytes IPv6 address */
506     len = 16;
507     break;
508   default:
509     logmsg("Unknown ATYP %d", type);
510     return CURL_SOCKET_BAD;
511   }
512   if(rc < (4 + len + 2)) {
513     logmsg("Request too short: %d, expected %d", rc, 4 + len + 2);
514     return CURL_SOCKET_BAD;
515   }
516 
517   if(!config.port) {
518     unsigned char *portp = &buffer[SOCKS5_DSTADDR + len];
519     s5port = (unsigned short)((portp[0]<<8) | (portp[1]));
520   }
521   else
522     s5port = config.port;
523 
524   if(!config.connectrep)
525     connfd = socksconnect(s5port, config.addr);
526 
527   if(connfd == CURL_SOCKET_BAD) {
528     /* failed */
529     rep = 1;
530   }
531   else {
532     rep = config.connectrep;
533   }
534 
535   /* */
536   response[SOCKS5_VERSION] = config.responseversion;
537 
538   /*
539     o  REP    Reply field:
540     o  X'00' succeeded
541     o  X'01' general SOCKS server failure
542     o  X'02' connection not allowed by ruleset
543     o  X'03' Network unreachable
544     o  X'04' Host unreachable
545     o  X'05' Connection refused
546     o  X'06' TTL expired
547     o  X'07' Command not supported
548     o  X'08' Address type not supported
549     o  X'09' to X'FF' unassigned
550   */
551   response[SOCKS5_REP] = rep;
552   response[SOCKS5_RESERVED] = 0; /* must be zero */
553   response[SOCKS5_ATYP] = type; /* address type */
554 
555   /* mirror back the original addr + port */
556 
557   /* address or hostname */
558   memcpy(&response[SOCKS5_BNDADDR], address, len);
559 
560   /* port number */
561   memcpy(&response[SOCKS5_BNDADDR + len],
562          &buffer[SOCKS5_DSTADDR + len], sizeof(socksport));
563 
564   rc = (send)(fd, (char *)response, len + 6, 0);
565   if(rc != (len + 6)) {
566     logmsg("Sending connect response failed!");
567     return CURL_SOCKET_BAD;
568   }
569   logmsg("Sent %d bytes", rc);
570   loghex(response, rc);
571 
572   if(!rep)
573     return connfd;
574 
575   if(connfd != CURL_SOCKET_BAD)
576     sclose(connfd);
577 
578   return CURL_SOCKET_BAD;
579 }
580 
581 struct perclient {
582   size_t fromremote;
583   size_t fromclient;
584   curl_socket_t remotefd;
585   curl_socket_t clientfd;
586   bool used;
587 };
588 
589 /* return non-zero when transfer is done */
tunnel(struct perclient * cp,fd_set * fds)590 static int tunnel(struct perclient *cp, fd_set *fds)
591 {
592   ssize_t nread;
593   ssize_t nwrite;
594   char buffer[512];
595   if(FD_ISSET(cp->clientfd, fds)) {
596     /* read from client, send to remote */
597     nread = recv(cp->clientfd, buffer, sizeof(buffer), 0);
598     if(nread > 0) {
599       nwrite = send(cp->remotefd, (char *)buffer,
600                     (SEND_TYPE_ARG3)nread, 0);
601       if(nwrite != nread)
602         return 1;
603       cp->fromclient += nwrite;
604     }
605     else
606       return 1;
607   }
608   if(FD_ISSET(cp->remotefd, fds)) {
609     /* read from remote, send to client */
610     nread = recv(cp->remotefd, buffer, sizeof(buffer), 0);
611     if(nread > 0) {
612       nwrite = send(cp->clientfd, (char *)buffer,
613                     (SEND_TYPE_ARG3)nread, 0);
614       if(nwrite != nread)
615         return 1;
616       cp->fromremote += nwrite;
617     }
618     else
619       return 1;
620   }
621   return 0;
622 }
623 
624 /*
625   sockfdp is a pointer to an established stream or CURL_SOCKET_BAD
626 
627   if sockfd is CURL_SOCKET_BAD, listendfd is a listening socket we must
628   accept()
629 */
incoming(curl_socket_t listenfd)630 static bool incoming(curl_socket_t listenfd)
631 {
632   fd_set fds_read;
633   fd_set fds_write;
634   fd_set fds_err;
635   int clients = 0; /* connected clients */
636   struct perclient c[2];
637 
638   memset(c, 0, sizeof(c));
639   if(got_exit_signal) {
640     logmsg("signalled to die, exiting...");
641     return FALSE;
642   }
643 
644 #ifdef HAVE_GETPPID
645   /* As a last resort, quit if socks5 process becomes orphan. */
646   if(getppid() <= 1) {
647     logmsg("process becomes orphan, exiting");
648     return FALSE;
649   }
650 #endif
651 
652   do {
653     int i;
654     ssize_t rc;
655     int error = 0;
656     curl_socket_t sockfd = listenfd;
657     int maxfd = (int)sockfd;
658 
659     FD_ZERO(&fds_read);
660     FD_ZERO(&fds_write);
661     FD_ZERO(&fds_err);
662 
663     /* there's always a socket to wait for */
664     FD_SET(sockfd, &fds_read);
665 
666     for(i = 0; i < 2; i++) {
667       if(c[i].used) {
668         curl_socket_t fd = c[i].clientfd;
669         FD_SET(fd, &fds_read);
670         if((int)fd > maxfd)
671           maxfd = (int)fd;
672         fd = c[i].remotefd;
673         FD_SET(fd, &fds_read);
674         if((int)fd > maxfd)
675           maxfd = (int)fd;
676       }
677     }
678 
679     do {
680       /* select() blocking behavior call on blocking descriptors please */
681       rc = select(maxfd + 1, &fds_read, &fds_write, &fds_err, NULL);
682       if(got_exit_signal) {
683         logmsg("signalled to die, exiting...");
684         return FALSE;
685       }
686     } while((rc == -1) && ((error = errno) == EINTR));
687 
688     if(rc < 0) {
689       logmsg("select() failed with error: (%d) %s",
690              error, strerror(error));
691       return FALSE;
692     }
693 
694     if((clients < 2) && FD_ISSET(sockfd, &fds_read)) {
695       curl_socket_t newfd = accept(sockfd, NULL, NULL);
696       if(CURL_SOCKET_BAD == newfd) {
697         error = SOCKERRNO;
698         logmsg("accept(%d, NULL, NULL) failed with error: (%d) %s",
699                sockfd, error, strerror(error));
700       }
701       else {
702         curl_socket_t remotefd;
703         logmsg("====> Client connect, fd %d. Read config from %s",
704                newfd, configfile);
705         remotefd = sockit(newfd); /* SOCKS until done */
706         if(remotefd == CURL_SOCKET_BAD) {
707           logmsg("====> Client disconnect");
708           sclose(newfd);
709         }
710         else {
711           struct perclient *cp = &c[0];
712           logmsg("====> Tunnel transfer");
713 
714           if(c[0].used)
715             cp = &c[1];
716           cp->fromremote = 0;
717           cp->fromclient = 0;
718           cp->clientfd = newfd;
719           cp->remotefd = remotefd;
720           cp->used = TRUE;
721           clients++;
722         }
723 
724       }
725     }
726     for(i = 0; i < 2; i++) {
727       struct perclient *cp = &c[i];
728       if(cp->used) {
729         if(tunnel(cp, &fds_read)) {
730           logmsg("SOCKS transfer completed. Bytes: < %zu > %zu",
731                  cp->fromremote, cp->fromclient);
732           sclose(cp->clientfd);
733           sclose(cp->remotefd);
734           cp->used = FALSE;
735           clients--;
736         }
737       }
738     }
739   } while(clients);
740 
741   return TRUE;
742 }
743 
sockdaemon(curl_socket_t sock,unsigned short * listenport)744 static curl_socket_t sockdaemon(curl_socket_t sock,
745                                 unsigned short *listenport)
746 {
747   /* passive daemon style */
748   srvr_sockaddr_union_t listener;
749   int flag;
750   int rc;
751   int totdelay = 0;
752   int maxretr = 10;
753   int delay = 20;
754   int attempt = 0;
755   int error = 0;
756 
757   do {
758     attempt++;
759     flag = 1;
760     rc = setsockopt(sock, SOL_SOCKET, SO_REUSEADDR,
761          (void *)&flag, sizeof(flag));
762     if(rc) {
763       error = SOCKERRNO;
764       logmsg("setsockopt(SO_REUSEADDR) failed with error: (%d) %s",
765              error, strerror(error));
766       if(maxretr) {
767         rc = wait_ms(delay);
768         if(rc) {
769           /* should not happen */
770           error = errno;
771           logmsg("wait_ms() failed with error: (%d) %s",
772                  error, strerror(error));
773           sclose(sock);
774           return CURL_SOCKET_BAD;
775         }
776         if(got_exit_signal) {
777           logmsg("signalled to die, exiting...");
778           sclose(sock);
779           return CURL_SOCKET_BAD;
780         }
781         totdelay += delay;
782         delay *= 2; /* double the sleep for next attempt */
783       }
784     }
785   } while(rc && maxretr--);
786 
787   if(rc) {
788     logmsg("setsockopt(SO_REUSEADDR) failed %d times in %d ms. Error: (%d) %s",
789            attempt, totdelay, error, strerror(error));
790     logmsg("Continuing anyway...");
791   }
792 
793   /* When the specified listener port is zero, it is actually a
794      request to let the system choose a non-zero available port. */
795 
796 #ifdef ENABLE_IPV6
797   if(!use_ipv6) {
798 #endif
799     memset(&listener.sa4, 0, sizeof(listener.sa4));
800     listener.sa4.sin_family = AF_INET;
801     listener.sa4.sin_addr.s_addr = INADDR_ANY;
802     listener.sa4.sin_port = htons(*listenport);
803     rc = bind(sock, &listener.sa, sizeof(listener.sa4));
804 #ifdef ENABLE_IPV6
805   }
806   else {
807     memset(&listener.sa6, 0, sizeof(listener.sa6));
808     listener.sa6.sin6_family = AF_INET6;
809     listener.sa6.sin6_addr = in6addr_any;
810     listener.sa6.sin6_port = htons(*listenport);
811     rc = bind(sock, &listener.sa, sizeof(listener.sa6));
812   }
813 #endif /* ENABLE_IPV6 */
814   if(rc) {
815     error = SOCKERRNO;
816     logmsg("Error binding socket on port %hu: (%d) %s",
817            *listenport, error, strerror(error));
818     sclose(sock);
819     return CURL_SOCKET_BAD;
820   }
821 
822   if(!*listenport) {
823     /* The system was supposed to choose a port number, figure out which
824        port we actually got and update the listener port value with it. */
825     curl_socklen_t la_size;
826     srvr_sockaddr_union_t localaddr;
827 #ifdef ENABLE_IPV6
828     if(!use_ipv6)
829 #endif
830       la_size = sizeof(localaddr.sa4);
831 #ifdef ENABLE_IPV6
832     else
833       la_size = sizeof(localaddr.sa6);
834 #endif
835     memset(&localaddr.sa, 0, (size_t)la_size);
836     if(getsockname(sock, &localaddr.sa, &la_size) < 0) {
837       error = SOCKERRNO;
838       logmsg("getsockname() failed with error: (%d) %s",
839              error, strerror(error));
840       sclose(sock);
841       return CURL_SOCKET_BAD;
842     }
843     switch(localaddr.sa.sa_family) {
844     case AF_INET:
845       *listenport = ntohs(localaddr.sa4.sin_port);
846       break;
847 #ifdef ENABLE_IPV6
848     case AF_INET6:
849       *listenport = ntohs(localaddr.sa6.sin6_port);
850       break;
851 #endif
852     default:
853       break;
854     }
855     if(!*listenport) {
856       /* Real failure, listener port shall not be zero beyond this point. */
857       logmsg("Apparently getsockname() succeeded, with listener port zero.");
858       logmsg("A valid reason for this failure is a binary built without");
859       logmsg("proper network library linkage. This might not be the only");
860       logmsg("reason, but double check it before anything else.");
861       sclose(sock);
862       return CURL_SOCKET_BAD;
863     }
864   }
865 
866   /* start accepting connections */
867   rc = listen(sock, 5);
868   if(0 != rc) {
869     error = SOCKERRNO;
870     logmsg("listen(%d, 5) failed with error: (%d) %s",
871            sock, error, strerror(error));
872     sclose(sock);
873     return CURL_SOCKET_BAD;
874   }
875 
876   return sock;
877 }
878 
879 
main(int argc,char * argv[])880 int main(int argc, char *argv[])
881 {
882   curl_socket_t sock = CURL_SOCKET_BAD;
883   curl_socket_t msgsock = CURL_SOCKET_BAD;
884   int wrotepidfile = 0;
885   const char *pidname = ".socksd.pid";
886   const char *portfile = NULL;
887   bool juggle_again;
888   int error;
889   int arg = 1;
890 
891   while(argc>arg) {
892     if(!strcmp("--version", argv[arg])) {
893       printf("socksd IPv4%s\n",
894 #ifdef ENABLE_IPV6
895              "/IPv6"
896 #else
897              ""
898 #endif
899              );
900       return 0;
901     }
902     else if(!strcmp("--pidfile", argv[arg])) {
903       arg++;
904       if(argc>arg)
905         pidname = argv[arg++];
906     }
907     else if(!strcmp("--portfile", argv[arg])) {
908       arg++;
909       if(argc>arg)
910         portfile = argv[arg++];
911     }
912     else if(!strcmp("--config", argv[arg])) {
913       arg++;
914       if(argc>arg)
915         configfile = argv[arg++];
916     }
917     else if(!strcmp("--backend", argv[arg])) {
918       arg++;
919       if(argc>arg)
920         backendaddr = argv[arg++];
921     }
922     else if(!strcmp("--backendport", argv[arg])) {
923       arg++;
924       if(argc>arg)
925         backendport = (unsigned short)atoi(argv[arg++]);
926     }
927     else if(!strcmp("--logfile", argv[arg])) {
928       arg++;
929       if(argc>arg)
930         serverlogfile = argv[arg++];
931     }
932     else if(!strcmp("--ipv6", argv[arg])) {
933 #ifdef ENABLE_IPV6
934       ipv_inuse = "IPv6";
935       use_ipv6 = TRUE;
936 #endif
937       arg++;
938     }
939     else if(!strcmp("--ipv4", argv[arg])) {
940       /* for completeness, we support this option as well */
941 #ifdef ENABLE_IPV6
942       ipv_inuse = "IPv4";
943       use_ipv6 = FALSE;
944 #endif
945       arg++;
946     }
947     else if(!strcmp("--port", argv[arg])) {
948       arg++;
949       if(argc>arg) {
950         char *endptr;
951         unsigned long ulnum = strtoul(argv[arg], &endptr, 10);
952         port = curlx_ultous(ulnum);
953         arg++;
954       }
955     }
956     else {
957       puts("Usage: socksd [option]\n"
958            " --backend [ipv4 addr]\n"
959            " --backendport [TCP port]\n"
960            " --config [file]\n"
961            " --version\n"
962            " --logfile [file]\n"
963            " --pidfile [file]\n"
964            " --portfile [file]\n"
965            " --ipv4\n"
966            " --ipv6\n"
967            " --bindonly\n"
968            " --port [port]\n");
969       return 0;
970     }
971   }
972 
973 #ifdef WIN32
974   win32_init();
975   atexit(win32_cleanup);
976 
977   setmode(fileno(stdin), O_BINARY);
978   setmode(fileno(stdout), O_BINARY);
979   setmode(fileno(stderr), O_BINARY);
980 #endif
981 
982   install_signal_handlers(false);
983 
984 #ifdef ENABLE_IPV6
985   if(!use_ipv6)
986 #endif
987     sock = socket(AF_INET, SOCK_STREAM, 0);
988 #ifdef ENABLE_IPV6
989   else
990     sock = socket(AF_INET6, SOCK_STREAM, 0);
991 #endif
992 
993   if(CURL_SOCKET_BAD == sock) {
994     error = SOCKERRNO;
995     logmsg("Error creating socket: (%d) %s",
996            error, strerror(error));
997     goto socks5_cleanup;
998   }
999 
1000   {
1001     /* passive daemon style */
1002     sock = sockdaemon(sock, &port);
1003     if(CURL_SOCKET_BAD == sock) {
1004       goto socks5_cleanup;
1005     }
1006     msgsock = CURL_SOCKET_BAD; /* no stream socket yet */
1007   }
1008 
1009   logmsg("Running %s version", ipv_inuse);
1010   logmsg("Listening on port %hu", port);
1011 
1012   wrotepidfile = write_pidfile(pidname);
1013   if(!wrotepidfile) {
1014     goto socks5_cleanup;
1015   }
1016 
1017   if(portfile) {
1018     wrotepidfile = write_portfile(portfile, port);
1019     if(!wrotepidfile) {
1020       goto socks5_cleanup;
1021     }
1022   }
1023 
1024   do {
1025     juggle_again = incoming(sock);
1026   } while(juggle_again);
1027 
1028 socks5_cleanup:
1029 
1030   if((msgsock != sock) && (msgsock != CURL_SOCKET_BAD))
1031     sclose(msgsock);
1032 
1033   if(sock != CURL_SOCKET_BAD)
1034     sclose(sock);
1035 
1036   if(wrotepidfile)
1037     unlink(pidname);
1038 
1039   restore_signal_handlers(false);
1040 
1041   if(got_exit_signal) {
1042     logmsg("============> socksd exits with signal (%d)", exit_signal);
1043     /*
1044      * To properly set the return status of the process we
1045      * must raise the same signal SIGINT or SIGTERM that we
1046      * caught and let the old handler take care of it.
1047      */
1048     raise(exit_signal);
1049   }
1050 
1051   logmsg("============> socksd quits");
1052   return 0;
1053 }
1054