зеркало из https://github.com/microsoft/git.git
sq_quote_argv and add_to_string rework with strbuf's.
* sq_quote_buf is made public, and works on a strbuf. * sq_quote_argv also works on a strbuf. * make sq_quote_argv take a "maxlen" argument to check the buffer won't grow too big. Signed-off-by: Pierre Habouzit <madcoder@debian.org> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Родитель
663af3422a
Коммит
7a33bcbe80
21
connect.c
21
connect.c
|
@ -577,16 +577,13 @@ pid_t git_connect(int fd[2], char *url, const char *prog, int flags)
|
||||||
if (pid < 0)
|
if (pid < 0)
|
||||||
die("unable to fork");
|
die("unable to fork");
|
||||||
if (!pid) {
|
if (!pid) {
|
||||||
char command[MAX_CMD_LEN];
|
struct strbuf cmd;
|
||||||
char *posn = command;
|
|
||||||
int size = MAX_CMD_LEN;
|
|
||||||
int of = 0;
|
|
||||||
|
|
||||||
of |= add_to_string(&posn, &size, prog, 0);
|
strbuf_init(&cmd, MAX_CMD_LEN);
|
||||||
of |= add_to_string(&posn, &size, " ", 0);
|
strbuf_addstr(&cmd, prog);
|
||||||
of |= add_to_string(&posn, &size, path, 1);
|
strbuf_addch(&cmd, ' ');
|
||||||
|
sq_quote_buf(&cmd, path);
|
||||||
if (of)
|
if (cmd.len >= MAX_CMD_LEN)
|
||||||
die("command line too long");
|
die("command line too long");
|
||||||
|
|
||||||
dup2(pipefd[1][0], 0);
|
dup2(pipefd[1][0], 0);
|
||||||
|
@ -606,10 +603,10 @@ pid_t git_connect(int fd[2], char *url, const char *prog, int flags)
|
||||||
ssh_basename++;
|
ssh_basename++;
|
||||||
|
|
||||||
if (!port)
|
if (!port)
|
||||||
execlp(ssh, ssh_basename, host, command, NULL);
|
execlp(ssh, ssh_basename, host, cmd.buf, NULL);
|
||||||
else
|
else
|
||||||
execlp(ssh, ssh_basename, "-p", port, host,
|
execlp(ssh, ssh_basename, "-p", port, host,
|
||||||
command, NULL);
|
cmd.buf, NULL);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
unsetenv(ALTERNATE_DB_ENVIRONMENT);
|
unsetenv(ALTERNATE_DB_ENVIRONMENT);
|
||||||
|
@ -618,7 +615,7 @@ pid_t git_connect(int fd[2], char *url, const char *prog, int flags)
|
||||||
unsetenv(GIT_WORK_TREE_ENVIRONMENT);
|
unsetenv(GIT_WORK_TREE_ENVIRONMENT);
|
||||||
unsetenv(GRAFT_ENVIRONMENT);
|
unsetenv(GRAFT_ENVIRONMENT);
|
||||||
unsetenv(INDEX_ENVIRONMENT);
|
unsetenv(INDEX_ENVIRONMENT);
|
||||||
execlp("sh", "sh", "-c", command, NULL);
|
execlp("sh", "sh", "-c", cmd.buf, NULL);
|
||||||
}
|
}
|
||||||
die("exec failed");
|
die("exec failed");
|
||||||
}
|
}
|
||||||
|
|
16
git.c
16
git.c
|
@ -187,19 +187,13 @@ static int handle_alias(int *argcp, const char ***argv)
|
||||||
if (alias_string) {
|
if (alias_string) {
|
||||||
if (alias_string[0] == '!') {
|
if (alias_string[0] == '!') {
|
||||||
if (*argcp > 1) {
|
if (*argcp > 1) {
|
||||||
int i, sz = PATH_MAX;
|
struct strbuf buf;
|
||||||
char *s = xmalloc(sz), *new_alias = s;
|
|
||||||
|
|
||||||
add_to_string(&s, &sz, alias_string, 0);
|
strbuf_init(&buf, PATH_MAX);
|
||||||
|
strbuf_addstr(&buf, alias_string);
|
||||||
|
sq_quote_argv(&buf, (*argv) + 1, *argcp - 1, PATH_MAX);
|
||||||
free(alias_string);
|
free(alias_string);
|
||||||
alias_string = new_alias;
|
alias_string = buf.buf;
|
||||||
for (i = 1; i < *argcp &&
|
|
||||||
!add_to_string(&s, &sz, " ", 0) &&
|
|
||||||
!add_to_string(&s, &sz, (*argv)[i], 1)
|
|
||||||
; i++)
|
|
||||||
; /* do nothing */
|
|
||||||
if (!sz)
|
|
||||||
die("Too many or long arguments");
|
|
||||||
}
|
}
|
||||||
trace_printf("trace: alias to shell cmd: %s => %s\n",
|
trace_printf("trace: alias to shell cmd: %s => %s\n",
|
||||||
alias_command, alias_string + 1);
|
alias_command, alias_string + 1);
|
||||||
|
|
92
quote.c
92
quote.c
|
@ -12,37 +12,31 @@
|
||||||
* a'b ==> a'\''b ==> 'a'\''b'
|
* a'b ==> a'\''b ==> 'a'\''b'
|
||||||
* a!b ==> a'\!'b ==> 'a'\!'b'
|
* a!b ==> a'\!'b ==> 'a'\!'b'
|
||||||
*/
|
*/
|
||||||
#undef EMIT
|
|
||||||
#define EMIT(x) do { if (++len < n) *bp++ = (x); } while(0)
|
|
||||||
|
|
||||||
static inline int need_bs_quote(char c)
|
static inline int need_bs_quote(char c)
|
||||||
{
|
{
|
||||||
return (c == '\'' || c == '!');
|
return (c == '\'' || c == '!');
|
||||||
}
|
}
|
||||||
|
|
||||||
static size_t sq_quote_buf(char *dst, size_t n, const char *src)
|
void sq_quote_buf(struct strbuf *dst, const char *src)
|
||||||
{
|
{
|
||||||
char c;
|
char *to_free = NULL;
|
||||||
char *bp = dst;
|
|
||||||
size_t len = 0;
|
|
||||||
|
|
||||||
EMIT('\'');
|
if (dst->buf == src)
|
||||||
while ((c = *src++)) {
|
to_free = strbuf_detach(dst);
|
||||||
if (need_bs_quote(c)) {
|
|
||||||
EMIT('\'');
|
strbuf_addch(dst, '\'');
|
||||||
EMIT('\\');
|
while (*src) {
|
||||||
EMIT(c);
|
size_t len = strcspn(src, "'\\");
|
||||||
EMIT('\'');
|
strbuf_add(dst, src, len);
|
||||||
} else {
|
src += len;
|
||||||
EMIT(c);
|
while (need_bs_quote(*src)) {
|
||||||
|
strbuf_addstr(dst, "'\\");
|
||||||
|
strbuf_addch(dst, *src++);
|
||||||
|
strbuf_addch(dst, '\'');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
EMIT('\'');
|
strbuf_addch(dst, '\'');
|
||||||
|
free(to_free);
|
||||||
if ( n )
|
|
||||||
*bp = 0;
|
|
||||||
|
|
||||||
return len;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void sq_quote_print(FILE *stream, const char *src)
|
void sq_quote_print(FILE *stream, const char *src)
|
||||||
|
@ -62,11 +56,10 @@ void sq_quote_print(FILE *stream, const char *src)
|
||||||
fputc('\'', stream);
|
fputc('\'', stream);
|
||||||
}
|
}
|
||||||
|
|
||||||
char *sq_quote_argv(const char** argv, int count)
|
void sq_quote_argv(struct strbuf *dst, const char** argv, int count,
|
||||||
|
size_t maxlen)
|
||||||
{
|
{
|
||||||
char *buf, *to;
|
|
||||||
int i;
|
int i;
|
||||||
size_t len = 0;
|
|
||||||
|
|
||||||
/* Count argv if needed. */
|
/* Count argv if needed. */
|
||||||
if (count < 0) {
|
if (count < 0) {
|
||||||
|
@ -74,53 +67,14 @@ char *sq_quote_argv(const char** argv, int count)
|
||||||
; /* just counting */
|
; /* just counting */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Special case: no argv. */
|
|
||||||
if (!count)
|
|
||||||
return xcalloc(1,1);
|
|
||||||
|
|
||||||
/* Get destination buffer length. */
|
|
||||||
for (i = 0; i < count; i++)
|
|
||||||
len += sq_quote_buf(NULL, 0, argv[i]) + 1;
|
|
||||||
|
|
||||||
/* Alloc destination buffer. */
|
|
||||||
to = buf = xmalloc(len + 1);
|
|
||||||
|
|
||||||
/* Copy into destination buffer. */
|
/* Copy into destination buffer. */
|
||||||
|
strbuf_grow(dst, 32 * count);
|
||||||
for (i = 0; i < count; ++i) {
|
for (i = 0; i < count; ++i) {
|
||||||
*to++ = ' ';
|
strbuf_addch(dst, ' ');
|
||||||
to += sq_quote_buf(to, len, argv[i]);
|
sq_quote_buf(dst, argv[i]);
|
||||||
|
if (maxlen && dst->len > maxlen)
|
||||||
|
die("Too many or long arguments");
|
||||||
}
|
}
|
||||||
|
|
||||||
return buf;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Append a string to a string buffer, with or without shell quoting.
|
|
||||||
* Return true if the buffer overflowed.
|
|
||||||
*/
|
|
||||||
int add_to_string(char **ptrp, int *sizep, const char *str, int quote)
|
|
||||||
{
|
|
||||||
char *p = *ptrp;
|
|
||||||
int size = *sizep;
|
|
||||||
int oc;
|
|
||||||
int err = 0;
|
|
||||||
|
|
||||||
if (quote)
|
|
||||||
oc = sq_quote_buf(p, size, str);
|
|
||||||
else {
|
|
||||||
oc = strlen(str);
|
|
||||||
memcpy(p, str, (size <= oc) ? size - 1 : oc);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (size <= oc) {
|
|
||||||
err = 1;
|
|
||||||
oc = size - 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
*ptrp += oc;
|
|
||||||
**ptrp = '\0';
|
|
||||||
*sizep -= oc;
|
|
||||||
return err;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
char *sq_dequote(char *arg)
|
char *sq_dequote(char *arg)
|
||||||
|
|
9
quote.h
9
quote.h
|
@ -29,13 +29,10 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
extern void sq_quote_print(FILE *stream, const char *src);
|
extern void sq_quote_print(FILE *stream, const char *src);
|
||||||
extern char *sq_quote_argv(const char** argv, int count);
|
|
||||||
|
|
||||||
/*
|
extern void sq_quote_buf(struct strbuf *, const char *src);
|
||||||
* Append a string to a string buffer, with or without shell quoting.
|
extern void sq_quote_argv(struct strbuf *, const char **argv, int count,
|
||||||
* Return true if the buffer overflowed.
|
size_t maxlen);
|
||||||
*/
|
|
||||||
extern int add_to_string(char **ptrp, int *sizep, const char *str, int quote);
|
|
||||||
|
|
||||||
/* This unwraps what sq_quote() produces in place, but returns
|
/* This unwraps what sq_quote() produces in place, but returns
|
||||||
* NULL if the input does not look like what sq_quote would have
|
* NULL if the input does not look like what sq_quote would have
|
||||||
|
|
34
rsh.c
34
rsh.c
|
@ -10,12 +10,9 @@ int setup_connection(int *fd_in, int *fd_out, const char *remote_prog,
|
||||||
char *host;
|
char *host;
|
||||||
char *path;
|
char *path;
|
||||||
int sv[2];
|
int sv[2];
|
||||||
char command[COMMAND_SIZE];
|
|
||||||
char *posn;
|
|
||||||
int sizen;
|
|
||||||
int of;
|
|
||||||
int i;
|
int i;
|
||||||
pid_t pid;
|
pid_t pid;
|
||||||
|
struct strbuf cmd;
|
||||||
|
|
||||||
if (!strcmp(url, "-")) {
|
if (!strcmp(url, "-")) {
|
||||||
*fd_in = 0;
|
*fd_in = 0;
|
||||||
|
@ -36,24 +33,23 @@ int setup_connection(int *fd_in, int *fd_out, const char *remote_prog,
|
||||||
if (!path) {
|
if (!path) {
|
||||||
return error("Bad URL: %s", url);
|
return error("Bad URL: %s", url);
|
||||||
}
|
}
|
||||||
/* $GIT_RSH <host> "env GIT_DIR=<path> <remote_prog> <args...>" */
|
|
||||||
sizen = COMMAND_SIZE;
|
|
||||||
posn = command;
|
|
||||||
of = 0;
|
|
||||||
of |= add_to_string(&posn, &sizen, "env ", 0);
|
|
||||||
of |= add_to_string(&posn, &sizen, GIT_DIR_ENVIRONMENT "=", 0);
|
|
||||||
of |= add_to_string(&posn, &sizen, path, 1);
|
|
||||||
of |= add_to_string(&posn, &sizen, " ", 0);
|
|
||||||
of |= add_to_string(&posn, &sizen, remote_prog, 1);
|
|
||||||
|
|
||||||
for ( i = 0 ; i < rmt_argc ; i++ ) {
|
/* $GIT_RSH <host> "env GIT_DIR=<path> <remote_prog> <args...>" */
|
||||||
of |= add_to_string(&posn, &sizen, " ", 0);
|
strbuf_init(&cmd, COMMAND_SIZE);
|
||||||
of |= add_to_string(&posn, &sizen, rmt_argv[i], 1);
|
strbuf_addstr(&cmd, "env ");
|
||||||
|
strbuf_addstr(&cmd, GIT_DIR_ENVIRONMENT "=");
|
||||||
|
sq_quote_buf(&cmd, path);
|
||||||
|
strbuf_addch(&cmd, ' ');
|
||||||
|
sq_quote_buf(&cmd, remote_prog);
|
||||||
|
|
||||||
|
for (i = 0 ; i < rmt_argc ; i++) {
|
||||||
|
strbuf_addch(&cmd, ' ');
|
||||||
|
sq_quote_buf(&cmd, rmt_argv[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
of |= add_to_string(&posn, &sizen, " -", 0);
|
strbuf_addstr(&cmd, " -");
|
||||||
|
|
||||||
if ( of )
|
if (cmd.len >= COMMAND_SIZE)
|
||||||
return error("Command line too long");
|
return error("Command line too long");
|
||||||
|
|
||||||
if (socketpair(AF_UNIX, SOCK_STREAM, 0, sv))
|
if (socketpair(AF_UNIX, SOCK_STREAM, 0, sv))
|
||||||
|
@ -74,7 +70,7 @@ int setup_connection(int *fd_in, int *fd_out, const char *remote_prog,
|
||||||
close(sv[1]);
|
close(sv[1]);
|
||||||
dup2(sv[0], 0);
|
dup2(sv[0], 0);
|
||||||
dup2(sv[0], 1);
|
dup2(sv[0], 1);
|
||||||
execlp(ssh, ssh_basename, host, command, NULL);
|
execlp(ssh, ssh_basename, host, cmd.buf, NULL);
|
||||||
}
|
}
|
||||||
close(sv[0]);
|
close(sv[0]);
|
||||||
*fd_in = sv[1];
|
*fd_in = sv[1];
|
||||||
|
|
51
trace.c
51
trace.c
|
@ -64,7 +64,7 @@ static const char err_msg[] = "Could not trace into fd given by "
|
||||||
|
|
||||||
void trace_printf(const char *fmt, ...)
|
void trace_printf(const char *fmt, ...)
|
||||||
{
|
{
|
||||||
char buf[8192];
|
struct strbuf buf;
|
||||||
va_list ap;
|
va_list ap;
|
||||||
int fd, len, need_close = 0;
|
int fd, len, need_close = 0;
|
||||||
|
|
||||||
|
@ -72,12 +72,22 @@ void trace_printf(const char *fmt, ...)
|
||||||
if (!fd)
|
if (!fd)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
strbuf_init(&buf, 0);
|
||||||
va_start(ap, fmt);
|
va_start(ap, fmt);
|
||||||
len = vsnprintf(buf, sizeof(buf), fmt, ap);
|
len = vsnprintf(buf.buf, strbuf_avail(&buf), fmt, ap);
|
||||||
va_end(ap);
|
va_end(ap);
|
||||||
if (len >= sizeof(buf))
|
if (len >= strbuf_avail(&buf)) {
|
||||||
die("unreasonnable trace length");
|
strbuf_grow(&buf, len - strbuf_avail(&buf) + 128);
|
||||||
write_or_whine_pipe(fd, buf, len, err_msg);
|
va_start(ap, fmt);
|
||||||
|
len = vsnprintf(buf.buf, strbuf_avail(&buf), fmt, ap);
|
||||||
|
va_end(ap);
|
||||||
|
if (len >= strbuf_avail(&buf))
|
||||||
|
die("broken vsnprintf");
|
||||||
|
}
|
||||||
|
strbuf_setlen(&buf, len);
|
||||||
|
|
||||||
|
write_or_whine_pipe(fd, buf.buf, buf.len, err_msg);
|
||||||
|
strbuf_release(&buf);
|
||||||
|
|
||||||
if (need_close)
|
if (need_close)
|
||||||
close(fd);
|
close(fd);
|
||||||
|
@ -85,31 +95,32 @@ void trace_printf(const char *fmt, ...)
|
||||||
|
|
||||||
void trace_argv_printf(const char **argv, int count, const char *fmt, ...)
|
void trace_argv_printf(const char **argv, int count, const char *fmt, ...)
|
||||||
{
|
{
|
||||||
char buf[8192];
|
struct strbuf buf;
|
||||||
va_list ap;
|
va_list ap;
|
||||||
char *argv_str;
|
|
||||||
size_t argv_len;
|
|
||||||
int fd, len, need_close = 0;
|
int fd, len, need_close = 0;
|
||||||
|
|
||||||
fd = get_trace_fd(&need_close);
|
fd = get_trace_fd(&need_close);
|
||||||
if (!fd)
|
if (!fd)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
strbuf_init(&buf, 0);
|
||||||
va_start(ap, fmt);
|
va_start(ap, fmt);
|
||||||
len = vsnprintf(buf, sizeof(buf), fmt, ap);
|
len = vsnprintf(buf.buf, strbuf_avail(&buf), fmt, ap);
|
||||||
va_end(ap);
|
va_end(ap);
|
||||||
if (len >= sizeof(buf))
|
if (len >= strbuf_avail(&buf)) {
|
||||||
die("unreasonnable trace length");
|
strbuf_grow(&buf, len - strbuf_avail(&buf) + 128);
|
||||||
|
va_start(ap, fmt);
|
||||||
|
len = vsnprintf(buf.buf, strbuf_avail(&buf), fmt, ap);
|
||||||
|
va_end(ap);
|
||||||
|
if (len >= strbuf_avail(&buf))
|
||||||
|
die("broken vsnprintf");
|
||||||
|
}
|
||||||
|
strbuf_setlen(&buf, len);
|
||||||
|
|
||||||
/* Get the argv string. */
|
sq_quote_argv(&buf, argv, count, 0);
|
||||||
argv_str = sq_quote_argv(argv, count);
|
strbuf_addch(&buf, '\n');
|
||||||
argv_len = strlen(argv_str);
|
write_or_whine_pipe(fd, buf.buf, buf.len, err_msg);
|
||||||
|
strbuf_release(&buf);
|
||||||
write_or_whine_pipe(fd, buf, len, err_msg);
|
|
||||||
write_or_whine_pipe(fd, argv_str, argv_len, err_msg);
|
|
||||||
write_or_whine_pipe(fd, "\n", 1, err_msg);
|
|
||||||
|
|
||||||
free(argv_str);
|
|
||||||
|
|
||||||
if (need_close)
|
if (need_close)
|
||||||
close(fd);
|
close(fd);
|
||||||
|
|
Загрузка…
Ссылка в новой задаче