mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-24 18:32:28 -05:00
Kernel: Avoid allocations when receiving network packets
This avoids two allocations when receiving network packets. One for inserting a PacketWithTimestamp into m_packet_queue and another one when inserting buffers into the list of unused packet buffers. With this fixed the only allocations in NetworkTask happen when initially allocating the PacketWithTimestamp structs and when switching contexts.
This commit is contained in:
parent
b66f91e7d3
commit
093818de62
2 changed files with 37 additions and 23 deletions
|
@ -153,27 +153,34 @@ void NetworkAdapter::did_receive(ReadonlyBytes payload)
|
||||||
m_packets_in++;
|
m_packets_in++;
|
||||||
m_bytes_in += payload.size();
|
m_bytes_in += payload.size();
|
||||||
|
|
||||||
Optional<KBuffer> buffer;
|
|
||||||
|
|
||||||
if (m_packet_queue_size == max_packet_buffers) {
|
if (m_packet_queue_size == max_packet_buffers) {
|
||||||
// FIXME: Keep track of the number of dropped packets
|
// FIXME: Keep track of the number of dropped packets
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (m_unused_packet_buffers.is_empty()) {
|
RefPtr<PacketWithTimestamp> packet;
|
||||||
buffer = KBuffer::copy(payload.data(), payload.size());
|
|
||||||
|
if (m_unused_packets.is_empty()) {
|
||||||
|
auto buffer = KBuffer::copy(payload.data(), payload.size());
|
||||||
|
packet = adopt_ref_if_nonnull(new PacketWithTimestamp { move(buffer), kgettimeofday() });
|
||||||
} else {
|
} else {
|
||||||
buffer = m_unused_packet_buffers.take_first();
|
packet = m_unused_packets.take_first();
|
||||||
--m_unused_packet_buffers_count;
|
if (payload.size() <= packet->buffer.capacity()) {
|
||||||
if (payload.size() <= buffer.value().capacity()) {
|
memcpy(packet->buffer.data(), payload.data(), payload.size());
|
||||||
memcpy(buffer.value().data(), payload.data(), payload.size());
|
packet->buffer.set_size(payload.size());
|
||||||
buffer.value().set_size(payload.size());
|
packet->timestamp = kgettimeofday();
|
||||||
} else {
|
} else {
|
||||||
buffer = KBuffer::copy(payload.data(), payload.size());
|
auto buffer = KBuffer::copy(payload.data(), payload.size());
|
||||||
|
packet = adopt_ref_if_nonnull(new PacketWithTimestamp { move(buffer), kgettimeofday() });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
m_packet_queue.append({ buffer.value(), kgettimeofday() });
|
if (!packet) {
|
||||||
|
dbgln("Discarding packet because we're out of memory");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_packet_queue.append(*packet);
|
||||||
m_packet_queue_size++;
|
m_packet_queue_size++;
|
||||||
|
|
||||||
if (on_receive)
|
if (on_receive)
|
||||||
|
@ -187,13 +194,12 @@ size_t NetworkAdapter::dequeue_packet(u8* buffer, size_t buffer_size, Time& pack
|
||||||
return 0;
|
return 0;
|
||||||
auto packet_with_timestamp = m_packet_queue.take_first();
|
auto packet_with_timestamp = m_packet_queue.take_first();
|
||||||
m_packet_queue_size--;
|
m_packet_queue_size--;
|
||||||
packet_timestamp = packet_with_timestamp.timestamp;
|
packet_timestamp = packet_with_timestamp->timestamp;
|
||||||
auto packet = move(packet_with_timestamp.packet);
|
auto& packet_buffer = packet_with_timestamp->buffer;
|
||||||
size_t packet_size = packet.size();
|
size_t packet_size = packet_buffer.size();
|
||||||
VERIFY(packet_size <= buffer_size);
|
VERIFY(packet_size <= buffer_size);
|
||||||
memcpy(buffer, packet.data(), packet_size);
|
memcpy(buffer, packet_buffer.data(), packet_size);
|
||||||
m_unused_packet_buffers.append(packet);
|
m_unused_packets.append(*packet_with_timestamp);
|
||||||
++m_unused_packet_buffers_count;
|
|
||||||
return packet_size;
|
return packet_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,8 +8,8 @@
|
||||||
|
|
||||||
#include <AK/ByteBuffer.h>
|
#include <AK/ByteBuffer.h>
|
||||||
#include <AK/Function.h>
|
#include <AK/Function.h>
|
||||||
|
#include <AK/IntrusiveList.h>
|
||||||
#include <AK/MACAddress.h>
|
#include <AK/MACAddress.h>
|
||||||
#include <AK/SinglyLinkedList.h>
|
|
||||||
#include <AK/Types.h>
|
#include <AK/Types.h>
|
||||||
#include <AK/WeakPtr.h>
|
#include <AK/WeakPtr.h>
|
||||||
#include <AK/Weakable.h>
|
#include <AK/Weakable.h>
|
||||||
|
@ -86,18 +86,26 @@ private:
|
||||||
IPv4Address m_ipv4_netmask;
|
IPv4Address m_ipv4_netmask;
|
||||||
IPv4Address m_ipv4_gateway;
|
IPv4Address m_ipv4_gateway;
|
||||||
|
|
||||||
struct PacketWithTimestamp {
|
struct PacketWithTimestamp : public RefCounted<PacketWithTimestamp> {
|
||||||
KBuffer packet;
|
PacketWithTimestamp(KBuffer buffer, Time timestamp)
|
||||||
|
: buffer(move(buffer))
|
||||||
|
, timestamp(timestamp)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
KBuffer buffer;
|
||||||
Time timestamp;
|
Time timestamp;
|
||||||
|
IntrusiveListNode<PacketWithTimestamp, RefPtr<PacketWithTimestamp>> packet_node;
|
||||||
};
|
};
|
||||||
|
|
||||||
// FIXME: Make this configurable
|
// FIXME: Make this configurable
|
||||||
static constexpr size_t max_packet_buffers = 1024;
|
static constexpr size_t max_packet_buffers = 1024;
|
||||||
|
|
||||||
SinglyLinkedList<PacketWithTimestamp> m_packet_queue;
|
using PacketList = IntrusiveList<PacketWithTimestamp, RefPtr<PacketWithTimestamp>, &PacketWithTimestamp::packet_node>;
|
||||||
|
|
||||||
|
PacketList m_packet_queue;
|
||||||
size_t m_packet_queue_size { 0 };
|
size_t m_packet_queue_size { 0 };
|
||||||
SinglyLinkedList<KBuffer> m_unused_packet_buffers;
|
PacketList m_unused_packets;
|
||||||
size_t m_unused_packet_buffers_count { 0 };
|
|
||||||
String m_name;
|
String m_name;
|
||||||
u32 m_packets_in { 0 };
|
u32 m_packets_in { 0 };
|
||||||
u32 m_bytes_in { 0 };
|
u32 m_bytes_in { 0 };
|
||||||
|
|
Loading…
Add table
Reference in a new issue