/*
* \brief Fast-bitmap allocator for NIC-session packet streams
* \author Sebastian Sumpf
* \date 2012-07-30
*
* This allocator can be used with a nic session. It is *not* required though.
*/
/*
* Copyright (C) 2012-2013 Genode Labs GmbH
*
* This file is part of the Genode OS framework, which is distributed
* under the terms of the GNU General Public License version 2.
*/
#ifndef _INCLUDE__NIC__PACKET_ALLOCATOR__
#define _INCLUDE__NIC__PACKET_ALLOCATOR__
#include
#include
namespace Nic {
class Packet_allocator : public Genode::Range_allocator
{
private:
Genode::Allocator *_md_alloc; /* meta-data allocator */
unsigned _block_size; /* network packet size */
Genode::addr_t _base; /* allocation base */
unsigned *_free; /* free list */
int _curr_idx; /* current index in free list */
int _count; /* number of elements */
public:
enum { DEFAULT_PACKET_SIZE = 1600 };
/**
* Constructor
*
* \param md_alloc Meta-data allocator
* \param block_size Size of network packet in stream
*/
Packet_allocator(Genode::Allocator *md_alloc,
unsigned block_size = DEFAULT_PACKET_SIZE)
: _md_alloc(md_alloc), _block_size(block_size), _base(0), _free(0),
_curr_idx(0), _count(0)
{}
~Packet_allocator()
{
if (_free)
_md_alloc->free(_free, sizeof(unsigned) * (_count / 32));
}
/*******************************
** Range-allocator interface **
*******************************/
int add_range(Genode::addr_t base, Genode::size_t size)
{
if (_count)
return -1;
_base = base;
_count = size / _block_size;
_free = (unsigned *)_md_alloc->alloc(sizeof(unsigned) * (_count / 32));
Genode::memset(_free, 0xff, sizeof(unsigned) * (_count / 32));
return 0;
}
Alloc_return alloc_aligned(Genode::size_t size, void **out_addr, int) {
return alloc(size, out_addr) ? Alloc_return::OK
: Alloc_return::RANGE_CONFLICT; }
bool alloc(Genode::size_t size, void **out_addr)
{
int free_cnt = _count / 32;
for (register int i = 0; i < free_cnt; i++) {
if (_free[_curr_idx] != 0) {
unsigned msb = Genode::log2(_free[_curr_idx]);
int elem_idx = (_curr_idx * 32) + msb;
if (elem_idx < _count) {
_free[_curr_idx] ^= (1 << msb);
*out_addr = reinterpret_cast(_base + (_block_size * elem_idx));
return true;
}
}
_curr_idx = (_curr_idx + 1) % free_cnt;
}
return false;
}
void free(void *addr)
{
Genode::addr_t a = reinterpret_cast(addr);
int elem_idx = (a - _base) / _block_size;
if (elem_idx >= _count)
return;
_curr_idx = elem_idx / 32;
_free[_curr_idx] |= (1 << (elem_idx % 32));
}
void free(void *addr, Genode::size_t) { free(addr); }
bool need_size_for_free() const override { return false; }
/*********************
** Dummy functions **
*********************/
Genode::size_t overhead(Genode::size_t) { return 0;}
int remove_range(Genode::addr_t, Genode::size_t) { return 0;}
Genode::size_t avail() { return 0; }
bool valid_addr(Genode::addr_t) { return 0; }
Alloc_return alloc_addr(Genode::size_t, Genode::addr_t) {
return Alloc_return(Alloc_return::OUT_OF_METADATA); }
};
};
#endif /* _INCLUDE__NIC__PACKET_ALLOCATOR__ */