mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-10 01:48:00 +00:00
New functions to shrink a strbuf.
These are better than my previous approach of just assigning to
sb->len, because firstly they check by assertion that the new length
is within range, and secondly they preserve the invariant that the
byte stored in the buffer just after the length runs out is \0.
Switched to using the new functions everywhere a grep could turn up
opportunities.
(cherry picked from commit 5891142aee
)
This commit is contained in:
parent
2c66217af8
commit
34a0460f05
3
misc.h
3
misc.h
@ -73,9 +73,12 @@ strbuf *strbuf_new_nm(void);
|
|||||||
|
|
||||||
void strbuf_free(strbuf *buf);
|
void strbuf_free(strbuf *buf);
|
||||||
void *strbuf_append(strbuf *buf, size_t len);
|
void *strbuf_append(strbuf *buf, size_t len);
|
||||||
|
void strbuf_shrink_to(strbuf *buf, size_t new_len);
|
||||||
|
void strbuf_shrink_by(strbuf *buf, size_t amount_to_remove);
|
||||||
char *strbuf_to_str(strbuf *buf); /* does free buf, but you must free result */
|
char *strbuf_to_str(strbuf *buf); /* does free buf, but you must free result */
|
||||||
void strbuf_catf(strbuf *buf, const char *fmt, ...);
|
void strbuf_catf(strbuf *buf, const char *fmt, ...);
|
||||||
void strbuf_catfv(strbuf *buf, const char *fmt, va_list ap);
|
void strbuf_catfv(strbuf *buf, const char *fmt, va_list ap);
|
||||||
|
static inline void strbuf_clear(strbuf *buf) { strbuf_shrink_to(buf, 0); }
|
||||||
|
|
||||||
strbuf *strbuf_new_for_agent_query(void);
|
strbuf *strbuf_new_for_agent_query(void);
|
||||||
void strbuf_finalise_agent_query(strbuf *buf);
|
void strbuf_finalise_agent_query(strbuf *buf);
|
||||||
|
8
pscp.c
8
pscp.c
@ -1306,7 +1306,7 @@ int scp_get_sink_action(struct scp_sink_action *act)
|
|||||||
act->action = SCP_SINK_RETRY;
|
act->action = SCP_SINK_RETRY;
|
||||||
} else {
|
} else {
|
||||||
act->action = SCP_SINK_DIR;
|
act->action = SCP_SINK_DIR;
|
||||||
act->buf->len = 0;
|
strbuf_clear(act->buf);
|
||||||
put_asciz(act->buf, stripslashes(fname, false));
|
put_asciz(act->buf, stripslashes(fname, false));
|
||||||
act->name = act->buf->s;
|
act->name = act->buf->s;
|
||||||
act->size = 0; /* duhh, it's a directory */
|
act->size = 0; /* duhh, it's a directory */
|
||||||
@ -1326,7 +1326,7 @@ int scp_get_sink_action(struct scp_sink_action *act)
|
|||||||
* It's a file. Return SCP_SINK_FILE.
|
* It's a file. Return SCP_SINK_FILE.
|
||||||
*/
|
*/
|
||||||
act->action = SCP_SINK_FILE;
|
act->action = SCP_SINK_FILE;
|
||||||
act->buf->len = 0;
|
strbuf_clear(act->buf);
|
||||||
put_asciz(act->buf, stripslashes(fname, false));
|
put_asciz(act->buf, stripslashes(fname, false));
|
||||||
act->name = act->buf->s;
|
act->name = act->buf->s;
|
||||||
if (attrs.flags & SSH_FILEXFER_ATTR_SIZE) {
|
if (attrs.flags & SSH_FILEXFER_ATTR_SIZE) {
|
||||||
@ -1354,7 +1354,7 @@ int scp_get_sink_action(struct scp_sink_action *act)
|
|||||||
char ch;
|
char ch;
|
||||||
|
|
||||||
act->settime = false;
|
act->settime = false;
|
||||||
act->buf->len = 0;
|
strbuf_clear(act->buf);
|
||||||
|
|
||||||
while (!done) {
|
while (!done) {
|
||||||
if (!ssh_scp_recv(&ch, 1))
|
if (!ssh_scp_recv(&ch, 1))
|
||||||
@ -1387,7 +1387,7 @@ int scp_get_sink_action(struct scp_sink_action *act)
|
|||||||
&act->mtime, &act->atime) == 2) {
|
&act->mtime, &act->atime) == 2) {
|
||||||
act->settime = true;
|
act->settime = true;
|
||||||
backend_send(backend, "", 1);
|
backend_send(backend, "", 1);
|
||||||
act->buf->len = 0;
|
strbuf_clear(act->buf);
|
||||||
continue; /* go round again */
|
continue; /* go round again */
|
||||||
}
|
}
|
||||||
bump("Protocol error: Illegal time format");
|
bump("Protocol error: Illegal time format");
|
||||||
|
@ -1074,7 +1074,7 @@ static void scp_sink_coroutine(ScpSink *scp)
|
|||||||
* Send an ack, and read a command.
|
* Send an ack, and read a command.
|
||||||
*/
|
*/
|
||||||
sshfwd_write(scp->sc, "\0", 1);
|
sshfwd_write(scp->sc, "\0", 1);
|
||||||
scp->command->len = 0;
|
strbuf_clear(scp->command);
|
||||||
while (1) {
|
while (1) {
|
||||||
crMaybeWaitUntilV(scp->input_eof || bufchain_size(&scp->data) > 0);
|
crMaybeWaitUntilV(scp->input_eof || bufchain_size(&scp->data) > 0);
|
||||||
if (scp->input_eof)
|
if (scp->input_eof)
|
||||||
@ -1095,7 +1095,7 @@ static void scp_sink_coroutine(ScpSink *scp)
|
|||||||
/*
|
/*
|
||||||
* Parse the command.
|
* Parse the command.
|
||||||
*/
|
*/
|
||||||
scp->command->len--; /* chomp the newline */
|
strbuf_shrink_by(scp->command, 1); /* chomp the newline */
|
||||||
scp->command_chr = scp->command->len > 0 ? scp->command->s[0] : '\0';
|
scp->command_chr = scp->command->len > 0 ? scp->command->s[0] : '\0';
|
||||||
if (scp->command_chr == 'T') {
|
if (scp->command_chr == 'T') {
|
||||||
unsigned long dummy1, dummy2;
|
unsigned long dummy1, dummy2;
|
||||||
@ -1131,7 +1131,7 @@ static void scp_sink_coroutine(ScpSink *scp)
|
|||||||
|
|
||||||
ptrlen leafname = make_ptrlen(
|
ptrlen leafname = make_ptrlen(
|
||||||
p, scp->command->len - (p - scp->command->s));
|
p, scp->command->len - (p - scp->command->s));
|
||||||
scp->filename_sb->len = 0;
|
strbuf_clear(scp->filename_sb);
|
||||||
put_datapl(scp->filename_sb, scp->head->destpath);
|
put_datapl(scp->filename_sb, scp->head->destpath);
|
||||||
if (scp->head->isdir) {
|
if (scp->head->isdir) {
|
||||||
if (scp->filename_sb->len > 0 &&
|
if (scp->filename_sb->len > 0 &&
|
||||||
|
@ -218,7 +218,7 @@ static void ssh1_login_server_process_queue(PacketProtocolLayer *ppl)
|
|||||||
if (rsa_ssh1_decrypt_pkcs1(s->sesskey, larger, data)) {
|
if (rsa_ssh1_decrypt_pkcs1(s->sesskey, larger, data)) {
|
||||||
mp_free(s->sesskey);
|
mp_free(s->sesskey);
|
||||||
s->sesskey = mp_from_bytes_be(ptrlen_from_strbuf(data));
|
s->sesskey = mp_from_bytes_be(ptrlen_from_strbuf(data));
|
||||||
data->len = 0;
|
strbuf_clear(data);
|
||||||
if (rsa_ssh1_decrypt_pkcs1(s->sesskey, smaller, data) &&
|
if (rsa_ssh1_decrypt_pkcs1(s->sesskey, smaller, data) &&
|
||||||
data->len == sizeof(s->session_key)) {
|
data->len == sizeof(s->session_key)) {
|
||||||
memcpy(s->session_key, data->u, sizeof(s->session_key));
|
memcpy(s->session_key, data->u, sizeof(s->session_key));
|
||||||
|
@ -516,7 +516,7 @@ static void ssh1_login_process_queue(PacketProtocolLayer *ppl)
|
|||||||
get_rsa_ssh1_pub(s->asrc, &s->key,
|
get_rsa_ssh1_pub(s->asrc, &s->key,
|
||||||
RSA_SSH1_EXPONENT_FIRST);
|
RSA_SSH1_EXPONENT_FIRST);
|
||||||
end = s->asrc->pos;
|
end = s->asrc->pos;
|
||||||
s->agent_comment->len = 0;
|
strbuf_clear(s->agent_comment);
|
||||||
put_datapl(s->agent_comment, get_string(s->asrc));
|
put_datapl(s->agent_comment, get_string(s->asrc));
|
||||||
if (get_err(s->asrc)) {
|
if (get_err(s->asrc)) {
|
||||||
ppl_logevent("Pageant key list packet was truncated");
|
ppl_logevent("Pageant key list packet was truncated");
|
||||||
|
@ -57,7 +57,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s, bool *aborted)
|
|||||||
assert(s->hkey);
|
assert(s->hkey);
|
||||||
}
|
}
|
||||||
|
|
||||||
s->hostkeyblob->len = 0;
|
strbuf_clear(s->hostkeyblob);
|
||||||
ssh_key_public_blob(s->hkey, BinarySink_UPCAST(s->hostkeyblob));
|
ssh_key_public_blob(s->hkey, BinarySink_UPCAST(s->hostkeyblob));
|
||||||
s->hostkeydata = ptrlen_from_strbuf(s->hostkeyblob);
|
s->hostkeydata = ptrlen_from_strbuf(s->hostkeyblob);
|
||||||
|
|
||||||
|
@ -265,7 +265,7 @@ static void ssh2_mkkey(
|
|||||||
*/
|
*/
|
||||||
keylen_padded = ((keylen + hlen - 1) / hlen) * hlen;
|
keylen_padded = ((keylen + hlen - 1) / hlen) * hlen;
|
||||||
|
|
||||||
out->len = 0;
|
strbuf_clear(out);
|
||||||
key = strbuf_append(out, keylen_padded);
|
key = strbuf_append(out, keylen_padded);
|
||||||
|
|
||||||
/* First hlen bytes. */
|
/* First hlen bytes. */
|
||||||
@ -1083,7 +1083,7 @@ static void ssh2_transport_process_queue(PacketProtocolLayer *ppl)
|
|||||||
* Construct our KEXINIT packet, in a strbuf so we can refer to it
|
* Construct our KEXINIT packet, in a strbuf so we can refer to it
|
||||||
* later.
|
* later.
|
||||||
*/
|
*/
|
||||||
s->client_kexinit->len = 0;
|
strbuf_clear(s->client_kexinit);
|
||||||
put_byte(s->outgoing_kexinit, SSH2_MSG_KEXINIT);
|
put_byte(s->outgoing_kexinit, SSH2_MSG_KEXINIT);
|
||||||
random_read(strbuf_append(s->outgoing_kexinit, 16), 16);
|
random_read(strbuf_append(s->outgoing_kexinit, 16), 16);
|
||||||
ssh2_write_kexinit_lists(
|
ssh2_write_kexinit_lists(
|
||||||
@ -1121,7 +1121,7 @@ static void ssh2_transport_process_queue(PacketProtocolLayer *ppl)
|
|||||||
s->ppl.bpp->pls->actx, pktin->type));
|
s->ppl.bpp->pls->actx, pktin->type));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
s->incoming_kexinit->len = 0;
|
strbuf_clear(s->incoming_kexinit);
|
||||||
put_byte(s->incoming_kexinit, SSH2_MSG_KEXINIT);
|
put_byte(s->incoming_kexinit, SSH2_MSG_KEXINIT);
|
||||||
put_data(s->incoming_kexinit, get_ptr(pktin), get_avail(pktin));
|
put_data(s->incoming_kexinit, get_ptr(pktin), get_avail(pktin));
|
||||||
|
|
||||||
|
@ -620,7 +620,7 @@ static void ssh2_userauth_process_queue(PacketProtocolLayer *ppl)
|
|||||||
/*
|
/*
|
||||||
* Save the methods string for use in error messages.
|
* Save the methods string for use in error messages.
|
||||||
*/
|
*/
|
||||||
s->last_methods_string->len = 0;
|
strbuf_clear(s->last_methods_string);
|
||||||
put_datapl(s->last_methods_string, methods);
|
put_datapl(s->last_methods_string, methods);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -303,8 +303,7 @@ void ssh_verstring_handle_input(BinaryPacketProtocol *bpp)
|
|||||||
while (s->vstring->len > 0 &&
|
while (s->vstring->len > 0 &&
|
||||||
(s->vstring->s[s->vstring->len-1] == '\r' ||
|
(s->vstring->s[s->vstring->len-1] == '\r' ||
|
||||||
s->vstring->s[s->vstring->len-1] == '\n'))
|
s->vstring->s[s->vstring->len-1] == '\n'))
|
||||||
s->vstring->len--;
|
strbuf_shrink_by(s->vstring, 1);
|
||||||
s->vstring->s[s->vstring->len] = '\0';
|
|
||||||
|
|
||||||
bpp_logevent("Remote version: %s", s->vstring->s);
|
bpp_logevent("Remote version: %s", s->vstring->s);
|
||||||
|
|
||||||
|
4
telnet.c
4
telnet.c
@ -580,7 +580,7 @@ static void do_telnet_read(Telnet *telnet, const char *buf, size_t len)
|
|||||||
break;
|
break;
|
||||||
case SEENSB:
|
case SEENSB:
|
||||||
telnet->sb_opt = c;
|
telnet->sb_opt = c;
|
||||||
telnet->sb_buf->len = 0;
|
strbuf_clear(telnet->sb_buf);
|
||||||
telnet->state = SUBNEGOT;
|
telnet->state = SUBNEGOT;
|
||||||
break;
|
break;
|
||||||
case SUBNEGOT:
|
case SUBNEGOT:
|
||||||
@ -604,7 +604,7 @@ static void do_telnet_read(Telnet *telnet, const char *buf, size_t len)
|
|||||||
|
|
||||||
if (outbuf->len >= 4096) {
|
if (outbuf->len >= 4096) {
|
||||||
c_write(telnet, outbuf->u, outbuf->len);
|
c_write(telnet, outbuf->u, outbuf->len);
|
||||||
outbuf->len = 0;
|
strbuf_clear(outbuf);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
10
terminal.c
10
terminal.c
@ -440,11 +440,11 @@ static void makerle(strbuf *b, termline *ldata,
|
|||||||
if (hdrsize == 0) {
|
if (hdrsize == 0) {
|
||||||
assert(prevpos == hdrpos + 1);
|
assert(prevpos == hdrpos + 1);
|
||||||
runpos = hdrpos;
|
runpos = hdrpos;
|
||||||
b->len = prevpos+prevlen;
|
strbuf_shrink_to(b, prevpos+prevlen);
|
||||||
} else {
|
} else {
|
||||||
memmove(b->u + prevpos+1, b->u + prevpos, prevlen);
|
memmove(b->u + prevpos+1, b->u + prevpos, prevlen);
|
||||||
runpos = prevpos;
|
runpos = prevpos;
|
||||||
b->len = prevpos+prevlen+1;
|
strbuf_shrink_to(b, prevpos+prevlen+1);
|
||||||
/*
|
/*
|
||||||
* Terminate the previous run of ordinary
|
* Terminate the previous run of ordinary
|
||||||
* literals.
|
* literals.
|
||||||
@ -461,7 +461,7 @@ static void makerle(strbuf *b, termline *ldata,
|
|||||||
oldstate = state;
|
oldstate = state;
|
||||||
makeliteral(b, c, &state);
|
makeliteral(b, c, &state);
|
||||||
tmplen = b->len - tmppos;
|
tmplen = b->len - tmppos;
|
||||||
b->len = tmppos;
|
strbuf_shrink_to(b, tmppos);
|
||||||
if (tmplen != thislen ||
|
if (tmplen != thislen ||
|
||||||
memcmp(b->u + runpos+1, b->u + tmppos, tmplen)) {
|
memcmp(b->u + runpos+1, b->u + tmppos, tmplen)) {
|
||||||
state = oldstate;
|
state = oldstate;
|
||||||
@ -519,7 +519,7 @@ static void makerle(strbuf *b, termline *ldata,
|
|||||||
assert(hdrsize <= 128);
|
assert(hdrsize <= 128);
|
||||||
b->u[hdrpos] = hdrsize - 1;
|
b->u[hdrpos] = hdrsize - 1;
|
||||||
} else {
|
} else {
|
||||||
b->len = hdrpos;
|
strbuf_shrink_to(b, hdrpos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
static void makeliteral_chr(strbuf *b, termchar *c, unsigned long *state)
|
static void makeliteral_chr(strbuf *b, termchar *c, unsigned long *state)
|
||||||
@ -3057,7 +3057,7 @@ static strbuf *term_input_data_from_unicode(
|
|||||||
int rv;
|
int rv;
|
||||||
rv = wc_to_mb(term->ucsdata->line_codepage, 0, widebuf, len,
|
rv = wc_to_mb(term->ucsdata->line_codepage, 0, widebuf, len,
|
||||||
bufptr, len + 1, NULL, term->ucsdata);
|
bufptr, len + 1, NULL, term->ucsdata);
|
||||||
buf->len = rv < 0 ? 0 : rv;
|
strbuf_shrink_to(buf, rv < 0 ? 0 : rv);
|
||||||
}
|
}
|
||||||
|
|
||||||
return buf;
|
return buf;
|
||||||
|
@ -760,7 +760,7 @@ strbuf *rsa_ssh1_decrypt_pkcs1_wrapper(mp_int *input, RSAKey *key)
|
|||||||
/* Again, return "" on failure */
|
/* Again, return "" on failure */
|
||||||
strbuf *sb = strbuf_new();
|
strbuf *sb = strbuf_new();
|
||||||
if (!rsa_ssh1_decrypt_pkcs1(input, key, sb))
|
if (!rsa_ssh1_decrypt_pkcs1(input, key, sb))
|
||||||
sb->len = 0;
|
strbuf_clear(sb);
|
||||||
return sb;
|
return sb;
|
||||||
}
|
}
|
||||||
#define rsa_ssh1_decrypt_pkcs1 rsa_ssh1_decrypt_pkcs1_wrapper
|
#define rsa_ssh1_decrypt_pkcs1 rsa_ssh1_decrypt_pkcs1_wrapper
|
||||||
|
@ -3835,7 +3835,7 @@ static void eventlog_list_handler(union control *ctrl, dlgparam *dp,
|
|||||||
/*
|
/*
|
||||||
* Construct the data to use as the selection.
|
* Construct the data to use as the selection.
|
||||||
*/
|
*/
|
||||||
es->seldata->len = 0;
|
strbuf_clear(es->seldata);
|
||||||
for (i = 0; i < es->ninitial; i++) {
|
for (i = 0; i < es->ninitial; i++) {
|
||||||
if (dlg_listbox_issel(ctrl, dp, i))
|
if (dlg_listbox_issel(ctrl, dp, i))
|
||||||
strbuf_catf(es->seldata, "%s\n", es->events_initial[i]);
|
strbuf_catf(es->seldata, "%s\n", es->events_initial[i]);
|
||||||
|
@ -252,7 +252,7 @@ SockAddr *sk_namelookup(const char *host, char **canonicalname, int address_fami
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
/* This way we are always sure the h->h_name is valid :) */
|
/* This way we are always sure the h->h_name is valid :) */
|
||||||
realhost->len = 0;
|
strbuf_clear(realhost);
|
||||||
strbuf_catf(realhost, "%s", h->h_name);
|
strbuf_catf(realhost, "%s", h->h_name);
|
||||||
for (n = 0; h->h_addr_list[n]; n++);
|
for (n = 0; h->h_addr_list[n]; n++);
|
||||||
ret->addresses = snewn(n, unsigned long);
|
ret->addresses = snewn(n, unsigned long);
|
||||||
@ -267,7 +267,7 @@ SockAddr *sk_namelookup(const char *host, char **canonicalname, int address_fami
|
|||||||
* success return from inet_addr.
|
* success return from inet_addr.
|
||||||
*/
|
*/
|
||||||
ret->superfamily = IP;
|
ret->superfamily = IP;
|
||||||
realhost->len = 0;
|
strbuf_clear(realhost);
|
||||||
strbuf_catf(realhost, "%s", host);
|
strbuf_catf(realhost, "%s", host);
|
||||||
ret->addresses = snew(unsigned long);
|
ret->addresses = snew(unsigned long);
|
||||||
ret->naddresses = 1;
|
ret->naddresses = 1;
|
||||||
|
@ -564,7 +564,7 @@ bool enum_settings_next(settings_e *handle, strbuf *out)
|
|||||||
size_t baselen = fullpath->len;
|
size_t baselen = fullpath->len;
|
||||||
|
|
||||||
while ( (de = readdir(handle->dp)) != NULL ) {
|
while ( (de = readdir(handle->dp)) != NULL ) {
|
||||||
fullpath->len = baselen;
|
strbuf_shrink_to(fullpath, baselen);
|
||||||
put_datapl(fullpath, ptrlen_from_asciz(de->d_name));
|
put_datapl(fullpath, ptrlen_from_asciz(de->d_name));
|
||||||
|
|
||||||
if (stat(fullpath->s, &st) < 0 || !S_ISREG(st.st_mode))
|
if (stat(fullpath->s, &st) < 0 || !S_ISREG(st.st_mode))
|
||||||
|
14
utils.c
14
utils.c
@ -429,6 +429,20 @@ void *strbuf_append(strbuf *buf_o, size_t len)
|
|||||||
return toret;
|
return toret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void strbuf_shrink_to(strbuf *buf, size_t new_len)
|
||||||
|
{
|
||||||
|
assert(new_len <= buf->len);
|
||||||
|
buf->len = new_len;
|
||||||
|
buf->s[buf->len] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
void strbuf_shrink_by(strbuf *buf, size_t amount_to_remove)
|
||||||
|
{
|
||||||
|
assert(amount_to_remove <= buf->len);
|
||||||
|
buf->len -= amount_to_remove;
|
||||||
|
buf->s[buf->len] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
static void strbuf_BinarySink_write(
|
static void strbuf_BinarySink_write(
|
||||||
BinarySink *bs, const void *data, size_t len)
|
BinarySink *bs, const void *data, size_t len)
|
||||||
{
|
{
|
||||||
|
@ -677,7 +677,7 @@ static void update_sessions(void)
|
|||||||
sb = strbuf_new();
|
sb = strbuf_new();
|
||||||
while(ERROR_SUCCESS == RegEnumKey(hkey, index_key, buf, MAX_PATH)) {
|
while(ERROR_SUCCESS == RegEnumKey(hkey, index_key, buf, MAX_PATH)) {
|
||||||
if(strcmp(buf, PUTTY_DEFAULT) != 0) {
|
if(strcmp(buf, PUTTY_DEFAULT) != 0) {
|
||||||
sb->len = 0;
|
strbuf_clear(sb);
|
||||||
unescape_registry_key(buf, sb);
|
unescape_registry_key(buf, sb);
|
||||||
|
|
||||||
memset(&mii, 0, sizeof(mii));
|
memset(&mii, 0, sizeof(mii));
|
||||||
|
Loading…
Reference in New Issue
Block a user