mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
146 lines
4.7 KiB
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;
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
/** @} */
|