ftp: reduce amount of realloc's done in cmdio_write

text    data     bss     dec     hex filename
 808562     476    7864  816902   c7706 busybox_old
 808568     476    7864  816908   c770c busybox_unstripped
This commit is contained in:
Denis Vlasenko 2009-03-09 04:38:37 +00:00
parent d42eb81206
commit 57a3b17498

View File

@ -102,7 +102,7 @@ struct globals {
int pasv_listen_fd; int pasv_listen_fd;
int data_fd; int data_fd;
off_t restart_pos; off_t restart_pos;
char *ftp_cmp; char *ftp_cmd;
char *ftp_arg; char *ftp_arg;
#if ENABLE_FEATURE_FTP_WRITE #if ENABLE_FEATURE_FTP_WRITE
char *rnfr_filename; char *rnfr_filename;
@ -112,8 +112,9 @@ struct globals {
#define G (*(struct globals*)&bb_common_bufsiz1) #define G (*(struct globals*)&bb_common_bufsiz1)
#define INIT_G() do { } while (0) #define INIT_G() do { } while (0)
static char * static char *
replace_text(const char *str, const char from, const char *to) escape_text(const char *prepend, const char *str, char from, const char *to, char append)
{ {
size_t retlen, remainlen, chunklen, tolen; size_t retlen, remainlen, chunklen, tolen;
const char *remain; const char *remain;
@ -124,29 +125,27 @@ replace_text(const char *str, const char from, const char *to)
tolen = strlen(to); tolen = strlen(to);
/* Simply alloc strlen(str)*strlen(to). "to" is max 2 so it's ok */ /* Simply alloc strlen(str)*strlen(to). "to" is max 2 so it's ok */
ret = xmalloc(remainlen * tolen + 1); retlen = strlen(prepend);
retlen = 0; ret = xmalloc(retlen + remainlen * tolen + 1 + 1);
strcpy(ret, prepend);
for (;;) { for (;;) {
found = strchr(remain, from); found = strchrnul(remain, from);
if (found != NULL) {
chunklen = found - remain; chunklen = found - remain;
/* Copy chunk which doesn't contain 'from' to ret */ /* Copy chunk which doesn't contain 'from' to ret */
memcpy(&ret[retlen], remain, chunklen); memcpy(&ret[retlen], remain, chunklen);
retlen += chunklen; retlen += chunklen;
if (*found != '\0') {
/* Now copy 'to' instead of 'from' */ /* Now copy 'to' instead of 'from' */
memcpy(&ret[retlen], to, tolen); memcpy(&ret[retlen], to, tolen);
retlen += tolen; retlen += tolen;
remain = found + 1; remain = found + 1;
} else { } else {
/* ret[retlen] = append;
* The last chunk. We are already sure that we have enough space ret[retlen+1] = '\0';
* so we can use strcpy.
*/
strcpy(&ret[retlen], remain);
break; break;
} }
} }
@ -163,15 +162,12 @@ replace_char(char *str, char from, char to)
static void static void
cmdio_write(unsigned status, const char *str) cmdio_write(unsigned status, const char *str)
{ {
char *escaped_str, *response; char *response;
int len; int len;
/* FTP allegedly uses telnet protocol for command link. /* FTP allegedly uses telnet protocol for command link.
* In telnet, 0xff is an escape char, and needs to be escaped: */ * In telnet, 0xff is an escape char, and needs to be escaped: */
escaped_str = replace_text(str, '\xff', "\xff\xff"); response = escape_text(utoa(status), str, '\xff', "\xff\xff", '\r');
response = xasprintf("%u%s\r", status, escaped_str);
free(escaped_str);
/* ?! does FTP send embedded LFs as NULs? wow */ /* ?! does FTP send embedded LFs as NULs? wow */
len = strlen(response); len = strlen(response);
@ -206,17 +202,15 @@ cmdio_write_raw(const char *p_text)
static void static void
handle_pwd(void) handle_pwd(void)
{ {
char *cwd, *promoted_cwd, *response; char *cwd, *response;
cwd = xrealloc_getcwd_or_warn(NULL); cwd = xrealloc_getcwd_or_warn(NULL);
if (cwd == NULL) if (cwd == NULL)
cwd = xstrdup(""); cwd = xstrdup("");
/* We have to promote each " to "" */ /* We have to promote each " to "" */
promoted_cwd = replace_text(cwd, '\"', "\"\""); response = escape_text(" \"", cwd, '\"', "\"\"", '\"');
free(cwd); free(cwd);
response = xasprintf(" \"%s\"", promoted_cwd);
free(promoted_cwd);
cmdio_write(FTP_PWDOK, response); cmdio_write(FTP_PWDOK, response);
free(response); free(response);
} }
@ -872,9 +866,9 @@ cmdio_get_cmd_and_arg(void)
uint32_t cmdval; uint32_t cmdval;
char *cmd; char *cmd;
free(G.ftp_cmp); free(G.ftp_cmd);
len = 8 * 1024; /* Paranoia. Peer may send 1 gigabyte long cmd... */ len = 8 * 1024; /* Paranoia. Peer may send 1 gigabyte long cmd... */
G.ftp_cmp = cmd = xmalloc_reads(STDIN_FILENO, NULL, &len); G.ftp_cmd = cmd = xmalloc_reads(STDIN_FILENO, NULL, &len);
if (!cmd) if (!cmd)
exit(0); exit(0);