mirror of
https://github.com/genodelabs/genode.git
synced 2025-01-11 15:33:04 +00:00
vbox: avoid stuck network during high tx load
- finished tx ack queue should be checked before new allocations - packets which got not sent must be released in packet stream, otherwise the network packet stream gets filled up and starves after a while - rRegister for ack avail packets and process them concurrently to EMT-* threads by nic_ep thread (thanks @Peter for the findings) + add synchronization. - add sigh_ready_to_submit to network adapter to improve latency by notifying the network model explicitly in case we had a full packet stream error case (_retry resp. VERR_TRY_LATER) Fixes #5045
This commit is contained in:
parent
5410ecf9ad
commit
c02aa759e6
@ -1 +1 @@
|
|||||||
a4d2f03ac0c0d8f7d7c3a4c3b284a68fb259cbc8
|
9d3de38fdb7c95517c4ce76dbd090a147fa7b79c
|
||||||
|
@ -69,8 +69,8 @@ typedef struct DRVNIC
|
|||||||
PPDMINETWORKCONFIG pIAboveConfig;
|
PPDMINETWORKCONFIG pIAboveConfig;
|
||||||
/** Pointer to the driver instance. */
|
/** Pointer to the driver instance. */
|
||||||
PPDMDRVINS pDrvIns;
|
PPDMDRVINS pDrvIns;
|
||||||
/** Receiver thread that handles all signals. */
|
/** Transmit lock used by pfnBeginXmit/pfnEndXmit. */
|
||||||
PPDMTHREAD pThread;
|
RTCRITSECT XmitLock;
|
||||||
/** Nic::Session client wrapper. */
|
/** Nic::Session client wrapper. */
|
||||||
Nic_client *nic_client;
|
Nic_client *nic_client;
|
||||||
} DRVNIC, *PDRVNIC;
|
} DRVNIC, *PDRVNIC;
|
||||||
@ -106,13 +106,17 @@ class Nic_client
|
|||||||
Genode::Signal_handler<Nic_client> _link_state_dispatcher;
|
Genode::Signal_handler<Nic_client> _link_state_dispatcher;
|
||||||
Genode::Signal_handler<Nic_client> _rx_packet_avail_dispatcher;
|
Genode::Signal_handler<Nic_client> _rx_packet_avail_dispatcher;
|
||||||
Genode::Signal_handler<Nic_client> _rx_ready_to_ack_dispatcher;
|
Genode::Signal_handler<Nic_client> _rx_ready_to_ack_dispatcher;
|
||||||
|
Genode::Signal_handler<Nic_client> _tx_ack_avail_dispatcher;
|
||||||
|
Genode::Signal_handler<Nic_client> _tx_ready_to_submit;
|
||||||
Genode::Signal_handler<Nic_client> _destruct_dispatcher;
|
Genode::Signal_handler<Nic_client> _destruct_dispatcher;
|
||||||
|
|
||||||
bool _link_up = false;
|
bool _link_up = false;
|
||||||
|
bool _retry = false;
|
||||||
|
|
||||||
/* VM <-> device driver (down) <-> nic_client (up) <-> nic session */
|
/* VM <-> device driver (down) <-> nic_client (up) <-> nic session */
|
||||||
PPDMINETWORKDOWN _down_rx;
|
PPDMINETWORKDOWN _down_net;
|
||||||
PPDMINETWORKCONFIG _down_rx_config;
|
PPDMINETWORKCONFIG _down_net_config;
|
||||||
|
PDRVNIC _drvnic;
|
||||||
|
|
||||||
void _handle_rx_packet_avail()
|
void _handle_rx_packet_avail()
|
||||||
{
|
{
|
||||||
@ -122,10 +126,10 @@ class Nic_client
|
|||||||
char *rx_content = _nic.rx()->packet_content(rx_packet);
|
char *rx_content = _nic.rx()->packet_content(rx_packet);
|
||||||
|
|
||||||
Libc::with_libc([&] () {
|
Libc::with_libc([&] () {
|
||||||
int rc = _down_rx->pfnWaitReceiveAvail(_down_rx, RT_INDEFINITE_WAIT);
|
int rc = _down_net->pfnWaitReceiveAvail(_down_net, RT_INDEFINITE_WAIT);
|
||||||
if (RT_FAILURE(rc)) return;
|
if (RT_FAILURE(rc)) return;
|
||||||
|
|
||||||
rc = _down_rx->pfnReceive(_down_rx, rx_content, rx_packet.size());
|
rc = _down_net->pfnReceive(_down_net, rx_content, rx_packet.size());
|
||||||
AssertRC(rc);
|
AssertRC(rc);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -135,12 +139,48 @@ class Nic_client
|
|||||||
|
|
||||||
void _handle_rx_ready_to_ack() { _handle_rx_packet_avail(); }
|
void _handle_rx_ready_to_ack() { _handle_rx_packet_avail(); }
|
||||||
|
|
||||||
|
template <typename F>
|
||||||
|
void _guard_with_xmit_lock(F const &fn)
|
||||||
|
{
|
||||||
|
int const rc = RTCritSectEnter(&_drvnic->XmitLock);
|
||||||
|
|
||||||
|
if (RT_FAILURE(rc)) {
|
||||||
|
RTLogPrintf("entering XmitLock failed %d\n", rc);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool const progress = fn();
|
||||||
|
|
||||||
|
RTCritSectLeave(&_drvnic->XmitLock);
|
||||||
|
|
||||||
|
/* pfnXmitPending takes the XmitLock again */
|
||||||
|
if (progress && _down_net->pfnXmitPending)
|
||||||
|
_down_net->pfnXmitPending(_down_net);
|
||||||
|
}
|
||||||
|
|
||||||
|
void _handle_tx_ack_avail()
|
||||||
|
{
|
||||||
|
_guard_with_xmit_lock([&](void){ return _tx_ack(); });
|
||||||
|
}
|
||||||
|
|
||||||
|
void _handle_tx_ready_to_submit()
|
||||||
|
{
|
||||||
|
_guard_with_xmit_lock([&](void){
|
||||||
|
|
||||||
|
bool const notify_network_model = _retry;
|
||||||
|
if (_retry)
|
||||||
|
_retry = false;
|
||||||
|
|
||||||
|
return notify_network_model;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
void _handle_link_state()
|
void _handle_link_state()
|
||||||
{
|
{
|
||||||
_link_up = _nic.link_state();
|
_link_up = _nic.link_state();
|
||||||
|
|
||||||
Libc::with_libc([&] () {
|
Libc::with_libc([&] () {
|
||||||
_down_rx_config->pfnSetLinkState(_down_rx_config,
|
_down_net_config->pfnSetLinkState(_down_net_config,
|
||||||
_link_up ? PDMNETWORKLINKSTATE_UP
|
_link_up ? PDMNETWORKLINKSTATE_UP
|
||||||
: PDMNETWORKLINKSTATE_DOWN);
|
: PDMNETWORKLINKSTATE_DOWN);
|
||||||
});
|
});
|
||||||
@ -151,19 +191,31 @@ class Nic_client
|
|||||||
_nic.link_state_sigh(Genode::Signal_context_capability());
|
_nic.link_state_sigh(Genode::Signal_context_capability());
|
||||||
_nic.rx_channel()->sigh_packet_avail(Genode::Signal_context_capability());
|
_nic.rx_channel()->sigh_packet_avail(Genode::Signal_context_capability());
|
||||||
_nic.rx_channel()->sigh_ready_to_ack(Genode::Signal_context_capability());
|
_nic.rx_channel()->sigh_ready_to_ack(Genode::Signal_context_capability());
|
||||||
|
_nic.tx_channel()->sigh_ack_avail(Genode::Signal_context_capability());
|
||||||
|
_nic.tx_channel()->sigh_ready_to_submit(Genode::Signal_context_capability());
|
||||||
|
|
||||||
destruct_blockade().wakeup();
|
destruct_blockade().wakeup();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _tx_ack()
|
bool _tx_ack()
|
||||||
{
|
{
|
||||||
|
bool progress = false;
|
||||||
|
|
||||||
/* check for acknowledgements */
|
/* check for acknowledgements */
|
||||||
while (_nic.tx()->ack_avail()) {
|
while (_nic.tx()->ack_avail()) {
|
||||||
Nic::Packet_descriptor acked_packet = _nic.tx()->get_acked_packet();
|
Nic::Packet_descriptor acked_packet = _nic.tx()->get_acked_packet();
|
||||||
auto packet_allocated_len = Nic::Packet_descriptor(acked_packet.offset(),
|
auto packet_allocated_len = Nic::Packet_descriptor(acked_packet.offset(),
|
||||||
Nic::Packet_allocator::OFFSET_PACKET_SIZE);
|
Nic::Packet_allocator::OFFSET_PACKET_SIZE);
|
||||||
|
|
||||||
_nic.tx()->release_packet(packet_allocated_len);
|
_nic.tx()->release_packet(packet_allocated_len);
|
||||||
|
|
||||||
|
if (_retry) {
|
||||||
|
progress = true;
|
||||||
|
_retry = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return progress;
|
||||||
}
|
}
|
||||||
|
|
||||||
static Nic::Packet_allocator* _packet_allocator()
|
static Nic::Packet_allocator* _packet_allocator()
|
||||||
@ -186,7 +238,7 @@ class Nic_client
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
Nic_client(Genode::Env &env, PDRVNIC drvtap,
|
Nic_client(Genode::Env &env, PDRVNIC drv,
|
||||||
Genode::Session::Label const &label)
|
Genode::Session::Label const &label)
|
||||||
:
|
:
|
||||||
_tx_block_alloc(_packet_allocator()),
|
_tx_block_alloc(_packet_allocator()),
|
||||||
@ -195,11 +247,14 @@ class Nic_client
|
|||||||
_link_state_dispatcher(_ep, *this, &Nic_client::_handle_link_state),
|
_link_state_dispatcher(_ep, *this, &Nic_client::_handle_link_state),
|
||||||
_rx_packet_avail_dispatcher(_ep, *this, &Nic_client::_handle_rx_packet_avail),
|
_rx_packet_avail_dispatcher(_ep, *this, &Nic_client::_handle_rx_packet_avail),
|
||||||
_rx_ready_to_ack_dispatcher(_ep, *this, &Nic_client::_handle_rx_ready_to_ack),
|
_rx_ready_to_ack_dispatcher(_ep, *this, &Nic_client::_handle_rx_ready_to_ack),
|
||||||
|
_tx_ack_avail_dispatcher(_ep, *this, &Nic_client::_handle_tx_ack_avail),
|
||||||
|
_tx_ready_to_submit(_ep, *this, &Nic_client::_handle_tx_ready_to_submit),
|
||||||
_destruct_dispatcher(_ep, *this, &Nic_client::_handle_destruct),
|
_destruct_dispatcher(_ep, *this, &Nic_client::_handle_destruct),
|
||||||
_down_rx(drvtap->pIAboveNet),
|
_down_net(drv->pIAboveNet),
|
||||||
_down_rx_config(drvtap->pIAboveConfig)
|
_down_net_config(drv->pIAboveConfig),
|
||||||
|
_drvnic(drv)
|
||||||
{
|
{
|
||||||
Genode::Signal_transmitter(_pthread_reg_sigh).submit();
|
_pthread_reg_sigh.local_submit();
|
||||||
}
|
}
|
||||||
|
|
||||||
~Nic_client()
|
~Nic_client()
|
||||||
@ -213,6 +268,8 @@ class Nic_client
|
|||||||
_nic.link_state_sigh(_link_state_dispatcher);
|
_nic.link_state_sigh(_link_state_dispatcher);
|
||||||
_nic.rx_channel()->sigh_packet_avail(_rx_packet_avail_dispatcher);
|
_nic.rx_channel()->sigh_packet_avail(_rx_packet_avail_dispatcher);
|
||||||
_nic.rx_channel()->sigh_ready_to_ack(_rx_ready_to_ack_dispatcher);
|
_nic.rx_channel()->sigh_ready_to_ack(_rx_ready_to_ack_dispatcher);
|
||||||
|
_nic.tx_channel()->sigh_ack_avail(_tx_ack_avail_dispatcher);
|
||||||
|
_nic.tx_channel()->sigh_ready_to_submit(_tx_ready_to_submit);
|
||||||
|
|
||||||
/* inform signal handler ep */
|
/* inform signal handler ep */
|
||||||
_link_state_dispatcher.local_submit();
|
_link_state_dispatcher.local_submit();
|
||||||
@ -223,12 +280,16 @@ class Nic_client
|
|||||||
|
|
||||||
bool alloc_packet(Nic::Packet_descriptor &pkg, uint32_t packet_len)
|
bool alloc_packet(Nic::Packet_descriptor &pkg, uint32_t packet_len)
|
||||||
{
|
{
|
||||||
|
/* check for acknowledgments */
|
||||||
|
_tx_ack();
|
||||||
|
|
||||||
auto const result = _nic.tx()->alloc_packet_attempt(packet_len);
|
auto const result = _nic.tx()->alloc_packet_attempt(packet_len);
|
||||||
|
|
||||||
return result.convert<bool>([&](auto &p) {
|
return result.convert<bool>([&](auto &p) {
|
||||||
pkg = p;
|
pkg = p;
|
||||||
return true;
|
return true;
|
||||||
}, [&] (auto &) {
|
}, [&] (auto &) {
|
||||||
|
_retry = true;
|
||||||
return false;
|
return false;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -237,9 +298,6 @@ class Nic_client
|
|||||||
{
|
{
|
||||||
if (!_link_up) { return VERR_NET_DOWN; }
|
if (!_link_up) { return VERR_NET_DOWN; }
|
||||||
|
|
||||||
/* check for acknowledgements */
|
|
||||||
_tx_ack();
|
|
||||||
|
|
||||||
if (tx_packet.size() < packet_len) {
|
if (tx_packet.size() < packet_len) {
|
||||||
RTLogPrintf("%s: packet too large\n", __func__);
|
RTLogPrintf("%s: packet too large\n", __func__);
|
||||||
_nic.tx()->release_packet(tx_packet);
|
_nic.tx()->release_packet(tx_packet);
|
||||||
@ -253,6 +311,13 @@ class Nic_client
|
|||||||
_nic.tx()->submit_packet(tx_packet_actual_len);
|
_nic.tx()->submit_packet(tx_packet_actual_len);
|
||||||
return VINF_SUCCESS;
|
return VINF_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void release_not_sent_packet(Nic::Packet_descriptor const &tx_not_sent)
|
||||||
|
{
|
||||||
|
auto const len = Nic::Packet_descriptor(tx_not_sent.offset(),
|
||||||
|
Nic::Packet_allocator::OFFSET_PACKET_SIZE);
|
||||||
|
_nic.tx()->release_packet(len);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -270,7 +335,21 @@ class Nic_client
|
|||||||
*/
|
*/
|
||||||
static DECLCALLBACK(int) drvNicNetworkUp_BeginXmit(PPDMINETWORKUP pInterface, bool fOnWorkerThread)
|
static DECLCALLBACK(int) drvNicNetworkUp_BeginXmit(PPDMINETWORKUP pInterface, bool fOnWorkerThread)
|
||||||
{
|
{
|
||||||
return VINF_SUCCESS;
|
PDRVNIC pThis = PDMINETWORKUP_2_DRVNIC(pInterface);
|
||||||
|
int rc = RTCritSectTryEnter(&pThis->XmitLock);
|
||||||
|
if (RT_FAILURE(rc))
|
||||||
|
rc = VERR_TRY_AGAIN;
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @interface_method_impl{PDMINETWORKUP,pfnEndXmit}
|
||||||
|
*/
|
||||||
|
static DECLCALLBACK(void) drvNicNetworkUp_EndXmit(PPDMINETWORKUP pInterface)
|
||||||
|
{
|
||||||
|
PDRVNIC pThis = PDMINETWORKUP_2_DRVNIC(pInterface);
|
||||||
|
RTCritSectLeave(&pThis->XmitLock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -336,13 +415,20 @@ static DECLCALLBACK(int) drvNicNetworkUp_AllocBuf(PPDMINETWORKUP pInterface, siz
|
|||||||
*/
|
*/
|
||||||
static DECLCALLBACK(int) drvNicNetworkUp_FreeBuf(PPDMINETWORKUP pInterface, PPDMSCATTERGATHER pSgBuf)
|
static DECLCALLBACK(int) drvNicNetworkUp_FreeBuf(PPDMINETWORKUP pInterface, PPDMSCATTERGATHER pSgBuf)
|
||||||
{
|
{
|
||||||
PDRVNIC pThis = PDMINETWORKUP_2_DRVNIC(pInterface);
|
|
||||||
if (pSgBuf)
|
if (pSgBuf)
|
||||||
{
|
{
|
||||||
Assert((pSgBuf->fFlags & PDMSCATTERGATHER_FLAGS_MAGIC_MASK) == PDMSCATTERGATHER_FLAGS_MAGIC);
|
Assert((pSgBuf->fFlags & PDMSCATTERGATHER_FLAGS_MAGIC_MASK) == PDMSCATTERGATHER_FLAGS_MAGIC);
|
||||||
|
|
||||||
pSgBuf->fFlags = 0;
|
pSgBuf->fFlags = 0;
|
||||||
if (pSgBuf->pvAllocator)
|
if (pSgBuf->pvAllocator) {
|
||||||
|
PDRVNIC pThis = PDMINETWORKUP_2_DRVNIC(pInterface);
|
||||||
|
Nic_client *nic_client = pThis->nic_client;
|
||||||
|
auto const &packet = *(Nic::Packet_descriptor *)pSgBuf->pvAllocator;
|
||||||
|
|
||||||
|
nic_client->release_not_sent_packet(packet);
|
||||||
|
|
||||||
RTMemFree(pSgBuf->pvAllocator);
|
RTMemFree(pSgBuf->pvAllocator);
|
||||||
|
}
|
||||||
RTMemFree(pSgBuf);
|
RTMemFree(pSgBuf);
|
||||||
}
|
}
|
||||||
return VINF_SUCCESS;
|
return VINF_SUCCESS;
|
||||||
@ -359,6 +445,7 @@ static DECLCALLBACK(int) drvNicNetworkUp_SendBuf(PPDMINETWORKUP pInterface, PPDM
|
|||||||
|
|
||||||
AssertPtr(pSgBuf);
|
AssertPtr(pSgBuf);
|
||||||
Assert((pSgBuf->fFlags & PDMSCATTERGATHER_FLAGS_MAGIC_MASK) == PDMSCATTERGATHER_FLAGS_MAGIC);
|
Assert((pSgBuf->fFlags & PDMSCATTERGATHER_FLAGS_MAGIC_MASK) == PDMSCATTERGATHER_FLAGS_MAGIC);
|
||||||
|
Assert(RTCritSectIsOwner(&pThis->XmitLock));
|
||||||
|
|
||||||
if (!pSgBuf->pvAllocator) {
|
if (!pSgBuf->pvAllocator) {
|
||||||
RTLogPrintf("%s: error in packet allocation\n", __func__);
|
RTLogPrintf("%s: error in packet allocation\n", __func__);
|
||||||
@ -410,15 +497,6 @@ static DECLCALLBACK(int) drvNicNetworkUp_SendBuf(PPDMINETWORKUP pInterface, PPDM
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @interface_method_impl{PDMINETWORKUP,pfnEndXmit}
|
|
||||||
*/
|
|
||||||
static DECLCALLBACK(void) drvNicNetworkUp_EndXmit(PPDMINETWORKUP pInterface)
|
|
||||||
{
|
|
||||||
PDRVNIC pThis = PDMINETWORKUP_2_DRVNIC(pInterface);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @interface_method_impl{PDMINETWORKUP,pfnSetPromiscuousMode}
|
* @interface_method_impl{PDMINETWORKUP,pfnSetPromiscuousMode}
|
||||||
*/
|
*/
|
||||||
@ -477,17 +555,20 @@ static DECLCALLBACK(void) drvNicDestruct(PPDMDRVINS pDrvIns)
|
|||||||
PDRVNIC pThis = PDMINS_2_DATA(pDrvIns, PDRVNIC);
|
PDRVNIC pThis = PDMINS_2_DATA(pDrvIns, PDRVNIC);
|
||||||
Nic_client *nic_client = pThis->nic_client;
|
Nic_client *nic_client = pThis->nic_client;
|
||||||
|
|
||||||
if (!nic_client)
|
if (!nic_client) {
|
||||||
Genode::error("nic_client not valid at destruction time");
|
Genode::error("nic_client not valid at destruction time");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (nic_client)
|
Genode::Signal_transmitter(nic_client->dispatcher()).submit();
|
||||||
Genode::Signal_transmitter(nic_client->dispatcher()).submit();
|
|
||||||
|
|
||||||
/* wait until the recv thread exits */
|
/* wait until the recv thread exits */
|
||||||
destruct_blockade().block();
|
destruct_blockade().block();
|
||||||
|
|
||||||
if (nic_client)
|
if (RTCritSectIsInitialized(&pThis->XmitLock))
|
||||||
destroy(vmm_heap(), nic_client);
|
RTCritSectDelete(&pThis->XmitLock);
|
||||||
|
|
||||||
|
destroy(vmm_heap(), nic_client);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -519,6 +600,8 @@ static DECLCALLBACK(int) drvNicConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uin
|
|||||||
/* INetworkConfig - used on Genode to request Mac address of nic_session */
|
/* INetworkConfig - used on Genode to request Mac address of nic_session */
|
||||||
pThis->INetworkConfig.pfnGetMac = drvGetMac;
|
pThis->INetworkConfig.pfnGetMac = drvGetMac;
|
||||||
|
|
||||||
|
RTCritSectInit(&pThis->XmitLock);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Check that no-one is attached to us.
|
* Check that no-one is attached to us.
|
||||||
*/
|
*/
|
||||||
|
@ -4,6 +4,15 @@ diff --git a/src/app/virtualbox/src/VBox/Devices/Network/DevE1000.cpp b/src/app/
|
|||||||
index aa3eb87..fc2660a 100644
|
index aa3eb87..fc2660a 100644
|
||||||
--- a/src/app/virtualbox/src/VBox/Devices/Network/DevE1000.cpp
|
--- a/src/app/virtualbox/src/VBox/Devices/Network/DevE1000.cpp
|
||||||
+++ b/src/app/virtualbox/src/VBox/Devices/Network/DevE1000.cpp
|
+++ b/src/app/virtualbox/src/VBox/Devices/Network/DevE1000.cpp
|
||||||
|
@@ -7629,7 +7629,7 @@
|
||||||
|
return PDMDEV_SET_ERROR(pDevIns, rc,
|
||||||
|
N_("Configuration error: Failed to get the value of 'EthernetCRC'"));
|
||||||
|
|
||||||
|
- rc = CFGMR3QueryBoolDef(pCfg, "GSOEnabled", &pThis->fGSOEnabled, true);
|
||||||
|
+ rc = CFGMR3QueryBoolDef(pCfg, "GSOEnabled", &pThis->fGSOEnabled, false);
|
||||||
|
if (RT_FAILURE(rc))
|
||||||
|
return PDMDEV_SET_ERROR(pDevIns, rc,
|
||||||
|
N_("Configuration error: Failed to get the value of 'GSOEnabled'"));
|
||||||
@@ -7519,6 +7519,35 @@ static DECLCALLBACK(int) e1kR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGM
|
@@ -7519,6 +7519,35 @@ static DECLCALLBACK(int) e1kR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGM
|
||||||
pThis->fR0Enabled ? "enabled" : "disabled",
|
pThis->fR0Enabled ? "enabled" : "disabled",
|
||||||
pThis->fRCEnabled ? "enabled" : "disabled"));
|
pThis->fRCEnabled ? "enabled" : "disabled"));
|
||||||
|
@ -122,13 +122,17 @@ class Nic_client
|
|||||||
Genode::Signal_handler<Nic_client> _link_state_dispatcher;
|
Genode::Signal_handler<Nic_client> _link_state_dispatcher;
|
||||||
Genode::Signal_handler<Nic_client> _rx_packet_avail_dispatcher;
|
Genode::Signal_handler<Nic_client> _rx_packet_avail_dispatcher;
|
||||||
Genode::Signal_handler<Nic_client> _rx_ready_to_ack_dispatcher;
|
Genode::Signal_handler<Nic_client> _rx_ready_to_ack_dispatcher;
|
||||||
|
Genode::Signal_handler<Nic_client> _tx_ack_avail_dispatcher;
|
||||||
|
Genode::Signal_handler<Nic_client> _tx_ready_to_submit;
|
||||||
Genode::Signal_handler<Nic_client> _destruct_dispatcher;
|
Genode::Signal_handler<Nic_client> _destruct_dispatcher;
|
||||||
|
|
||||||
bool _link_up = false;
|
bool _link_up = false;
|
||||||
|
bool _retry = false;
|
||||||
|
|
||||||
/* VM <-> device driver (down) <-> nic_client (up) <-> nic session */
|
/* VM <-> device driver (down) <-> nic_client (up) <-> nic session */
|
||||||
PPDMINETWORKDOWN _down_rx;
|
PPDMINETWORKDOWN _down_net;
|
||||||
PPDMINETWORKCONFIG _down_rx_config;
|
PPDMINETWORKCONFIG _down_net_config;
|
||||||
|
PDRVNIC _drvnic;
|
||||||
|
|
||||||
void _handle_rx_packet_avail()
|
void _handle_rx_packet_avail()
|
||||||
{
|
{
|
||||||
@ -138,10 +142,10 @@ class Nic_client
|
|||||||
char *rx_content = _nic.rx()->packet_content(rx_packet);
|
char *rx_content = _nic.rx()->packet_content(rx_packet);
|
||||||
|
|
||||||
Libc::with_libc([&] () {
|
Libc::with_libc([&] () {
|
||||||
int rc = _down_rx->pfnWaitReceiveAvail(_down_rx, RT_INDEFINITE_WAIT);
|
int rc = _down_net->pfnWaitReceiveAvail(_down_net, RT_INDEFINITE_WAIT);
|
||||||
if (RT_FAILURE(rc)) return;
|
if (RT_FAILURE(rc)) return;
|
||||||
|
|
||||||
rc = _down_rx->pfnReceive(_down_rx, rx_content, rx_packet.size());
|
rc = _down_net->pfnReceive(_down_net, rx_content, rx_packet.size());
|
||||||
AssertRC(rc);
|
AssertRC(rc);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -151,12 +155,48 @@ class Nic_client
|
|||||||
|
|
||||||
void _handle_rx_ready_to_ack() { _handle_rx_packet_avail(); }
|
void _handle_rx_ready_to_ack() { _handle_rx_packet_avail(); }
|
||||||
|
|
||||||
|
template <typename F>
|
||||||
|
void _guard_with_xmit_lock(F const &fn)
|
||||||
|
{
|
||||||
|
int const rc = RTCritSectEnter(&_drvnic->XmitLock);
|
||||||
|
|
||||||
|
if (RT_FAILURE(rc)) {
|
||||||
|
RTLogPrintf("entering XmitLock failed %d\n", rc);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool const progress = fn();
|
||||||
|
|
||||||
|
RTCritSectLeave(&_drvnic->XmitLock);
|
||||||
|
|
||||||
|
/* pfnXmitPending takes the XmitLock again */
|
||||||
|
if (progress && _down_net->pfnXmitPending)
|
||||||
|
_down_net->pfnXmitPending(_down_net);
|
||||||
|
}
|
||||||
|
|
||||||
|
void _handle_tx_ack_avail()
|
||||||
|
{
|
||||||
|
_guard_with_xmit_lock([&](void){ return _tx_ack(); });
|
||||||
|
}
|
||||||
|
|
||||||
|
void _handle_tx_ready_to_submit()
|
||||||
|
{
|
||||||
|
_guard_with_xmit_lock([&](void){
|
||||||
|
|
||||||
|
bool const notify_network_model = _retry;
|
||||||
|
if (_retry)
|
||||||
|
_retry = false;
|
||||||
|
|
||||||
|
return notify_network_model;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
void _handle_link_state()
|
void _handle_link_state()
|
||||||
{
|
{
|
||||||
_link_up = _nic.link_state();
|
_link_up = _nic.link_state();
|
||||||
|
|
||||||
Libc::with_libc([&] () {
|
Libc::with_libc([&] () {
|
||||||
_down_rx_config->pfnSetLinkState(_down_rx_config,
|
_down_net_config->pfnSetLinkState(_down_net_config,
|
||||||
_link_up ? PDMNETWORKLINKSTATE_UP
|
_link_up ? PDMNETWORKLINKSTATE_UP
|
||||||
: PDMNETWORKLINKSTATE_DOWN);
|
: PDMNETWORKLINKSTATE_DOWN);
|
||||||
});
|
});
|
||||||
@ -167,19 +207,31 @@ class Nic_client
|
|||||||
_nic.link_state_sigh(Genode::Signal_context_capability());
|
_nic.link_state_sigh(Genode::Signal_context_capability());
|
||||||
_nic.rx_channel()->sigh_packet_avail(Genode::Signal_context_capability());
|
_nic.rx_channel()->sigh_packet_avail(Genode::Signal_context_capability());
|
||||||
_nic.rx_channel()->sigh_ready_to_ack(Genode::Signal_context_capability());
|
_nic.rx_channel()->sigh_ready_to_ack(Genode::Signal_context_capability());
|
||||||
|
_nic.tx_channel()->sigh_ack_avail(Genode::Signal_context_capability());
|
||||||
|
_nic.tx_channel()->sigh_ready_to_submit(Genode::Signal_context_capability());
|
||||||
|
|
||||||
destruct_blockade().wakeup();
|
destruct_blockade().wakeup();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _tx_ack()
|
bool _tx_ack()
|
||||||
{
|
{
|
||||||
|
bool progress = false;
|
||||||
|
|
||||||
/* check for acknowledgements */
|
/* check for acknowledgements */
|
||||||
while (_nic.tx()->ack_avail()) {
|
while (_nic.tx()->ack_avail()) {
|
||||||
Nic::Packet_descriptor acked_packet = _nic.tx()->get_acked_packet();
|
Nic::Packet_descriptor acked_packet = _nic.tx()->get_acked_packet();
|
||||||
auto packet_allocated_len = Nic::Packet_descriptor(acked_packet.offset(),
|
auto packet_allocated_len = Nic::Packet_descriptor(acked_packet.offset(),
|
||||||
Nic::Packet_allocator::OFFSET_PACKET_SIZE);
|
Nic::Packet_allocator::OFFSET_PACKET_SIZE);
|
||||||
|
|
||||||
_nic.tx()->release_packet(packet_allocated_len);
|
_nic.tx()->release_packet(packet_allocated_len);
|
||||||
|
|
||||||
|
if (_retry) {
|
||||||
|
progress = true;
|
||||||
|
_retry = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return progress;
|
||||||
}
|
}
|
||||||
|
|
||||||
static Nic::Packet_allocator* _packet_allocator()
|
static Nic::Packet_allocator* _packet_allocator()
|
||||||
@ -202,7 +254,7 @@ class Nic_client
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
Nic_client(Genode::Env &env, PDRVNIC drvtap, char const *label)
|
Nic_client(Genode::Env &env, PDRVNIC drv, char const *label)
|
||||||
:
|
:
|
||||||
_tx_block_alloc(_packet_allocator()),
|
_tx_block_alloc(_packet_allocator()),
|
||||||
_nic(env, _tx_block_alloc, BUF_SIZE, BUF_SIZE, label),
|
_nic(env, _tx_block_alloc, BUF_SIZE, BUF_SIZE, label),
|
||||||
@ -210,9 +262,12 @@ class Nic_client
|
|||||||
_link_state_dispatcher(_ep, *this, &Nic_client::_handle_link_state),
|
_link_state_dispatcher(_ep, *this, &Nic_client::_handle_link_state),
|
||||||
_rx_packet_avail_dispatcher(_ep, *this, &Nic_client::_handle_rx_packet_avail),
|
_rx_packet_avail_dispatcher(_ep, *this, &Nic_client::_handle_rx_packet_avail),
|
||||||
_rx_ready_to_ack_dispatcher(_ep, *this, &Nic_client::_handle_rx_ready_to_ack),
|
_rx_ready_to_ack_dispatcher(_ep, *this, &Nic_client::_handle_rx_ready_to_ack),
|
||||||
|
_tx_ack_avail_dispatcher(_ep, *this, &Nic_client::_handle_tx_ack_avail),
|
||||||
|
_tx_ready_to_submit(_ep, *this, &Nic_client::_handle_tx_ready_to_submit),
|
||||||
_destruct_dispatcher(_ep, *this, &Nic_client::_handle_destruct),
|
_destruct_dispatcher(_ep, *this, &Nic_client::_handle_destruct),
|
||||||
_down_rx(drvtap->pIAboveNet),
|
_down_net(drv->pIAboveNet),
|
||||||
_down_rx_config(drvtap->pIAboveConfig)
|
_down_net_config(drv->pIAboveConfig),
|
||||||
|
_drvnic(drv)
|
||||||
{
|
{
|
||||||
_pthread_reg_sigh.local_submit();
|
_pthread_reg_sigh.local_submit();
|
||||||
}
|
}
|
||||||
@ -228,6 +283,8 @@ class Nic_client
|
|||||||
_nic.link_state_sigh(_link_state_dispatcher);
|
_nic.link_state_sigh(_link_state_dispatcher);
|
||||||
_nic.rx_channel()->sigh_packet_avail(_rx_packet_avail_dispatcher);
|
_nic.rx_channel()->sigh_packet_avail(_rx_packet_avail_dispatcher);
|
||||||
_nic.rx_channel()->sigh_ready_to_ack(_rx_ready_to_ack_dispatcher);
|
_nic.rx_channel()->sigh_ready_to_ack(_rx_ready_to_ack_dispatcher);
|
||||||
|
_nic.tx_channel()->sigh_ack_avail(_tx_ack_avail_dispatcher);
|
||||||
|
_nic.tx_channel()->sigh_ready_to_submit(_tx_ready_to_submit);
|
||||||
|
|
||||||
/* inform signal handler ep */
|
/* inform signal handler ep */
|
||||||
_link_state_dispatcher.local_submit();
|
_link_state_dispatcher.local_submit();
|
||||||
@ -238,12 +295,16 @@ class Nic_client
|
|||||||
|
|
||||||
bool alloc_packet(Nic::Packet_descriptor &pkg, uint32_t packet_len)
|
bool alloc_packet(Nic::Packet_descriptor &pkg, uint32_t packet_len)
|
||||||
{
|
{
|
||||||
|
/* check for acknowledgments */
|
||||||
|
_tx_ack();
|
||||||
|
|
||||||
auto const result = _nic.tx()->alloc_packet_attempt(packet_len);
|
auto const result = _nic.tx()->alloc_packet_attempt(packet_len);
|
||||||
|
|
||||||
return result.convert<bool>([&](auto &p) {
|
return result.convert<bool>([&](auto &p) {
|
||||||
pkg = p;
|
pkg = p;
|
||||||
return true;
|
return true;
|
||||||
}, [&] (auto &) {
|
}, [&] (auto &) {
|
||||||
|
_retry = true;
|
||||||
return false;
|
return false;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -252,9 +313,6 @@ class Nic_client
|
|||||||
{
|
{
|
||||||
if (!_link_up) { return VERR_NET_DOWN; }
|
if (!_link_up) { return VERR_NET_DOWN; }
|
||||||
|
|
||||||
/* check for acknowledgements */
|
|
||||||
_tx_ack();
|
|
||||||
|
|
||||||
if (tx_packet.size() < packet_len) {
|
if (tx_packet.size() < packet_len) {
|
||||||
RTLogPrintf("%s: packet too large\n", __func__);
|
RTLogPrintf("%s: packet too large\n", __func__);
|
||||||
_nic.tx()->release_packet(tx_packet);
|
_nic.tx()->release_packet(tx_packet);
|
||||||
@ -268,6 +326,13 @@ class Nic_client
|
|||||||
_nic.tx()->submit_packet(tx_packet_actual_len);
|
_nic.tx()->submit_packet(tx_packet_actual_len);
|
||||||
return VINF_SUCCESS;
|
return VINF_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void release_not_sent_packet(Nic::Packet_descriptor const &tx_not_sent)
|
||||||
|
{
|
||||||
|
auto const len = Nic::Packet_descriptor(tx_not_sent.offset(),
|
||||||
|
Nic::Packet_allocator::OFFSET_PACKET_SIZE);
|
||||||
|
_nic.tx()->release_packet(len);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -368,9 +433,17 @@ static DECLCALLBACK(int) drvNicNetworkUp_FreeBuf(PPDMINETWORKUP pInterface, PPDM
|
|||||||
if (pSgBuf)
|
if (pSgBuf)
|
||||||
{
|
{
|
||||||
Assert((pSgBuf->fFlags & PDMSCATTERGATHER_FLAGS_MAGIC_MASK) == PDMSCATTERGATHER_FLAGS_MAGIC);
|
Assert((pSgBuf->fFlags & PDMSCATTERGATHER_FLAGS_MAGIC_MASK) == PDMSCATTERGATHER_FLAGS_MAGIC);
|
||||||
|
|
||||||
pSgBuf->fFlags = 0;
|
pSgBuf->fFlags = 0;
|
||||||
if (pSgBuf->pvAllocator)
|
if (pSgBuf->pvAllocator) {
|
||||||
|
PDRVNIC pThis = PDMINETWORKUP_2_DRVNIC(pInterface);
|
||||||
|
Nic_client *nic_client = pThis->nic_client;
|
||||||
|
auto const &packet = *(Nic::Packet_descriptor *)pSgBuf->pvAllocator;
|
||||||
|
|
||||||
|
nic_client->release_not_sent_packet(packet);
|
||||||
|
|
||||||
RTMemFree(pSgBuf->pvAllocator);
|
RTMemFree(pSgBuf->pvAllocator);
|
||||||
|
}
|
||||||
RTMemFree(pSgBuf);
|
RTMemFree(pSgBuf);
|
||||||
}
|
}
|
||||||
return VINF_SUCCESS;
|
return VINF_SUCCESS;
|
||||||
|
Loading…
Reference in New Issue
Block a user