[rules] Rewrite rule matching more efficiently

Signed-off-by: Julien Danjou <julien@danjou.info>
This commit is contained in:
Julien Danjou 2008-04-09 20:32:39 +02:00
parent 97dc830db5
commit 111a89d53f
1 changed files with 40 additions and 49 deletions

89
rules.c
View File

@ -41,54 +41,6 @@ rules_compile_regex(char *val)
return NULL; return NULL;
} }
static bool
client_match_rule(Client *c, Rule *r)
{
char *prop, buf[512];
regmatch_t tmp;
ssize_t len;
class_hint_t *ch = NULL;
bool ret = false;
if(r->prop_r)
{
/* first try to match on name */
ch = xutil_get_class_hint(globalconf.connection, c->win);
if (!ch)
return false;
len = a_strlen(ch->res_class) + a_strlen(ch->res_name) + a_strlen(c->name);
prop = p_new(char, len + 3);
/* rule matching */
snprintf(prop, len + 3, "%s:%s:%s",
ch->res_class ? ch->res_class : "", ch->res_name ? ch->res_name : "", c->name);
ret = !regexec(r->prop_r, prop, 1, &tmp, 0);
p_delete(&prop);
if(ch->res_class)
p_delete(&ch->res_class);
if(ch->res_name)
p_delete(&ch->res_name);
p_delete(&ch);
if(ret)
return true;
}
if(r->xprop
&& r->xpropval_r
&& xutil_gettextprop(globalconf.connection, c->win,
xutil_intern_atom(globalconf.connection, r->xprop),
buf, ssizeof(buf)))
ret = !regexec(r->xpropval_r, buf, 1, &tmp, 0);
return ret;
}
bool bool
tag_match_rule(Tag *t, Rule *r) tag_match_rule(Tag *t, Rule *r)
{ {
@ -104,9 +56,48 @@ Rule *
rule_matching_client(Client *c) rule_matching_client(Client *c)
{ {
Rule *r; Rule *r;
char *prop = NULL, buf[512];
regmatch_t tmp;
ssize_t len;
class_hint_t *ch = NULL;
bool ret = false;
if(!(ch = xutil_get_class_hint(globalconf.connection, c->win)))
return NULL;
len = a_strlen(ch->res_class) + a_strlen(ch->res_name) + a_strlen(c->name);
prop = p_new(char, len + 3);
snprintf(prop, len + 3, "%s:%s:%s",
ch->res_class ? ch->res_class : "", ch->res_name ? ch->res_name : "", c->name);
if(ch->res_class)
p_delete(&ch->res_class);
if(ch->res_name)
p_delete(&ch->res_name);
p_delete(&ch);
for(r = globalconf.rules; r; r = r->next) for(r = globalconf.rules; r; r = r->next)
if(client_match_rule(c, r)) {
if(r->prop_r && prop)
ret = !regexec(r->prop_r, prop, 1, &tmp, 0);
if(!ret
&& r->xprop
&& r->xpropval_r
&& xutil_gettextprop(globalconf.connection, c->win,
xutil_intern_atom(globalconf.connection, r->xprop),
buf, ssizeof(buf)))
ret = !regexec(r->xpropval_r, buf, 1, &tmp, 0);
if(ret)
{
p_delete(&prop);
return r; return r;
}
}
p_delete(&prop);
return NULL; return NULL;
} }