diff --git a/src/network/model/packet-tag-list.cc b/src/network/model/packet-tag-list.cc --- a/src/network/model/packet-tag-list.cc +++ b/src/network/model/packet-tag-list.cc @@ -34,6 +34,21 @@ NS_LOG_COMPONENT_DEFINE ("PacketTagList"); +PacketTagList::TagData::TagData (size_t dataSize) + : size (dataSize) +{ + NS_ASSERT_MSG (dataSize < std::numeric_limits::max (), + "Requested TagData size " << dataSize + << " exceeds maximum " + << std::numeric_limits::max () ); + data = (uint8_t *) std::malloc (size); +} + +PacketTagList::TagData::~TagData (void) +{ + free (data); +} + bool PacketTagList::COWTraverse (Tag & tag, PacketTagList::COWWriter Writer) { @@ -133,10 +148,10 @@ NS_ASSERT (cur != 0); NS_ASSERT (cur->count > 1); cur->count--; // unmerge cur - struct TagData * copy = new struct TagData (); + struct TagData * copy = new struct TagData (cur->size); copy->tid = cur->tid; copy->count = 1; - memcpy (copy->data, cur->data, TagData::MAX_SIZE); + memcpy (copy->data, cur->data, copy->size); copy->next = cur->next; // merge into tail copy->next->count++; // mark new merge *prevNext = copy; // point prior list at copy @@ -170,8 +185,7 @@ // found tid bool found = true; - tag.Deserialize (TagBuffer (cur->data, - cur->data + TagData::MAX_SIZE)); + tag.Deserialize (TagBuffer (cur->data, cur->data + cur->size)); *prevNext = cur->next; // link around cur if (preMerge) @@ -217,19 +231,17 @@ if (preMerge) { // found tid before first merge, so just rewrite - tag.Serialize (TagBuffer (cur->data, - cur->data + tag.GetSerializedSize ())); + tag.Serialize (TagBuffer (cur->data, cur->data + cur->size)); } else { // cur is always a merge at this point // need to copy, replace, and link past cur cur->count--; // unmerge cur - struct TagData * copy = new struct TagData (); + struct TagData * copy = new struct TagData (tag.GetSerializedSize ()); copy->tid = tag.GetInstanceTypeId (); copy->count = 1; - tag.Serialize (TagBuffer (copy->data, - copy->data + tag.GetSerializedSize ())); + tag.Serialize (TagBuffer (copy->data, copy->data + copy->size)); copy->next = cur->next; // merge into tail if (copy->next != 0) { @@ -249,13 +261,12 @@ { NS_ASSERT_MSG (cur->tid != tag.GetInstanceTypeId (), "Error: cannot add the same kind of tag twice."); } - struct TagData * head = new struct TagData (); + struct TagData * head = new struct TagData (tag.GetSerializedSize ()); head->count = 1; head->next = 0; head->tid = tag.GetInstanceTypeId (); head->next = m_next; - NS_ASSERT (tag.GetSerializedSize () <= TagData::MAX_SIZE); - tag.Serialize (TagBuffer (head->data, head->data + tag.GetSerializedSize ())); + tag.Serialize (TagBuffer (head->data, head->data + head->size)); const_cast (this)->m_next = head; } @@ -270,7 +281,7 @@ if (cur->tid == tid) { /* found tag */ - tag.Deserialize (TagBuffer (cur->data, cur->data + TagData::MAX_SIZE)); + tag.Deserialize (TagBuffer (cur->data, cur->data + cur->size)); return true; } } diff --git a/src/network/model/packet-tag-list.h b/src/network/model/packet-tag-list.h --- a/src/network/model/packet-tag-list.h +++ b/src/network/model/packet-tag-list.h @@ -134,37 +134,22 @@ * * See PacketTagList for a discussion of the data structure. * - * See TagData::TagData_e for a discussion of the size limit on - * tag serialization. + * \internal + * Unfortunately this has to be public, because + * PacketTagIterator::Item::GetTag() needs the data and size values. + * The Item nested class can't be forward declared, so friending isn't + * possible. */ struct TagData { - /** - * \brief Packet Tag maximum size - * - * The maximum size (in bytes) of a Tag is stored - * in this constant. - * - * \internal - * Ideally, TagData would be 32 bytes in size, so they require - * no padding on 64-bit architectures. (The architecture - * affects the size because of the \c #next pointer.) - * This would leave 18 bytes for \c #data. However, - * ns3:Ipv6PacketInfoTag needs 19 bytes! The current - * implementation allows 20 bytes, which gives TagData - * a size of 30 bytes on 32-byte machines (which gets - * padded with 2 bytes), and 34 bytes on 64-bit machines, which - * gets padded to 40 bytes. - */ - enum TagData_e - { - MAX_SIZE = 21 /**< Size of serialization buffer #data */ - }; - - uint8_t data[MAX_SIZE]; /**< Serialization buffer */ - struct TagData * next; /**< Pointer to next in list */ - TypeId tid; /**< Type of the tag serialized into #data */ - uint32_t count; /**< Number of incoming links */ + TagData (size_t dataSize); /**< Construct with dataSize bytes */ + ~TagData (void); /**< Destructor */ + + struct TagData * next; /**< Pointer to next in list */ + uint32_t count; /**< Number of incoming links */ + TypeId tid; /**< Type of the tag serialized into #data */ + uint32_t size; /**< Size of the \c data buffer */ + uint8_t * data; /**< Serialization buffer */ }; /* struct TagData */ /** @@ -272,7 +257,7 @@ * pointing to \pname{cur}. * \returns True, since tag will definitely be removed. */ - bool RemoveWriter (Tag & tag, bool preMerge, + bool RemoveWriter (Tag & tag, bool preMerge, struct TagData * cur, struct TagData ** prevNext); /** * Copy-on-write implementing Replace @@ -285,7 +270,8 @@ * pointing to \pname{cur}. * \returns True, since tag value will definitely be replaced. */ - bool ReplaceWriter (Tag & tag, bool preMerge, struct TagData * cur, struct TagData ** prevNext); + bool ReplaceWriter (Tag & tag, bool preMerge, + struct TagData * cur, struct TagData ** prevNext); /** * Pointer to first \ref TagData on the list diff --git a/src/network/model/packet.cc b/src/network/model/packet.cc --- a/src/network/model/packet.cc +++ b/src/network/model/packet.cc @@ -113,8 +113,7 @@ { NS_ASSERT (tag.GetInstanceTypeId () == m_data->tid); tag.Deserialize (TagBuffer ((uint8_t*)m_data->data, - (uint8_t*)m_data->data - + PacketTagList::TagData::MAX_SIZE)); + (uint8_t*)m_data->data + m_data->size)); } diff --git a/src/network/utils/packet-socket.cc b/src/network/utils/packet-socket.cc --- a/src/network/utils/packet-socket.cc +++ b/src/network/utils/packet-socket.cc @@ -643,7 +643,6 @@ DeviceNameTag::GetSerializedSize (void) const { uint32_t s = 1 + m_deviceName.size(); // +1 for name length field - s = std::min (s, (uint32_t)PacketTagList::TagData::MAX_SIZE); return s; } void @@ -652,8 +651,6 @@ const char *n = m_deviceName.c_str(); uint8_t l = (uint8_t) m_deviceName.size (); - l = std::min ((uint32_t)l, (uint32_t)PacketTagList::TagData::MAX_SIZE - 1); - i.WriteU8 (l); i.Write ( (uint8_t*) n , (uint32_t) l); }