cf-socket.c 55 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930
  1. /***************************************************************************
  2. * _ _ ____ _
  3. * Project ___| | | | _ \| |
  4. * / __| | | | |_) | |
  5. * | (__| |_| | _ <| |___
  6. * \___|\___/|_| \_\_____|
  7. *
  8. * Copyright (C) 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. * SPDX-License-Identifier: curl
  22. *
  23. ***************************************************************************/
  24. #include "curl_setup.h"
  25. #ifdef HAVE_NETINET_IN_H
  26. #include <netinet/in.h> /* <netinet/tcp.h> may need it */
  27. #endif
  28. #ifdef HAVE_SYS_UN_H
  29. #include <sys/un.h> /* for sockaddr_un */
  30. #endif
  31. #ifdef HAVE_LINUX_TCP_H
  32. #include <linux/tcp.h>
  33. #elif defined(HAVE_NETINET_TCP_H)
  34. #include <netinet/tcp.h>
  35. #endif
  36. #ifdef HAVE_SYS_IOCTL_H
  37. #include <sys/ioctl.h>
  38. #endif
  39. #ifdef HAVE_NETDB_H
  40. #include <netdb.h>
  41. #endif
  42. #ifdef HAVE_FCNTL_H
  43. #include <fcntl.h>
  44. #endif
  45. #ifdef HAVE_ARPA_INET_H
  46. #include <arpa/inet.h>
  47. #endif
  48. #ifdef __VMS
  49. #include <in.h>
  50. #include <inet.h>
  51. #endif
  52. #include "urldata.h"
  53. #include "sendf.h"
  54. #include "if2ip.h"
  55. #include "strerror.h"
  56. #include "cfilters.h"
  57. #include "cf-socket.h"
  58. #include "connect.h"
  59. #include "select.h"
  60. #include "url.h" /* for Curl_safefree() */
  61. #include "multiif.h"
  62. #include "sockaddr.h" /* required for Curl_sockaddr_storage */
  63. #include "inet_ntop.h"
  64. #include "inet_pton.h"
  65. #include "progress.h"
  66. #include "warnless.h"
  67. #include "conncache.h"
  68. #include "multihandle.h"
  69. #include "share.h"
  70. #include "version_win32.h"
  71. /* The last 3 #include files should be in this order */
  72. #include "curl_printf.h"
  73. #include "curl_memory.h"
  74. #include "memdebug.h"
  75. static void tcpnodelay(struct Curl_easy *data, curl_socket_t sockfd)
  76. {
  77. #if defined(TCP_NODELAY)
  78. curl_socklen_t onoff = (curl_socklen_t) 1;
  79. int level = IPPROTO_TCP;
  80. #if !defined(CURL_DISABLE_VERBOSE_STRINGS)
  81. char buffer[STRERROR_LEN];
  82. #else
  83. (void) data;
  84. #endif
  85. if(setsockopt(sockfd, level, TCP_NODELAY, (void *)&onoff,
  86. sizeof(onoff)) < 0)
  87. infof(data, "Could not set TCP_NODELAY: %s",
  88. Curl_strerror(SOCKERRNO, buffer, sizeof(buffer)));
  89. #else
  90. (void)data;
  91. (void)sockfd;
  92. #endif
  93. }
  94. #ifdef SO_NOSIGPIPE
  95. /* The preferred method on Mac OS X (10.2 and later) to prevent SIGPIPEs when
  96. sending data to a dead peer (instead of relying on the 4th argument to send
  97. being MSG_NOSIGNAL). Possibly also existing and in use on other BSD
  98. systems? */
  99. static void nosigpipe(struct Curl_easy *data,
  100. curl_socket_t sockfd)
  101. {
  102. int onoff = 1;
  103. if(setsockopt(sockfd, SOL_SOCKET, SO_NOSIGPIPE, (void *)&onoff,
  104. sizeof(onoff)) < 0) {
  105. #if !defined(CURL_DISABLE_VERBOSE_STRINGS)
  106. char buffer[STRERROR_LEN];
  107. infof(data, "Could not set SO_NOSIGPIPE: %s",
  108. Curl_strerror(SOCKERRNO, buffer, sizeof(buffer)));
  109. #endif
  110. }
  111. }
  112. #else
  113. #define nosigpipe(x,y) Curl_nop_stmt
  114. #endif
  115. #if defined(__DragonFly__) || defined(HAVE_WINSOCK2_H)
  116. /* DragonFlyBSD and Windows use millisecond units */
  117. #define KEEPALIVE_FACTOR(x) (x *= 1000)
  118. #else
  119. #define KEEPALIVE_FACTOR(x)
  120. #endif
  121. #if defined(HAVE_WINSOCK2_H) && !defined(SIO_KEEPALIVE_VALS)
  122. #define SIO_KEEPALIVE_VALS _WSAIOW(IOC_VENDOR,4)
  123. struct tcp_keepalive {
  124. u_long onoff;
  125. u_long keepalivetime;
  126. u_long keepaliveinterval;
  127. };
  128. #endif
  129. static void
  130. tcpkeepalive(struct Curl_easy *data,
  131. curl_socket_t sockfd)
  132. {
  133. int optval = data->set.tcp_keepalive?1:0;
  134. /* only set IDLE and INTVL if setting KEEPALIVE is successful */
  135. if(setsockopt(sockfd, SOL_SOCKET, SO_KEEPALIVE,
  136. (void *)&optval, sizeof(optval)) < 0) {
  137. infof(data, "Failed to set SO_KEEPALIVE on fd %d", sockfd);
  138. }
  139. else {
  140. #if defined(SIO_KEEPALIVE_VALS)
  141. struct tcp_keepalive vals;
  142. DWORD dummy;
  143. vals.onoff = 1;
  144. optval = curlx_sltosi(data->set.tcp_keepidle);
  145. KEEPALIVE_FACTOR(optval);
  146. vals.keepalivetime = optval;
  147. optval = curlx_sltosi(data->set.tcp_keepintvl);
  148. KEEPALIVE_FACTOR(optval);
  149. vals.keepaliveinterval = optval;
  150. if(WSAIoctl(sockfd, SIO_KEEPALIVE_VALS, (LPVOID) &vals, sizeof(vals),
  151. NULL, 0, &dummy, NULL, NULL) != 0) {
  152. infof(data, "Failed to set SIO_KEEPALIVE_VALS on fd %d: %d",
  153. (int)sockfd, WSAGetLastError());
  154. }
  155. #else
  156. #ifdef TCP_KEEPIDLE
  157. optval = curlx_sltosi(data->set.tcp_keepidle);
  158. KEEPALIVE_FACTOR(optval);
  159. if(setsockopt(sockfd, IPPROTO_TCP, TCP_KEEPIDLE,
  160. (void *)&optval, sizeof(optval)) < 0) {
  161. infof(data, "Failed to set TCP_KEEPIDLE on fd %d", sockfd);
  162. }
  163. #elif defined(TCP_KEEPALIVE)
  164. /* Mac OS X style */
  165. optval = curlx_sltosi(data->set.tcp_keepidle);
  166. KEEPALIVE_FACTOR(optval);
  167. if(setsockopt(sockfd, IPPROTO_TCP, TCP_KEEPALIVE,
  168. (void *)&optval, sizeof(optval)) < 0) {
  169. infof(data, "Failed to set TCP_KEEPALIVE on fd %d", sockfd);
  170. }
  171. #endif
  172. #ifdef TCP_KEEPINTVL
  173. optval = curlx_sltosi(data->set.tcp_keepintvl);
  174. KEEPALIVE_FACTOR(optval);
  175. if(setsockopt(sockfd, IPPROTO_TCP, TCP_KEEPINTVL,
  176. (void *)&optval, sizeof(optval)) < 0) {
  177. infof(data, "Failed to set TCP_KEEPINTVL on fd %d", sockfd);
  178. }
  179. #endif
  180. #endif
  181. }
  182. }
  183. void Curl_sock_assign_addr(struct Curl_sockaddr_ex *dest,
  184. const struct Curl_addrinfo *ai,
  185. int transport)
  186. {
  187. /*
  188. * The Curl_sockaddr_ex structure is basically libcurl's external API
  189. * curl_sockaddr structure with enough space available to directly hold
  190. * any protocol-specific address structures. The variable declared here
  191. * will be used to pass / receive data to/from the fopensocket callback
  192. * if this has been set, before that, it is initialized from parameters.
  193. */
  194. dest->family = ai->ai_family;
  195. switch(transport) {
  196. case TRNSPRT_TCP:
  197. dest->socktype = SOCK_STREAM;
  198. dest->protocol = IPPROTO_TCP;
  199. break;
  200. case TRNSPRT_UNIX:
  201. dest->socktype = SOCK_STREAM;
  202. dest->protocol = IPPROTO_IP;
  203. break;
  204. default: /* UDP and QUIC */
  205. dest->socktype = SOCK_DGRAM;
  206. dest->protocol = IPPROTO_UDP;
  207. break;
  208. }
  209. dest->addrlen = ai->ai_addrlen;
  210. if(dest->addrlen > sizeof(struct Curl_sockaddr_storage))
  211. dest->addrlen = sizeof(struct Curl_sockaddr_storage);
  212. memcpy(&dest->sa_addr, ai->ai_addr, dest->addrlen);
  213. }
  214. static CURLcode socket_open(struct Curl_easy *data,
  215. struct Curl_sockaddr_ex *addr,
  216. curl_socket_t *sockfd)
  217. {
  218. DEBUGASSERT(data);
  219. DEBUGASSERT(data->conn);
  220. if(data->set.fopensocket) {
  221. /*
  222. * If the opensocket callback is set, all the destination address
  223. * information is passed to the callback. Depending on this information the
  224. * callback may opt to abort the connection, this is indicated returning
  225. * CURL_SOCKET_BAD; otherwise it will return a not-connected socket. When
  226. * the callback returns a valid socket the destination address information
  227. * might have been changed and this 'new' address will actually be used
  228. * here to connect.
  229. */
  230. Curl_set_in_callback(data, true);
  231. *sockfd = data->set.fopensocket(data->set.opensocket_client,
  232. CURLSOCKTYPE_IPCXN,
  233. (struct curl_sockaddr *)addr);
  234. Curl_set_in_callback(data, false);
  235. }
  236. else {
  237. /* opensocket callback not set, so simply create the socket now */
  238. *sockfd = socket(addr->family, addr->socktype, addr->protocol);
  239. }
  240. if(*sockfd == CURL_SOCKET_BAD)
  241. /* no socket, no connection */
  242. return CURLE_COULDNT_CONNECT;
  243. #if defined(ENABLE_IPV6) && defined(HAVE_SOCKADDR_IN6_SIN6_SCOPE_ID)
  244. if(data->conn->scope_id && (addr->family == AF_INET6)) {
  245. struct sockaddr_in6 * const sa6 = (void *)&addr->sa_addr;
  246. sa6->sin6_scope_id = data->conn->scope_id;
  247. }
  248. #endif
  249. return CURLE_OK;
  250. }
  251. /*
  252. * Create a socket based on info from 'conn' and 'ai'.
  253. *
  254. * 'addr' should be a pointer to the correct struct to get data back, or NULL.
  255. * 'sockfd' must be a pointer to a socket descriptor.
  256. *
  257. * If the open socket callback is set, used that!
  258. *
  259. */
  260. CURLcode Curl_socket_open(struct Curl_easy *data,
  261. const struct Curl_addrinfo *ai,
  262. struct Curl_sockaddr_ex *addr,
  263. int transport,
  264. curl_socket_t *sockfd)
  265. {
  266. struct Curl_sockaddr_ex dummy;
  267. if(!addr)
  268. /* if the caller doesn't want info back, use a local temp copy */
  269. addr = &dummy;
  270. Curl_sock_assign_addr(addr, ai, transport);
  271. return socket_open(data, addr, sockfd);
  272. }
  273. static int socket_close(struct Curl_easy *data, struct connectdata *conn,
  274. int use_callback, curl_socket_t sock)
  275. {
  276. if(use_callback && conn && conn->fclosesocket) {
  277. int rc;
  278. Curl_multi_closed(data, sock);
  279. Curl_set_in_callback(data, true);
  280. rc = conn->fclosesocket(conn->closesocket_client, sock);
  281. Curl_set_in_callback(data, false);
  282. return rc;
  283. }
  284. if(conn)
  285. /* tell the multi-socket code about this */
  286. Curl_multi_closed(data, sock);
  287. sclose(sock);
  288. return 0;
  289. }
  290. /*
  291. * Close a socket.
  292. *
  293. * 'conn' can be NULL, beware!
  294. */
  295. int Curl_socket_close(struct Curl_easy *data, struct connectdata *conn,
  296. curl_socket_t sock)
  297. {
  298. return socket_close(data, conn, FALSE, sock);
  299. }
  300. bool Curl_socket_is_dead(curl_socket_t sock)
  301. {
  302. int sval;
  303. bool ret_val = TRUE;
  304. sval = SOCKET_READABLE(sock, 0);
  305. if(sval == 0)
  306. /* timeout */
  307. ret_val = FALSE;
  308. return ret_val;
  309. }
  310. #ifdef USE_WINSOCK
  311. /* When you run a program that uses the Windows Sockets API, you may
  312. experience slow performance when you copy data to a TCP server.
  313. https://support.microsoft.com/kb/823764
  314. Work-around: Make the Socket Send Buffer Size Larger Than the Program Send
  315. Buffer Size
  316. The problem described in this knowledge-base is applied only to pre-Vista
  317. Windows. Following function trying to detect OS version and skips
  318. SO_SNDBUF adjustment for Windows Vista and above.
  319. */
  320. #define DETECT_OS_NONE 0
  321. #define DETECT_OS_PREVISTA 1
  322. #define DETECT_OS_VISTA_OR_LATER 2
  323. void Curl_sndbufset(curl_socket_t sockfd)
  324. {
  325. int val = CURL_MAX_WRITE_SIZE + 32;
  326. int curval = 0;
  327. int curlen = sizeof(curval);
  328. static int detectOsState = DETECT_OS_NONE;
  329. if(detectOsState == DETECT_OS_NONE) {
  330. if(curlx_verify_windows_version(6, 0, 0, PLATFORM_WINNT,
  331. VERSION_GREATER_THAN_EQUAL))
  332. detectOsState = DETECT_OS_VISTA_OR_LATER;
  333. else
  334. detectOsState = DETECT_OS_PREVISTA;
  335. }
  336. if(detectOsState == DETECT_OS_VISTA_OR_LATER)
  337. return;
  338. if(getsockopt(sockfd, SOL_SOCKET, SO_SNDBUF, (char *)&curval, &curlen) == 0)
  339. if(curval > val)
  340. return;
  341. setsockopt(sockfd, SOL_SOCKET, SO_SNDBUF, (const char *)&val, sizeof(val));
  342. }
  343. #endif
  344. static CURLcode bindlocal(struct Curl_easy *data, struct connectdata *conn,
  345. curl_socket_t sockfd, int af, unsigned int scope)
  346. {
  347. struct Curl_sockaddr_storage sa;
  348. struct sockaddr *sock = (struct sockaddr *)&sa; /* bind to this address */
  349. curl_socklen_t sizeof_sa = 0; /* size of the data sock points to */
  350. struct sockaddr_in *si4 = (struct sockaddr_in *)&sa;
  351. #ifdef ENABLE_IPV6
  352. struct sockaddr_in6 *si6 = (struct sockaddr_in6 *)&sa;
  353. #endif
  354. struct Curl_dns_entry *h = NULL;
  355. unsigned short port = data->set.localport; /* use this port number, 0 for
  356. "random" */
  357. /* how many port numbers to try to bind to, increasing one at a time */
  358. int portnum = data->set.localportrange;
  359. const char *dev = data->set.str[STRING_DEVICE];
  360. int error;
  361. #ifdef IP_BIND_ADDRESS_NO_PORT
  362. int on = 1;
  363. #endif
  364. #ifndef ENABLE_IPV6
  365. (void)scope;
  366. #endif
  367. /*************************************************************
  368. * Select device to bind socket to
  369. *************************************************************/
  370. if(!dev && !port)
  371. /* no local kind of binding was requested */
  372. return CURLE_OK;
  373. memset(&sa, 0, sizeof(struct Curl_sockaddr_storage));
  374. if(dev && (strlen(dev)<255) ) {
  375. char myhost[256] = "";
  376. int done = 0; /* -1 for error, 1 for address found */
  377. bool is_interface = FALSE;
  378. bool is_host = FALSE;
  379. static const char *if_prefix = "if!";
  380. static const char *host_prefix = "host!";
  381. if(strncmp(if_prefix, dev, strlen(if_prefix)) == 0) {
  382. dev += strlen(if_prefix);
  383. is_interface = TRUE;
  384. }
  385. else if(strncmp(host_prefix, dev, strlen(host_prefix)) == 0) {
  386. dev += strlen(host_prefix);
  387. is_host = TRUE;
  388. }
  389. /* interface */
  390. if(!is_host) {
  391. #ifdef SO_BINDTODEVICE
  392. /* I am not sure any other OSs than Linux that provide this feature,
  393. * and at the least I cannot test. --Ben
  394. *
  395. * This feature allows one to tightly bind the local socket to a
  396. * particular interface. This will force even requests to other
  397. * local interfaces to go out the external interface.
  398. *
  399. *
  400. * Only bind to the interface when specified as interface, not just
  401. * as a hostname or ip address.
  402. *
  403. * interface might be a VRF, eg: vrf-blue, which means it cannot be
  404. * converted to an IP address and would fail Curl_if2ip. Simply try
  405. * to use it straight away.
  406. */
  407. if(setsockopt(sockfd, SOL_SOCKET, SO_BINDTODEVICE,
  408. dev, (curl_socklen_t)strlen(dev) + 1) == 0) {
  409. /* This is typically "errno 1, error: Operation not permitted" if
  410. * you're not running as root or another suitable privileged
  411. * user.
  412. * If it succeeds it means the parameter was a valid interface and
  413. * not an IP address. Return immediately.
  414. */
  415. return CURLE_OK;
  416. }
  417. #endif
  418. switch(Curl_if2ip(af,
  419. #ifdef ENABLE_IPV6
  420. scope, conn->scope_id,
  421. #endif
  422. dev, myhost, sizeof(myhost))) {
  423. case IF2IP_NOT_FOUND:
  424. if(is_interface) {
  425. /* Do not fall back to treating it as a host name */
  426. failf(data, "Couldn't bind to interface '%s'", dev);
  427. return CURLE_INTERFACE_FAILED;
  428. }
  429. break;
  430. case IF2IP_AF_NOT_SUPPORTED:
  431. /* Signal the caller to try another address family if available */
  432. return CURLE_UNSUPPORTED_PROTOCOL;
  433. case IF2IP_FOUND:
  434. is_interface = TRUE;
  435. /*
  436. * We now have the numerical IP address in the 'myhost' buffer
  437. */
  438. infof(data, "Local Interface %s is ip %s using address family %i",
  439. dev, myhost, af);
  440. done = 1;
  441. break;
  442. }
  443. }
  444. if(!is_interface) {
  445. /*
  446. * This was not an interface, resolve the name as a host name
  447. * or IP number
  448. *
  449. * Temporarily force name resolution to use only the address type
  450. * of the connection. The resolve functions should really be changed
  451. * to take a type parameter instead.
  452. */
  453. unsigned char ipver = conn->ip_version;
  454. int rc;
  455. if(af == AF_INET)
  456. conn->ip_version = CURL_IPRESOLVE_V4;
  457. #ifdef ENABLE_IPV6
  458. else if(af == AF_INET6)
  459. conn->ip_version = CURL_IPRESOLVE_V6;
  460. #endif
  461. rc = Curl_resolv(data, dev, 0, FALSE, &h);
  462. if(rc == CURLRESOLV_PENDING)
  463. (void)Curl_resolver_wait_resolv(data, &h);
  464. conn->ip_version = ipver;
  465. if(h) {
  466. /* convert the resolved address, sizeof myhost >= INET_ADDRSTRLEN */
  467. Curl_printable_address(h->addr, myhost, sizeof(myhost));
  468. infof(data, "Name '%s' family %i resolved to '%s' family %i",
  469. dev, af, myhost, h->addr->ai_family);
  470. Curl_resolv_unlock(data, h);
  471. if(af != h->addr->ai_family) {
  472. /* bad IP version combo, signal the caller to try another address
  473. family if available */
  474. return CURLE_UNSUPPORTED_PROTOCOL;
  475. }
  476. done = 1;
  477. }
  478. else {
  479. /*
  480. * provided dev was no interface (or interfaces are not supported
  481. * e.g. solaris) no ip address and no domain we fail here
  482. */
  483. done = -1;
  484. }
  485. }
  486. if(done > 0) {
  487. #ifdef ENABLE_IPV6
  488. /* IPv6 address */
  489. if(af == AF_INET6) {
  490. #ifdef HAVE_SOCKADDR_IN6_SIN6_SCOPE_ID
  491. char *scope_ptr = strchr(myhost, '%');
  492. if(scope_ptr)
  493. *(scope_ptr++) = '\0';
  494. #endif
  495. if(Curl_inet_pton(AF_INET6, myhost, &si6->sin6_addr) > 0) {
  496. si6->sin6_family = AF_INET6;
  497. si6->sin6_port = htons(port);
  498. #ifdef HAVE_SOCKADDR_IN6_SIN6_SCOPE_ID
  499. if(scope_ptr) {
  500. /* The "myhost" string either comes from Curl_if2ip or from
  501. Curl_printable_address. The latter returns only numeric scope
  502. IDs and the former returns none at all. So the scope ID, if
  503. present, is known to be numeric */
  504. unsigned long scope_id = strtoul(scope_ptr, NULL, 10);
  505. if(scope_id > UINT_MAX)
  506. return CURLE_UNSUPPORTED_PROTOCOL;
  507. si6->sin6_scope_id = (unsigned int)scope_id;
  508. }
  509. #endif
  510. }
  511. sizeof_sa = sizeof(struct sockaddr_in6);
  512. }
  513. else
  514. #endif
  515. /* IPv4 address */
  516. if((af == AF_INET) &&
  517. (Curl_inet_pton(AF_INET, myhost, &si4->sin_addr) > 0)) {
  518. si4->sin_family = AF_INET;
  519. si4->sin_port = htons(port);
  520. sizeof_sa = sizeof(struct sockaddr_in);
  521. }
  522. }
  523. if(done < 1) {
  524. /* errorbuf is set false so failf will overwrite any message already in
  525. the error buffer, so the user receives this error message instead of a
  526. generic resolve error. */
  527. data->state.errorbuf = FALSE;
  528. failf(data, "Couldn't bind to '%s'", dev);
  529. return CURLE_INTERFACE_FAILED;
  530. }
  531. }
  532. else {
  533. /* no device was given, prepare sa to match af's needs */
  534. #ifdef ENABLE_IPV6
  535. if(af == AF_INET6) {
  536. si6->sin6_family = AF_INET6;
  537. si6->sin6_port = htons(port);
  538. sizeof_sa = sizeof(struct sockaddr_in6);
  539. }
  540. else
  541. #endif
  542. if(af == AF_INET) {
  543. si4->sin_family = AF_INET;
  544. si4->sin_port = htons(port);
  545. sizeof_sa = sizeof(struct sockaddr_in);
  546. }
  547. }
  548. #ifdef IP_BIND_ADDRESS_NO_PORT
  549. (void)setsockopt(sockfd, SOL_IP, IP_BIND_ADDRESS_NO_PORT, &on, sizeof(on));
  550. #endif
  551. for(;;) {
  552. if(bind(sockfd, sock, sizeof_sa) >= 0) {
  553. /* we succeeded to bind */
  554. struct Curl_sockaddr_storage add;
  555. curl_socklen_t size = sizeof(add);
  556. memset(&add, 0, sizeof(struct Curl_sockaddr_storage));
  557. if(getsockname(sockfd, (struct sockaddr *) &add, &size) < 0) {
  558. char buffer[STRERROR_LEN];
  559. data->state.os_errno = error = SOCKERRNO;
  560. failf(data, "getsockname() failed with errno %d: %s",
  561. error, Curl_strerror(error, buffer, sizeof(buffer)));
  562. return CURLE_INTERFACE_FAILED;
  563. }
  564. infof(data, "Local port: %hu", port);
  565. conn->bits.bound = TRUE;
  566. return CURLE_OK;
  567. }
  568. if(--portnum > 0) {
  569. port++; /* try next port */
  570. if(port == 0)
  571. break;
  572. infof(data, "Bind to local port %hu failed, trying next", port - 1);
  573. /* We re-use/clobber the port variable here below */
  574. if(sock->sa_family == AF_INET)
  575. si4->sin_port = ntohs(port);
  576. #ifdef ENABLE_IPV6
  577. else
  578. si6->sin6_port = ntohs(port);
  579. #endif
  580. }
  581. else
  582. break;
  583. }
  584. {
  585. char buffer[STRERROR_LEN];
  586. data->state.os_errno = error = SOCKERRNO;
  587. failf(data, "bind failed with errno %d: %s",
  588. error, Curl_strerror(error, buffer, sizeof(buffer)));
  589. }
  590. return CURLE_INTERFACE_FAILED;
  591. }
  592. /*
  593. * verifyconnect() returns TRUE if the connect really has happened.
  594. */
  595. static bool verifyconnect(curl_socket_t sockfd, int *error)
  596. {
  597. bool rc = TRUE;
  598. #ifdef SO_ERROR
  599. int err = 0;
  600. curl_socklen_t errSize = sizeof(err);
  601. #ifdef WIN32
  602. /*
  603. * In October 2003 we effectively nullified this function on Windows due to
  604. * problems with it using all CPU in multi-threaded cases.
  605. *
  606. * In May 2004, we bring it back to offer more info back on connect failures.
  607. * Gisle Vanem could reproduce the former problems with this function, but
  608. * could avoid them by adding this SleepEx() call below:
  609. *
  610. * "I don't have Rational Quantify, but the hint from his post was
  611. * ntdll::NtRemoveIoCompletion(). So I'd assume the SleepEx (or maybe
  612. * just Sleep(0) would be enough?) would release whatever
  613. * mutex/critical-section the ntdll call is waiting on.
  614. *
  615. * Someone got to verify this on Win-NT 4.0, 2000."
  616. */
  617. #ifdef _WIN32_WCE
  618. Sleep(0);
  619. #else
  620. SleepEx(0, FALSE);
  621. #endif
  622. #endif
  623. if(0 != getsockopt(sockfd, SOL_SOCKET, SO_ERROR, (void *)&err, &errSize))
  624. err = SOCKERRNO;
  625. #ifdef _WIN32_WCE
  626. /* Old WinCE versions don't support SO_ERROR */
  627. if(WSAENOPROTOOPT == err) {
  628. SET_SOCKERRNO(0);
  629. err = 0;
  630. }
  631. #endif
  632. #if defined(EBADIOCTL) && defined(__minix)
  633. /* Minix 3.1.x doesn't support getsockopt on UDP sockets */
  634. if(EBADIOCTL == err) {
  635. SET_SOCKERRNO(0);
  636. err = 0;
  637. }
  638. #endif
  639. if((0 == err) || (EISCONN == err))
  640. /* we are connected, awesome! */
  641. rc = TRUE;
  642. else
  643. /* This wasn't a successful connect */
  644. rc = FALSE;
  645. if(error)
  646. *error = err;
  647. #else
  648. (void)sockfd;
  649. if(error)
  650. *error = SOCKERRNO;
  651. #endif
  652. return rc;
  653. }
  654. CURLcode Curl_socket_connect_result(struct Curl_easy *data,
  655. const char *ipaddress, int error)
  656. {
  657. char buffer[STRERROR_LEN];
  658. switch(error) {
  659. case EINPROGRESS:
  660. case EWOULDBLOCK:
  661. #if defined(EAGAIN)
  662. #if (EAGAIN) != (EWOULDBLOCK)
  663. /* On some platforms EAGAIN and EWOULDBLOCK are the
  664. * same value, and on others they are different, hence
  665. * the odd #if
  666. */
  667. case EAGAIN:
  668. #endif
  669. #endif
  670. return CURLE_OK;
  671. default:
  672. /* unknown error, fallthrough and try another address! */
  673. infof(data, "Immediate connect fail for %s: %s",
  674. ipaddress, Curl_strerror(error, buffer, sizeof(buffer)));
  675. data->state.os_errno = error;
  676. /* connect failed */
  677. return CURLE_COULDNT_CONNECT;
  678. }
  679. }
  680. #ifdef USE_RECV_BEFORE_SEND_WORKAROUND
  681. struct io_buffer {
  682. char *bufr;
  683. size_t allc; /* size of the current allocation */
  684. size_t head; /* bufr index for next read */
  685. size_t tail; /* bufr index for next write */
  686. };
  687. static void io_buffer_reset(struct io_buffer *iob)
  688. {
  689. if(iob->bufr)
  690. free(iob->bufr);
  691. memset(iob, 0, sizeof(*iob));
  692. }
  693. #endif /* USE_RECV_BEFORE_SEND_WORKAROUND */
  694. struct cf_socket_ctx {
  695. int transport;
  696. struct Curl_sockaddr_ex addr; /* address to connect to */
  697. curl_socket_t sock; /* current attempt socket */
  698. #ifdef USE_RECV_BEFORE_SEND_WORKAROUND
  699. struct io_buffer recv_buffer;
  700. #endif
  701. char r_ip[MAX_IPADR_LEN]; /* remote IP as string */
  702. int r_port; /* remote port number */
  703. char l_ip[MAX_IPADR_LEN]; /* local IP as string */
  704. int l_port; /* local port number */
  705. struct curltime started_at; /* when socket was created */
  706. struct curltime connected_at; /* when socket connected/got first byte */
  707. struct curltime first_byte_at; /* when first byte was recvd */
  708. int error; /* errno of last failure or 0 */
  709. BIT(got_first_byte); /* if first byte was received */
  710. BIT(accepted); /* socket was accepted, not connected */
  711. BIT(active);
  712. };
  713. static void cf_socket_ctx_init(struct cf_socket_ctx *ctx,
  714. const struct Curl_addrinfo *ai,
  715. int transport)
  716. {
  717. memset(ctx, 0, sizeof(*ctx));
  718. ctx->sock = CURL_SOCKET_BAD;
  719. ctx->transport = transport;
  720. Curl_sock_assign_addr(&ctx->addr, ai, transport);
  721. }
  722. static void cf_socket_close(struct Curl_cfilter *cf, struct Curl_easy *data)
  723. {
  724. struct cf_socket_ctx *ctx = cf->ctx;
  725. if(ctx && CURL_SOCKET_BAD != ctx->sock) {
  726. if(ctx->active) {
  727. /* We share our socket at cf->conn->sock[cf->sockindex] when active.
  728. * If it is no longer there, someone has stolen (and hopefully
  729. * closed it) and we just forget about it.
  730. */
  731. if(ctx->sock == cf->conn->sock[cf->sockindex]) {
  732. DEBUGF(LOG_CF(data, cf, "cf_socket_close(%d, active)",
  733. (int)ctx->sock));
  734. socket_close(data, cf->conn, !ctx->accepted, ctx->sock);
  735. cf->conn->sock[cf->sockindex] = CURL_SOCKET_BAD;
  736. }
  737. else {
  738. DEBUGF(LOG_CF(data, cf, "cf_socket_close(%d) no longer at "
  739. "conn->sock[], discarding", (int)ctx->sock));
  740. /* TODO: we do not want this to happen. Need to check which
  741. * code is messing with conn->sock[cf->sockindex] */
  742. }
  743. ctx->sock = CURL_SOCKET_BAD;
  744. if(cf->sockindex == FIRSTSOCKET)
  745. cf->conn->remote_addr = NULL;
  746. }
  747. else {
  748. /* this is our local socket, we did never publish it */
  749. DEBUGF(LOG_CF(data, cf, "cf_socket_close(%d, not active)",
  750. (int)ctx->sock));
  751. sclose(ctx->sock);
  752. ctx->sock = CURL_SOCKET_BAD;
  753. }
  754. #ifdef USE_RECV_BEFORE_SEND_WORKAROUND
  755. io_buffer_reset(&ctx->recv_buffer);
  756. #endif
  757. ctx->active = FALSE;
  758. memset(&ctx->started_at, 0, sizeof(ctx->started_at));
  759. memset(&ctx->connected_at, 0, sizeof(ctx->connected_at));
  760. }
  761. cf->connected = FALSE;
  762. }
  763. static void cf_socket_destroy(struct Curl_cfilter *cf, struct Curl_easy *data)
  764. {
  765. struct cf_socket_ctx *ctx = cf->ctx;
  766. cf_socket_close(cf, data);
  767. DEBUGF(LOG_CF(data, cf, "destroy"));
  768. free(ctx);
  769. cf->ctx = NULL;
  770. }
  771. static CURLcode set_local_ip(struct Curl_cfilter *cf,
  772. struct Curl_easy *data)
  773. {
  774. struct cf_socket_ctx *ctx = cf->ctx;
  775. #ifdef HAVE_GETSOCKNAME
  776. char buffer[STRERROR_LEN];
  777. struct Curl_sockaddr_storage ssloc;
  778. curl_socklen_t slen = sizeof(struct Curl_sockaddr_storage);
  779. memset(&ssloc, 0, sizeof(ssloc));
  780. if(getsockname(ctx->sock, (struct sockaddr*) &ssloc, &slen)) {
  781. int error = SOCKERRNO;
  782. failf(data, "getsockname() failed with errno %d: %s",
  783. error, Curl_strerror(error, buffer, sizeof(buffer)));
  784. return CURLE_FAILED_INIT;
  785. }
  786. if(!Curl_addr2string((struct sockaddr*)&ssloc, slen,
  787. ctx->l_ip, &ctx->l_port)) {
  788. failf(data, "ssloc inet_ntop() failed with errno %d: %s",
  789. errno, Curl_strerror(errno, buffer, sizeof(buffer)));
  790. return CURLE_FAILED_INIT;
  791. }
  792. #else
  793. (void)data;
  794. ctx->l_ip[0] = 0;
  795. ctx->l_port = -1;
  796. #endif
  797. return CURLE_OK;
  798. }
  799. static CURLcode set_remote_ip(struct Curl_cfilter *cf,
  800. struct Curl_easy *data)
  801. {
  802. struct cf_socket_ctx *ctx = cf->ctx;
  803. /* store remote address and port used in this connection attempt */
  804. if(!Curl_addr2string(&ctx->addr.sa_addr, ctx->addr.addrlen,
  805. ctx->r_ip, &ctx->r_port)) {
  806. char buffer[STRERROR_LEN];
  807. ctx->error = errno;
  808. /* malformed address or bug in inet_ntop, try next address */
  809. failf(data, "sa_addr inet_ntop() failed with errno %d: %s",
  810. errno, Curl_strerror(errno, buffer, sizeof(buffer)));
  811. return CURLE_FAILED_INIT;
  812. }
  813. return CURLE_OK;
  814. }
  815. static CURLcode cf_socket_open(struct Curl_cfilter *cf,
  816. struct Curl_easy *data)
  817. {
  818. struct cf_socket_ctx *ctx = cf->ctx;
  819. int error = 0;
  820. bool isconnected = FALSE;
  821. CURLcode result = CURLE_COULDNT_CONNECT;
  822. bool is_tcp;
  823. const char *ipmsg;
  824. (void)data;
  825. DEBUGASSERT(ctx->sock == CURL_SOCKET_BAD);
  826. ctx->started_at = Curl_now();
  827. result = socket_open(data, &ctx->addr, &ctx->sock);
  828. if(result)
  829. goto out;
  830. result = set_remote_ip(cf, data);
  831. if(result)
  832. goto out;
  833. #ifdef ENABLE_IPV6
  834. if(ctx->addr.family == AF_INET6)
  835. ipmsg = " Trying [%s]:%d...";
  836. else
  837. #endif
  838. ipmsg = " Trying %s:%d...";
  839. infof(data, ipmsg, ctx->r_ip, ctx->r_port);
  840. #ifdef ENABLE_IPV6
  841. is_tcp = (ctx->addr.family == AF_INET
  842. || ctx->addr.family == AF_INET6) &&
  843. ctx->addr.socktype == SOCK_STREAM;
  844. #else
  845. is_tcp = (ctx->addr.family == AF_INET) &&
  846. ctx->addr.socktype == SOCK_STREAM;
  847. #endif
  848. if(is_tcp && data->set.tcp_nodelay)
  849. tcpnodelay(data, ctx->sock);
  850. nosigpipe(data, ctx->sock);
  851. Curl_sndbufset(ctx->sock);
  852. if(is_tcp && data->set.tcp_keepalive)
  853. tcpkeepalive(data, ctx->sock);
  854. if(data->set.fsockopt) {
  855. /* activate callback for setting socket options */
  856. Curl_set_in_callback(data, true);
  857. error = data->set.fsockopt(data->set.sockopt_client,
  858. ctx->sock,
  859. CURLSOCKTYPE_IPCXN);
  860. Curl_set_in_callback(data, false);
  861. if(error == CURL_SOCKOPT_ALREADY_CONNECTED)
  862. isconnected = TRUE;
  863. else if(error) {
  864. result = CURLE_ABORTED_BY_CALLBACK;
  865. goto out;
  866. }
  867. }
  868. /* possibly bind the local end to an IP, interface or port */
  869. if(ctx->addr.family == AF_INET
  870. #ifdef ENABLE_IPV6
  871. || ctx->addr.family == AF_INET6
  872. #endif
  873. ) {
  874. result = bindlocal(data, cf->conn, ctx->sock, ctx->addr.family,
  875. Curl_ipv6_scope(&ctx->addr.sa_addr));
  876. if(result) {
  877. if(result == CURLE_UNSUPPORTED_PROTOCOL) {
  878. /* The address family is not supported on this interface.
  879. We can continue trying addresses */
  880. result = CURLE_COULDNT_CONNECT;
  881. }
  882. goto out;
  883. }
  884. }
  885. /* set socket non-blocking */
  886. (void)curlx_nonblock(ctx->sock, TRUE);
  887. out:
  888. if(result) {
  889. if(ctx->sock != CURL_SOCKET_BAD) {
  890. socket_close(data, cf->conn, TRUE, ctx->sock);
  891. ctx->sock = CURL_SOCKET_BAD;
  892. }
  893. }
  894. else if(isconnected) {
  895. set_local_ip(cf, data);
  896. ctx->connected_at = Curl_now();
  897. cf->connected = TRUE;
  898. }
  899. DEBUGF(LOG_CF(data, cf, "cf_socket_open() -> %d, fd=%d", result, ctx->sock));
  900. return result;
  901. }
  902. static int do_connect(struct Curl_cfilter *cf, struct Curl_easy *data,
  903. bool is_tcp_fastopen)
  904. {
  905. struct cf_socket_ctx *ctx = cf->ctx;
  906. #ifdef TCP_FASTOPEN_CONNECT
  907. int optval = 1;
  908. #endif
  909. int rc = -1;
  910. (void)data;
  911. if(is_tcp_fastopen) {
  912. #if defined(CONNECT_DATA_IDEMPOTENT) /* Darwin */
  913. # if defined(HAVE_BUILTIN_AVAILABLE)
  914. /* while connectx function is available since macOS 10.11 / iOS 9,
  915. it did not have the interface declared correctly until
  916. Xcode 9 / macOS SDK 10.13 */
  917. if(__builtin_available(macOS 10.11, iOS 9.0, tvOS 9.0, watchOS 2.0, *)) {
  918. sa_endpoints_t endpoints;
  919. endpoints.sae_srcif = 0;
  920. endpoints.sae_srcaddr = NULL;
  921. endpoints.sae_srcaddrlen = 0;
  922. endpoints.sae_dstaddr = &ctx->addr.sa_addr;
  923. endpoints.sae_dstaddrlen = ctx->addr.addrlen;
  924. rc = connectx(ctx->sock, &endpoints, SAE_ASSOCID_ANY,
  925. CONNECT_RESUME_ON_READ_WRITE | CONNECT_DATA_IDEMPOTENT,
  926. NULL, 0, NULL, NULL);
  927. }
  928. else {
  929. rc = connect(ctx->sock, &ctx->addr.sa_addr, ctx->addr.addrlen);
  930. }
  931. # else
  932. rc = connect(ctx->sock, &ctx->addr.sa_addr, ctx->addr.addrlen);
  933. # endif /* HAVE_BUILTIN_AVAILABLE */
  934. #elif defined(TCP_FASTOPEN_CONNECT) /* Linux >= 4.11 */
  935. if(setsockopt(ctx->sock, IPPROTO_TCP, TCP_FASTOPEN_CONNECT,
  936. (void *)&optval, sizeof(optval)) < 0)
  937. infof(data, "Failed to enable TCP Fast Open on fd %d", ctx->sock);
  938. rc = connect(ctx->sock, &ctx->addr.sa_addr, ctx->addr.addrlen);
  939. #elif defined(MSG_FASTOPEN) /* old Linux */
  940. if(cf->conn->given->flags & PROTOPT_SSL)
  941. rc = connect(ctx->sock, &ctx->addr.sa_addr, ctx->addr.addrlen);
  942. else
  943. rc = 0; /* Do nothing */
  944. #endif
  945. }
  946. else {
  947. rc = connect(ctx->sock, &ctx->addr.sa_addr, ctx->addr.addrlen);
  948. }
  949. return rc;
  950. }
  951. static CURLcode cf_tcp_connect(struct Curl_cfilter *cf,
  952. struct Curl_easy *data,
  953. bool blocking, bool *done)
  954. {
  955. struct cf_socket_ctx *ctx = cf->ctx;
  956. CURLcode result = CURLE_COULDNT_CONNECT;
  957. int rc = 0;
  958. (void)data;
  959. if(cf->connected) {
  960. *done = TRUE;
  961. return CURLE_OK;
  962. }
  963. /* TODO: need to support blocking connect? */
  964. if(blocking)
  965. return CURLE_UNSUPPORTED_PROTOCOL;
  966. *done = FALSE; /* a very negative world view is best */
  967. if(ctx->sock == CURL_SOCKET_BAD) {
  968. result = cf_socket_open(cf, data);
  969. if(result)
  970. goto out;
  971. if(cf->connected) {
  972. *done = TRUE;
  973. return CURLE_OK;
  974. }
  975. /* Connect TCP socket */
  976. rc = do_connect(cf, data, cf->conn->bits.tcp_fastopen);
  977. if(-1 == rc) {
  978. result = Curl_socket_connect_result(data, ctx->r_ip, SOCKERRNO);
  979. goto out;
  980. }
  981. }
  982. #ifdef mpeix
  983. /* Call this function once now, and ignore the results. We do this to
  984. "clear" the error state on the socket so that we can later read it
  985. reliably. This is reported necessary on the MPE/iX operating
  986. system. */
  987. (void)verifyconnect(ctx->sock, NULL);
  988. #endif
  989. /* check socket for connect */
  990. rc = SOCKET_WRITABLE(ctx->sock, 0);
  991. if(rc == 0) { /* no connection yet */
  992. DEBUGF(LOG_CF(data, cf, "not connected yet"));
  993. return CURLE_OK;
  994. }
  995. else if(rc == CURL_CSELECT_OUT || cf->conn->bits.tcp_fastopen) {
  996. if(verifyconnect(ctx->sock, &ctx->error)) {
  997. /* we are connected with TCP, awesome! */
  998. ctx->connected_at = Curl_now();
  999. set_local_ip(cf, data);
  1000. *done = TRUE;
  1001. cf->connected = TRUE;
  1002. DEBUGF(LOG_CF(data, cf, "connected"));
  1003. return CURLE_OK;
  1004. }
  1005. }
  1006. else if(rc & CURL_CSELECT_ERR) {
  1007. (void)verifyconnect(ctx->sock, &ctx->error);
  1008. result = CURLE_COULDNT_CONNECT;
  1009. }
  1010. out:
  1011. if(result) {
  1012. if(ctx->error) {
  1013. data->state.os_errno = ctx->error;
  1014. SET_SOCKERRNO(ctx->error);
  1015. #ifndef CURL_DISABLE_VERBOSE_STRINGS
  1016. {
  1017. char buffer[STRERROR_LEN];
  1018. infof(data, "connect to %s port %u failed: %s",
  1019. ctx->r_ip, ctx->r_port,
  1020. Curl_strerror(ctx->error, buffer, sizeof(buffer)));
  1021. }
  1022. #endif
  1023. }
  1024. if(ctx->sock != CURL_SOCKET_BAD) {
  1025. socket_close(data, cf->conn, TRUE, ctx->sock);
  1026. ctx->sock = CURL_SOCKET_BAD;
  1027. }
  1028. *done = FALSE;
  1029. }
  1030. return result;
  1031. }
  1032. static void cf_socket_get_host(struct Curl_cfilter *cf,
  1033. struct Curl_easy *data,
  1034. const char **phost,
  1035. const char **pdisplay_host,
  1036. int *pport)
  1037. {
  1038. (void)data;
  1039. *phost = cf->conn->host.name;
  1040. *pdisplay_host = cf->conn->host.dispname;
  1041. *pport = cf->conn->port;
  1042. }
  1043. static int cf_socket_get_select_socks(struct Curl_cfilter *cf,
  1044. struct Curl_easy *data,
  1045. curl_socket_t *socks)
  1046. {
  1047. struct cf_socket_ctx *ctx = cf->ctx;
  1048. int rc = GETSOCK_BLANK;
  1049. (void)data;
  1050. if(!cf->connected && ctx->sock != CURL_SOCKET_BAD) {
  1051. socks[0] = ctx->sock;
  1052. rc |= GETSOCK_WRITESOCK(0);
  1053. }
  1054. return rc;
  1055. }
  1056. #ifdef USE_RECV_BEFORE_SEND_WORKAROUND
  1057. static CURLcode pre_receive_plain(struct Curl_cfilter *cf,
  1058. struct Curl_easy *data)
  1059. {
  1060. struct cf_socket_ctx *ctx = cf->ctx;
  1061. struct io_buffer * const iob = &ctx->recv_buffer;
  1062. /* WinSock will destroy unread received data if send() is
  1063. failed.
  1064. To avoid lossage of received data, recv() must be
  1065. performed before every send() if any incoming data is
  1066. available. However, skip this, if buffer is already full. */
  1067. if((cf->conn->handler->protocol&PROTO_FAMILY_HTTP) != 0 &&
  1068. cf->conn->recv[cf->sockindex] == Curl_conn_recv &&
  1069. (!iob->bufr || (iob->allc > iob->tail))) {
  1070. const int readymask = Curl_socket_check(ctx->sock, CURL_SOCKET_BAD,
  1071. CURL_SOCKET_BAD, 0);
  1072. if(readymask != -1 && (readymask & CURL_CSELECT_IN) != 0) {
  1073. size_t bytestorecv = iob->allc - iob->tail;
  1074. ssize_t nread;
  1075. /* Have some incoming data */
  1076. if(!iob->bufr) {
  1077. /* Use buffer double default size for intermediate buffer */
  1078. iob->allc = 2 * data->set.buffer_size;
  1079. iob->bufr = malloc(iob->allc);
  1080. if(!iob->bufr)
  1081. return CURLE_OUT_OF_MEMORY;
  1082. iob->tail = 0;
  1083. iob->head = 0;
  1084. bytestorecv = iob->allc;
  1085. }
  1086. nread = sread(ctx->sock, iob->bufr + iob->tail, bytestorecv);
  1087. if(nread > 0)
  1088. iob->tail += (size_t)nread;
  1089. }
  1090. }
  1091. return CURLE_OK;
  1092. }
  1093. static ssize_t get_pre_recved(struct Curl_cfilter *cf, char *buf, size_t len)
  1094. {
  1095. struct cf_socket_ctx *ctx = cf->ctx;
  1096. struct io_buffer * const iob = &ctx->recv_buffer;
  1097. size_t copysize;
  1098. if(!iob->bufr)
  1099. return 0;
  1100. DEBUGASSERT(iob->allc > 0);
  1101. DEBUGASSERT(iob->tail <= iob->allc);
  1102. DEBUGASSERT(iob->head <= iob->tail);
  1103. /* Check and process data that already received and storied in internal
  1104. intermediate buffer */
  1105. if(iob->tail > iob->head) {
  1106. copysize = CURLMIN(len, iob->tail - iob->head);
  1107. memcpy(buf, iob->bufr + iob->head, copysize);
  1108. iob->head += copysize;
  1109. }
  1110. else
  1111. copysize = 0; /* buffer was allocated, but nothing was received */
  1112. /* Free intermediate buffer if it has no unprocessed data */
  1113. if(iob->head == iob->tail)
  1114. io_buffer_reset(iob);
  1115. return (ssize_t)copysize;
  1116. }
  1117. #endif /* USE_RECV_BEFORE_SEND_WORKAROUND */
  1118. static bool cf_socket_data_pending(struct Curl_cfilter *cf,
  1119. const struct Curl_easy *data)
  1120. {
  1121. struct cf_socket_ctx *ctx = cf->ctx;
  1122. int readable;
  1123. #ifdef USE_RECV_BEFORE_SEND_WORKAROUND
  1124. if(ctx->recv_buffer.bufr && ctx->recv_buffer.allc &&
  1125. ctx->recv_buffer.tail > ctx->recv_buffer.head)
  1126. return TRUE;
  1127. #endif
  1128. (void)data;
  1129. readable = SOCKET_READABLE(ctx->sock, 0);
  1130. return (readable > 0 && (readable & CURL_CSELECT_IN));
  1131. }
  1132. static ssize_t cf_socket_send(struct Curl_cfilter *cf, struct Curl_easy *data,
  1133. const void *buf, size_t len, CURLcode *err)
  1134. {
  1135. struct cf_socket_ctx *ctx = cf->ctx;
  1136. curl_socket_t fdsave;
  1137. ssize_t nwritten;
  1138. *err = CURLE_OK;
  1139. #ifdef USE_RECV_BEFORE_SEND_WORKAROUND
  1140. /* WinSock will destroy unread received data if send() is
  1141. failed.
  1142. To avoid lossage of received data, recv() must be
  1143. performed before every send() if any incoming data is
  1144. available. */
  1145. if(pre_receive_plain(cf, data)) {
  1146. *err = CURLE_OUT_OF_MEMORY;
  1147. return -1;
  1148. }
  1149. #endif
  1150. fdsave = cf->conn->sock[cf->sockindex];
  1151. cf->conn->sock[cf->sockindex] = ctx->sock;
  1152. #if defined(MSG_FASTOPEN) && !defined(TCP_FASTOPEN_CONNECT) /* Linux */
  1153. if(cf->conn->bits.tcp_fastopen) {
  1154. nwritten = sendto(ctx->sock, buf, len, MSG_FASTOPEN,
  1155. &cf->conn->remote_addr->sa_addr,
  1156. cf->conn->remote_addr->addrlen);
  1157. cf->conn->bits.tcp_fastopen = FALSE;
  1158. }
  1159. else
  1160. #endif
  1161. nwritten = swrite(ctx->sock, buf, len);
  1162. if(-1 == nwritten) {
  1163. int sockerr = SOCKERRNO;
  1164. if(
  1165. #ifdef WSAEWOULDBLOCK
  1166. /* This is how Windows does it */
  1167. (WSAEWOULDBLOCK == sockerr)
  1168. #else
  1169. /* errno may be EWOULDBLOCK or on some systems EAGAIN when it returned
  1170. due to its inability to send off data without blocking. We therefore
  1171. treat both error codes the same here */
  1172. (EWOULDBLOCK == sockerr) || (EAGAIN == sockerr) || (EINTR == sockerr) ||
  1173. (EINPROGRESS == sockerr)
  1174. #endif
  1175. ) {
  1176. /* this is just a case of EWOULDBLOCK */
  1177. *err = CURLE_AGAIN;
  1178. }
  1179. else {
  1180. char buffer[STRERROR_LEN];
  1181. failf(data, "Send failure: %s",
  1182. Curl_strerror(sockerr, buffer, sizeof(buffer)));
  1183. data->state.os_errno = sockerr;
  1184. *err = CURLE_SEND_ERROR;
  1185. }
  1186. }
  1187. DEBUGF(LOG_CF(data, cf, "send(len=%zu) -> %d, err=%d",
  1188. len, (int)nwritten, *err));
  1189. cf->conn->sock[cf->sockindex] = fdsave;
  1190. return nwritten;
  1191. }
  1192. static ssize_t cf_socket_recv(struct Curl_cfilter *cf, struct Curl_easy *data,
  1193. char *buf, size_t len, CURLcode *err)
  1194. {
  1195. struct cf_socket_ctx *ctx = cf->ctx;
  1196. curl_socket_t fdsave;
  1197. ssize_t nread;
  1198. *err = CURLE_OK;
  1199. #ifdef USE_RECV_BEFORE_SEND_WORKAROUND
  1200. /* Check and return data that already received and storied in internal
  1201. intermediate buffer */
  1202. nread = get_pre_recved(cf, buf, len);
  1203. if(nread > 0) {
  1204. *err = CURLE_OK;
  1205. return nread;
  1206. }
  1207. #endif
  1208. fdsave = cf->conn->sock[cf->sockindex];
  1209. cf->conn->sock[cf->sockindex] = ctx->sock;
  1210. nread = sread(ctx->sock, buf, len);
  1211. if(-1 == nread) {
  1212. int sockerr = SOCKERRNO;
  1213. if(
  1214. #ifdef WSAEWOULDBLOCK
  1215. /* This is how Windows does it */
  1216. (WSAEWOULDBLOCK == sockerr)
  1217. #else
  1218. /* errno may be EWOULDBLOCK or on some systems EAGAIN when it returned
  1219. due to its inability to send off data without blocking. We therefore
  1220. treat both error codes the same here */
  1221. (EWOULDBLOCK == sockerr) || (EAGAIN == sockerr) || (EINTR == sockerr)
  1222. #endif
  1223. ) {
  1224. /* this is just a case of EWOULDBLOCK */
  1225. *err = CURLE_AGAIN;
  1226. }
  1227. else {
  1228. char buffer[STRERROR_LEN];
  1229. failf(data, "Recv failure: %s",
  1230. Curl_strerror(sockerr, buffer, sizeof(buffer)));
  1231. data->state.os_errno = sockerr;
  1232. *err = CURLE_RECV_ERROR;
  1233. }
  1234. }
  1235. DEBUGF(LOG_CF(data, cf, "recv(len=%zu) -> %d, err=%d", len, (int)nread,
  1236. *err));
  1237. if(nread > 0 && !ctx->got_first_byte) {
  1238. ctx->first_byte_at = Curl_now();
  1239. ctx->got_first_byte = TRUE;
  1240. }
  1241. cf->conn->sock[cf->sockindex] = fdsave;
  1242. return nread;
  1243. }
  1244. static void conn_set_primary_ip(struct Curl_cfilter *cf,
  1245. struct Curl_easy *data)
  1246. {
  1247. struct cf_socket_ctx *ctx = cf->ctx;
  1248. #ifdef HAVE_GETPEERNAME
  1249. char buffer[STRERROR_LEN];
  1250. struct Curl_sockaddr_storage ssrem;
  1251. curl_socklen_t plen;
  1252. int port;
  1253. plen = sizeof(ssrem);
  1254. memset(&ssrem, 0, plen);
  1255. if(getpeername(ctx->sock, (struct sockaddr*) &ssrem, &plen)) {
  1256. int error = SOCKERRNO;
  1257. failf(data, "getpeername() failed with errno %d: %s",
  1258. error, Curl_strerror(error, buffer, sizeof(buffer)));
  1259. return;
  1260. }
  1261. if(!Curl_addr2string((struct sockaddr*)&ssrem, plen,
  1262. cf->conn->primary_ip, &port)) {
  1263. failf(data, "ssrem inet_ntop() failed with errno %d: %s",
  1264. errno, Curl_strerror(errno, buffer, sizeof(buffer)));
  1265. return;
  1266. }
  1267. #else
  1268. cf->conn->primary_ip[0] = 0;
  1269. (void)data;
  1270. #endif
  1271. }
  1272. static void cf_socket_active(struct Curl_cfilter *cf, struct Curl_easy *data)
  1273. {
  1274. struct cf_socket_ctx *ctx = cf->ctx;
  1275. /* use this socket from now on */
  1276. cf->conn->sock[cf->sockindex] = ctx->sock;
  1277. /* the first socket info gets set at conn and data */
  1278. if(cf->sockindex == FIRSTSOCKET) {
  1279. cf->conn->remote_addr = &ctx->addr;
  1280. #ifdef ENABLE_IPV6
  1281. cf->conn->bits.ipv6 = (ctx->addr.family == AF_INET6)? TRUE : FALSE;
  1282. #endif
  1283. conn_set_primary_ip(cf, data);
  1284. set_local_ip(cf, data);
  1285. Curl_persistconninfo(data, cf->conn, ctx->l_ip, ctx->l_port);
  1286. }
  1287. ctx->active = TRUE;
  1288. }
  1289. static CURLcode cf_socket_cntrl(struct Curl_cfilter *cf,
  1290. struct Curl_easy *data,
  1291. int event, int arg1, void *arg2)
  1292. {
  1293. struct cf_socket_ctx *ctx = cf->ctx;
  1294. (void)arg1;
  1295. (void)arg2;
  1296. switch(event) {
  1297. case CF_CTRL_CONN_INFO_UPDATE:
  1298. cf_socket_active(cf, data);
  1299. break;
  1300. case CF_CTRL_CONN_REPORT_STATS:
  1301. switch(ctx->transport) {
  1302. case TRNSPRT_UDP:
  1303. case TRNSPRT_QUIC:
  1304. /* Since UDP connected sockets work different from TCP, we use the
  1305. * time of the first byte from the peer as the "connect" time. */
  1306. if(ctx->got_first_byte) {
  1307. Curl_pgrsTimeWas(data, TIMER_CONNECT, ctx->first_byte_at);
  1308. break;
  1309. }
  1310. /* FALLTHROUGH */
  1311. default:
  1312. Curl_pgrsTimeWas(data, TIMER_CONNECT, ctx->connected_at);
  1313. break;
  1314. }
  1315. break;
  1316. case CF_CTRL_DATA_SETUP:
  1317. Curl_persistconninfo(data, cf->conn, ctx->l_ip, ctx->l_port);
  1318. break;
  1319. }
  1320. return CURLE_OK;
  1321. }
  1322. static bool cf_socket_conn_is_alive(struct Curl_cfilter *cf,
  1323. struct Curl_easy *data)
  1324. {
  1325. struct cf_socket_ctx *ctx = cf->ctx;
  1326. struct pollfd pfd[1];
  1327. int r;
  1328. (void)data;
  1329. if(!ctx || ctx->sock == CURL_SOCKET_BAD)
  1330. return FALSE;
  1331. /* Check with 0 timeout if there are any events pending on the socket */
  1332. pfd[0].fd = ctx->sock;
  1333. pfd[0].events = POLLRDNORM|POLLIN|POLLRDBAND|POLLPRI;
  1334. pfd[0].revents = 0;
  1335. r = Curl_poll(pfd, 1, 0);
  1336. if(r < 0) {
  1337. DEBUGF(LOG_CF(data, cf, "is_alive: poll error, assume dead"));
  1338. return FALSE;
  1339. }
  1340. else if(r == 0) {
  1341. DEBUGF(LOG_CF(data, cf, "is_alive: poll timeout, assume alive"));
  1342. return TRUE;
  1343. }
  1344. else if(pfd[0].revents & (POLLERR|POLLHUP|POLLPRI|POLLNVAL)) {
  1345. DEBUGF(LOG_CF(data, cf, "is_alive: err/hup/etc events, assume dead"));
  1346. return FALSE;
  1347. }
  1348. DEBUGF(LOG_CF(data, cf, "is_alive: valid events, looks alive"));
  1349. return TRUE;
  1350. }
  1351. static CURLcode cf_socket_query(struct Curl_cfilter *cf,
  1352. struct Curl_easy *data,
  1353. int query, int *pres1, void *pres2)
  1354. {
  1355. struct cf_socket_ctx *ctx = cf->ctx;
  1356. switch(query) {
  1357. case CF_QUERY_SOCKET:
  1358. DEBUGASSERT(pres2);
  1359. *((curl_socket_t *)pres2) = ctx->sock;
  1360. return CURLE_OK;
  1361. case CF_QUERY_CONNECT_REPLY_MS:
  1362. if(ctx->got_first_byte) {
  1363. timediff_t ms = Curl_timediff(ctx->first_byte_at, ctx->started_at);
  1364. *pres1 = (ms < INT_MAX)? (int)ms : INT_MAX;
  1365. }
  1366. else
  1367. *pres1 = -1;
  1368. return CURLE_OK;
  1369. default:
  1370. break;
  1371. }
  1372. return cf->next?
  1373. cf->next->cft->query(cf->next, data, query, pres1, pres2) :
  1374. CURLE_UNKNOWN_OPTION;
  1375. }
  1376. struct Curl_cftype Curl_cft_tcp = {
  1377. "TCP",
  1378. CF_TYPE_IP_CONNECT,
  1379. CURL_LOG_DEFAULT,
  1380. cf_socket_destroy,
  1381. cf_tcp_connect,
  1382. cf_socket_close,
  1383. cf_socket_get_host,
  1384. cf_socket_get_select_socks,
  1385. cf_socket_data_pending,
  1386. cf_socket_send,
  1387. cf_socket_recv,
  1388. cf_socket_cntrl,
  1389. cf_socket_conn_is_alive,
  1390. Curl_cf_def_conn_keep_alive,
  1391. cf_socket_query,
  1392. };
  1393. CURLcode Curl_cf_tcp_create(struct Curl_cfilter **pcf,
  1394. struct Curl_easy *data,
  1395. struct connectdata *conn,
  1396. const struct Curl_addrinfo *ai,
  1397. int transport)
  1398. {
  1399. struct cf_socket_ctx *ctx = NULL;
  1400. struct Curl_cfilter *cf = NULL;
  1401. CURLcode result;
  1402. (void)data;
  1403. (void)conn;
  1404. DEBUGASSERT(transport == TRNSPRT_TCP);
  1405. ctx = calloc(sizeof(*ctx), 1);
  1406. if(!ctx) {
  1407. result = CURLE_OUT_OF_MEMORY;
  1408. goto out;
  1409. }
  1410. cf_socket_ctx_init(ctx, ai, transport);
  1411. result = Curl_cf_create(&cf, &Curl_cft_tcp, ctx);
  1412. out:
  1413. *pcf = (!result)? cf : NULL;
  1414. if(result) {
  1415. Curl_safefree(cf);
  1416. Curl_safefree(ctx);
  1417. }
  1418. return result;
  1419. }
  1420. static CURLcode cf_udp_setup_quic(struct Curl_cfilter *cf,
  1421. struct Curl_easy *data)
  1422. {
  1423. struct cf_socket_ctx *ctx = cf->ctx;
  1424. int rc;
  1425. /* QUIC needs a connected socket, nonblocking */
  1426. DEBUGASSERT(ctx->sock != CURL_SOCKET_BAD);
  1427. rc = connect(ctx->sock, &ctx->addr.sa_addr, ctx->addr.addrlen);
  1428. if(-1 == rc) {
  1429. return Curl_socket_connect_result(data, ctx->r_ip, SOCKERRNO);
  1430. }
  1431. set_local_ip(cf, data);
  1432. DEBUGF(LOG_CF(data, cf, "%s socket %d connected: [%s:%d] -> [%s:%d]",
  1433. (ctx->transport == TRNSPRT_QUIC)? "QUIC" : "UDP",
  1434. ctx->sock, ctx->l_ip, ctx->l_port, ctx->r_ip, ctx->r_port));
  1435. (void)curlx_nonblock(ctx->sock, TRUE);
  1436. switch(ctx->addr.family) {
  1437. #if defined(__linux__) && defined(IP_MTU_DISCOVER)
  1438. case AF_INET: {
  1439. int val = IP_PMTUDISC_DO;
  1440. (void)setsockopt(ctx->sock, IPPROTO_IP, IP_MTU_DISCOVER, &val,
  1441. sizeof(val));
  1442. break;
  1443. }
  1444. #endif
  1445. #if defined(__linux__) && defined(IPV6_MTU_DISCOVER)
  1446. case AF_INET6: {
  1447. int val = IPV6_PMTUDISC_DO;
  1448. (void)setsockopt(ctx->sock, IPPROTO_IPV6, IPV6_MTU_DISCOVER, &val,
  1449. sizeof(val));
  1450. break;
  1451. }
  1452. #endif
  1453. }
  1454. return CURLE_OK;
  1455. }
  1456. static CURLcode cf_udp_connect(struct Curl_cfilter *cf,
  1457. struct Curl_easy *data,
  1458. bool blocking, bool *done)
  1459. {
  1460. struct cf_socket_ctx *ctx = cf->ctx;
  1461. CURLcode result = CURLE_COULDNT_CONNECT;
  1462. (void)blocking;
  1463. if(cf->connected) {
  1464. *done = TRUE;
  1465. return CURLE_OK;
  1466. }
  1467. *done = FALSE;
  1468. if(ctx->sock == CURL_SOCKET_BAD) {
  1469. result = cf_socket_open(cf, data);
  1470. if(result) {
  1471. DEBUGF(LOG_CF(data, cf, "cf_udp_connect(), open failed -> %d", result));
  1472. if(ctx->sock != CURL_SOCKET_BAD) {
  1473. socket_close(data, cf->conn, TRUE, ctx->sock);
  1474. ctx->sock = CURL_SOCKET_BAD;
  1475. }
  1476. goto out;
  1477. }
  1478. if(ctx->transport == TRNSPRT_QUIC) {
  1479. result = cf_udp_setup_quic(cf, data);
  1480. if(result)
  1481. goto out;
  1482. DEBUGF(LOG_CF(data, cf, "cf_udp_connect(), opened socket=%d (%s:%d)",
  1483. ctx->sock, ctx->l_ip, ctx->l_port));
  1484. }
  1485. else {
  1486. DEBUGF(LOG_CF(data, cf, "cf_udp_connect(), opened socket=%d "
  1487. "(unconnected)", ctx->sock));
  1488. }
  1489. *done = TRUE;
  1490. cf->connected = TRUE;
  1491. }
  1492. out:
  1493. return result;
  1494. }
  1495. struct Curl_cftype Curl_cft_udp = {
  1496. "UDP",
  1497. CF_TYPE_IP_CONNECT,
  1498. CURL_LOG_DEFAULT,
  1499. cf_socket_destroy,
  1500. cf_udp_connect,
  1501. cf_socket_close,
  1502. cf_socket_get_host,
  1503. cf_socket_get_select_socks,
  1504. cf_socket_data_pending,
  1505. cf_socket_send,
  1506. cf_socket_recv,
  1507. cf_socket_cntrl,
  1508. cf_socket_conn_is_alive,
  1509. Curl_cf_def_conn_keep_alive,
  1510. cf_socket_query,
  1511. };
  1512. CURLcode Curl_cf_udp_create(struct Curl_cfilter **pcf,
  1513. struct Curl_easy *data,
  1514. struct connectdata *conn,
  1515. const struct Curl_addrinfo *ai,
  1516. int transport)
  1517. {
  1518. struct cf_socket_ctx *ctx = NULL;
  1519. struct Curl_cfilter *cf = NULL;
  1520. CURLcode result;
  1521. (void)data;
  1522. (void)conn;
  1523. DEBUGASSERT(transport == TRNSPRT_UDP || transport == TRNSPRT_QUIC);
  1524. ctx = calloc(sizeof(*ctx), 1);
  1525. if(!ctx) {
  1526. result = CURLE_OUT_OF_MEMORY;
  1527. goto out;
  1528. }
  1529. cf_socket_ctx_init(ctx, ai, transport);
  1530. result = Curl_cf_create(&cf, &Curl_cft_udp, ctx);
  1531. out:
  1532. *pcf = (!result)? cf : NULL;
  1533. if(result) {
  1534. Curl_safefree(cf);
  1535. Curl_safefree(ctx);
  1536. }
  1537. return result;
  1538. }
  1539. /* this is the TCP filter which can also handle this case */
  1540. struct Curl_cftype Curl_cft_unix = {
  1541. "UNIX",
  1542. CF_TYPE_IP_CONNECT,
  1543. CURL_LOG_DEFAULT,
  1544. cf_socket_destroy,
  1545. cf_tcp_connect,
  1546. cf_socket_close,
  1547. cf_socket_get_host,
  1548. cf_socket_get_select_socks,
  1549. cf_socket_data_pending,
  1550. cf_socket_send,
  1551. cf_socket_recv,
  1552. cf_socket_cntrl,
  1553. cf_socket_conn_is_alive,
  1554. Curl_cf_def_conn_keep_alive,
  1555. cf_socket_query,
  1556. };
  1557. CURLcode Curl_cf_unix_create(struct Curl_cfilter **pcf,
  1558. struct Curl_easy *data,
  1559. struct connectdata *conn,
  1560. const struct Curl_addrinfo *ai,
  1561. int transport)
  1562. {
  1563. struct cf_socket_ctx *ctx = NULL;
  1564. struct Curl_cfilter *cf = NULL;
  1565. CURLcode result;
  1566. (void)data;
  1567. (void)conn;
  1568. DEBUGASSERT(transport == TRNSPRT_UNIX);
  1569. ctx = calloc(sizeof(*ctx), 1);
  1570. if(!ctx) {
  1571. result = CURLE_OUT_OF_MEMORY;
  1572. goto out;
  1573. }
  1574. cf_socket_ctx_init(ctx, ai, transport);
  1575. result = Curl_cf_create(&cf, &Curl_cft_unix, ctx);
  1576. out:
  1577. *pcf = (!result)? cf : NULL;
  1578. if(result) {
  1579. Curl_safefree(cf);
  1580. Curl_safefree(ctx);
  1581. }
  1582. return result;
  1583. }
  1584. static CURLcode cf_tcp_accept_connect(struct Curl_cfilter *cf,
  1585. struct Curl_easy *data,
  1586. bool blocking, bool *done)
  1587. {
  1588. /* we start accepted, if we ever close, we cannot go on */
  1589. (void)data;
  1590. (void)blocking;
  1591. if(cf->connected) {
  1592. *done = TRUE;
  1593. return CURLE_OK;
  1594. }
  1595. return CURLE_FAILED_INIT;
  1596. }
  1597. struct Curl_cftype Curl_cft_tcp_accept = {
  1598. "TCP-ACCEPT",
  1599. CF_TYPE_IP_CONNECT,
  1600. CURL_LOG_DEFAULT,
  1601. cf_socket_destroy,
  1602. cf_tcp_accept_connect,
  1603. cf_socket_close,
  1604. cf_socket_get_host, /* TODO: not accurate */
  1605. cf_socket_get_select_socks,
  1606. cf_socket_data_pending,
  1607. cf_socket_send,
  1608. cf_socket_recv,
  1609. cf_socket_cntrl,
  1610. cf_socket_conn_is_alive,
  1611. Curl_cf_def_conn_keep_alive,
  1612. cf_socket_query,
  1613. };
  1614. CURLcode Curl_conn_tcp_listen_set(struct Curl_easy *data,
  1615. struct connectdata *conn,
  1616. int sockindex, curl_socket_t *s)
  1617. {
  1618. CURLcode result;
  1619. struct Curl_cfilter *cf = NULL;
  1620. struct cf_socket_ctx *ctx = NULL;
  1621. /* replace any existing */
  1622. Curl_conn_cf_discard_all(data, conn, sockindex);
  1623. DEBUGASSERT(conn->sock[sockindex] == CURL_SOCKET_BAD);
  1624. ctx = calloc(sizeof(*ctx), 1);
  1625. if(!ctx) {
  1626. result = CURLE_OUT_OF_MEMORY;
  1627. goto out;
  1628. }
  1629. ctx->transport = conn->transport;
  1630. ctx->sock = *s;
  1631. ctx->accepted = FALSE;
  1632. result = Curl_cf_create(&cf, &Curl_cft_tcp_accept, ctx);
  1633. if(result)
  1634. goto out;
  1635. Curl_conn_cf_add(data, conn, sockindex, cf);
  1636. conn->sock[sockindex] = ctx->sock;
  1637. set_local_ip(cf, data);
  1638. ctx->active = TRUE;
  1639. ctx->connected_at = Curl_now();
  1640. cf->connected = TRUE;
  1641. DEBUGF(LOG_CF(data, cf, "Curl_conn_tcp_listen_set(%d)", (int)ctx->sock));
  1642. out:
  1643. if(result) {
  1644. Curl_safefree(cf);
  1645. Curl_safefree(ctx);
  1646. }
  1647. return result;
  1648. }
  1649. static void set_accepted_remote_ip(struct Curl_cfilter *cf,
  1650. struct Curl_easy *data)
  1651. {
  1652. struct cf_socket_ctx *ctx = cf->ctx;
  1653. #ifdef HAVE_GETPEERNAME
  1654. char buffer[STRERROR_LEN];
  1655. struct Curl_sockaddr_storage ssrem;
  1656. curl_socklen_t plen;
  1657. ctx->r_ip[0] = 0;
  1658. ctx->r_port = 0;
  1659. plen = sizeof(ssrem);
  1660. memset(&ssrem, 0, plen);
  1661. if(getpeername(ctx->sock, (struct sockaddr*) &ssrem, &plen)) {
  1662. int error = SOCKERRNO;
  1663. failf(data, "getpeername() failed with errno %d: %s",
  1664. error, Curl_strerror(error, buffer, sizeof(buffer)));
  1665. return;
  1666. }
  1667. if(!Curl_addr2string((struct sockaddr*)&ssrem, plen,
  1668. ctx->r_ip, &ctx->r_port)) {
  1669. failf(data, "ssrem inet_ntop() failed with errno %d: %s",
  1670. errno, Curl_strerror(errno, buffer, sizeof(buffer)));
  1671. return;
  1672. }
  1673. #else
  1674. ctx->r_ip[0] = 0;
  1675. ctx->r_port = 0;
  1676. (void)data;
  1677. #endif
  1678. }
  1679. CURLcode Curl_conn_tcp_accepted_set(struct Curl_easy *data,
  1680. struct connectdata *conn,
  1681. int sockindex, curl_socket_t *s)
  1682. {
  1683. struct Curl_cfilter *cf = NULL;
  1684. struct cf_socket_ctx *ctx = NULL;
  1685. cf = conn->cfilter[sockindex];
  1686. if(!cf || cf->cft != &Curl_cft_tcp_accept)
  1687. return CURLE_FAILED_INIT;
  1688. ctx = cf->ctx;
  1689. /* discard the listen socket */
  1690. socket_close(data, conn, TRUE, ctx->sock);
  1691. ctx->sock = *s;
  1692. conn->sock[sockindex] = ctx->sock;
  1693. set_accepted_remote_ip(cf, data);
  1694. set_local_ip(cf, data);
  1695. ctx->active = TRUE;
  1696. ctx->accepted = TRUE;
  1697. ctx->connected_at = Curl_now();
  1698. cf->connected = TRUE;
  1699. DEBUGF(LOG_CF(data, cf, "accepted_set(sock=%d, remote=%s port=%d)",
  1700. (int)ctx->sock, ctx->r_ip, ctx->r_port));
  1701. return CURLE_OK;
  1702. }
  1703. bool Curl_cf_is_socket(struct Curl_cfilter *cf)
  1704. {
  1705. return cf && (cf->cft == &Curl_cft_tcp ||
  1706. cf->cft == &Curl_cft_udp ||
  1707. cf->cft == &Curl_cft_unix ||
  1708. cf->cft == &Curl_cft_tcp_accept);
  1709. }
  1710. CURLcode Curl_cf_socket_peek(struct Curl_cfilter *cf,
  1711. struct Curl_easy *data,
  1712. curl_socket_t *psock,
  1713. const struct Curl_sockaddr_ex **paddr,
  1714. const char **pr_ip_str, int *pr_port,
  1715. const char **pl_ip_str, int *pl_port)
  1716. {
  1717. if(Curl_cf_is_socket(cf) && cf->ctx) {
  1718. struct cf_socket_ctx *ctx = cf->ctx;
  1719. if(psock)
  1720. *psock = ctx->sock;
  1721. if(paddr)
  1722. *paddr = &ctx->addr;
  1723. if(pr_ip_str)
  1724. *pr_ip_str = ctx->r_ip;
  1725. if(pr_port)
  1726. *pr_port = ctx->r_port;
  1727. if(pl_port ||pl_ip_str) {
  1728. set_local_ip(cf, data);
  1729. if(pl_ip_str)
  1730. *pl_ip_str = ctx->l_ip;
  1731. if(pl_port)
  1732. *pl_port = ctx->l_port;
  1733. }
  1734. return CURLE_OK;
  1735. }
  1736. return CURLE_FAILED_INIT;
  1737. }