summaryrefslogtreecommitdiffstats
path: root/src/common
diff options
context:
space:
mode:
authorimarom <imarom@cisco.com>2016-09-28 14:57:43 +0300
committerimarom <imarom@cisco.com>2016-09-28 15:51:18 +0300
commit5963a0384f1317d03f65247e96a82bef402b0be0 (patch)
tree5746845621826bf07309e09aa174e3452d4620a2 /src/common
parent5497c271302aa417814aa2e368b1ab2cf1fcef62 (diff)
performance tweak - faster checksum
Diffstat (limited to 'src/common')
-rwxr-xr-xsrc/common/Network/Packet/IPHeader.h57
1 files changed, 57 insertions, 0 deletions
diff --git a/src/common/Network/Packet/IPHeader.h b/src/common/Network/Packet/IPHeader.h
index b9ef8a2e..dd9f5096 100755
--- a/src/common/Network/Packet/IPHeader.h
+++ b/src/common/Network/Packet/IPHeader.h
@@ -19,6 +19,10 @@ limitations under the License.
#include "PacketHeaderBase.h"
+#ifndef likely
+#define likely(x) __builtin_expect((x),1)
+#endif /* likely */
+
#define IPV4_HDR_LEN 20
class IPHeader
@@ -140,6 +144,59 @@ public:
inline void swapSrcDest ();
+
+ inline void updateCheckSumFast() {
+ myChecksum = 0;
+
+ if (likely(myVer_HeaderLength == 0x45)) {
+ myChecksum = calc_cksum_fixed();
+ } else {
+ myChecksum = calc_cksum_nonfixed();
+ }
+
+ }
+
+protected:
+
+ /* fast inline checksum calculation */
+ inline uint16_t calc_cksum_fixed() {
+ const uint16_t *ipv4 = (const uint16_t *)this;
+
+ int sum = 0;
+
+ /* calcualte 20 bytes unrolled loop */
+ sum += ipv4[0];
+ sum += ipv4[1];
+ sum += ipv4[2];
+ sum += ipv4[3];
+ sum += ipv4[4];
+ sum += ipv4[5];
+ sum += ipv4[6];
+ sum += ipv4[7];
+ sum += ipv4[8];
+ sum += ipv4[9];
+
+ sum = (sum & 0xffff) + (sum >> 16);
+
+ return (uint16_t)(~sum);
+ }
+
+ /* a slow non-frequent call - never inline */
+ uint16_t calc_cksum_nonfixed() __attribute__ ((noinline)) {
+ const uint16_t *ipv4 = (const uint16_t *)this;
+
+ int sum = 0;
+ int hlen = getHeaderLength();
+
+ for (int i = 0; i < (hlen / 2); i++) {
+ sum += ipv4[i];
+ }
+
+ sum = (sum & 0xffff) + (sum >> 16);
+
+ return (uint16_t)(~sum);
+ }
+
////////////////////////////////////////////////////////////////////////////////////////
// Common Header Interface
////////////////////////////////////////////////////////////////////////////////////////