49a3a0e0d0
Introduce the uplink tag: ! <config> ! <uplink label="wifi" domain="uplink"> ! <uplink label="wired" domain="wired_bridge"> ! <uplink domain="wired_bridge"> ! <config/> For each uplink tag, the NIC router requests a NIC session with the corresponding label or an empty label if there is no label attribute. These NIC sessions get attached to the domain that is set in their uplink tag as soon as the domain appears. This means their lifetime is not bound to the domain. Uplink NIC sessions can be safely moved from one domain to another without being closed by reconfiguring the corresponding domain attribute. Attention: This may render previously valid NIC router configurations useless. A domain named "uplink" doesn't automatically request a NIC session anymore. To fix these configurations, just add ! <uplink domain="uplink"/> or ! <uplink label="[LABEL]" domain="uplink"/> as direct subtag of the <config> tag. Issue #2840
169 lines
5.6 KiB
C++
169 lines
5.6 KiB
C++
/*
|
|
* \brief Downlink interface in form of a NIC session component
|
|
* \author Martin Stein
|
|
* \date 2016-08-23
|
|
*/
|
|
|
|
/*
|
|
* Copyright (C) 2016-2017 Genode Labs GmbH
|
|
*
|
|
* This file is part of the Genode OS framework, which is distributed
|
|
* under the terms of the GNU Affero General Public License version 3.
|
|
*/
|
|
|
|
/* Genode includes */
|
|
#include <os/session_policy.h>
|
|
|
|
/* local includes */
|
|
#include <component.h>
|
|
#include <configuration.h>
|
|
|
|
using namespace Net;
|
|
using namespace Genode;
|
|
|
|
|
|
/**************************
|
|
** Communication_buffer **
|
|
**************************/
|
|
|
|
Communication_buffer::Communication_buffer(Ram_session &ram,
|
|
Genode::size_t const size)
|
|
:
|
|
Ram_dataspace_capability(ram.alloc(size)), _ram(ram)
|
|
{ }
|
|
|
|
|
|
/****************************
|
|
** Session_component_base **
|
|
****************************/
|
|
|
|
Session_component_base::
|
|
Session_component_base(Allocator &guarded_alloc_backing,
|
|
size_t const guarded_alloc_amount,
|
|
Ram_session &buf_ram,
|
|
size_t const tx_buf_size,
|
|
size_t const rx_buf_size)
|
|
:
|
|
_guarded_alloc(&guarded_alloc_backing, guarded_alloc_amount),
|
|
_range_alloc(&_guarded_alloc), _tx_buf(buf_ram, tx_buf_size),
|
|
_rx_buf(buf_ram, rx_buf_size)
|
|
{ }
|
|
|
|
|
|
/*****************************************
|
|
** Session_component::Interface_policy **
|
|
*****************************************/
|
|
|
|
Net::Session_component::
|
|
Interface_policy::Interface_policy(Genode::Session_label const &label,
|
|
Configuration const &config)
|
|
: _label(label), _config(config) { }
|
|
|
|
|
|
Domain_name
|
|
Net::Session_component::Interface_policy::determine_domain_name() const
|
|
{
|
|
Domain_name domain_name;
|
|
try {
|
|
Session_policy policy(_label, _config().node());
|
|
domain_name = policy.attribute_value("domain", Domain_name());
|
|
}
|
|
catch (Session_policy::No_policy_defined) { if (_config().verbose()) { log("No matching policy"); } }
|
|
catch (Xml_node::Nonexistent_attribute) { if (_config().verbose()) { log("No domain attribute in policy"); } }
|
|
return domain_name;
|
|
}
|
|
|
|
|
|
/***********************
|
|
** Session_component **
|
|
***********************/
|
|
|
|
Net::Session_component::Session_component(Allocator &alloc,
|
|
Timer::Connection &timer,
|
|
size_t const amount,
|
|
Ram_session &buf_ram,
|
|
size_t const tx_buf_size,
|
|
size_t const rx_buf_size,
|
|
Region_map ®ion_map,
|
|
Mac_address const mac,
|
|
Entrypoint &ep,
|
|
Mac_address const &router_mac,
|
|
Session_label const &label,
|
|
Interface_list &interfaces,
|
|
Configuration &config)
|
|
:
|
|
Session_component_base { alloc, amount, buf_ram, tx_buf_size,
|
|
rx_buf_size },
|
|
Session_rpc_object { region_map, _tx_buf, _rx_buf, &_range_alloc,
|
|
ep.rpc_ep() },
|
|
_interface_policy { label, config },
|
|
_interface { ep, timer, router_mac, _guarded_alloc, mac,
|
|
config, interfaces, *_tx.sink(), *_rx.source(),
|
|
_link_state, _interface_policy }
|
|
{
|
|
_interface.attach_to_domain();
|
|
|
|
_tx.sigh_ready_to_ack (_interface.sink_ack());
|
|
_tx.sigh_packet_avail (_interface.sink_submit());
|
|
_rx.sigh_ack_avail (_interface.source_ack());
|
|
_rx.sigh_ready_to_submit(_interface.source_submit());
|
|
}
|
|
|
|
|
|
/**********
|
|
** Root **
|
|
**********/
|
|
|
|
Net::Root::Root(Entrypoint &ep,
|
|
Timer::Connection &timer,
|
|
Allocator &alloc,
|
|
Configuration &config,
|
|
Ram_session &buf_ram,
|
|
Interface_list &interfaces,
|
|
Region_map ®ion_map)
|
|
:
|
|
Root_component<Session_component>(&ep.rpc_ep(), &alloc), _timer(timer),
|
|
_mac_alloc(MAC_ALLOC_BASE), _ep(ep), _router_mac(_mac_alloc.alloc()),
|
|
_config(config), _buf_ram(buf_ram), _region_map(region_map),
|
|
_interfaces(interfaces)
|
|
{ }
|
|
|
|
|
|
Session_component *Net::Root::_create_session(char const *args)
|
|
{
|
|
try {
|
|
size_t const ram_quota =
|
|
Arg_string::find_arg(args, "ram_quota").ulong_value(0);
|
|
|
|
size_t const tx_buf_size =
|
|
Arg_string::find_arg(args, "tx_buf_size").ulong_value(0);
|
|
|
|
size_t const rx_buf_size =
|
|
Arg_string::find_arg(args, "rx_buf_size").ulong_value(0);
|
|
|
|
size_t const session_size =
|
|
max((size_t)4096, sizeof(Session_component));
|
|
|
|
if (ram_quota < session_size) {
|
|
throw Insufficient_ram_quota(); }
|
|
|
|
if (tx_buf_size > ram_quota - session_size ||
|
|
rx_buf_size > ram_quota - session_size ||
|
|
tx_buf_size + rx_buf_size > ram_quota - session_size)
|
|
{
|
|
error("insufficient 'ram_quota' for session creation");
|
|
throw Insufficient_ram_quota();
|
|
}
|
|
Session_label const label(label_from_args(args));
|
|
return new (md_alloc())
|
|
Session_component(*md_alloc(), _timer, ram_quota - session_size,
|
|
_buf_ram, tx_buf_size, rx_buf_size, _region_map,
|
|
_mac_alloc.alloc(), _ep, _router_mac, label,
|
|
_interfaces, _config());
|
|
}
|
|
catch (Mac_allocator::Alloc_failed) {
|
|
error("failed to allocate MAC address");
|
|
}
|
|
throw Service_denied();
|
|
}
|