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:
parent
e2431c3ef8
commit
876e1589f8
84
conf.c
84
conf.c
@ -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);
|
|
||||||
}
|
}
|
||||||
|
6
putty.h
6
putty.h
@ -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 */
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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)
|
||||||
|
@ -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()) {
|
||||||
|
@ -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)
|
||||||
|
Loading…
Reference in New Issue
Block a user