mirror of
https://github.com/genodelabs/genode.git
synced 2025-06-16 14:18:27 +00:00
committed by
Christian Helmuth
parent
dffc1b0497
commit
e32b78d95d
@ -26,34 +26,127 @@ class Genode::Ram_session_guard : public Genode::Ram_session
|
|||||||
private:
|
private:
|
||||||
|
|
||||||
Ram_session &_session;
|
Ram_session &_session;
|
||||||
|
Ram_session_capability _session_cap;
|
||||||
|
|
||||||
size_t _quota;
|
size_t _quota;
|
||||||
size_t _used;
|
size_t _used = 0;
|
||||||
|
size_t _withdraw = 0;
|
||||||
|
|
||||||
|
/* XXX should be either a exception or a enum in rm_session */
|
||||||
|
enum { RM_SESSION_INSUFFICIENT_QUOTA = -3 };
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
Ram_session_guard(Ram_session &session, size_t quota)
|
Ram_session_guard(Ram_session &session, Ram_session_capability cap,
|
||||||
: _session(session) { }
|
size_t quota)
|
||||||
|
: _session(session), _session_cap(cap), _quota(quota) { }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenient transfer_quota method throwing a exception iif the
|
||||||
|
* quota is insufficient.
|
||||||
|
*/
|
||||||
|
template <typename T>
|
||||||
|
int transfer_quota(Ram_session_capability ram_session, size_t amount)
|
||||||
|
{
|
||||||
|
int const error = transfer_quota(ram_session, amount);
|
||||||
|
|
||||||
|
if (error == RM_SESSION_INSUFFICIENT_QUOTA)
|
||||||
|
throw T();
|
||||||
|
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extend allocation limit
|
||||||
|
*/
|
||||||
|
void upgrade(size_t additional_amount) {
|
||||||
|
_quota += additional_amount; }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Consume bytes without actually allocating them
|
||||||
|
*/
|
||||||
|
bool withdraw(size_t size)
|
||||||
|
{
|
||||||
|
if ((_quota - _used) < size)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
_used += size;
|
||||||
|
_withdraw += size;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Revert withdraw
|
||||||
|
*/
|
||||||
|
bool revert_withdraw(size_t size)
|
||||||
|
{
|
||||||
|
if (size > _withdraw)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
_used -= size;
|
||||||
|
_withdraw -= size;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Revert transfer quota
|
||||||
|
*/
|
||||||
|
int revert_transfer_quota(Ram_session &ram_session,
|
||||||
|
size_t amount)
|
||||||
|
{
|
||||||
|
if (amount > _used)
|
||||||
|
return -4;
|
||||||
|
|
||||||
|
int error = ram_session.transfer_quota(_session_cap, amount);
|
||||||
|
if (!error)
|
||||||
|
_used -= amount;
|
||||||
|
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
/***************************
|
||||||
|
** Ram_session interface **
|
||||||
|
***************************/
|
||||||
|
|
||||||
|
|
||||||
Ram_dataspace_capability alloc(size_t size,
|
Ram_dataspace_capability alloc(size_t size,
|
||||||
Cache_attribute cached = CACHED) override
|
Cache_attribute cached = CACHED) override
|
||||||
{
|
{
|
||||||
if (_used + size > _used) throw Quota_exceeded();
|
if (_used + size <= _used || _used + size > _quota)
|
||||||
|
throw Quota_exceeded();
|
||||||
|
|
||||||
|
Ram_dataspace_capability cap = _session.alloc(size, cached);
|
||||||
|
|
||||||
|
if (cap.valid())
|
||||||
_used += size;
|
_used += size;
|
||||||
return _session.alloc(size, cached);
|
|
||||||
|
return cap;
|
||||||
}
|
}
|
||||||
|
|
||||||
void free(Ram_dataspace_capability ds) override
|
void free(Ram_dataspace_capability ds) override
|
||||||
{
|
{
|
||||||
size_t size = Dataspace_client(ds).size();
|
size_t size = Dataspace_client(ds).size();
|
||||||
_used -= size;
|
|
||||||
_session.free(ds);
|
_session.free(ds);
|
||||||
|
_used -= size;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ref_account(Ram_session_capability ram_session) override {
|
int ref_account(Ram_session_capability ram_session) override {
|
||||||
return -1; }
|
return _session.ref_account(ram_session); }
|
||||||
|
|
||||||
int transfer_quota(Ram_session_capability ram_session, size_t amount) override {
|
int transfer_quota(Ram_session_capability ram_session,
|
||||||
return -1; }
|
size_t amount) override
|
||||||
|
{
|
||||||
|
if (_used + amount <= _used || _used + amount > _quota)
|
||||||
|
return RM_SESSION_INSUFFICIENT_QUOTA;
|
||||||
|
|
||||||
|
int const error = _session.transfer_quota(ram_session, amount);
|
||||||
|
|
||||||
|
if (!error)
|
||||||
|
_used += amount;
|
||||||
|
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
size_t quota() override { return _quota; }
|
size_t quota() override { return _quota; }
|
||||||
|
|
||||||
|
@ -56,7 +56,7 @@ class Net::Stream_allocator
|
|||||||
Stream_allocator(Genode::Ram_session &ram,
|
Stream_allocator(Genode::Ram_session &ram,
|
||||||
Genode::Region_map &rm,
|
Genode::Region_map &rm,
|
||||||
Genode::size_t amount)
|
Genode::size_t amount)
|
||||||
: _ram(ram, amount),
|
: _ram(ram, Genode::Ram_session_capability(), amount),
|
||||||
_heap(ram, rm),
|
_heap(ram, rm),
|
||||||
_range_alloc(&_heap) {}
|
_range_alloc(&_heap) {}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user