summaryrefslogtreecommitdiffstats
path: root/src/VBox/VMM/VMMAll/PDMAllNetShaper.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/VBox/VMM/VMMAll/PDMAllNetShaper.cpp')
-rw-r--r--src/VBox/VMM/VMMAll/PDMAllNetShaper.cpp77
1 files changed, 77 insertions, 0 deletions
diff --git a/src/VBox/VMM/VMMAll/PDMAllNetShaper.cpp b/src/VBox/VMM/VMMAll/PDMAllNetShaper.cpp
new file mode 100644
index 00000000..7d1ea83a
--- /dev/null
+++ b/src/VBox/VMM/VMMAll/PDMAllNetShaper.cpp
@@ -0,0 +1,77 @@
+/* $Id: PDMAllNetShaper.cpp $ */
+/** @file
+ * PDM Network Shaper - Limit network traffic according to bandwidth group settings.
+ */
+
+/*
+ * Copyright (C) 2011-2019 Oracle Corporation
+ *
+ * This file is part of VirtualBox Open Source Edition (OSE), as
+ * available from http://www.virtualbox.org. This file is free software;
+ * you can redistribute it and/or modify it under the terms of the GNU
+ * General Public License (GPL) as published by the Free Software
+ * Foundation, in version 2 as it comes in the "COPYING" file of the
+ * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
+ * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
+ */
+
+
+/*********************************************************************************************************************************
+* Header Files *
+*********************************************************************************************************************************/
+#define LOG_GROUP LOG_GROUP_NET_SHAPER
+#include <VBox/vmm/pdm.h>
+#include <VBox/log.h>
+#include <iprt/time.h>
+
+#include <VBox/vmm/pdmnetshaper.h>
+#include "PDMNetShaperInternal.h"
+
+
+/**
+ * Obtain bandwidth in a bandwidth group.
+ *
+ * @returns True if bandwidth was allocated, false if not.
+ * @param pFilter Pointer to the filter that allocates bandwidth.
+ * @param cbTransfer Number of bytes to allocate.
+ */
+VMMDECL(bool) PDMNsAllocateBandwidth(PPDMNSFILTER pFilter, size_t cbTransfer)
+{
+ AssertPtrReturn(pFilter, true);
+ if (!VALID_PTR(pFilter->CTX_SUFF(pBwGroup)))
+ return true;
+
+ PPDMNSBWGROUP pBwGroup = ASMAtomicReadPtrT(&pFilter->CTX_SUFF(pBwGroup), PPDMNSBWGROUP);
+ int rc = PDMCritSectEnter(&pBwGroup->Lock, VERR_SEM_BUSY); AssertRC(rc);
+ if (RT_UNLIKELY(rc == VERR_SEM_BUSY))
+ return true;
+
+ bool fAllowed = true;
+ if (pBwGroup->cbPerSecMax)
+ {
+ /* Re-fill the bucket first */
+ uint64_t tsNow = RTTimeSystemNanoTS();
+ uint32_t uTokensAdded = (tsNow - pBwGroup->tsUpdatedLast) * pBwGroup->cbPerSecMax / (1000 * 1000 * 1000);
+ uint32_t uTokens = RT_MIN(pBwGroup->cbBucket, uTokensAdded + pBwGroup->cbTokensLast);
+
+ if (cbTransfer > uTokens)
+ {
+ fAllowed = false;
+ ASMAtomicWriteBool(&pFilter->fChoked, true);
+ }
+ else
+ {
+ pBwGroup->tsUpdatedLast = tsNow;
+ pBwGroup->cbTokensLast = uTokens - (uint32_t)cbTransfer;
+ }
+ Log2(("pdmNsAllocateBandwidth: BwGroup=%#p{%s} cbTransfer=%u uTokens=%u uTokensAdded=%u fAllowed=%RTbool\n",
+ pBwGroup, R3STRING(pBwGroup->pszNameR3), cbTransfer, uTokens, uTokensAdded, fAllowed));
+ }
+ else
+ Log2(("pdmNsAllocateBandwidth: BwGroup=%#p{%s} disabled fAllowed=%RTbool\n",
+ pBwGroup, R3STRING(pBwGroup->pszNameR3), fAllowed));
+
+ rc = PDMCritSectLeave(&pBwGroup->Lock); AssertRC(rc);
+ return fAllowed;
+}
+