1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/net/protocol-multiplex.c

146 lines
4.7 KiB
C

/******************************************************************************
Copyright 2009-2010, Freie Universitaet Berlin (FUB). All rights reserved.
These sources were developed at the Freie Universitaet Berlin, Computer Systems
and Telematics group (http://cst.mi.fu-berlin.de).
-------------------------------------------------------------------------------
This file is part of FeuerWare.
This program is free software: you can redistribute it and/or modify it under
the terms of the GNU General Public License as published by the Free Software
Foundation, either version 3 of the License, or (at your option) any later
version.
FeuerWare is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
this program. If not, see http://www.gnu.org/licenses/ .
--------------------------------------------------------------------------------
For further information and questions please use the web site
http://scatterweb.mi.fu-berlin.de
and the mailinglist (subscription via web site)
scatterweb@lists.spline.inf.fu-berlin.de
*******************************************************************************/
/**
* @addtogroup net_mmstack
* @{
*/
/**
* @file
* @internal
* @brief Protocol handler multiplexing
*
* @author Freie Universität Berlin, Computer Systems & Telematics, FeuerWhere project
* @author Thomas Hillebrandt <hillebra@inf.fu-berlin.de>
* @author Heiko Will <hwill@inf.fu-berlin.de>
* @author Michael Baar <baar@inf.fu-berlin.de>
* @version $Revision: 1938 $
*
* @note $Id: protocol-multiplex.c 1938 2010-03-02 12:14:29Z baar $
*/
#include <stddef.h>
#include <string.h>
#include "protocol-multiplex.h"
//#define DEBUG
#ifdef DEBUG
#include <stdio.h>
#define PRINTF(fmt,args...) printf("pm: " fmt "\n", ##args)
#else
#define PRINTF(...)
#endif
/*---------------------------------------------------------------------------*/
void
pm_init_table(pm_table_t* table, uint8_t size, handler_entry_t* handler)
{
table->size = size;
table->handler = handler;
memset(handler, 0, sizeof(handler_entry_t) * size);
}
/*---------------------------------------------------------------------------*/
int
pm_find_handler_index(const pm_table_t* table, protocol_t protocol, unsigned int start)
{
int i;
handler_entry_t* e = &table->handler[start];
for(i = start; i < table->size; i++, e++)
{
if (e->protocol == protocol)
return i;
}
return -1;
}
/*---------------------------------------------------------------------------*/
int
pm_set_handler(const pm_table_t* table, protocol_t protocol, packet_handler_t handler)
{
// Reject illegal values
if (protocol == 0 || handler == NULL) {
PRINTF("proto %u rejected", protocol);
return -1;
}
// Check if there is already a handler for given protocol
int index = pm_find_handler_index(table, protocol, 0);
if (index >= 0)
{
// Two handlers for same protocol not allowed because only
// one gets called. This hasn't to be the last one who
// registered!
PRINTF("proto %u handler found, reset", protocol);
table->handler[index].protocol = 0;
table->handler[index].handler = NULL;
}
// Find free position for handler (protocol value is 0)
index = pm_find_handler_index(table, 0, 0);
// Store handler if free index found
if (index >= 0)
{
PRINTF("proto %u, set", protocol);
table->handler[index].protocol = protocol;
table->handler[index].handler = handler;
}
// Return result (-1 on error or no free index)
return index;
}
/*---------------------------------------------------------------------------*/
void
pm_remove_handler(const pm_table_t* table, protocol_t protocol, packet_handler_t handler)
{
int i;
for (i = 0; i < table->size; i++)
{
if (table->handler[i].protocol == protocol && table->handler[i].handler == handler)
{
PRINTF("proto %u handler found, reset", protocol);
table->handler[i].protocol = 0;
table->handler[i].handler = NULL;
}
}
}
/*---------------------------------------------------------------------------*/
int
pm_invoke(const pm_table_t* table, protocol_t protocol, void* payload, int payload_size, packet_info_t* packet_info)
{
int index = 0;
// Reject illegal values
if (protocol == 0) return -1;
if ( (index = pm_find_handler_index(table, protocol, index)) != -1 ) {
PRINTF("proto %u, invoke", protocol);
table->handler[index].handler(payload, payload_size, packet_info);
} else {
PRINTF("proto %u invoke failed (no handler)", protocol);
}
return index;
}
/*---------------------------------------------------------------------------*/
/** @} */