summaryrefslogtreecommitdiff
path: root/Kernel/IPv4Socket.cpp
diff options
context:
space:
mode:
authorAndreas Kling <awesomekling@gmail.com>2019-03-13 16:05:56 +0100
committerAndreas Kling <awesomekling@gmail.com>2019-03-13 16:05:56 +0100
commit209a16bb7f2b62742a5edaddf391697073bdafd6 (patch)
tree5550fd445a17d36b2c765e91ff57307ea4938e3c /Kernel/IPv4Socket.cpp
parent4dddf949c83579df866ad95ae3cea16a0b651a0b (diff)
downloadserenity-209a16bb7f2b62742a5edaddf391697073bdafd6.zip
IPv4: Dynamically allocate the UDP source port if needed.
Diffstat (limited to 'Kernel/IPv4Socket.cpp')
-rw-r--r--Kernel/IPv4Socket.cpp37
1 files changed, 36 insertions, 1 deletions
diff --git a/Kernel/IPv4Socket.cpp b/Kernel/IPv4Socket.cpp
index 601f200414..8e46bc334c 100644
--- a/Kernel/IPv4Socket.cpp
+++ b/Kernel/IPv4Socket.cpp
@@ -10,6 +10,22 @@
#define IPV4_SOCKET_DEBUG
+Lockable<HashMap<word, IPv4Socket*>>& IPv4Socket::sockets_by_udp_port()
+{
+ static Lockable<HashMap<word, IPv4Socket*>>* s_map;
+ if (!s_map)
+ s_map = new Lockable<HashMap<word, IPv4Socket*>>;
+ return *s_map;
+}
+
+Lockable<HashMap<word, IPv4Socket*>>& IPv4Socket::sockets_by_tcp_port()
+{
+ static Lockable<HashMap<word, IPv4Socket*>>* s_map;
+ if (!s_map)
+ s_map = new Lockable<HashMap<word, IPv4Socket*>>;
+ return *s_map;
+}
+
Lockable<HashTable<IPv4Socket*>>& IPv4Socket::all_sockets()
{
static Lockable<HashTable<IPv4Socket*>>* s_table;
@@ -100,6 +116,25 @@ bool IPv4Socket::can_write(SocketRole role) const
ASSERT_NOT_REACHED();
}
+void IPv4Socket::allocate_source_port_if_needed()
+{
+ if (m_source_port)
+ return;
+ if (type() == SOCK_DGRAM) {
+ // This is not a very efficient allocation algorithm.
+ // FIXME: Replace it with a bitmap or some other fast-paced looker-upper.
+ LOCKER(sockets_by_udp_port().lock());
+ for (word port = 2000; port < 60000; ++port) {
+ auto it = sockets_by_udp_port().resource().find(port);
+ if (it == sockets_by_udp_port().resource().end()) {
+ m_source_port = port;
+ return;
+ }
+ }
+ ASSERT_NOT_REACHED();
+ }
+}
+
ssize_t IPv4Socket::sendto(const void* data, size_t data_length, int flags, const sockaddr* addr, socklen_t addr_length)
{
(void)flags;
@@ -121,7 +156,7 @@ ssize_t IPv4Socket::sendto(const void* data, size_t data_length, int flags, cons
m_destination_address = IPv4Address((const byte*)&ia.sin_addr.s_addr);
m_destination_port = ntohs(ia.sin_port);
- m_source_port = 2413;
+ allocate_source_port_if_needed();
kprintf("sendto: destination=%s:%u\n", m_destination_address.to_string().characters(), m_destination_port);