mirror of
https://github.com/irssi/irssi.git
synced 2025-01-03 14:56:47 -05:00
Merge pull request #294 from dequis/key-states-rescan-recursion-limit
Limit recursion depth of key/combo expansion in key_states_scan()
This commit is contained in:
commit
c3e4664870
@ -31,6 +31,8 @@
|
|||||||
#include "fe-windows.h"
|
#include "fe-windows.h"
|
||||||
#include "printtext.h"
|
#include "printtext.h"
|
||||||
|
|
||||||
|
#define MAX_EXPAND_RECURSION 100
|
||||||
|
|
||||||
GSList *keyinfos;
|
GSList *keyinfos;
|
||||||
static GHashTable *keys, *default_keys;
|
static GHashTable *keys, *default_keys;
|
||||||
|
|
||||||
@ -171,7 +173,7 @@ KEYINFO_REC *key_info_find(const char *id)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int expand_key(const char *key, GSList **out);
|
static int expand_key(const char *key, GSList **out, int *limit);
|
||||||
|
|
||||||
#define expand_out_char(out, c) \
|
#define expand_out_char(out, c) \
|
||||||
{ \
|
{ \
|
||||||
@ -188,13 +190,17 @@ static int expand_key(const char *key, GSList **out);
|
|||||||
g_slist_free(out); out = NULL; \
|
g_slist_free(out); out = NULL; \
|
||||||
}
|
}
|
||||||
|
|
||||||
static int expand_combo(const char *start, const char *end, GSList **out)
|
static int expand_combo(const char *start, const char *end, GSList **out, int *limit)
|
||||||
{
|
{
|
||||||
KEY_REC *rec;
|
KEY_REC *rec;
|
||||||
KEYINFO_REC *info;
|
KEYINFO_REC *info;
|
||||||
GSList *tmp, *tmp2, *list, *copy, *newout;
|
GSList *tmp, *tmp2, *list, *copy, *newout;
|
||||||
char *str, *p;
|
char *str, *p;
|
||||||
|
|
||||||
|
if ((*limit)-- < 0) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
if (start == end) {
|
if (start == end) {
|
||||||
/* single key */
|
/* single key */
|
||||||
expand_out_char(*out, *start);
|
expand_out_char(*out, *start);
|
||||||
@ -229,7 +235,7 @@ static int expand_combo(const char *start, const char *end, GSList **out)
|
|||||||
/* only one way to generate the combo, good */
|
/* only one way to generate the combo, good */
|
||||||
rec = list->data;
|
rec = list->data;
|
||||||
g_slist_free(list);
|
g_slist_free(list);
|
||||||
return expand_key(rec->key, out);
|
return expand_key(rec->key, out, limit);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* multiple ways to generate the combo -
|
/* multiple ways to generate the combo -
|
||||||
@ -244,7 +250,11 @@ static int expand_combo(const char *start, const char *end, GSList **out)
|
|||||||
copy = g_slist_append(copy, g_string_new(str->str));
|
copy = g_slist_append(copy, g_string_new(str->str));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!expand_key(rec->key, ©)) {
|
if (!expand_key(rec->key, ©, limit)) {
|
||||||
|
if (*limit < 0) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
/* illegal key combo, remove from list */
|
/* illegal key combo, remove from list */
|
||||||
expand_out_free(copy);
|
expand_out_free(copy);
|
||||||
} else {
|
} else {
|
||||||
@ -254,7 +264,11 @@ static int expand_combo(const char *start, const char *end, GSList **out)
|
|||||||
|
|
||||||
rec = list->data;
|
rec = list->data;
|
||||||
g_slist_free(list);
|
g_slist_free(list);
|
||||||
if (!expand_key(rec->key, out)) {
|
if (!expand_key(rec->key, out, limit)) {
|
||||||
|
if (*limit < 0) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
/* illegal key combo, remove from list */
|
/* illegal key combo, remove from list */
|
||||||
expand_out_free(*out);
|
expand_out_free(*out);
|
||||||
}
|
}
|
||||||
@ -264,12 +278,16 @@ static int expand_combo(const char *start, const char *end, GSList **out)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Expand key code - returns TRUE if successful. */
|
/* Expand key code - returns TRUE if successful. */
|
||||||
static int expand_key(const char *key, GSList **out)
|
static int expand_key(const char *key, GSList **out, int *limit)
|
||||||
{
|
{
|
||||||
GSList *tmp;
|
GSList *tmp;
|
||||||
const char *start;
|
const char *start;
|
||||||
int last_hyphen;
|
int last_hyphen;
|
||||||
|
|
||||||
|
if ((*limit)-- < 0) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
/* meta-^W^Gf -> ^[-^W-^G-f */
|
/* meta-^W^Gf -> ^[-^W-^G-f */
|
||||||
start = NULL; last_hyphen = TRUE;
|
start = NULL; last_hyphen = TRUE;
|
||||||
for (; *key != '\0'; key++) {
|
for (; *key != '\0'; key++) {
|
||||||
@ -279,7 +297,7 @@ static int expand_key(const char *key, GSList **out)
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!expand_combo(start, key-1, out))
|
if (!expand_combo(start, key-1, out, limit))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
expand_out_char(*out, '-');
|
expand_out_char(*out, '-');
|
||||||
start = NULL;
|
start = NULL;
|
||||||
@ -332,7 +350,7 @@ static int expand_key(const char *key, GSList **out)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (start != NULL)
|
if (start != NULL)
|
||||||
return expand_combo(start, key-1, out);
|
return expand_combo(start, key-1, out, limit);
|
||||||
|
|
||||||
for (tmp = *out; tmp != NULL; tmp = tmp->next) {
|
for (tmp = *out; tmp != NULL; tmp = tmp->next) {
|
||||||
GString *str = tmp->data;
|
GString *str = tmp->data;
|
||||||
@ -346,12 +364,13 @@ static int expand_key(const char *key, GSList **out)
|
|||||||
static void key_states_scan_key(const char *key, KEY_REC *rec)
|
static void key_states_scan_key(const char *key, KEY_REC *rec)
|
||||||
{
|
{
|
||||||
GSList *tmp, *out;
|
GSList *tmp, *out;
|
||||||
|
int limit = MAX_EXPAND_RECURSION;
|
||||||
|
|
||||||
if (g_strcmp0(rec->info->id, "key") == 0)
|
if (g_strcmp0(rec->info->id, "key") == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
out = g_slist_append(NULL, g_string_new(NULL));
|
out = g_slist_append(NULL, g_string_new(NULL));
|
||||||
if (expand_key(key, &out)) {
|
if (expand_key(key, &out, &limit)) {
|
||||||
for (tmp = out; tmp != NULL; tmp = tmp->next) {
|
for (tmp = out; tmp != NULL; tmp = tmp->next) {
|
||||||
GString *str = tmp->data;
|
GString *str = tmp->data;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user