Compare commits

...

7 Commits

Author SHA1 Message Date
Vftdan 4922b7f304 Add "modifiers" node type 2024-08-17 15:46:48 +02:00
Vftdan 9ac5a0b957 Check failed modifier resize 2024-08-17 15:45:10 +02:00
Vftdan 5048728022 Modifier operation enumeration 2024-08-17 15:17:49 +02:00
Vftdan f67a8842cc Modifier event predicate 2024-08-17 15:04:20 +02:00
Vftdan 8e6795a051 Single modifier type 2024-08-17 14:47:53 +02:00
Vftdan 4f71351796 Fix ModifierSet buffer overrun 2024-08-17 14:47:22 +02:00
Vftdan 20a97dbba7 Input index event predicate 2024-08-17 14:28:38 +02:00
7 changed files with 287 additions and 4 deletions

View File

@ -12,7 +12,7 @@ CPPFLAGS += $(shell pkg-config --cflags $(DEPS))
LDLIBS += $(shell pkg-config --libs $(DEPS)) LDLIBS += $(shell pkg-config --libs $(DEPS))
INTERP ?= INTERP ?=
MAIN = main MAIN = main
OBJS = main.o events.o processing.o graph.o config.o event_code_names.o hash_table.o module_registry.o event_predicate.o nodes/getchar.o nodes/print.o nodes/evdev.o nodes/tee.o nodes/router.o OBJS = main.o events.o processing.o graph.o config.o event_code_names.o hash_table.o module_registry.o event_predicate.o nodes/getchar.o nodes/print.o nodes/evdev.o nodes/tee.o nodes/router.o nodes/modifiers.o
all: $(MAIN) all: $(MAIN)

View File

@ -266,12 +266,18 @@ parse_event_predicate_type(const char *name)
if (strcmp(name, "payload") == 0) { if (strcmp(name, "payload") == 0) {
return EVPRED_PAYLOAD; return EVPRED_PAYLOAD;
} }
if (strcmp(name, "input_index") == 0) {
return EVPRED_INPUT_INDEX;
}
if (strcmp(name, "conjunction") == 0 || strcmp(name, "and") == 0) { if (strcmp(name, "conjunction") == 0 || strcmp(name, "and") == 0) {
return EVPRED_CONJUNCTION; return EVPRED_CONJUNCTION;
} }
if (strcmp(name, "disjunction") == 0 || strcmp(name, "or") == 0) { if (strcmp(name, "disjunction") == 0 || strcmp(name, "or") == 0) {
return EVPRED_DISJUNCTION; return EVPRED_DISJUNCTION;
} }
if (strcmp(name, "modifier") == 0) {
return EVPRED_MODIFIER;
}
return EVPRED_INVALID; return EVPRED_INVALID;
} }
@ -295,7 +301,7 @@ load_single_predicate(const config_setting_t * setting, EventPredicateHandleRegi
return -1; return -1;
case EVPRED_ACCEPT: case EVPRED_ACCEPT:
break; break;
case EVPRED_CODE_NS...EVPRED_PAYLOAD: case EVPRED_CODE_NS...EVPRED_INPUT_INDEX:
{ {
int64_t min_value = INT64_MIN; int64_t min_value = INT64_MIN;
int64_t max_value = INT64_MAX; int64_t max_value = INT64_MAX;
@ -321,6 +327,15 @@ load_single_predicate(const config_setting_t * setting, EventPredicateHandleRegi
predicate.aggregate_data.handles = handles; predicate.aggregate_data.handles = handles;
} }
break; break;
case EVPRED_MODIFIER:
{
long long modifier = resolve_constant_or(constants, config_setting_get_member(setting, "modifier"), -1);
if (modifier < 0 || modifier > MODIFIER_MAX) {
return -1;
}
predicate.single_modifier = modifier;
}
break;
default: default:
return -1; return -1;
} }

View File

@ -1,3 +1,7 @@
constants = {
my_modifier = 9;
};
enums = { enums = {
namespaces: ["stdin", "clickpad"]; namespaces: ["stdin", "clickpad"];
}; };
@ -42,7 +46,21 @@ nodes = {
select_key_events = { select_key_events = {
type = "router"; type = "router";
options = { options = {
predicates = ["key_event"]; predicates = (
"key_event",
{
type = "or";
inverted = 1;
args = ["key_event"];
}
);
};
};
set_modifier = {
type = "modifiers";
options = {
operation = "set";
modifiers = ["my_modifier"];
}; };
}; };
}; };
@ -55,7 +73,13 @@ channels = ({
to: ("select_key_events", 1); to: ("select_key_events", 1);
}, { }, {
from: ("select_key_events", 0); from: ("select_key_events", 0);
to: ("set_modifier", 0);
}, {
from: ("set_modifier", 0);
to: ("print", 0); to: ("print", 0);
}, {
from: ("select_key_events", 1);
to: ("print", 1);
}); });
// vim: ft=libconfig // vim: ft=libconfig

View File

@ -109,7 +109,7 @@ event_predicate_apply(EventPredicateHandle handle, EventNode * event)
case EVPRED_ACCEPT: case EVPRED_ACCEPT:
accepted = true; accepted = true;
break; break;
case EVPRED_CODE_NS...EVPRED_PAYLOAD: case EVPRED_CODE_NS...EVPRED_INPUT_INDEX:
if (!event) { if (!event) {
return EVPREDRES_DISABLED; return EVPREDRES_DISABLED;
} }
@ -128,6 +128,9 @@ event_predicate_apply(EventPredicateHandle handle, EventNode * event)
case EVPRED_PAYLOAD: case EVPRED_PAYLOAD:
actual = event->data.payload; actual = event->data.payload;
break; break;
case EVPRED_INPUT_INDEX:
actual = event->input_index;
break;
default: default:
return EVPREDRES_DISABLED; return EVPREDRES_DISABLED;
} }
@ -165,6 +168,14 @@ event_predicate_apply(EventPredicateHandle handle, EventNode * event)
} }
} }
break; break;
case EVPRED_MODIFIER:
if (!event) {
return EVPREDRES_DISABLED;
}
{
accepted = modifier_set_has(event->data.modifiers, ptr->single_modifier);
}
break;
default: default:
return EVPREDRES_DISABLED; return EVPREDRES_DISABLED;
} }

View File

@ -14,9 +14,12 @@ typedef enum {
EVPRED_CODE_MAJOR, EVPRED_CODE_MAJOR,
EVPRED_CODE_MINOR, EVPRED_CODE_MINOR,
EVPRED_PAYLOAD, EVPRED_PAYLOAD,
EVPRED_INPUT_INDEX,
// Aggregation // Aggregation
EVPRED_CONJUNCTION, EVPRED_CONJUNCTION,
EVPRED_DISJUNCTION, EVPRED_DISJUNCTION,
// Modifier
EVPRED_MODIFIER,
} EventPredicateType; } EventPredicateType;
typedef enum { typedef enum {
@ -38,6 +41,7 @@ struct event_predicate {
size_t length; size_t length;
EventPredicateHandle *handles; EventPredicateHandle *handles;
} aggregate_data; } aggregate_data;
Modifier single_modifier;
}; };
}; };

View File

@ -9,7 +9,16 @@ typedef struct {
uint8_t *bits; uint8_t *bits;
} ModifierSet; } ModifierSet;
typedef int32_t Modifier;
typedef enum {
MODOP_SET,
MODOP_UNSET,
MODOP_TOGGLE,
} ModifierOperation;
#define EMPTY_MODIFIER_SET ((ModifierSet) {.byte_length = 0, .bits = NULL}) #define EMPTY_MODIFIER_SET ((ModifierSet) {.byte_length = 0, .bits = NULL})
#define MODIFIER_MAX ((Modifier) 0xFFFFF)
__attribute__((unused)) inline static ModifierSet __attribute__((unused)) inline static ModifierSet
modifier_set_copy(const ModifierSet old) modifier_set_copy(const ModifierSet old)
@ -54,6 +63,9 @@ modifier_set_set_from(ModifierSet * target, const ModifierSet source)
{ {
modifier_set_extend(target, source.byte_length); modifier_set_extend(target, source.byte_length);
for (size_t i = 0; i < target->byte_length; ++i) { for (size_t i = 0; i < target->byte_length; ++i) {
if (i >= source.byte_length) {
return;
}
target->bits[i] |= source.bits[i]; target->bits[i] |= source.bits[i];
} }
} }
@ -62,6 +74,9 @@ __attribute__((unused)) inline static void
modifier_set_unset_from(ModifierSet * target, const ModifierSet source) modifier_set_unset_from(ModifierSet * target, const ModifierSet source)
{ {
for (size_t i = 0; i < target->byte_length; ++i) { for (size_t i = 0; i < target->byte_length; ++i) {
if (i >= source.byte_length) {
return;
}
target->bits[i] &= ~source.bits[i]; target->bits[i] &= ~source.bits[i];
} }
} }
@ -71,8 +86,123 @@ modifier_set_toggle_from(ModifierSet * target, const ModifierSet source)
{ {
modifier_set_extend(target, source.byte_length); modifier_set_extend(target, source.byte_length);
for (size_t i = 0; i < target->byte_length; ++i) { for (size_t i = 0; i < target->byte_length; ++i) {
if (i >= source.byte_length) {
return;
}
target->bits[i] ^= source.bits[i]; target->bits[i] ^= source.bits[i];
} }
} }
__attribute__((unused)) inline static void
modifier_index_and_mask(Modifier modifier, size_t * byte_index, uint8_t * mask)
{
if (byte_index) {
*byte_index = modifier >> 3;
}
if (mask) {
*mask = 1 << (modifier & 7);
}
}
__attribute__((unused)) inline static bool
modifier_set_has(ModifierSet collection, Modifier element)
{
size_t byte_index;
uint8_t mask;
modifier_index_and_mask(element, &byte_index, &mask);
if (byte_index >= collection.byte_length) {
return false;
}
return (collection.bits[byte_index] & mask) != 0;
}
__attribute__((unused)) inline static void
modifier_set_set(ModifierSet * target, Modifier element)
{
size_t byte_index;
uint8_t mask;
modifier_index_and_mask(element, &byte_index, &mask);
if (!modifier_set_extend(target, byte_index + 1)) {
return;
}
target->bits[byte_index] |= mask;
}
__attribute__((unused)) inline static void
modifier_set_unset(ModifierSet * target, Modifier element)
{
size_t byte_index;
uint8_t mask;
modifier_index_and_mask(element, &byte_index, &mask);
if (byte_index >= target->byte_length) {
return;
}
target->bits[byte_index] &= ~mask;
}
__attribute__((unused)) inline static void
modifier_set_toggle(ModifierSet * target, Modifier element)
{
size_t byte_index;
uint8_t mask;
modifier_index_and_mask(element, &byte_index, &mask);
if (!modifier_set_extend(target, byte_index + 1)) {
return;
}
target->bits[byte_index] ^= mask;
}
__attribute__((unused)) inline static void
modifier_set_operation_from(ModifierSet * target, const ModifierSet source, ModifierOperation op)
{
switch (op) {
case MODOP_SET:
modifier_set_set_from(target, source);
return;
case MODOP_UNSET:
modifier_set_unset_from(target, source);
return;
case MODOP_TOGGLE:
modifier_set_toggle_from(target, source);
return;
}
}
__attribute__((unused)) inline static void
modifier_set_operation(ModifierSet * target, Modifier element, ModifierOperation op)
{
switch (op) {
case MODOP_SET:
modifier_set_set(target, element);
return;
case MODOP_UNSET:
modifier_set_unset(target, element);
return;
case MODOP_TOGGLE:
modifier_set_toggle(target, element);
return;
}
}
__attribute__((unused)) inline static ModifierOperation
modifier_operation_parse(const char* name)
{
if (!name) {
return -1;
}
if (strcmp(name, "set") == 0) {
return MODOP_SET;
}
if (strcmp(name, "unset") == 0) {
return MODOP_UNSET;
}
if (strcmp(name, "reset") == 0) {
return MODOP_UNSET;
}
if (strcmp(name, "togggle") == 0) {
return MODOP_TOGGLE;
}
return -1;
}
#endif /* end of include guard: MODIFIERS_H_ */ #endif /* end of include guard: MODIFIERS_H_ */

99
nodes/modifiers.c Normal file
View File

@ -0,0 +1,99 @@
#include "../graph.h"
#include "../module_registry.h"
typedef struct {
GraphNode as_GraphNode;
ModifierSet modifiers;
ModifierOperation operation;
} ModifiersGraphNode;
static bool
handle_event(EventPositionBase * self, EventNode * event)
{
ModifiersGraphNode *node = DOWNCAST(ModifiersGraphNode, GraphNode, DOWNCAST(GraphNode, EventPositionBase, self));
size_t count = node->as_GraphNode.outputs.length;
if (!count) {
event_destroy(event);
return true;
}
modifier_set_operation_from(&event->data.modifiers, node->modifiers, node->operation);
if (count > 1) {
count = event_replicate(event, count - 1) + 1;
}
for (size_t i = 0; i < count; ++i) {
event->position = &node->as_GraphNode.outputs.elements[i]->as_EventPositionBase;
event = event->next;
}
return true;
}
static GraphNode *
create(GraphNodeSpecification * spec, GraphNodeConfig * config, InitializationEnvironment * env)
{
(void) config;
(void) env;
ModifiersGraphNode * node = T_ALLOC(1, ModifiersGraphNode);
if (!node) {
return NULL;
}
const char *operation_name;
config_setting_lookup_string(config->options, "operation", &operation_name);
ModifierOperation op = modifier_operation_parse(operation_name);
if (op < 0) {
free(node);
return NULL;
}
config_setting_t *modifiers_setting = config_setting_get_member(config->options, "modifiers");
if (!modifiers_setting) {
free(node);
return NULL;
}
ModifierSet modifier_set = EMPTY_MODIFIER_SET;
size_t length = config_setting_length(modifiers_setting);
for (size_t i = 0; i < length; ++i) {
long long mod = env_resolve_constant_or(env, config_setting_get_elem(modifiers_setting, i), -1);
if (mod < 0 || mod > MODIFIER_MAX) {
continue;
}
modifier_set_set(&modifier_set, mod);
}
*node = (ModifiersGraphNode) {
.as_GraphNode = {
.as_EventPositionBase = {
.handle_event = &handle_event,
.waiting_new_event = false,
},
.specification = spec,
.inputs = EMPTY_GRAPH_CHANNEL_LIST,
.outputs = EMPTY_GRAPH_CHANNEL_LIST,
},
.modifiers = modifier_set,
.operation = op,
};
return &node->as_GraphNode;
}
static void destroy
(GraphNodeSpecification * self, GraphNode * target)
{
(void) self;
ModifiersGraphNode * node = DOWNCAST(ModifiersGraphNode, GraphNode, target);
modifier_set_destruct(&node->modifiers);
free(target);
}
GraphNodeSpecification nodespec_modifiers = (GraphNodeSpecification) {
.create = &create,
.destroy = &destroy,
.register_io = NULL,
.name = "modifiers",
};
MODULE_CONSTRUCTOR(init)
{
register_graph_node_specification(&nodespec_modifiers);
}