2018-10-26 21:23:51 +00:00
|
|
|
/* radare2 - MIT - Copyright 2018 - pancake */
|
|
|
|
|
|
|
|
#include <r_util.h>
|
|
|
|
#include <r_vector.h>
|
|
|
|
|
|
|
|
R_API REvent *r_event_new(void *user) {
|
|
|
|
REvent *ev = R_NEW0 (REvent);
|
|
|
|
if (!ev) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-11-14 09:47:28 +00:00
|
|
|
ev->callbacks = ht_up_new0 ();
|
|
|
|
|
2018-10-26 21:23:51 +00:00
|
|
|
// skip R_EVENT_ALL and R_EVENT_MAX, so that they don't have a mapping
|
|
|
|
// and if used in hook/unhook/send APIs will raise a warning
|
2018-11-14 09:47:28 +00:00
|
|
|
ut64 i;
|
2018-10-29 09:53:35 +00:00
|
|
|
for (i = 1; i < R_EVENT_MAX; ++i) {
|
2018-11-14 09:47:28 +00:00
|
|
|
ht_up_insert (ev->callbacks, (ut64)i, r_pvector_new (NULL));
|
2018-10-26 21:23:51 +00:00
|
|
|
}
|
|
|
|
ev->user = user;
|
|
|
|
return ev;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API void r_event_free(REvent *ev) {
|
|
|
|
if (!ev) {
|
|
|
|
return;
|
|
|
|
}
|
2018-11-14 09:47:28 +00:00
|
|
|
ut64 i;
|
2018-11-05 15:01:41 +00:00
|
|
|
for (i = 1; i < R_EVENT_MAX; ++i) {
|
2018-11-14 09:47:28 +00:00
|
|
|
RVector *entry = ht_up_find (ev->callbacks, i, NULL);
|
2018-11-05 15:01:41 +00:00
|
|
|
r_vector_free (entry);
|
|
|
|
}
|
2018-11-14 09:47:28 +00:00
|
|
|
ht_up_free (ev->callbacks);
|
2018-10-26 21:23:51 +00:00
|
|
|
free (ev);
|
|
|
|
}
|
|
|
|
|
2018-11-14 09:47:28 +00:00
|
|
|
static bool add_hook(void *cb, const ut64 k, const void *v) {
|
2018-10-26 21:23:51 +00:00
|
|
|
RPVector *cbs = (RPVector *)v;
|
|
|
|
r_return_val_if_fail (cbs, false);
|
|
|
|
r_pvector_push (cbs, cb);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API void r_event_hook(REvent *ev, REventType type, REventCallback cb) {
|
|
|
|
r_return_if_fail (ev);
|
|
|
|
if (type == R_EVENT_ALL) {
|
2018-11-14 09:47:28 +00:00
|
|
|
ht_up_foreach (ev->callbacks, add_hook, cb);
|
2018-10-26 21:23:51 +00:00
|
|
|
} else {
|
2018-11-14 09:47:28 +00:00
|
|
|
RPVector *cbs = ht_up_find (ev->callbacks, type, NULL);
|
|
|
|
add_hook (cb, 0, cbs);
|
2018-10-26 21:23:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-14 09:47:28 +00:00
|
|
|
static bool del_hook(void *cb, const ut64 k, const void *v) {
|
2018-10-26 21:23:51 +00:00
|
|
|
RPVector *cbs = (RPVector *)v;
|
|
|
|
r_return_val_if_fail (cbs, false);
|
|
|
|
r_pvector_remove_data (cbs, cb);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API void r_event_unhook(REvent *ev, REventType type, REventCallback cb) {
|
|
|
|
r_return_if_fail (ev);
|
|
|
|
if (type == R_EVENT_ALL) {
|
2018-11-14 09:47:28 +00:00
|
|
|
ht_up_foreach (ev->callbacks, del_hook, cb);
|
2018-10-26 21:23:51 +00:00
|
|
|
} else {
|
2018-11-14 09:47:28 +00:00
|
|
|
RPVector *cbs = ht_up_find (ev->callbacks, type, NULL);
|
|
|
|
del_hook (cb, 0, cbs);
|
2018-10-26 21:23:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API void r_event_send(REvent *ev, REventType type, void *data) {
|
|
|
|
r_return_if_fail (ev && !ev->incall);
|
|
|
|
|
|
|
|
void **it;
|
2018-11-14 09:47:28 +00:00
|
|
|
RPVector *cbs = ht_up_find (ev->callbacks, type, NULL);
|
2018-10-26 21:23:51 +00:00
|
|
|
r_return_if_fail (cbs);
|
|
|
|
ev->incall = true;
|
|
|
|
r_pvector_foreach (cbs, it) {
|
|
|
|
REventCallback cb = *it;
|
|
|
|
cb (ev, type, data);
|
|
|
|
}
|
|
|
|
ev->incall = false;
|
|
|
|
}
|