提交 24fe899c 编写于 作者: W Wolfgang Bumiller 提交者: Michael Roth

hmp: fix sendkey out of bounds write (CVE-2015-8619)

When processing 'sendkey' command, hmp_sendkey routine null
terminates the 'keyname_buf' array. This results in an OOB
write issue, if 'keyname_len' was to fall outside of
'keyname_buf' array.

Since the keyname's length is known the keyname_buf can be
removed altogether by adding a length parameter to
index_from_key() and using it for the error output as well.
Reported-by: NLing Liu <liuling-it@360.cn>
Signed-off-by: NWolfgang Bumiller <w.bumiller@proxmox.com>
Message-Id: <20160113080958.GA18934@olga>
[Comparison with "<" dumbed down, test for junk after strtoul()
tweaked]
Signed-off-by: NMarkus Armbruster <armbru@redhat.com>

(cherry picked from commit 64ffbe04)

Conflicts:
	hmp.c

*removed dependency on 7fb1cf16Signed-off-by: NMichael Roth <mdroth@linux.vnet.ibm.com>
上级 aaf4fb6a
...@@ -1734,21 +1734,18 @@ void hmp_sendkey(Monitor *mon, const QDict *qdict) ...@@ -1734,21 +1734,18 @@ void hmp_sendkey(Monitor *mon, const QDict *qdict)
int has_hold_time = qdict_haskey(qdict, "hold-time"); int has_hold_time = qdict_haskey(qdict, "hold-time");
int hold_time = qdict_get_try_int(qdict, "hold-time", -1); int hold_time = qdict_get_try_int(qdict, "hold-time", -1);
Error *err = NULL; Error *err = NULL;
char keyname_buf[16];
char *separator; char *separator;
int keyname_len; int keyname_len;
while (1) { while (1) {
separator = strchr(keys, '-'); separator = strchr(keys, '-');
keyname_len = separator ? separator - keys : strlen(keys); keyname_len = separator ? separator - keys : strlen(keys);
pstrcpy(keyname_buf, sizeof(keyname_buf), keys);
/* Be compatible with old interface, convert user inputted "<" */ /* Be compatible with old interface, convert user inputted "<" */
if (!strncmp(keyname_buf, "<", 1) && keyname_len == 1) { if (keys[0] == '<' && keyname_len == 1) {
pstrcpy(keyname_buf, sizeof(keyname_buf), "less"); keys = "less";
keyname_len = 4; keyname_len = 4;
} }
keyname_buf[keyname_len] = 0;
keylist = g_malloc0(sizeof(*keylist)); keylist = g_malloc0(sizeof(*keylist));
keylist->value = g_malloc0(sizeof(*keylist->value)); keylist->value = g_malloc0(sizeof(*keylist->value));
...@@ -1761,16 +1758,17 @@ void hmp_sendkey(Monitor *mon, const QDict *qdict) ...@@ -1761,16 +1758,17 @@ void hmp_sendkey(Monitor *mon, const QDict *qdict)
} }
tmp = keylist; tmp = keylist;
if (strstart(keyname_buf, "0x", NULL)) { if (strstart(keys, "0x", NULL)) {
char *endp; char *endp;
int value = strtoul(keyname_buf, &endp, 0); int value = strtoul(keys, &endp, 0);
if (*endp != '\0') { assert(endp <= keys + keyname_len);
if (endp != keys + keyname_len) {
goto err_out; goto err_out;
} }
keylist->value->type = KEY_VALUE_KIND_NUMBER; keylist->value->type = KEY_VALUE_KIND_NUMBER;
keylist->value->u.number = value; keylist->value->u.number = value;
} else { } else {
int idx = index_from_key(keyname_buf); int idx = index_from_key(keys, keyname_len);
if (idx == Q_KEY_CODE_MAX) { if (idx == Q_KEY_CODE_MAX) {
goto err_out; goto err_out;
} }
...@@ -1792,7 +1790,7 @@ out: ...@@ -1792,7 +1790,7 @@ out:
return; return;
err_out: err_out:
monitor_printf(mon, "invalid parameter: %s\n", keyname_buf); monitor_printf(mon, "invalid parameter: %.*s\n", keyname_len, keys);
goto out; goto out;
} }
......
...@@ -433,7 +433,7 @@ static inline int vnc_display_pw_expire(const char *id, time_t expires) ...@@ -433,7 +433,7 @@ static inline int vnc_display_pw_expire(const char *id, time_t expires)
void curses_display_init(DisplayState *ds, int full_screen); void curses_display_init(DisplayState *ds, int full_screen);
/* input.c */ /* input.c */
int index_from_key(const char *key); int index_from_key(const char *key, size_t key_length);
/* gtk.c */ /* gtk.c */
void early_gtk_display_init(int opengl); void early_gtk_display_init(int opengl);
......
...@@ -57,12 +57,13 @@ struct QEMUPutLEDEntry { ...@@ -57,12 +57,13 @@ struct QEMUPutLEDEntry {
static QTAILQ_HEAD(, QEMUPutLEDEntry) led_handlers = static QTAILQ_HEAD(, QEMUPutLEDEntry) led_handlers =
QTAILQ_HEAD_INITIALIZER(led_handlers); QTAILQ_HEAD_INITIALIZER(led_handlers);
int index_from_key(const char *key) int index_from_key(const char *key, size_t key_length)
{ {
int i; int i;
for (i = 0; QKeyCode_lookup[i] != NULL; i++) { for (i = 0; QKeyCode_lookup[i] != NULL; i++) {
if (!strcmp(key, QKeyCode_lookup[i])) { if (!strncmp(key, QKeyCode_lookup[i], key_length) &&
!QKeyCode_lookup[i][key_length]) {
break; break;
} }
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册