1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 01:48:00 +00:00

Rewrite conf deserialisation using BinarySource.

Like the corresponding rewrite of conf serialisation, this affects not
just conf_deserialise itself but also the per-platform filename and
fontspec deserialisers.
This commit is contained in:
Simon Tatham 2018-05-28 15:36:15 +01:00
parent e2431c3ef8
commit 876e1589f8
6 changed files with 68 additions and 134 deletions

84
conf.c
View File

@ -504,97 +504,53 @@ void conf_serialise(BinarySink *bs, Conf *conf)
put_uint32(bs, 0xFFFFFFFFU); put_uint32(bs, 0xFFFFFFFFU);
} }
int conf_deserialise(Conf *conf, void *vdata, int maxsize) int conf_deserialise(Conf *conf, BinarySource *src)
{ {
unsigned char *data = (unsigned char *)vdata;
unsigned char *start = data;
struct conf_entry *entry; struct conf_entry *entry;
unsigned primary; unsigned primary;
int used;
unsigned char *zero;
while (maxsize >= 4) { while (1) {
primary = GET_32BIT_MSB_FIRST(data); primary = get_uint32(src);
data += 4, maxsize -= 4;
if (get_err(src))
return FALSE;
if (primary == 0xFFFFFFFFU)
return TRUE;
if (primary >= N_CONFIG_OPTIONS) if (primary >= N_CONFIG_OPTIONS)
break; return FALSE;
entry = snew(struct conf_entry); entry = snew(struct conf_entry);
entry->key.primary = primary; entry->key.primary = primary;
switch (subkeytypes[entry->key.primary]) { switch (subkeytypes[entry->key.primary]) {
case TYPE_INT: case TYPE_INT:
if (maxsize < 4) { entry->key.secondary.i = toint(get_uint32(src));
sfree(entry);
goto done;
}
entry->key.secondary.i = toint(GET_32BIT_MSB_FIRST(data));
data += 4, maxsize -= 4;
break; break;
case TYPE_STR: case TYPE_STR:
zero = memchr(data, 0, maxsize); entry->key.secondary.s = dupstr(get_asciz(src));
if (!zero) {
sfree(entry);
goto done;
}
entry->key.secondary.s = dupstr((char *)data);
maxsize -= (zero + 1 - data);
data = zero + 1;
break; break;
} }
switch (valuetypes[entry->key.primary]) { switch (valuetypes[entry->key.primary]) {
case TYPE_INT: case TYPE_INT:
if (maxsize < 4) { entry->value.u.intval = toint(get_uint32(src));
if (subkeytypes[entry->key.primary] == TYPE_STR)
sfree(entry->key.secondary.s);
sfree(entry);
goto done;
}
entry->value.u.intval = toint(GET_32BIT_MSB_FIRST(data));
data += 4, maxsize -= 4;
break; break;
case TYPE_STR: case TYPE_STR:
zero = memchr(data, 0, maxsize); entry->value.u.stringval = dupstr(get_asciz(src));
if (!zero) {
if (subkeytypes[entry->key.primary] == TYPE_STR)
sfree(entry->key.secondary.s);
sfree(entry);
goto done;
}
entry->value.u.stringval = dupstr((char *)data);
maxsize -= (zero + 1 - data);
data = zero + 1;
break; break;
case TYPE_FILENAME: case TYPE_FILENAME:
entry->value.u.fileval = entry->value.u.fileval = filename_deserialise(src);
filename_deserialise(data, maxsize, &used);
if (!entry->value.u.fileval) {
if (subkeytypes[entry->key.primary] == TYPE_STR)
sfree(entry->key.secondary.s);
sfree(entry);
goto done;
}
data += used;
maxsize -= used;
break; break;
case TYPE_FONT: case TYPE_FONT:
entry->value.u.fontval = entry->value.u.fontval = fontspec_deserialise(src);
fontspec_deserialise(data, maxsize, &used);
if (!entry->value.u.fontval) {
if (subkeytypes[entry->key.primary] == TYPE_STR)
sfree(entry->key.secondary.s);
sfree(entry);
goto done;
}
data += used;
maxsize -= used;
break; break;
} }
if (get_err(src)) {
free_entry(entry);
return FALSE;
}
conf_insert(conf, entry); conf_insert(conf, entry);
} }
done:
return (int)(data - start);
} }

View File

@ -1015,7 +1015,7 @@ void conf_set_filename(Conf *conf, int key, const Filename *val);
void conf_set_fontspec(Conf *conf, int key, const FontSpec *val); void conf_set_fontspec(Conf *conf, int key, const FontSpec *val);
/* Serialisation functions for Duplicate Session */ /* Serialisation functions for Duplicate Session */
void conf_serialise(BinarySink *bs, Conf *conf); void conf_serialise(BinarySink *bs, Conf *conf);
int conf_deserialise(Conf *conf, void *data, int maxsize);/*returns size used*/ int conf_deserialise(Conf *conf, BinarySource *src);/*returns true on success*/
/* /*
* Functions to copy, free, serialise and deserialise FontSpecs. * Functions to copy, free, serialise and deserialise FontSpecs.
@ -1029,7 +1029,7 @@ int conf_deserialise(Conf *conf, void *data, int maxsize);/*returns size used*/
FontSpec *fontspec_copy(const FontSpec *f); FontSpec *fontspec_copy(const FontSpec *f);
void fontspec_free(FontSpec *f); void fontspec_free(FontSpec *f);
void fontspec_serialise(BinarySink *bs, FontSpec *f); void fontspec_serialise(BinarySink *bs, FontSpec *f);
FontSpec *fontspec_deserialise(void *data, int maxsize, int *used); FontSpec *fontspec_deserialise(BinarySource *src);
/* /*
* Exports from noise.c. * Exports from noise.c.
@ -1456,7 +1456,7 @@ int filename_is_null(const Filename *fn);
Filename *filename_copy(const Filename *fn); Filename *filename_copy(const Filename *fn);
void filename_free(Filename *fn); void filename_free(Filename *fn);
void filename_serialise(BinarySink *bs, const Filename *f); void filename_serialise(BinarySink *bs, const Filename *f);
Filename *filename_deserialise(void *data, int maxsize, int *used); Filename *filename_deserialise(BinarySource *src);
char *get_username(void); /* return value needs freeing */ char *get_username(void); /* return value needs freeing */
char *get_random_data(int bytes, const char *device); /* used in cmdgen.c */ char *get_random_data(int bytes, const char *device); /* used in cmdgen.c */
char filename_char_sanitise(char c); /* rewrite special pathname chars */ char filename_char_sanitise(char c); /* rewrite special pathname chars */

View File

@ -201,8 +201,9 @@ void launch_saved_session(const char *str)
int read_dupsession_data(Conf *conf, char *arg) int read_dupsession_data(Conf *conf, char *arg)
{ {
int fd, i, ret, size, size_used; int fd, i, ret, size;
char *data; char *data;
BinarySource src[1];
if (sscanf(arg, "---[%d,%d]", &fd, &size) != 2) { if (sscanf(arg, "---[%d,%d]", &fd, &size) != 2) {
fprintf(stderr, "%s: malformed magic argument `%s'\n", appname, arg); fprintf(stderr, "%s: malformed magic argument `%s'\n", appname, arg);
@ -222,35 +223,36 @@ int read_dupsession_data(Conf *conf, char *arg)
exit(1); exit(1);
} }
size_used = conf_deserialise(conf, data, size); BinarySource_BARE_INIT(src, data, size);
if (use_pty_argv && size > size_used) { if (!conf_deserialise(conf, src)) {
int n = 0; fprintf(stderr, "%s: malformed Duplicate Session data\n", appname);
i = size_used; exit(1);
while (i < size) { }
while (i < size && data[i]) i++; if (use_pty_argv) {
if (i >= size) { int pty_argc = 0;
fprintf(stderr, "%s: malformed Duplicate Session data\n", size_t argv_startpos = src->pos;
appname);
exit(1); while (get_asciz(src), !get_err(src))
} pty_argc++;
i++;
n++; src->err = BSE_NO_ERROR;
}
pty_argv = snewn(n+1, char *); if (pty_argc > 0) {
pty_argv[n] = NULL; src->pos = argv_startpos;
n = 0;
i = size_used; pty_argv = snewn(pty_argc + 1, char *);
while (i < size) { pty_argv[pty_argc] = NULL;
char *p = data + i; for (i = 0; i < pty_argc; i++)
while (i < size && data[i]) i++; pty_argv[i] = dupstr(get_asciz(src));
assert(i < size); }
i++; }
pty_argv[n++] = dupstr(p);
} if (get_err(src) || get_avail(src) > 0) {
fprintf(stderr, "%s: malformed Duplicate Session data\n", appname);
exit(1);
} }
sfree(data); sfree(data);
return 0; return 0;
} }

View File

@ -78,16 +78,9 @@ void filename_serialise(BinarySink *bs, const Filename *f)
{ {
put_asciz(bs, f->path); put_asciz(bs, f->path);
} }
Filename *filename_deserialise(void *vdata, int maxsize, int *used) Filename *filename_deserialise(BinarySource *src)
{ {
char *data = (char *)vdata; return filename_from_str(get_asciz(src));
char *end;
end = memchr(data, '\0', maxsize);
if (!end)
return NULL;
end++;
*used = end - data;
return filename_from_str(data);
} }
char filename_char_sanitise(char c) char filename_char_sanitise(char c)
@ -274,14 +267,9 @@ void fontspec_serialise(BinarySink *bs, FontSpec *f)
{ {
put_asciz(bs, f->name); put_asciz(bs, f->name);
} }
FontSpec *fontspec_deserialise(void *vdata, int maxsize, int *used) FontSpec *fontspec_deserialise(BinarySource *src)
{ {
char *data = (char *)vdata; return fontspec_new(get_asciz(src));
char *end = memchr(data, '\0', maxsize);
if (!end)
return NULL;
*used = end - data + 1;
return fontspec_new(data);
} }
char *make_dir_and_check_ours(const char *dirname) char *make_dir_and_check_ours(const char *dirname)

View File

@ -480,7 +480,10 @@ int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show)
if (sscanf(p + 1, "%p:%u", &filemap, &cpsize) == 2 && if (sscanf(p + 1, "%p:%u", &filemap, &cpsize) == 2 &&
(cp = MapViewOfFile(filemap, FILE_MAP_READ, (cp = MapViewOfFile(filemap, FILE_MAP_READ,
0, 0, cpsize)) != NULL) { 0, 0, cpsize)) != NULL) {
conf_deserialise(conf, cp, cpsize); BinarySource src[1];
BinarySource_BARE_INIT(src, cp, cpsize);
if (!conf_deserialise(conf, src))
modalfatalbox("Serialised configuration data was invalid");
UnmapViewOfFile(cp); UnmapViewOfFile(cp);
CloseHandle(filemap); CloseHandle(filemap);
} else if (!do_config()) { } else if (!do_config()) {

View File

@ -54,16 +54,9 @@ void filename_serialise(BinarySink *bs, const Filename *f)
{ {
put_asciz(bs, f->path); put_asciz(bs, f->path);
} }
Filename *filename_deserialise(void *vdata, int maxsize, int *used) Filename *filename_deserialise(BinarySource *src)
{ {
char *data = (char *)vdata; return filename_from_str(get_asciz(src));
char *end;
end = memchr(data, '\0', maxsize);
if (!end)
return NULL;
end++;
*used = end - data;
return filename_from_str(data);
} }
char filename_char_sanitise(char c) char filename_char_sanitise(char c)
@ -561,21 +554,13 @@ void fontspec_serialise(BinarySink *bs, FontSpec *f)
put_uint32(bs, f->height); put_uint32(bs, f->height);
put_uint32(bs, f->charset); put_uint32(bs, f->charset);
} }
FontSpec *fontspec_deserialise(void *vdata, int maxsize, int *used) FontSpec *fontspec_deserialise(BinarySource *src)
{ {
char *data = (char *)vdata; const char *name = get_asciz(src);
char *end; unsigned isbold = get_uint32(src);
if (maxsize < 13) unsigned height = get_uint32(src);
return NULL; unsigned charset = get_uint32(src);
end = memchr(data, '\0', maxsize-12); return fontspec_new(name, isbold, height, charset);
if (!end)
return NULL;
end++;
*used = end - data + 12;
return fontspec_new(data,
GET_32BIT_MSB_FIRST(end),
GET_32BIT_MSB_FIRST(end + 4),
GET_32BIT_MSB_FIRST(end + 8));
} }
int open_for_write_would_lose_data(const Filename *fn) int open_for_write_would_lose_data(const Filename *fn)