unleashed-firmware/targets/f7/ble_glue/furi_ble/event_dispatcher.c

97 lines
2.7 KiB
C

#include "event_dispatcher.h"
#include <core/check.h>
#include <furi.h>
#include <ble/ble.h>
#include <m-list.h>
struct GapEventHandler {
void* context;
BleSvcEventHandlerCb callback;
};
LIST_DEF(GapSvcEventHandlerList, GapSvcEventHandler, M_POD_OPLIST);
static GapSvcEventHandlerList_t handlers;
static bool initialized = false;
BleEventFlowStatus ble_event_dispatcher_process_event(void* payload) {
furi_check(initialized);
GapSvcEventHandlerList_it_t it;
BleEventAckStatus ack_status = BleEventNotAck;
for(GapSvcEventHandlerList_it(it, handlers); !GapSvcEventHandlerList_end_p(it);
GapSvcEventHandlerList_next(it)) {
const GapSvcEventHandler* item = GapSvcEventHandlerList_cref(it);
ack_status = item->callback(payload, item->context);
if(ack_status == BleEventNotAck) {
/* Keep going */
continue;
} else if((ack_status == BleEventAckFlowEnable) || (ack_status == BleEventAckFlowDisable)) {
break;
}
}
/* Handlers for client-mode events are also to be implemented here. But not today. */
/* Now, decide on a flow control action based on results of all handlers */
switch(ack_status) {
case BleEventNotAck:
/* The event has NOT been managed yet. Pass to app for processing */
return ble_event_app_notification(payload);
case BleEventAckFlowEnable:
return BleEventFlowEnable;
case BleEventAckFlowDisable:
return BleEventFlowDisable;
default:
return BleEventFlowEnable;
}
}
void ble_event_dispatcher_init(void) {
if(!initialized) {
GapSvcEventHandlerList_init(handlers);
initialized = true;
}
}
void ble_event_dispatcher_reset(void) {
furi_check(initialized);
furi_check(GapSvcEventHandlerList_size(handlers) == 0);
GapSvcEventHandlerList_clear(handlers);
}
GapSvcEventHandler*
ble_event_dispatcher_register_svc_handler(BleSvcEventHandlerCb handler, void* context) {
furi_check(handler);
furi_check(context);
furi_check(initialized);
GapSvcEventHandler* item = GapSvcEventHandlerList_push_raw(handlers);
item->context = context;
item->callback = handler;
return item;
}
void ble_event_dispatcher_unregister_svc_handler(GapSvcEventHandler* handler) {
furi_check(handler);
bool found = false;
GapSvcEventHandlerList_it_t it;
for(GapSvcEventHandlerList_it(it, handlers); !GapSvcEventHandlerList_end_p(it);
GapSvcEventHandlerList_next(it)) {
const GapSvcEventHandler* item = GapSvcEventHandlerList_cref(it);
if(item == handler) {
GapSvcEventHandlerList_remove(handlers, it);
found = true;
break;
}
}
furi_check(found);
}