blob: e38138c0a85b1b4d8ef0e753597ab323c5113ae9 [file] [log] [blame]
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001/* vi: set sw=4 ts=4: */
2/*
3 * Simple FTP daemon, based on vsftpd 2.0.7 (written by Chris Evans)
4 *
5 * Author: Adam Tkac <vonsch@gmail.com>
6 *
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02007 * Licensed under GPLv2, see file LICENSE in this source tree.
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00008 *
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00009 * Only subset of FTP protocol is implemented but vast majority of clients
Denis Vlasenko1432cb42009-03-18 00:45:00 +000010 * should not have any problem.
11 *
12 * You have to run this daemon via inetd.
Denis Vlasenkobf9d1792009-03-08 09:31:28 +000013 */
14
Denys Vlasenko115c35d2011-03-08 03:01:10 +010015//usage:#define ftpd_trivial_usage
16//usage: "[-wvS] [-t N] [-T N] [DIR]"
17//usage:#define ftpd_full_usage "\n\n"
18//usage: "Anonymous FTP server\n"
19//usage: "\n"
20//usage: "ftpd should be used as an inetd service.\n"
21//usage: "ftpd's line for inetd.conf:\n"
22//usage: " 21 stream tcp nowait root ftpd ftpd /files/to/serve\n"
23//usage: "It also can be ran from tcpsvd:\n"
24//usage: " tcpsvd -vE 0.0.0.0 21 ftpd /files/to/serve\n"
Denys Vlasenko115c35d2011-03-08 03:01:10 +010025//usage: "\n -w Allow upload"
26//usage: "\n -v Log errors to stderr. -vv: verbose log"
27//usage: "\n -S Log errors to syslog. -SS: verbose log"
28//usage: "\n -t,-T Idle and absolute timeouts"
29//usage: "\n DIR Change root to this directory"
30
Denis Vlasenkobf9d1792009-03-08 09:31:28 +000031#include "libbb.h"
Denis Vlasenko5e4fda02009-03-08 23:46:48 +000032#include <syslog.h>
Denis Vlasenkobf9d1792009-03-08 09:31:28 +000033#include <netinet/tcp.h>
34
Denis Vlasenkof1a11b52009-03-09 04:13:59 +000035#define FTP_DATACONN 150
36#define FTP_NOOPOK 200
37#define FTP_TYPEOK 200
38#define FTP_PORTOK 200
39#define FTP_STRUOK 200
40#define FTP_MODEOK 200
41#define FTP_ALLOOK 202
42#define FTP_STATOK 211
43#define FTP_STATFILE_OK 213
44#define FTP_HELP 214
45#define FTP_SYSTOK 215
46#define FTP_GREET 220
47#define FTP_GOODBYE 221
48#define FTP_TRANSFEROK 226
49#define FTP_PASVOK 227
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +000050/*#define FTP_EPRTOK 228*/
51#define FTP_EPSVOK 229
Denis Vlasenkof1a11b52009-03-09 04:13:59 +000052#define FTP_LOGINOK 230
53#define FTP_CWDOK 250
54#define FTP_RMDIROK 250
55#define FTP_DELEOK 250
56#define FTP_RENAMEOK 250
57#define FTP_PWDOK 257
58#define FTP_MKDIROK 257
59#define FTP_GIVEPWORD 331
60#define FTP_RESTOK 350
61#define FTP_RNFROK 350
Denis Vlasenko20c82162009-03-16 16:19:53 +000062#define FTP_TIMEOUT 421
Denis Vlasenkof1a11b52009-03-09 04:13:59 +000063#define FTP_BADSENDCONN 425
64#define FTP_BADSENDNET 426
65#define FTP_BADSENDFILE 451
66#define FTP_BADCMD 500
67#define FTP_COMMANDNOTIMPL 502
68#define FTP_NEEDUSER 503
69#define FTP_NEEDRNFR 503
70#define FTP_BADSTRU 504
71#define FTP_BADMODE 504
72#define FTP_LOGINERR 530
73#define FTP_FILEFAIL 550
74#define FTP_NOPERM 550
75#define FTP_UPLOADFAIL 553
76
77#define STR1(s) #s
78#define STR(s) STR1(s)
79
Denis Vlasenkoc41cba52009-03-09 15:46:07 +000080/* Convert a constant to 3-digit string, packed into uint32_t */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +000081enum {
Denis Vlasenkoc41cba52009-03-09 15:46:07 +000082 /* Shift for Nth decimal digit */
Denis Vlasenkofbf58462009-03-16 20:54:45 +000083 SHIFT2 = 0 * BB_LITTLE_ENDIAN + 24 * BB_BIG_ENDIAN,
84 SHIFT1 = 8 * BB_LITTLE_ENDIAN + 16 * BB_BIG_ENDIAN,
85 SHIFT0 = 16 * BB_LITTLE_ENDIAN + 8 * BB_BIG_ENDIAN,
86 /* And for 4th position (space) */
87 SHIFTsp = 24 * BB_LITTLE_ENDIAN + 0 * BB_BIG_ENDIAN,
Denis Vlasenkoc41cba52009-03-09 15:46:07 +000088};
89#define STRNUM32(s) (uint32_t)(0 \
90 | (('0' + ((s) / 1 % 10)) << SHIFT0) \
91 | (('0' + ((s) / 10 % 10)) << SHIFT1) \
92 | (('0' + ((s) / 100 % 10)) << SHIFT2) \
93)
Denis Vlasenkofbf58462009-03-16 20:54:45 +000094#define STRNUM32sp(s) (uint32_t)(0 \
95 | (' ' << SHIFTsp) \
96 | (('0' + ((s) / 1 % 10)) << SHIFT0) \
97 | (('0' + ((s) / 10 % 10)) << SHIFT1) \
98 | (('0' + ((s) / 100 % 10)) << SHIFT2) \
99)
100
101#define MSG_OK "Operation successful\r\n"
102#define MSG_ERR "Error\r\n"
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000103
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000104struct globals {
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000105 int pasv_listen_fd;
Denis Vlasenko5b492ee2009-03-18 14:12:22 +0000106#if !BB_MMU
107 int root_fd;
108#endif
Denis Vlasenko20c82162009-03-16 16:19:53 +0000109 int local_file_fd;
Denis Vlasenkoe6c94a62009-03-17 05:11:51 +0000110 unsigned end_time;
111 unsigned timeout;
Denis Vlasenkoedb0de42009-03-17 12:23:24 +0000112 unsigned verbose;
Denis Vlasenko20c82162009-03-16 16:19:53 +0000113 off_t local_file_pos;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000114 off_t restart_pos;
Denis Vlasenko20c82162009-03-16 16:19:53 +0000115 len_and_sockaddr *local_addr;
116 len_and_sockaddr *port_addr;
Denis Vlasenko57a3b172009-03-09 04:38:37 +0000117 char *ftp_cmd;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000118 char *ftp_arg;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000119#if ENABLE_FEATURE_FTP_WRITE
120 char *rnfr_filename;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000121#endif
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000122 /* We need these aligned to uint32_t */
123 char msg_ok [(sizeof("NNN " MSG_OK ) + 3) & 0xfffc];
124 char msg_err[(sizeof("NNN " MSG_ERR) + 3) & 0xfffc];
Denys Vlasenko98a4c7c2010-02-04 15:00:15 +0100125} FIX_ALIASING;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000126#define G (*(struct globals*)&bb_common_bufsiz1)
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000127#define INIT_G() do { \
Denis Vlasenkoe6c94a62009-03-17 05:11:51 +0000128 /* Moved to main */ \
129 /*strcpy(G.msg_ok + 4, MSG_OK );*/ \
130 /*strcpy(G.msg_err + 4, MSG_ERR);*/ \
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000131} while (0)
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000132
Denis Vlasenko57a3b172009-03-09 04:38:37 +0000133
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000134static char *
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000135escape_text(const char *prepend, const char *str, unsigned escapee)
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000136{
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000137 unsigned retlen, remainlen, chunklen;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000138 char *ret, *found;
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000139 char append;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000140
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000141 append = (char)escapee;
142 escapee >>= 8;
143
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000144 remainlen = strlen(str);
Denis Vlasenko57a3b172009-03-09 04:38:37 +0000145 retlen = strlen(prepend);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000146 ret = xmalloc(retlen + remainlen * 2 + 1 + 1);
Denis Vlasenko57a3b172009-03-09 04:38:37 +0000147 strcpy(ret, prepend);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000148
149 for (;;) {
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000150 found = strchrnul(str, escapee);
151 chunklen = found - str + 1;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000152
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000153 /* Copy chunk up to and including escapee (or NUL) to ret */
154 memcpy(ret + retlen, str, chunklen);
Denis Vlasenko57a3b172009-03-09 04:38:37 +0000155 retlen += chunklen;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000156
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000157 if (*found == '\0') {
158 /* It wasn't escapee, it was NUL! */
159 ret[retlen - 1] = append; /* replace NUL */
160 ret[retlen] = '\0'; /* add NUL */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000161 break;
162 }
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000163 ret[retlen++] = escapee; /* duplicate escapee */
164 str = found + 1;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000165 }
166 return ret;
167}
168
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000169/* Returns strlen as a bonus */
170static unsigned
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000171replace_char(char *str, char from, char to)
172{
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000173 char *p = str;
174 while (*p) {
175 if (*p == from)
176 *p = to;
177 p++;
178 }
179 return p - str;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000180}
181
Denis Vlasenkoedb0de42009-03-17 12:23:24 +0000182static void
183verbose_log(const char *str)
184{
185 bb_error_msg("%.*s", (int)strcspn(str, "\r\n"), str);
186}
187
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000188/* NB: status_str is char[4] packed into uint32_t */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000189static void
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000190cmdio_write(uint32_t status_str, const char *str)
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000191{
Denis Vlasenko57a3b172009-03-09 04:38:37 +0000192 char *response;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000193 int len;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000194
Denis Vlasenko1432cb42009-03-18 00:45:00 +0000195 /* FTP uses telnet protocol for command link.
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000196 * In telnet, 0xff is an escape char, and needs to be escaped: */
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000197 response = escape_text((char *) &status_str, str, (0xff << 8) + '\r');
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000198
Denis Vlasenko1432cb42009-03-18 00:45:00 +0000199 /* FTP sends embedded LFs as NULs */
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000200 len = replace_char(response, '\n', '\0');
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000201
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000202 response[len++] = '\n'; /* tack on trailing '\n' */
Denis Vlasenkof1a11b52009-03-09 04:13:59 +0000203 xwrite(STDOUT_FILENO, response, len);
Denis Vlasenkoedb0de42009-03-17 12:23:24 +0000204 if (G.verbose > 1)
205 verbose_log(response);
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000206 free(response);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000207}
208
209static void
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000210cmdio_write_ok(unsigned status)
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000211{
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000212 *(uint32_t *) G.msg_ok = status;
213 xwrite(STDOUT_FILENO, G.msg_ok, sizeof("NNN " MSG_OK) - 1);
Denis Vlasenkoedb0de42009-03-17 12:23:24 +0000214 if (G.verbose > 1)
215 verbose_log(G.msg_ok);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000216}
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000217#define WRITE_OK(a) cmdio_write_ok(STRNUM32sp(a))
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000218
Denis Vlasenkof1a11b52009-03-09 04:13:59 +0000219/* TODO: output strerr(errno) if errno != 0? */
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000220static void
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000221cmdio_write_error(unsigned status)
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000222{
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000223 *(uint32_t *) G.msg_err = status;
224 xwrite(STDOUT_FILENO, G.msg_err, sizeof("NNN " MSG_ERR) - 1);
Denys Vlasenko115c35d2011-03-08 03:01:10 +0100225 if (G.verbose > 0)
Denis Vlasenkoedb0de42009-03-17 12:23:24 +0000226 verbose_log(G.msg_err);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000227}
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000228#define WRITE_ERR(a) cmdio_write_error(STRNUM32sp(a))
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000229
230static void
231cmdio_write_raw(const char *p_text)
232{
Denis Vlasenkof1a11b52009-03-09 04:13:59 +0000233 xwrite_str(STDOUT_FILENO, p_text);
Denis Vlasenkoedb0de42009-03-17 12:23:24 +0000234 if (G.verbose > 1)
235 verbose_log(p_text);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000236}
237
Denis Vlasenko20c82162009-03-16 16:19:53 +0000238static void
239timeout_handler(int sig UNUSED_PARAM)
240{
241 off_t pos;
242 int sv_errno = errno;
243
Denis Vlasenkoe6c94a62009-03-17 05:11:51 +0000244 if ((int)(monotonic_sec() - G.end_time) >= 0)
Denis Vlasenko20c82162009-03-16 16:19:53 +0000245 goto timed_out;
246
247 if (!G.local_file_fd)
248 goto timed_out;
249
250 pos = xlseek(G.local_file_fd, 0, SEEK_CUR);
251 if (pos == G.local_file_pos)
252 goto timed_out;
253 G.local_file_pos = pos;
254
255 alarm(G.timeout);
256 errno = sv_errno;
257 return;
258
259 timed_out:
260 cmdio_write_raw(STR(FTP_TIMEOUT)" Timeout\r\n");
261/* TODO: do we need to abort (as opposed to usual shutdown) data transfer? */
262 exit(1);
263}
264
Denis Vlasenko9e959202009-03-09 03:15:05 +0000265/* Simple commands */
266
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000267static void
268handle_pwd(void)
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000269{
Denis Vlasenko57a3b172009-03-09 04:38:37 +0000270 char *cwd, *response;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000271
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000272 cwd = xrealloc_getcwd_or_warn(NULL);
273 if (cwd == NULL)
274 cwd = xstrdup("");
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000275
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000276 /* We have to promote each " to "" */
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000277 response = escape_text(" \"", cwd, ('"' << 8) + '"');
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000278 free(cwd);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000279 cmdio_write(STRNUM32(FTP_PWDOK), response);
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000280 free(response);
281}
282
283static void
284handle_cwd(void)
285{
286 if (!G.ftp_arg || chdir(G.ftp_arg) != 0) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000287 WRITE_ERR(FTP_FILEFAIL);
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000288 return;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000289 }
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000290 WRITE_OK(FTP_CWDOK);
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000291}
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000292
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000293static void
294handle_cdup(void)
295{
296 G.ftp_arg = (char*)"..";
297 handle_cwd();
298}
299
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000300static void
Denis Vlasenko9e959202009-03-09 03:15:05 +0000301handle_stat(void)
302{
Denis Vlasenkoe3b840c2009-03-18 14:25:28 +0000303 cmdio_write_raw(STR(FTP_STATOK)"-Server status:\r\n"
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000304 " TYPE: BINARY\r\n"
Denis Vlasenkof1a11b52009-03-09 04:13:59 +0000305 STR(FTP_STATOK)" Ok\r\n");
Denis Vlasenko9e959202009-03-09 03:15:05 +0000306}
307
Denis Vlasenko1a825552009-03-17 12:40:34 +0000308/* Examples of HELP and FEAT:
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000309# nc -vvv ftp.kernel.org 21
310ftp.kernel.org (130.239.17.4:21) open
311220 Welcome to ftp.kernel.org.
312FEAT
313211-Features:
314 EPRT
315 EPSV
316 MDTM
317 PASV
318 REST STREAM
319 SIZE
320 TVFS
321 UTF8
322211 End
323HELP
324214-The following commands are recognized.
325 ABOR ACCT ALLO APPE CDUP CWD DELE EPRT EPSV FEAT HELP LIST MDTM MKD
326 MODE NLST NOOP OPTS PASS PASV PORT PWD QUIT REIN REST RETR RMD RNFR
327 RNTO SITE SIZE SMNT STAT STOR STOU STRU SYST TYPE USER XCUP XCWD XMKD
328 XPWD XRMD
329214 Help OK.
330*/
Denis Vlasenko9e959202009-03-09 03:15:05 +0000331static void
Denis Vlasenko1a825552009-03-17 12:40:34 +0000332handle_feat(unsigned status)
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000333{
Denis Vlasenko1a825552009-03-17 12:40:34 +0000334 cmdio_write(status, "-Features:");
335 cmdio_write_raw(" EPSV\r\n"
336 " PASV\r\n"
337 " REST STREAM\r\n"
Denis Vlasenko1432cb42009-03-18 00:45:00 +0000338 " MDTM\r\n"
Denis Vlasenko1a825552009-03-17 12:40:34 +0000339 " SIZE\r\n");
340 cmdio_write(status, " Ok");
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000341}
342
Denis Vlasenko9e959202009-03-09 03:15:05 +0000343/* Download commands */
344
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000345static inline int
346port_active(void)
347{
348 return (G.port_addr != NULL);
349}
350
351static inline int
352pasv_active(void)
353{
354 return (G.pasv_listen_fd > STDOUT_FILENO);
355}
356
357static void
358port_pasv_cleanup(void)
359{
360 free(G.port_addr);
361 G.port_addr = NULL;
362 if (G.pasv_listen_fd > STDOUT_FILENO)
363 close(G.pasv_listen_fd);
364 G.pasv_listen_fd = -1;
365}
366
367/* On error, emits error code to the peer */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000368static int
369ftpdataio_get_pasv_fd(void)
370{
371 int remote_fd;
372
373 remote_fd = accept(G.pasv_listen_fd, NULL, 0);
374
375 if (remote_fd < 0) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000376 WRITE_ERR(FTP_BADSENDCONN);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000377 return remote_fd;
378 }
379
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000380 setsockopt(remote_fd, SOL_SOCKET, SO_KEEPALIVE, &const_int_1, sizeof(const_int_1));
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000381 return remote_fd;
382}
383
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000384/* Clears port/pasv data.
385 * This means we dont waste resources, for example, keeping
386 * PASV listening socket open when it is no longer needed.
387 * On error, emits error code to the peer (or exits).
388 * On success, emits p_status_msg to the peer.
389 */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000390static int
391get_remote_transfer_fd(const char *p_status_msg)
392{
393 int remote_fd;
394
395 if (pasv_active())
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000396 /* On error, emits error code to the peer */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000397 remote_fd = ftpdataio_get_pasv_fd();
398 else
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000399 /* Exits on error */
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000400 remote_fd = xconnect_stream(G.port_addr);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000401
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000402 port_pasv_cleanup();
403
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000404 if (remote_fd < 0)
405 return remote_fd;
406
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000407 cmdio_write(STRNUM32(FTP_DATACONN), p_status_msg);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000408 return remote_fd;
409}
410
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000411/* If there were neither PASV nor PORT, emits error code to the peer */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000412static int
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000413port_or_pasv_was_seen(void)
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000414{
415 if (!pasv_active() && !port_active()) {
Denis Vlasenkoe3b840c2009-03-18 14:25:28 +0000416 cmdio_write_raw(STR(FTP_BADSENDCONN)" Use PORT/PASV first\r\n");
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000417 return 0;
418 }
419
420 return 1;
421}
422
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000423/* Exits on error */
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000424static unsigned
425bind_for_passive_mode(void)
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000426{
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000427 int fd;
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000428 unsigned port;
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000429
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000430 port_pasv_cleanup();
431
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000432 G.pasv_listen_fd = fd = xsocket(G.local_addr->u.sa.sa_family, SOCK_STREAM, 0);
433 setsockopt_reuseaddr(fd);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000434
Denys Vlasenkoca183112011-04-07 17:52:20 +0200435 set_nport(&G.local_addr->u.sa, 0);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000436 xbind(fd, &G.local_addr->u.sa, G.local_addr->len);
437 xlisten(fd, 1);
438 getsockname(fd, &G.local_addr->u.sa, &G.local_addr->len);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000439
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000440 port = get_nport(&G.local_addr->u.sa);
441 port = ntohs(port);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000442 return port;
443}
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000444
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000445/* Exits on error */
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000446static void
447handle_pasv(void)
448{
449 unsigned port;
450 char *addr, *response;
451
452 port = bind_for_passive_mode();
453
454 if (G.local_addr->u.sa.sa_family == AF_INET)
455 addr = xmalloc_sockaddr2dotted_noport(&G.local_addr->u.sa);
456 else /* seen this in the wild done by other ftp servers: */
457 addr = xstrdup("0.0.0.0");
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000458 replace_char(addr, '.', ',');
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000459
Denis Vlasenkoe3b840c2009-03-18 14:25:28 +0000460 response = xasprintf(STR(FTP_PASVOK)" PASV ok (%s,%u,%u)\r\n",
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000461 addr, (int)(port >> 8), (int)(port & 255));
462 free(addr);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000463 cmdio_write_raw(response);
464 free(response);
465}
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000466
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000467/* Exits on error */
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000468static void
469handle_epsv(void)
470{
471 unsigned port;
472 char *response;
473
474 port = bind_for_passive_mode();
Denis Vlasenkoe3b840c2009-03-18 14:25:28 +0000475 response = xasprintf(STR(FTP_EPSVOK)" EPSV ok (|||%u|)\r\n", port);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000476 cmdio_write_raw(response);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000477 free(response);
478}
479
480static void
481handle_port(void)
482{
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000483 unsigned port, port_hi;
484 char *raw, *comma;
Denis Vlasenko43bb7bb2009-03-16 19:54:06 +0000485#ifdef WHY_BOTHER_WE_CAN_ASSUME_IP_MATCHES
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000486 socklen_t peer_ipv4_len;
487 struct sockaddr_in peer_ipv4;
488 struct in_addr port_ipv4_sin_addr;
Denis Vlasenko43bb7bb2009-03-16 19:54:06 +0000489#endif
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000490
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000491 port_pasv_cleanup();
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000492
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000493 raw = G.ftp_arg;
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000494
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000495 /* PORT command format makes sense only over IPv4 */
Denis Vlasenko43bb7bb2009-03-16 19:54:06 +0000496 if (!raw
497#ifdef WHY_BOTHER_WE_CAN_ASSUME_IP_MATCHES
498 || G.local_addr->u.sa.sa_family != AF_INET
499#endif
500 ) {
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000501 bail:
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000502 WRITE_ERR(FTP_BADCMD);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000503 return;
504 }
505
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000506 comma = strrchr(raw, ',');
507 if (comma == NULL)
508 goto bail;
509 *comma = '\0';
510 port = bb_strtou(&comma[1], NULL, 10);
511 if (errno || port > 0xff)
512 goto bail;
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000513
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000514 comma = strrchr(raw, ',');
515 if (comma == NULL)
516 goto bail;
517 *comma = '\0';
518 port_hi = bb_strtou(&comma[1], NULL, 10);
519 if (errno || port_hi > 0xff)
520 goto bail;
521 port |= port_hi << 8;
522
Denis Vlasenko43bb7bb2009-03-16 19:54:06 +0000523#ifdef WHY_BOTHER_WE_CAN_ASSUME_IP_MATCHES
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000524 replace_char(raw, ',', '.');
525
526 /* We are verifying that PORT's IP matches getpeername().
527 * Otherwise peer can make us open data connections
528 * to other hosts (security problem!)
529 * This code would be too simplistic:
530 * lsa = xdotted2sockaddr(raw, port);
531 * if (lsa == NULL) goto bail;
532 */
533 if (!inet_aton(raw, &port_ipv4_sin_addr))
534 goto bail;
535 peer_ipv4_len = sizeof(peer_ipv4);
536 if (getpeername(STDIN_FILENO, &peer_ipv4, &peer_ipv4_len) != 0)
537 goto bail;
538 if (memcmp(&port_ipv4_sin_addr, &peer_ipv4.sin_addr, sizeof(struct in_addr)) != 0)
539 goto bail;
540
541 G.port_addr = xdotted2sockaddr(raw, port);
Denis Vlasenko43bb7bb2009-03-16 19:54:06 +0000542#else
543 G.port_addr = get_peer_lsa(STDIN_FILENO);
Denys Vlasenkoca183112011-04-07 17:52:20 +0200544 set_nport(&G.port_addr->u.sa, htons(port));
Denis Vlasenko43bb7bb2009-03-16 19:54:06 +0000545#endif
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000546 WRITE_OK(FTP_PORTOK);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000547}
548
549static void
550handle_rest(void)
551{
552 /* When ftp_arg == NULL simply restart from beginning */
Denys Vlasenko77832482010-08-12 14:14:45 +0200553 G.restart_pos = G.ftp_arg ? xatoi_positive(G.ftp_arg) : 0;
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000554 WRITE_OK(FTP_RESTOK);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000555}
556
557static void
558handle_retr(void)
559{
560 struct stat statbuf;
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000561 off_t bytes_transferred;
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000562 int remote_fd;
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000563 int local_file_fd;
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000564 off_t offset = G.restart_pos;
565 char *response;
566
567 G.restart_pos = 0;
568
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000569 if (!port_or_pasv_was_seen())
570 return; /* port_or_pasv_was_seen emitted error response */
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000571
572 /* O_NONBLOCK is useful if file happens to be a device node */
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000573 local_file_fd = G.ftp_arg ? open(G.ftp_arg, O_RDONLY | O_NONBLOCK) : -1;
574 if (local_file_fd < 0) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000575 WRITE_ERR(FTP_FILEFAIL);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000576 return;
577 }
578
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000579 if (fstat(local_file_fd, &statbuf) != 0 || !S_ISREG(statbuf.st_mode)) {
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000580 /* Note - pretend open failed */
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000581 WRITE_ERR(FTP_FILEFAIL);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000582 goto file_close_out;
583 }
Denis Vlasenko20c82162009-03-16 16:19:53 +0000584 G.local_file_fd = local_file_fd;
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000585
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000586 /* Now deactive O_NONBLOCK, otherwise we have a problem
587 * on DMAPI filesystems such as XFS DMAPI.
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000588 */
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000589 ndelay_off(local_file_fd);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000590
591 /* Set the download offset (from REST) if any */
592 if (offset != 0)
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000593 xlseek(local_file_fd, offset, SEEK_SET);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000594
595 response = xasprintf(
Denis Vlasenkoe3b840c2009-03-18 14:25:28 +0000596 " Opening BINARY connection for %s (%"OFF_FMT"u bytes)",
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000597 G.ftp_arg, statbuf.st_size);
598 remote_fd = get_remote_transfer_fd(response);
599 free(response);
600 if (remote_fd < 0)
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000601 goto file_close_out;
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000602
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000603 bytes_transferred = bb_copyfd_eof(local_file_fd, remote_fd);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000604 close(remote_fd);
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000605 if (bytes_transferred < 0)
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000606 WRITE_ERR(FTP_BADSENDFILE);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000607 else
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000608 WRITE_OK(FTP_TRANSFEROK);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000609
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000610 file_close_out:
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000611 close(local_file_fd);
Denis Vlasenko20c82162009-03-16 16:19:53 +0000612 G.local_file_fd = 0;
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000613}
614
Denis Vlasenko9e959202009-03-09 03:15:05 +0000615/* List commands */
616
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000617static int
618popen_ls(const char *opt)
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000619{
Denys Vlasenko33f9dc02010-01-17 22:32:22 +0100620 const char *argv[5];
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000621 struct fd_pair outfd;
622 pid_t pid;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000623
Denys Vlasenko33f9dc02010-01-17 22:32:22 +0100624 argv[0] = "ftpd";
625 argv[1] = opt; /* "-l" or "-1" */
626#if BB_MMU
627 argv[2] = "--";
628#else
629 /* NOMMU ftpd ls helper chdirs to argv[2],
630 * preventing peer from seeing real root. */
631 argv[2] = xrealloc_getcwd_or_warn(NULL);
632#endif
633 argv[3] = G.ftp_arg;
634 argv[4] = NULL;
635
Stefan Seyfriedf3fc9ac2010-01-18 02:08:30 +0100636 /* Improve compatibility with non-RFC conforming FTP clients
Denys Vlasenko238c83b2010-04-21 18:05:10 -0400637 * which send e.g. "LIST -l", "LIST -la", "LIST -aL".
Stefan Seyfriedf3fc9ac2010-01-18 02:08:30 +0100638 * See https://bugs.kde.org/show_bug.cgi?id=195578 */
639 if (ENABLE_FEATURE_FTPD_ACCEPT_BROKEN_LIST
Denys Vlasenko238c83b2010-04-21 18:05:10 -0400640 && G.ftp_arg && G.ftp_arg[0] == '-'
Stefan Seyfriedf3fc9ac2010-01-18 02:08:30 +0100641 ) {
642 const char *tmp = strchr(G.ftp_arg, ' ');
643 if (tmp) /* skip the space */
644 tmp++;
645 argv[3] = tmp;
646 }
647
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000648 xpiped_pair(outfd);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000649
Denys Vlasenko8131eea2009-11-02 14:19:51 +0100650 /*fflush_all(); - so far we dont use stdio on output */
Pascal Bellard926031b2010-07-04 15:32:38 +0200651 pid = BB_MMU ? xfork() : xvfork();
Denis Vlasenko5b492ee2009-03-18 14:12:22 +0000652 if (pid == 0) {
653 /* child */
654#if !BB_MMU
Denys Vlasenko33f9dc02010-01-17 22:32:22 +0100655 /* On NOMMU, we want to execute a child - copy of ourself.
656 * In chroot we usually can't do it. Thus we chdir
657 * out of the chroot back to original root,
658 * and (see later below) execute bb_busybox_exec_path
659 * relative to current directory */
Denis Vlasenko5b492ee2009-03-18 14:12:22 +0000660 if (fchdir(G.root_fd) != 0)
661 _exit(127);
Denys Vlasenko33f9dc02010-01-17 22:32:22 +0100662 /*close(G.root_fd); - close_on_exec_on() took care of this */
Denis Vlasenko5b492ee2009-03-18 14:12:22 +0000663#endif
664 /* NB: close _first_, then move fd! */
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000665 close(outfd.rd);
666 xmove_fd(outfd.wr, STDOUT_FILENO);
Denis Vlasenko5b492ee2009-03-18 14:12:22 +0000667 /* Opening /dev/null in chroot is hard.
668 * Just making sure STDIN_FILENO is opened
669 * to something harmless. Paranoia,
670 * ls won't read it anyway */
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000671 close(STDIN_FILENO);
Denis Vlasenko5b492ee2009-03-18 14:12:22 +0000672 dup(STDOUT_FILENO); /* copy will become STDIN_FILENO */
Denys Vlasenko33f9dc02010-01-17 22:32:22 +0100673#if BB_MMU
674 /* memset(&G, 0, sizeof(G)); - ls_main does it */
675 exit(ls_main(ARRAY_SIZE(argv) - 1, (char**) argv));
676#else
Denis Vlasenko5b492ee2009-03-18 14:12:22 +0000677 /* + 1: we must use relative path here if in chroot.
678 * For example, execv("/proc/self/exe") will fail, since
679 * it looks for "/proc/self/exe" _relative to chroot!_ */
Denis Vlasenko42e78b92009-04-04 20:34:22 +0000680 execv(bb_busybox_exec_path + 1, (char**) argv);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000681 _exit(127);
Denis Vlasenko5b492ee2009-03-18 14:12:22 +0000682#endif
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000683 }
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000684
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000685 /* parent */
686 close(outfd.wr);
Denys Vlasenko33f9dc02010-01-17 22:32:22 +0100687#if !BB_MMU
688 free((char*)argv[2]);
689#endif
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000690 return outfd.rd;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000691}
692
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000693enum {
694 USE_CTRL_CONN = 1,
695 LONG_LISTING = 2,
696};
697
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000698static void
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000699handle_dir_common(int opts)
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000700{
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000701 FILE *ls_fp;
702 char *line;
703 int ls_fd;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000704
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000705 if (!(opts & USE_CTRL_CONN) && !port_or_pasv_was_seen())
706 return; /* port_or_pasv_was_seen emitted error response */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000707
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000708 /* -n prevents user/groupname display,
709 * which can be problematic in chroot */
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000710 ls_fd = popen_ls((opts & LONG_LISTING) ? "-l" : "-1");
Denys Vlasenkoa7ccdee2009-11-15 23:28:11 +0100711 ls_fp = xfdopen_for_read(ls_fd);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000712
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000713 if (opts & USE_CTRL_CONN) {
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000714 /* STAT <filename> */
Denis Vlasenkoe3b840c2009-03-18 14:25:28 +0000715 cmdio_write_raw(STR(FTP_STATFILE_OK)"-File status:\r\n");
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000716 while (1) {
Denys Vlasenko3581c622010-01-25 13:39:24 +0100717 line = xmalloc_fgetline(ls_fp);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000718 if (!line)
719 break;
Denis Vlasenkofce4a942009-03-18 16:02:54 +0000720 /* Hack: 0 results in no status at all */
721 /* Note: it's ok that we don't prepend space,
722 * ftp.kernel.org doesn't do that too */
723 cmdio_write(0, line);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000724 free(line);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000725 }
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000726 WRITE_OK(FTP_STATFILE_OK);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000727 } else {
728 /* LIST/NLST [<filename>] */
Denis Vlasenkoe3b840c2009-03-18 14:25:28 +0000729 int remote_fd = get_remote_transfer_fd(" Directory listing");
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000730 if (remote_fd >= 0) {
731 while (1) {
Denys Vlasenko3581c622010-01-25 13:39:24 +0100732 line = xmalloc_fgetline(ls_fp);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000733 if (!line)
734 break;
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000735 /* I've seen clients complaining when they
736 * are fed with ls output with bare '\n'.
737 * Pity... that would be much simpler.
738 */
Denis Vlasenko3a7a1eb2009-03-17 13:03:06 +0000739/* TODO: need to s/LF/NUL/g here */
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000740 xwrite_str(remote_fd, line);
741 xwrite(remote_fd, "\r\n", 2);
742 free(line);
743 }
744 }
745 close(remote_fd);
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000746 WRITE_OK(FTP_TRANSFEROK);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000747 }
748 fclose(ls_fp); /* closes ls_fd too */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000749}
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000750static void
751handle_list(void)
752{
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000753 handle_dir_common(LONG_LISTING);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000754}
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000755static void
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000756handle_nlst(void)
757{
Denis Vlasenko3a7a1eb2009-03-17 13:03:06 +0000758 /* NLST returns list of names, "\r\n" terminated without regard
759 * to the current binary flag. Names may start with "/",
760 * then they represent full names (we don't produce such names),
761 * otherwise names are relative to current directory.
762 * Embedded "\n" are replaced by NULs. This is safe since names
763 * can never contain NUL.
764 */
Denis Vlasenkoc41cba52009-03-09 15:46:07 +0000765 handle_dir_common(0);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000766}
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000767static void
768handle_stat_file(void)
769{
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000770 handle_dir_common(LONG_LISTING + USE_CTRL_CONN);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000771}
772
Denis Vlasenko1432cb42009-03-18 00:45:00 +0000773/* This can be extended to handle MLST, as all info is available
774 * in struct stat for that:
775 * MLST file_name
776 * 250-Listing file_name
777 * type=file;size=4161;modify=19970214165800; /dir/dir/file_name
778 * 250 End
779 * Nano-doc:
780 * MLST [<file or dir name, "." assumed if not given>]
781 * Returned name should be either the same as requested, or fully qualified.
782 * If there was no parameter, return "" or (preferred) fully-qualified name.
783 * Returned "facts" (case is not important):
784 * size - size in octets
785 * modify - last modification time
786 * type - entry type (file,dir,OS.unix=block)
787 * (+ cdir and pdir types for MLSD)
788 * unique - unique id of file/directory (inode#)
789 * perm -
790 * a: can be appended to (APPE)
791 * d: can be deleted (RMD/DELE)
792 * f: can be renamed (RNFR)
793 * r: can be read (RETR)
794 * w: can be written (STOR)
795 * e: can CWD into this dir
796 * l: this dir can be listed (dir only!)
797 * c: can create files in this dir
798 * m: can create dirs in this dir (MKD)
799 * p: can delete files in this dir
800 * UNIX.mode - unix file mode
801 */
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000802static void
Denis Vlasenko1432cb42009-03-18 00:45:00 +0000803handle_size_or_mdtm(int need_size)
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000804{
805 struct stat statbuf;
Denis Vlasenko1432cb42009-03-18 00:45:00 +0000806 struct tm broken_out;
807 char buf[(sizeof("NNN %"OFF_FMT"u\r\n") + sizeof(off_t) * 3)
808 | sizeof("NNN YYYYMMDDhhmmss\r\n")
809 ];
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000810
811 if (!G.ftp_arg
812 || stat(G.ftp_arg, &statbuf) != 0
813 || !S_ISREG(statbuf.st_mode)
814 ) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000815 WRITE_ERR(FTP_FILEFAIL);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000816 return;
817 }
Denis Vlasenko1432cb42009-03-18 00:45:00 +0000818 if (need_size) {
819 sprintf(buf, STR(FTP_STATFILE_OK)" %"OFF_FMT"u\r\n", statbuf.st_size);
820 } else {
821 gmtime_r(&statbuf.st_mtime, &broken_out);
822 sprintf(buf, STR(FTP_STATFILE_OK)" %04u%02u%02u%02u%02u%02u\r\n",
823 broken_out.tm_year + 1900,
824 broken_out.tm_mon,
825 broken_out.tm_mday,
826 broken_out.tm_hour,
827 broken_out.tm_min,
828 broken_out.tm_sec);
829 }
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +0000830 cmdio_write_raw(buf);
831}
832
Denis Vlasenko9e959202009-03-09 03:15:05 +0000833/* Upload commands */
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000834
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000835#if ENABLE_FEATURE_FTP_WRITE
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000836static void
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000837handle_mkd(void)
838{
839 if (!G.ftp_arg || mkdir(G.ftp_arg, 0777) != 0) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000840 WRITE_ERR(FTP_FILEFAIL);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000841 return;
842 }
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000843 WRITE_OK(FTP_MKDIROK);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000844}
845
846static void
847handle_rmd(void)
848{
849 if (!G.ftp_arg || rmdir(G.ftp_arg) != 0) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000850 WRITE_ERR(FTP_FILEFAIL);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000851 return;
852 }
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000853 WRITE_OK(FTP_RMDIROK);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000854}
855
856static void
857handle_dele(void)
858{
859 if (!G.ftp_arg || unlink(G.ftp_arg) != 0) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000860 WRITE_ERR(FTP_FILEFAIL);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000861 return;
862 }
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000863 WRITE_OK(FTP_DELEOK);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000864}
865
866static void
867handle_rnfr(void)
868{
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000869 free(G.rnfr_filename);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000870 G.rnfr_filename = xstrdup(G.ftp_arg);
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000871 WRITE_OK(FTP_RNFROK);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000872}
873
874static void
875handle_rnto(void)
876{
877 int retval;
878
879 /* If we didn't get a RNFR, throw a wobbly */
880 if (G.rnfr_filename == NULL || G.ftp_arg == NULL) {
Denis Vlasenkoe3b840c2009-03-18 14:25:28 +0000881 cmdio_write_raw(STR(FTP_NEEDRNFR)" Use RNFR first\r\n");
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000882 return;
883 }
884
885 retval = rename(G.rnfr_filename, G.ftp_arg);
886 free(G.rnfr_filename);
887 G.rnfr_filename = NULL;
888
889 if (retval) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000890 WRITE_ERR(FTP_FILEFAIL);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000891 return;
892 }
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000893 WRITE_OK(FTP_RENAMEOK);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000894}
895
896static void
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000897handle_upload_common(int is_append, int is_unique)
898{
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000899 struct stat statbuf;
900 char *tempname;
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000901 off_t bytes_transferred;
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000902 off_t offset;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000903 int local_file_fd;
904 int remote_fd;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000905
906 offset = G.restart_pos;
907 G.restart_pos = 0;
908
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000909 if (!port_or_pasv_was_seen())
910 return; /* port_or_pasv_was_seen emitted error response */
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000911
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000912 tempname = NULL;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000913 local_file_fd = -1;
914 if (is_unique) {
Denis Vlasenkof1a11b52009-03-09 04:13:59 +0000915 tempname = xstrdup(" FILE: uniq.XXXXXX");
916 local_file_fd = mkstemp(tempname + 7);
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000917 } else if (G.ftp_arg) {
918 int flags = O_WRONLY | O_CREAT | O_TRUNC;
919 if (is_append)
920 flags = O_WRONLY | O_CREAT | O_APPEND;
921 if (offset)
922 flags = O_WRONLY | O_CREAT;
923 local_file_fd = open(G.ftp_arg, flags, 0666);
924 }
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000925
926 if (local_file_fd < 0
927 || fstat(local_file_fd, &statbuf) != 0
928 || !S_ISREG(statbuf.st_mode)
929 ) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000930 WRITE_ERR(FTP_UPLOADFAIL);
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000931 if (local_file_fd >= 0)
932 goto close_local_and_bail;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000933 return;
934 }
Denis Vlasenko20c82162009-03-16 16:19:53 +0000935 G.local_file_fd = local_file_fd;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000936
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000937 if (offset)
938 xlseek(local_file_fd, offset, SEEK_SET);
939
Denis Vlasenkof1a11b52009-03-09 04:13:59 +0000940 remote_fd = get_remote_transfer_fd(tempname ? tempname : " Ok to send data");
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000941 free(tempname);
942
943 if (remote_fd < 0)
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000944 goto close_local_and_bail;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000945
Denis Vlasenkofc58ba12009-03-15 15:54:58 +0000946 bytes_transferred = bb_copyfd_eof(remote_fd, local_file_fd);
947 close(remote_fd);
948 if (bytes_transferred < 0)
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000949 WRITE_ERR(FTP_BADSENDFILE);
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000950 else
Denis Vlasenkofbf58462009-03-16 20:54:45 +0000951 WRITE_OK(FTP_TRANSFEROK);
Denis Vlasenkof2160b62009-03-16 14:53:54 +0000952
953 close_local_and_bail:
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000954 close(local_file_fd);
Denis Vlasenko20c82162009-03-16 16:19:53 +0000955 G.local_file_fd = 0;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000956}
957
958static void
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000959handle_stor(void)
960{
961 handle_upload_common(0, 0);
962}
963
964static void
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000965handle_appe(void)
966{
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000967 G.restart_pos = 0;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000968 handle_upload_common(1, 0);
969}
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000970
971static void
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000972handle_stou(void)
973{
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000974 G.restart_pos = 0;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000975 handle_upload_common(0, 1);
976}
Denis Vlasenko5e4fda02009-03-08 23:46:48 +0000977#endif /* ENABLE_FEATURE_FTP_WRITE */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +0000978
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000979static uint32_t
980cmdio_get_cmd_and_arg(void)
981{
Denys Vlasenko4a1884d2010-04-03 14:59:12 +0200982 int len;
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000983 uint32_t cmdval;
984 char *cmd;
985
Denis Vlasenko20c82162009-03-16 16:19:53 +0000986 alarm(G.timeout);
987
Denis Vlasenko57a3b172009-03-09 04:38:37 +0000988 free(G.ftp_cmd);
Denys Vlasenko4a1884d2010-04-03 14:59:12 +0200989 {
990 /* Paranoia. Peer may send 1 gigabyte long cmd... */
991 /* Using separate len_on_stk instead of len optimizes
992 * code size (allows len to be in CPU register) */
993 size_t len_on_stk = 8 * 1024;
994 G.ftp_cmd = cmd = xmalloc_fgets_str_len(stdin, "\r\n", &len_on_stk);
995 if (!cmd)
996 exit(0);
997 len = len_on_stk;
998 }
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +0000999
Denis Vlasenko9f57cf62009-03-18 17:32:44 +00001000 /* De-escape telnet: 0xff,0xff => 0xff */
1001 /* RFC959 says that ABOR, STAT, QUIT may be sent even during
1002 * data transfer, and may be preceded by telnet's "Interrupt Process"
1003 * code (two-byte sequence 255,244) and then by telnet "Synch" code
1004 * 255,242 (byte 242 is sent with TCP URG bit using send(MSG_OOB)
1005 * and may generate SIGURG on our side. See RFC854).
1006 * So far we don't support that (may install SIGURG handler if we'd want to),
1007 * but we need to at least remove 255,xxx pairs. lftp sends those. */
1008 /* Then de-escape FTP: NUL => '\n' */
1009 /* Testing for \xff:
1010 * Create file named '\xff': echo Hello >`echo -ne "\xff"`
1011 * Try to get it: ftpget -v 127.0.0.1 Eff `echo -ne "\xff\xff"`
1012 * (need "\xff\xff" until ftpget applet is fixed to do escaping :)
1013 * Testing for embedded LF:
1014 * LF_HERE=`echo -ne "LF\nHERE"`
1015 * echo Hello >"$LF_HERE"
1016 * ftpget -v 127.0.0.1 LF_HERE "$LF_HERE"
1017 */
1018 {
1019 int dst, src;
Denis Vlasenkofce4a942009-03-18 16:02:54 +00001020
Denis Vlasenko9f57cf62009-03-18 17:32:44 +00001021 /* Strip "\r\n" if it is there */
1022 if (len != 0 && cmd[len - 1] == '\n') {
1023 len--;
1024 if (len != 0 && cmd[len - 1] == '\r')
1025 len--;
1026 cmd[len] = '\0';
1027 }
1028 src = strchrnul(cmd, 0xff) - cmd;
1029 /* 99,99% there are neither NULs nor 255s and src == len */
1030 if (src < len) {
1031 dst = src;
1032 do {
1033 if ((unsigned char)(cmd[src]) == 255) {
1034 src++;
1035 /* 255,xxx - skip 255 */
1036 if ((unsigned char)(cmd[src]) != 255) {
1037 /* 255,!255 - skip both */
1038 src++;
1039 continue;
1040 }
1041 /* 255,255 - retain one 255 */
1042 }
1043 /* NUL => '\n' */
1044 cmd[dst++] = cmd[src] ? cmd[src] : '\n';
1045 src++;
1046 } while (src < len);
1047 cmd[dst] = '\0';
1048 }
1049 }
Denis Vlasenkoedb0de42009-03-17 12:23:24 +00001050
1051 if (G.verbose > 1)
Denis Vlasenko1a825552009-03-17 12:40:34 +00001052 verbose_log(cmd);
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +00001053
1054 G.ftp_arg = strchr(cmd, ' ');
Denis Vlasenkof2160b62009-03-16 14:53:54 +00001055 if (G.ftp_arg != NULL)
1056 *G.ftp_arg++ = '\0';
1057
1058 /* Uppercase and pack into uint32_t first word of the command */
Denis Vlasenkoffb4bb32009-03-09 02:23:45 +00001059 cmdval = 0;
1060 while (*cmd)
1061 cmdval = (cmdval << 8) + ((unsigned char)*cmd++ & (unsigned char)~0x20);
1062
1063 return cmdval;
1064}
1065
Denis Vlasenkof2160b62009-03-16 14:53:54 +00001066#define mk_const4(a,b,c,d) (((a * 0x100 + b) * 0x100 + c) * 0x100 + d)
1067#define mk_const3(a,b,c) ((a * 0x100 + b) * 0x100 + c)
1068enum {
1069 const_ALLO = mk_const4('A', 'L', 'L', 'O'),
1070 const_APPE = mk_const4('A', 'P', 'P', 'E'),
1071 const_CDUP = mk_const4('C', 'D', 'U', 'P'),
1072 const_CWD = mk_const3('C', 'W', 'D'),
1073 const_DELE = mk_const4('D', 'E', 'L', 'E'),
1074 const_EPSV = mk_const4('E', 'P', 'S', 'V'),
Denis Vlasenko1a825552009-03-17 12:40:34 +00001075 const_FEAT = mk_const4('F', 'E', 'A', 'T'),
Denis Vlasenkof2160b62009-03-16 14:53:54 +00001076 const_HELP = mk_const4('H', 'E', 'L', 'P'),
1077 const_LIST = mk_const4('L', 'I', 'S', 'T'),
Denis Vlasenko1432cb42009-03-18 00:45:00 +00001078 const_MDTM = mk_const4('M', 'D', 'T', 'M'),
Denis Vlasenkof2160b62009-03-16 14:53:54 +00001079 const_MKD = mk_const3('M', 'K', 'D'),
1080 const_MODE = mk_const4('M', 'O', 'D', 'E'),
1081 const_NLST = mk_const4('N', 'L', 'S', 'T'),
1082 const_NOOP = mk_const4('N', 'O', 'O', 'P'),
1083 const_PASS = mk_const4('P', 'A', 'S', 'S'),
1084 const_PASV = mk_const4('P', 'A', 'S', 'V'),
1085 const_PORT = mk_const4('P', 'O', 'R', 'T'),
1086 const_PWD = mk_const3('P', 'W', 'D'),
1087 const_QUIT = mk_const4('Q', 'U', 'I', 'T'),
1088 const_REST = mk_const4('R', 'E', 'S', 'T'),
1089 const_RETR = mk_const4('R', 'E', 'T', 'R'),
1090 const_RMD = mk_const3('R', 'M', 'D'),
1091 const_RNFR = mk_const4('R', 'N', 'F', 'R'),
1092 const_RNTO = mk_const4('R', 'N', 'T', 'O'),
1093 const_SIZE = mk_const4('S', 'I', 'Z', 'E'),
1094 const_STAT = mk_const4('S', 'T', 'A', 'T'),
1095 const_STOR = mk_const4('S', 'T', 'O', 'R'),
1096 const_STOU = mk_const4('S', 'T', 'O', 'U'),
1097 const_STRU = mk_const4('S', 'T', 'R', 'U'),
1098 const_SYST = mk_const4('S', 'Y', 'S', 'T'),
1099 const_TYPE = mk_const4('T', 'Y', 'P', 'E'),
1100 const_USER = mk_const4('U', 'S', 'E', 'R'),
1101
Denis Vlasenko5b492ee2009-03-18 14:12:22 +00001102#if !BB_MMU
Denis Vlasenkof2160b62009-03-16 14:53:54 +00001103 OPT_l = (1 << 0),
1104 OPT_1 = (1 << 1),
Denis Vlasenko5b492ee2009-03-18 14:12:22 +00001105#endif
1106 OPT_v = (1 << ((!BB_MMU) * 2 + 0)),
1107 OPT_S = (1 << ((!BB_MMU) * 2 + 1)),
1108 OPT_w = (1 << ((!BB_MMU) * 2 + 2)) * ENABLE_FEATURE_FTP_WRITE,
Denis Vlasenkof2160b62009-03-16 14:53:54 +00001109};
1110
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001111int ftpd_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Denis Vlasenko5b492ee2009-03-18 14:12:22 +00001112#if !BB_MMU
Denis Vlasenkoc41cba52009-03-09 15:46:07 +00001113int ftpd_main(int argc, char **argv)
Denis Vlasenko5b492ee2009-03-18 14:12:22 +00001114#else
1115int ftpd_main(int argc UNUSED_PARAM, char **argv)
1116#endif
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001117{
Denis Vlasenkoe6c94a62009-03-17 05:11:51 +00001118 unsigned abs_timeout;
Denys Vlasenko035b4d72009-06-05 20:42:40 +02001119 unsigned verbose_S;
Denis Vlasenkoc41cba52009-03-09 15:46:07 +00001120 smallint opts;
1121
Denis Vlasenko20c82162009-03-16 16:19:53 +00001122 INIT_G();
1123
Denis Vlasenkoe6c94a62009-03-17 05:11:51 +00001124 abs_timeout = 1 * 60 * 60;
Denys Vlasenko035b4d72009-06-05 20:42:40 +02001125 verbose_S = 0;
Denis Vlasenko20c82162009-03-16 16:19:53 +00001126 G.timeout = 2 * 60;
Denys Vlasenko035b4d72009-06-05 20:42:40 +02001127 opt_complementary = "t+:T+:vv:SS";
Denis Vlasenko5b492ee2009-03-18 14:12:22 +00001128#if BB_MMU
Denys Vlasenko035b4d72009-06-05 20:42:40 +02001129 opts = getopt32(argv, "vS" IF_FEATURE_FTP_WRITE("w") "t:T:", &G.timeout, &abs_timeout, &G.verbose, &verbose_S);
Denis Vlasenko5b492ee2009-03-18 14:12:22 +00001130#else
Denys Vlasenko035b4d72009-06-05 20:42:40 +02001131 opts = getopt32(argv, "l1vS" IF_FEATURE_FTP_WRITE("w") "t:T:", &G.timeout, &abs_timeout, &G.verbose, &verbose_S);
Denis Vlasenkoc41cba52009-03-09 15:46:07 +00001132 if (opts & (OPT_l|OPT_1)) {
1133 /* Our secret backdoor to ls */
Denys Vlasenko80318482010-02-24 14:27:55 +01001134/* TODO: pass -n? It prevents user/group resolution, which may not work in chroot anyway */
Denys Vlasenko33f9dc02010-01-17 22:32:22 +01001135/* TODO: pass -A? It shows dot files */
Denys Vlasenko80318482010-02-24 14:27:55 +01001136/* TODO: pass --group-directories-first? would be nice, but ls doesn't do that yet */
Denis Vlasenkoc41cba52009-03-09 15:46:07 +00001137 xchdir(argv[2]);
1138 argv[2] = (char*)"--";
Denys Vlasenko1b34d4f2009-09-30 02:39:57 +02001139 /* memset(&G, 0, sizeof(G)); - ls_main does it */
Denis Vlasenkoc41cba52009-03-09 15:46:07 +00001140 return ls_main(argc, argv);
1141 }
Denis Vlasenko5b492ee2009-03-18 14:12:22 +00001142#endif
Denys Vlasenko035b4d72009-06-05 20:42:40 +02001143 if (G.verbose < verbose_S)
1144 G.verbose = verbose_S;
Denis Vlasenkoedb0de42009-03-17 12:23:24 +00001145 if (abs_timeout | G.timeout) {
1146 if (abs_timeout == 0)
1147 abs_timeout = INT_MAX;
1148 G.end_time = monotonic_sec() + abs_timeout;
1149 if (G.timeout > abs_timeout)
1150 G.timeout = abs_timeout;
1151 }
Denis Vlasenkoe6c94a62009-03-17 05:11:51 +00001152 strcpy(G.msg_ok + 4, MSG_OK );
1153 strcpy(G.msg_err + 4, MSG_ERR);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001154
1155 G.local_addr = get_sock_lsa(STDIN_FILENO);
1156 if (!G.local_addr) {
1157 /* This is confusing:
1158 * bb_error_msg_and_die("stdin is not a socket");
1159 * Better: */
1160 bb_show_usage();
1161 /* Help text says that ftpd must be used as inetd service,
1162 * which is by far the most usual cause of get_sock_lsa
1163 * failure */
1164 }
1165
Denis Vlasenkoc41cba52009-03-09 15:46:07 +00001166 if (!(opts & OPT_v))
Denis Vlasenko4221e902009-03-11 15:07:44 +00001167 logmode = LOGMODE_NONE;
1168 if (opts & OPT_S) {
1169 /* LOG_NDELAY is needed since we may chroot later */
1170 openlog(applet_name, LOG_PID | LOG_NDELAY, LOG_DAEMON);
1171 logmode |= LOGMODE_SYSLOG;
1172 }
Denis Vlasenkoedb0de42009-03-17 12:23:24 +00001173 if (logmode)
1174 applet_name = xasprintf("%s[%u]", applet_name, (int)getpid());
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001175
Denis Vlasenko5b492ee2009-03-18 14:12:22 +00001176#if !BB_MMU
1177 G.root_fd = xopen("/", O_RDONLY | O_DIRECTORY);
Denys Vlasenko33f9dc02010-01-17 22:32:22 +01001178 close_on_exec_on(G.root_fd);
Denis Vlasenko5b492ee2009-03-18 14:12:22 +00001179#endif
Denis Vlasenkoc41cba52009-03-09 15:46:07 +00001180
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001181 if (argv[optind]) {
1182 xchdir(argv[optind]);
1183 chroot(".");
1184 }
1185
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001186 //umask(077); - admin can set umask before starting us
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001187
1188 /* Signals. We'll always take -EPIPE rather than a rude signal, thanks */
1189 signal(SIGPIPE, SIG_IGN);
1190
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001191 /* Set up options on the command socket (do we need these all? why?) */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001192 setsockopt(STDIN_FILENO, IPPROTO_TCP, TCP_NODELAY, &const_int_1, sizeof(const_int_1));
1193 setsockopt(STDIN_FILENO, SOL_SOCKET, SO_KEEPALIVE, &const_int_1, sizeof(const_int_1));
Denis Vlasenkofce4a942009-03-18 16:02:54 +00001194 /* Telnet protocol over command link may send "urgent" data,
1195 * we prefer it to be received in the "normal" data stream: */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001196 setsockopt(STDIN_FILENO, SOL_SOCKET, SO_OOBINLINE, &const_int_1, sizeof(const_int_1));
1197
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001198 WRITE_OK(FTP_GREET);
Denis Vlasenko20c82162009-03-16 16:19:53 +00001199 signal(SIGALRM, timeout_handler);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001200
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001201#ifdef IF_WE_WANT_TO_REQUIRE_LOGIN
1202 {
1203 smallint user_was_specified = 0;
1204 while (1) {
1205 uint32_t cmdval = cmdio_get_cmd_and_arg();
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001206
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001207 if (cmdval == const_USER) {
1208 if (G.ftp_arg == NULL || strcasecmp(G.ftp_arg, "anonymous") != 0)
Denis Vlasenkof1a11b52009-03-09 04:13:59 +00001209 cmdio_write_raw(STR(FTP_LOGINERR)" Server is anonymous only\r\n");
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001210 else {
1211 user_was_specified = 1;
Denis Vlasenkof1a11b52009-03-09 04:13:59 +00001212 cmdio_write_raw(STR(FTP_GIVEPWORD)" Please specify the password\r\n");
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001213 }
1214 } else if (cmdval == const_PASS) {
1215 if (user_was_specified)
1216 break;
Denis Vlasenkof1a11b52009-03-09 04:13:59 +00001217 cmdio_write_raw(STR(FTP_NEEDUSER)" Login with USER\r\n");
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001218 } else if (cmdval == const_QUIT) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001219 WRITE_OK(FTP_GOODBYE);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001220 return 0;
1221 } else {
Denis Vlasenkof1a11b52009-03-09 04:13:59 +00001222 cmdio_write_raw(STR(FTP_LOGINERR)" Login with USER and PASS\r\n");
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001223 }
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001224 }
1225 }
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001226 WRITE_OK(FTP_LOGINOK);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001227#endif
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001228
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001229 /* RFC-959 Section 5.1
1230 * The following commands and options MUST be supported by every
1231 * server-FTP and user-FTP, except in cases where the underlying
1232 * file system or operating system does not allow or support
1233 * a particular command.
1234 * Type: ASCII Non-print, IMAGE, LOCAL 8
1235 * Mode: Stream
1236 * Structure: File, Record*
1237 * (Record structure is REQUIRED only for hosts whose file
1238 * systems support record structure).
1239 * Commands:
1240 * USER, PASS, ACCT, [bbox: ACCT not supported]
1241 * PORT, PASV,
1242 * TYPE, MODE, STRU,
1243 * RETR, STOR, APPE,
1244 * RNFR, RNTO, DELE,
1245 * CWD, CDUP, RMD, MKD, PWD,
1246 * LIST, NLST,
1247 * SYST, STAT,
1248 * HELP, NOOP, QUIT.
1249 */
1250 /* ACCOUNT (ACCT)
Denis Vlasenko34552852009-03-09 04:18:00 +00001251 * "The argument field is a Telnet string identifying the user's account.
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001252 * The command is not necessarily related to the USER command, as some
1253 * sites may require an account for login and others only for specific
1254 * access, such as storing files. In the latter case the command may
1255 * arrive at any time.
1256 * There are reply codes to differentiate these cases for the automation:
1257 * when account information is required for login, the response to
1258 * a successful PASSword command is reply code 332. On the other hand,
1259 * if account information is NOT required for login, the reply to
1260 * a successful PASSword command is 230; and if the account information
1261 * is needed for a command issued later in the dialogue, the server
1262 * should return a 332 or 532 reply depending on whether it stores
1263 * (pending receipt of the ACCounT command) or discards the command,
Denis Vlasenko34552852009-03-09 04:18:00 +00001264 * respectively."
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001265 */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001266
1267 while (1) {
1268 uint32_t cmdval = cmdio_get_cmd_and_arg();
1269
1270 if (cmdval == const_QUIT) {
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001271 WRITE_OK(FTP_GOODBYE);
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001272 return 0;
1273 }
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00001274 else if (cmdval == const_USER)
Denis Vlasenko1432cb42009-03-18 00:45:00 +00001275 /* This would mean "ok, now give me PASS". */
1276 /*WRITE_OK(FTP_GIVEPWORD);*/
1277 /* vsftpd can be configured to not require that,
1278 * and this also saves one roundtrip:
1279 */
1280 WRITE_OK(FTP_LOGINOK);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00001281 else if (cmdval == const_PASS)
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001282 WRITE_OK(FTP_LOGINOK);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00001283 else if (cmdval == const_NOOP)
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001284 WRITE_OK(FTP_NOOPOK);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00001285 else if (cmdval == const_TYPE)
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001286 WRITE_OK(FTP_TYPEOK);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00001287 else if (cmdval == const_STRU)
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001288 WRITE_OK(FTP_STRUOK);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00001289 else if (cmdval == const_MODE)
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001290 WRITE_OK(FTP_MODEOK);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00001291 else if (cmdval == const_ALLO)
Denis Vlasenkofbf58462009-03-16 20:54:45 +00001292 WRITE_OK(FTP_ALLOOK);
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00001293 else if (cmdval == const_SYST)
1294 cmdio_write_raw(STR(FTP_SYSTOK)" UNIX Type: L8\r\n");
1295 else if (cmdval == const_PWD)
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001296 handle_pwd();
1297 else if (cmdval == const_CWD)
1298 handle_cwd();
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001299 else if (cmdval == const_CDUP) /* cd .. */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001300 handle_cdup();
Denis Vlasenko1a825552009-03-17 12:40:34 +00001301 /* HELP is nearly useless, but we can reuse FEAT for it */
Denis Vlasenko1432cb42009-03-18 00:45:00 +00001302 /* lftp uses FEAT */
Denis Vlasenko1a825552009-03-17 12:40:34 +00001303 else if (cmdval == const_HELP || cmdval == const_FEAT)
Denis Vlasenko1432cb42009-03-18 00:45:00 +00001304 handle_feat(cmdval == const_HELP
1305 ? STRNUM32(FTP_HELP)
1306 : STRNUM32(FTP_STATOK)
1307 );
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001308 else if (cmdval == const_LIST) /* ls -l */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001309 handle_list();
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001310 else if (cmdval == const_NLST) /* "name list", bare ls */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001311 handle_nlst();
Denis Vlasenko1432cb42009-03-18 00:45:00 +00001312 /* SIZE is crucial for wget's download indicator etc */
1313 /* Mozilla, lftp use MDTM (presumably for caching) */
1314 else if (cmdval == const_SIZE || cmdval == const_MDTM)
1315 handle_size_or_mdtm(cmdval == const_SIZE);
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001316 else if (cmdval == const_STAT) {
1317 if (G.ftp_arg == NULL)
1318 handle_stat();
1319 else
1320 handle_stat_file();
Denis Vlasenko9b2fbda2009-03-09 13:01:08 +00001321 }
1322 else if (cmdval == const_PASV)
1323 handle_pasv();
1324 else if (cmdval == const_EPSV)
1325 handle_epsv();
1326 else if (cmdval == const_RETR)
1327 handle_retr();
1328 else if (cmdval == const_PORT)
1329 handle_port();
1330 else if (cmdval == const_REST)
1331 handle_rest();
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001332#if ENABLE_FEATURE_FTP_WRITE
Denis Vlasenkoc41cba52009-03-09 15:46:07 +00001333 else if (opts & OPT_w) {
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001334 if (cmdval == const_STOR)
1335 handle_stor();
1336 else if (cmdval == const_MKD)
1337 handle_mkd();
1338 else if (cmdval == const_RMD)
1339 handle_rmd();
1340 else if (cmdval == const_DELE)
1341 handle_dele();
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001342 else if (cmdval == const_RNFR) /* "rename from" */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001343 handle_rnfr();
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001344 else if (cmdval == const_RNTO) /* "rename to" */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001345 handle_rnto();
1346 else if (cmdval == const_APPE)
1347 handle_appe();
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001348 else if (cmdval == const_STOU) /* "store unique" */
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001349 handle_stou();
Denys Vlasenko8507e1f2009-06-04 19:03:20 +02001350 else
1351 goto bad_cmd;
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001352 }
1353#endif
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001354#if 0
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001355 else if (cmdval == const_STOR
1356 || cmdval == const_MKD
1357 || cmdval == const_RMD
1358 || cmdval == const_DELE
1359 || cmdval == const_RNFR
1360 || cmdval == const_RNTO
1361 || cmdval == const_APPE
1362 || cmdval == const_STOU
1363 ) {
Denis Vlasenkof1a11b52009-03-09 04:13:59 +00001364 cmdio_write_raw(STR(FTP_NOPERM)" Permission denied\r\n");
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001365 }
1366#endif
1367 else {
Denis Vlasenko51c9bb12009-03-09 02:51:46 +00001368 /* Which unsupported commands were seen in the wild?
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001369 * (doesn't necessarily mean "we must support them")
Denis Vlasenko1432cb42009-03-18 00:45:00 +00001370 * foo 1.2.3: XXXX - comment
Denis Vlasenko5e4fda02009-03-08 23:46:48 +00001371 */
Denys Vlasenko8507e1f2009-06-04 19:03:20 +02001372#if ENABLE_FEATURE_FTP_WRITE
1373 bad_cmd:
1374#endif
Denis Vlasenkof1a11b52009-03-09 04:13:59 +00001375 cmdio_write_raw(STR(FTP_BADCMD)" Unknown command\r\n");
Denis Vlasenkobf9d1792009-03-08 09:31:28 +00001376 }
1377 }
1378}