2018-02-06 15:48:12 +01:00
|
|
|
/*
|
|
|
|
* Utility routines.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2018 Denys Vlasenko
|
|
|
|
*
|
|
|
|
* Licensed under GPLv2 or later, see file LICENSE in this source tree.
|
|
|
|
*/
|
2018-02-06 17:11:15 +01:00
|
|
|
//kbuild:lib-$(CONFIG_FTPGET) += parse_pasv_epsv.o
|
|
|
|
//kbuild:lib-$(CONFIG_FTPPUT) += parse_pasv_epsv.o
|
2018-02-06 15:48:12 +01:00
|
|
|
//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 <some text>(|||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, ')');
|
2020-07-19 20:59:35 +02:00
|
|
|
if (!ptr) ptr = strrchr(buf, '\r'); /* for PASV responses not ending with ')' */
|
|
|
|
if (!ptr) ptr = strrchr(buf, '\n'); /* for PASV responses not ending with ')' */
|
2018-02-06 15:48:12 +01:00
|
|
|
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;
|
|
|
|
}
|