1*98e87f76SDavid Spickett //===-- PortMapTest.cpp ---------------------------------------------------===//
2*98e87f76SDavid Spickett //
3*98e87f76SDavid Spickett // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*98e87f76SDavid Spickett // See https://llvm.org/LICENSE.txt for license information.
5*98e87f76SDavid Spickett // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*98e87f76SDavid Spickett //
7*98e87f76SDavid Spickett //===----------------------------------------------------------------------===//
8*98e87f76SDavid Spickett 
9*98e87f76SDavid Spickett #include "llvm/Testing/Support/Error.h"
10*98e87f76SDavid Spickett #include "gmock/gmock.h"
11*98e87f76SDavid Spickett #include "gtest/gtest.h"
12*98e87f76SDavid Spickett 
13*98e87f76SDavid Spickett #include "Plugins/Process/gdb-remote/GDBRemoteCommunicationServerPlatform.h"
14*98e87f76SDavid Spickett 
15*98e87f76SDavid Spickett using namespace lldb_private::process_gdb_remote;
16*98e87f76SDavid Spickett 
TEST(PortMapTest,Constructors)17*98e87f76SDavid Spickett TEST(PortMapTest, Constructors) {
18*98e87f76SDavid Spickett   // Default construct to empty map
19*98e87f76SDavid Spickett   GDBRemoteCommunicationServerPlatform::PortMap p1;
20*98e87f76SDavid Spickett   ASSERT_TRUE(p1.empty());
21*98e87f76SDavid Spickett 
22*98e87f76SDavid Spickett   // Empty means no restrictions, return 0 and and bind to get a port
23*98e87f76SDavid Spickett   llvm::Expected<uint16_t> available_port = p1.GetNextAvailablePort();
24*98e87f76SDavid Spickett   ASSERT_THAT_EXPECTED(available_port, llvm::HasValue(0));
25*98e87f76SDavid Spickett 
26*98e87f76SDavid Spickett   // Adding any port makes it not empty
27*98e87f76SDavid Spickett   p1.AllowPort(1);
28*98e87f76SDavid Spickett   ASSERT_FALSE(p1.empty());
29*98e87f76SDavid Spickett 
30*98e87f76SDavid Spickett   // So we will return the added port this time
31*98e87f76SDavid Spickett   available_port = p1.GetNextAvailablePort();
32*98e87f76SDavid Spickett   ASSERT_THAT_EXPECTED(available_port, llvm::HasValue(1));
33*98e87f76SDavid Spickett 
34*98e87f76SDavid Spickett   // Construct from a range of ports
35*98e87f76SDavid Spickett   GDBRemoteCommunicationServerPlatform::PortMap p2(1, 4);
36*98e87f76SDavid Spickett   ASSERT_FALSE(p2.empty());
37*98e87f76SDavid Spickett 
38*98e87f76SDavid Spickett   // Use up all the ports
39*98e87f76SDavid Spickett   for (uint16_t expected = 1; expected < 4; ++expected) {
40*98e87f76SDavid Spickett     available_port = p2.GetNextAvailablePort();
41*98e87f76SDavid Spickett     ASSERT_THAT_EXPECTED(available_port, llvm::HasValue(expected));
42*98e87f76SDavid Spickett     p2.AssociatePortWithProcess(*available_port, 1);
43*98e87f76SDavid Spickett   }
44*98e87f76SDavid Spickett 
45*98e87f76SDavid Spickett   // Now we fail since we're not an empty port map but all ports are used
46*98e87f76SDavid Spickett   available_port = p2.GetNextAvailablePort();
47*98e87f76SDavid Spickett   ASSERT_THAT_EXPECTED(available_port, llvm::Failed());
48*98e87f76SDavid Spickett }
49*98e87f76SDavid Spickett 
TEST(PortMapTest,FreePort)50*98e87f76SDavid Spickett TEST(PortMapTest, FreePort) {
51*98e87f76SDavid Spickett   GDBRemoteCommunicationServerPlatform::PortMap p(1, 4);
52*98e87f76SDavid Spickett   // Use up all the ports
53*98e87f76SDavid Spickett   for (uint16_t port = 1; port < 4; ++port) {
54*98e87f76SDavid Spickett     p.AssociatePortWithProcess(port, 1);
55*98e87f76SDavid Spickett   }
56*98e87f76SDavid Spickett 
57*98e87f76SDavid Spickett   llvm::Expected<uint16_t> available_port = p.GetNextAvailablePort();
58*98e87f76SDavid Spickett   ASSERT_THAT_EXPECTED(available_port, llvm::Failed());
59*98e87f76SDavid Spickett 
60*98e87f76SDavid Spickett   // Can't free a port that isn't in the map
61*98e87f76SDavid Spickett   ASSERT_FALSE(p.FreePort(0));
62*98e87f76SDavid Spickett   ASSERT_FALSE(p.FreePort(4));
63*98e87f76SDavid Spickett 
64*98e87f76SDavid Spickett   // After freeing a port it becomes available
65*98e87f76SDavid Spickett   ASSERT_TRUE(p.FreePort(2));
66*98e87f76SDavid Spickett   available_port = p.GetNextAvailablePort();
67*98e87f76SDavid Spickett   ASSERT_THAT_EXPECTED(available_port, llvm::HasValue(2));
68*98e87f76SDavid Spickett }
69*98e87f76SDavid Spickett 
TEST(PortMapTest,FreePortForProcess)70*98e87f76SDavid Spickett TEST(PortMapTest, FreePortForProcess) {
71*98e87f76SDavid Spickett   GDBRemoteCommunicationServerPlatform::PortMap p;
72*98e87f76SDavid Spickett   p.AllowPort(1);
73*98e87f76SDavid Spickett   p.AllowPort(2);
74*98e87f76SDavid Spickett   ASSERT_TRUE(p.AssociatePortWithProcess(1, 11));
75*98e87f76SDavid Spickett   ASSERT_TRUE(p.AssociatePortWithProcess(2, 22));
76*98e87f76SDavid Spickett 
77*98e87f76SDavid Spickett   // All ports have been used
78*98e87f76SDavid Spickett   llvm::Expected<uint16_t> available_port = p.GetNextAvailablePort();
79*98e87f76SDavid Spickett   ASSERT_THAT_EXPECTED(available_port, llvm::Failed());
80*98e87f76SDavid Spickett 
81*98e87f76SDavid Spickett   // Can't free a port for a process that doesn't have any
82*98e87f76SDavid Spickett   ASSERT_FALSE(p.FreePortForProcess(33));
83*98e87f76SDavid Spickett 
84*98e87f76SDavid Spickett   // You can move a used port to a new pid
85*98e87f76SDavid Spickett   ASSERT_TRUE(p.AssociatePortWithProcess(1, 99));
86*98e87f76SDavid Spickett 
87*98e87f76SDavid Spickett   ASSERT_TRUE(p.FreePortForProcess(22));
88*98e87f76SDavid Spickett   available_port = p.GetNextAvailablePort();
89*98e87f76SDavid Spickett   ASSERT_THAT_EXPECTED(available_port, llvm::Succeeded());
90*98e87f76SDavid Spickett   ASSERT_EQ(2, *available_port);
91*98e87f76SDavid Spickett 
92*98e87f76SDavid Spickett   // proces 22 no longer has a port
93*98e87f76SDavid Spickett   ASSERT_FALSE(p.FreePortForProcess(22));
94*98e87f76SDavid Spickett }
95*98e87f76SDavid Spickett 
TEST(PortMapTest,AllowPort)96*98e87f76SDavid Spickett TEST(PortMapTest, AllowPort) {
97*98e87f76SDavid Spickett   GDBRemoteCommunicationServerPlatform::PortMap p;
98*98e87f76SDavid Spickett 
99*98e87f76SDavid Spickett   // Allow port 1 and tie it to process 11
100*98e87f76SDavid Spickett   p.AllowPort(1);
101*98e87f76SDavid Spickett   ASSERT_TRUE(p.AssociatePortWithProcess(1, 11));
102*98e87f76SDavid Spickett 
103*98e87f76SDavid Spickett   // Allowing it a second time shouldn't change existing mapping
104*98e87f76SDavid Spickett   p.AllowPort(1);
105*98e87f76SDavid Spickett   llvm::Expected<uint16_t> available_port = p.GetNextAvailablePort();
106*98e87f76SDavid Spickett   ASSERT_THAT_EXPECTED(available_port, llvm::Failed());
107*98e87f76SDavid Spickett 
108*98e87f76SDavid Spickett   // A new port is marked as free when allowed
109*98e87f76SDavid Spickett   p.AllowPort(2);
110*98e87f76SDavid Spickett   available_port = p.GetNextAvailablePort();
111*98e87f76SDavid Spickett   ASSERT_THAT_EXPECTED(available_port, llvm::HasValue(2));
112*98e87f76SDavid Spickett 
113*98e87f76SDavid Spickett   // 11 should still be tied to port 1
114*98e87f76SDavid Spickett   ASSERT_TRUE(p.FreePortForProcess(11));
115*98e87f76SDavid Spickett }
116