/* * Utility routines. * * Copyright (C) 2018 Denys Vlasenko * * Licensed under GPLv2 or later, see file LICENSE in this source tree. */ //kbuild:lib-$(CONFIG_FTPGET) += parse_pasv_epsv.o //kbuild:lib-$(CONFIG_FTPPUT) += parse_pasv_epsv.o //kbuild:lib-$(CONFIG_WGET) += parse_pasv_epsv.o #include "libbb.h" int FAST_FUNC parse_pasv_epsv(char *buf) { /* * PASV command will not work for IPv6. RFC2428 describes * IPv6-capable "extended PASV" - EPSV. * * "EPSV [protocol]" asks server to bind to and listen on a data port * in specified protocol. Protocol is 1 for IPv4, 2 for IPv6. * If not specified, defaults to "same as used for control connection". * If server understood you, it should answer "229 (|||port|)" * where "|" are literal pipe chars and "port" is ASCII decimal port#. * * There is also an IPv6-capable replacement for PORT (EPRT), * but we don't need that. * * NB: PASV may still work for some servers even over IPv6. * For example, vsftp happily answers * "227 Entering Passive Mode (0,0,0,0,n,n)" and proceeds as usual. */ char *ptr; int port; if (!ENABLE_FEATURE_IPV6 || buf[2] == '7' /* "227" */) { /* Response is "227 garbageN1,N2,N3,N4,P1,P2[)garbage]" * Server's IP is N1.N2.N3.N4 (we ignore it) * Server's port for data connection is P1*256+P2 */ ptr = strrchr(buf, ')'); if (!ptr) ptr = strrchr(buf, '\r'); /* for PASV responses not ending with ')' */ if (!ptr) ptr = strrchr(buf, '\n'); /* for PASV responses not ending with ')' */ if (ptr) *ptr = '\0'; ptr = strrchr(buf, ','); if (!ptr) return -1; *ptr = '\0'; port = xatou_range(ptr + 1, 0, 255); ptr = strrchr(buf, ','); if (!ptr) return -1; *ptr = '\0'; port += xatou_range(ptr + 1, 0, 255) * 256; } else { /* Response is "229 garbage(|||P1|)" * Server's port for data connection is P1 */ ptr = strrchr(buf, '|'); if (!ptr) return -1; *ptr = '\0'; ptr = strrchr(buf, '|'); if (!ptr) return -1; *ptr = '\0'; port = xatou_range(ptr + 1, 0, 65535); } return port; }