xref: /aosp_15_r20/external/cronet/net/socket/udp_socket_perftest.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2015 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #include "base/functional/bind.h"
6*6777b538SAndroid Build Coastguard Worker #include "base/memory/weak_ptr.h"
7*6777b538SAndroid Build Coastguard Worker #include "base/run_loop.h"
8*6777b538SAndroid Build Coastguard Worker #include "base/test/task_environment.h"
9*6777b538SAndroid Build Coastguard Worker #include "base/timer/elapsed_timer.h"
10*6777b538SAndroid Build Coastguard Worker #include "net/base/io_buffer.h"
11*6777b538SAndroid Build Coastguard Worker #include "net/base/ip_endpoint.h"
12*6777b538SAndroid Build Coastguard Worker #include "net/base/net_errors.h"
13*6777b538SAndroid Build Coastguard Worker #include "net/base/test_completion_callback.h"
14*6777b538SAndroid Build Coastguard Worker #include "net/log/net_log_source.h"
15*6777b538SAndroid Build Coastguard Worker #include "net/socket/udp_client_socket.h"
16*6777b538SAndroid Build Coastguard Worker #include "net/socket/udp_server_socket.h"
17*6777b538SAndroid Build Coastguard Worker #include "net/socket/udp_socket.h"
18*6777b538SAndroid Build Coastguard Worker #include "net/test/gtest_util.h"
19*6777b538SAndroid Build Coastguard Worker #include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
20*6777b538SAndroid Build Coastguard Worker #include "testing/gmock/include/gmock/gmock.h"
21*6777b538SAndroid Build Coastguard Worker #include "testing/gtest/include/gtest/gtest.h"
22*6777b538SAndroid Build Coastguard Worker #include "testing/perf/perf_result_reporter.h"
23*6777b538SAndroid Build Coastguard Worker #include "testing/platform_test.h"
24*6777b538SAndroid Build Coastguard Worker 
25*6777b538SAndroid Build Coastguard Worker using net::test::IsOk;
26*6777b538SAndroid Build Coastguard Worker 
27*6777b538SAndroid Build Coastguard Worker namespace net {
28*6777b538SAndroid Build Coastguard Worker 
29*6777b538SAndroid Build Coastguard Worker namespace {
30*6777b538SAndroid Build Coastguard Worker 
31*6777b538SAndroid Build Coastguard Worker static constexpr char kMetricPrefixUDPSocket[] = "UDPSocketWrite.";
32*6777b538SAndroid Build Coastguard Worker static constexpr char kMetricElapsedTimeMs[] = "elapsed_time";
33*6777b538SAndroid Build Coastguard Worker static constexpr char kMetricWriteSpeedBytesPerSecond[] = "write_speed";
34*6777b538SAndroid Build Coastguard Worker 
SetUpUDPSocketReporter(const std::string & story)35*6777b538SAndroid Build Coastguard Worker perf_test::PerfResultReporter SetUpUDPSocketReporter(const std::string& story) {
36*6777b538SAndroid Build Coastguard Worker   perf_test::PerfResultReporter reporter(kMetricPrefixUDPSocket, story);
37*6777b538SAndroid Build Coastguard Worker   reporter.RegisterImportantMetric(kMetricElapsedTimeMs, "ms");
38*6777b538SAndroid Build Coastguard Worker   reporter.RegisterImportantMetric(kMetricWriteSpeedBytesPerSecond,
39*6777b538SAndroid Build Coastguard Worker                                    "bytesPerSecond_biggerIsBetter");
40*6777b538SAndroid Build Coastguard Worker   return reporter;
41*6777b538SAndroid Build Coastguard Worker }
42*6777b538SAndroid Build Coastguard Worker 
43*6777b538SAndroid Build Coastguard Worker class UDPSocketPerfTest : public PlatformTest {
44*6777b538SAndroid Build Coastguard Worker  public:
UDPSocketPerfTest()45*6777b538SAndroid Build Coastguard Worker   UDPSocketPerfTest()
46*6777b538SAndroid Build Coastguard Worker       : buffer_(base::MakeRefCounted<IOBufferWithSize>(kPacketSize)) {}
47*6777b538SAndroid Build Coastguard Worker 
DoneWritePacketsToSocket(UDPClientSocket * socket,int num_of_packets,base::OnceClosure * done_callback,int error)48*6777b538SAndroid Build Coastguard Worker   void DoneWritePacketsToSocket(UDPClientSocket* socket,
49*6777b538SAndroid Build Coastguard Worker                                 int num_of_packets,
50*6777b538SAndroid Build Coastguard Worker                                 base::OnceClosure* done_callback,
51*6777b538SAndroid Build Coastguard Worker                                 int error) {
52*6777b538SAndroid Build Coastguard Worker     WritePacketsToSocket(socket, num_of_packets, done_callback);
53*6777b538SAndroid Build Coastguard Worker   }
54*6777b538SAndroid Build Coastguard Worker 
55*6777b538SAndroid Build Coastguard Worker   // Send |num_of_packets| to |socket|. Invoke |done_callback| when done.
56*6777b538SAndroid Build Coastguard Worker   void WritePacketsToSocket(UDPClientSocket* socket,
57*6777b538SAndroid Build Coastguard Worker                             int num_of_packets,
58*6777b538SAndroid Build Coastguard Worker                             base::OnceClosure* done_callback);
59*6777b538SAndroid Build Coastguard Worker 
60*6777b538SAndroid Build Coastguard Worker   // Use non-blocking IO if |use_nonblocking_io| is true. This variable only
61*6777b538SAndroid Build Coastguard Worker   // has effect on Windows.
62*6777b538SAndroid Build Coastguard Worker   void WriteBenchmark(bool use_nonblocking_io);
63*6777b538SAndroid Build Coastguard Worker 
64*6777b538SAndroid Build Coastguard Worker  protected:
65*6777b538SAndroid Build Coastguard Worker   static const int kPacketSize = 1024;
66*6777b538SAndroid Build Coastguard Worker   scoped_refptr<IOBufferWithSize> buffer_;
67*6777b538SAndroid Build Coastguard Worker   base::WeakPtrFactory<UDPSocketPerfTest> weak_factory_{this};
68*6777b538SAndroid Build Coastguard Worker };
69*6777b538SAndroid Build Coastguard Worker 
70*6777b538SAndroid Build Coastguard Worker const int UDPSocketPerfTest::kPacketSize;
71*6777b538SAndroid Build Coastguard Worker 
72*6777b538SAndroid Build Coastguard Worker // Creates and address from an ip/port and returns it in |address|.
CreateUDPAddress(const std::string & ip_str,uint16_t port,IPEndPoint * address)73*6777b538SAndroid Build Coastguard Worker void CreateUDPAddress(const std::string& ip_str,
74*6777b538SAndroid Build Coastguard Worker                       uint16_t port,
75*6777b538SAndroid Build Coastguard Worker                       IPEndPoint* address) {
76*6777b538SAndroid Build Coastguard Worker   IPAddress ip_address;
77*6777b538SAndroid Build Coastguard Worker   if (!ip_address.AssignFromIPLiteral(ip_str))
78*6777b538SAndroid Build Coastguard Worker     return;
79*6777b538SAndroid Build Coastguard Worker   *address = IPEndPoint(ip_address, port);
80*6777b538SAndroid Build Coastguard Worker }
81*6777b538SAndroid Build Coastguard Worker 
WritePacketsToSocket(UDPClientSocket * socket,int num_of_packets,base::OnceClosure * done_callback)82*6777b538SAndroid Build Coastguard Worker void UDPSocketPerfTest::WritePacketsToSocket(UDPClientSocket* socket,
83*6777b538SAndroid Build Coastguard Worker                                              int num_of_packets,
84*6777b538SAndroid Build Coastguard Worker                                              base::OnceClosure* done_callback) {
85*6777b538SAndroid Build Coastguard Worker   scoped_refptr<IOBufferWithSize> io_buffer =
86*6777b538SAndroid Build Coastguard Worker       base::MakeRefCounted<IOBufferWithSize>(kPacketSize);
87*6777b538SAndroid Build Coastguard Worker   memset(io_buffer->data(), 'G', kPacketSize);
88*6777b538SAndroid Build Coastguard Worker 
89*6777b538SAndroid Build Coastguard Worker   while (num_of_packets) {
90*6777b538SAndroid Build Coastguard Worker     int rv = socket->Write(
91*6777b538SAndroid Build Coastguard Worker         io_buffer.get(), io_buffer->size(),
92*6777b538SAndroid Build Coastguard Worker         base::BindOnce(&UDPSocketPerfTest::DoneWritePacketsToSocket,
93*6777b538SAndroid Build Coastguard Worker                        weak_factory_.GetWeakPtr(), socket, num_of_packets - 1,
94*6777b538SAndroid Build Coastguard Worker                        done_callback),
95*6777b538SAndroid Build Coastguard Worker         TRAFFIC_ANNOTATION_FOR_TESTS);
96*6777b538SAndroid Build Coastguard Worker     if (rv == ERR_IO_PENDING)
97*6777b538SAndroid Build Coastguard Worker       break;
98*6777b538SAndroid Build Coastguard Worker     --num_of_packets;
99*6777b538SAndroid Build Coastguard Worker   }
100*6777b538SAndroid Build Coastguard Worker   if (!num_of_packets) {
101*6777b538SAndroid Build Coastguard Worker     std::move(*done_callback).Run();
102*6777b538SAndroid Build Coastguard Worker     return;
103*6777b538SAndroid Build Coastguard Worker   }
104*6777b538SAndroid Build Coastguard Worker }
105*6777b538SAndroid Build Coastguard Worker 
WriteBenchmark(bool use_nonblocking_io)106*6777b538SAndroid Build Coastguard Worker void UDPSocketPerfTest::WriteBenchmark(bool use_nonblocking_io) {
107*6777b538SAndroid Build Coastguard Worker   base::ElapsedTimer total_elapsed_timer;
108*6777b538SAndroid Build Coastguard Worker   base::test::SingleThreadTaskEnvironment task_environment(
109*6777b538SAndroid Build Coastguard Worker       base::test::SingleThreadTaskEnvironment::MainThreadType::IO);
110*6777b538SAndroid Build Coastguard Worker   const uint16_t kPort = 9999;
111*6777b538SAndroid Build Coastguard Worker 
112*6777b538SAndroid Build Coastguard Worker   // Setup the server to listen.
113*6777b538SAndroid Build Coastguard Worker   IPEndPoint bind_address;
114*6777b538SAndroid Build Coastguard Worker   CreateUDPAddress("127.0.0.1", kPort, &bind_address);
115*6777b538SAndroid Build Coastguard Worker   auto server = std::make_unique<UDPServerSocket>(nullptr, NetLogSource());
116*6777b538SAndroid Build Coastguard Worker   if (use_nonblocking_io)
117*6777b538SAndroid Build Coastguard Worker     server->UseNonBlockingIO();
118*6777b538SAndroid Build Coastguard Worker   int rv = server->Listen(bind_address);
119*6777b538SAndroid Build Coastguard Worker   ASSERT_THAT(rv, IsOk());
120*6777b538SAndroid Build Coastguard Worker 
121*6777b538SAndroid Build Coastguard Worker   // Setup the client.
122*6777b538SAndroid Build Coastguard Worker   IPEndPoint server_address;
123*6777b538SAndroid Build Coastguard Worker   CreateUDPAddress("127.0.0.1", kPort, &server_address);
124*6777b538SAndroid Build Coastguard Worker   auto client = std::make_unique<UDPClientSocket>(DatagramSocket::DEFAULT_BIND,
125*6777b538SAndroid Build Coastguard Worker                                                   nullptr, NetLogSource());
126*6777b538SAndroid Build Coastguard Worker   if (use_nonblocking_io)
127*6777b538SAndroid Build Coastguard Worker     client->UseNonBlockingIO();
128*6777b538SAndroid Build Coastguard Worker   rv = client->Connect(server_address);
129*6777b538SAndroid Build Coastguard Worker   EXPECT_THAT(rv, IsOk());
130*6777b538SAndroid Build Coastguard Worker 
131*6777b538SAndroid Build Coastguard Worker   base::RunLoop run_loop;
132*6777b538SAndroid Build Coastguard Worker   base::OnceClosure done_callback = run_loop.QuitClosure();
133*6777b538SAndroid Build Coastguard Worker   base::ElapsedTimer write_elapsed_timer;
134*6777b538SAndroid Build Coastguard Worker   int packets = 100000;
135*6777b538SAndroid Build Coastguard Worker   client->SetSendBufferSize(1024);
136*6777b538SAndroid Build Coastguard Worker   WritePacketsToSocket(client.get(), packets, &done_callback);
137*6777b538SAndroid Build Coastguard Worker   run_loop.Run();
138*6777b538SAndroid Build Coastguard Worker 
139*6777b538SAndroid Build Coastguard Worker   double write_elapsed = write_elapsed_timer.Elapsed().InSecondsF();
140*6777b538SAndroid Build Coastguard Worker   double total_elapsed = total_elapsed_timer.Elapsed().InMillisecondsF();
141*6777b538SAndroid Build Coastguard Worker   auto reporter =
142*6777b538SAndroid Build Coastguard Worker       SetUpUDPSocketReporter(use_nonblocking_io ? "nonblocking" : "blocking");
143*6777b538SAndroid Build Coastguard Worker   reporter.AddResult(kMetricElapsedTimeMs, total_elapsed);
144*6777b538SAndroid Build Coastguard Worker   reporter.AddResult(kMetricWriteSpeedBytesPerSecond,
145*6777b538SAndroid Build Coastguard Worker                      packets * 1024 / write_elapsed);
146*6777b538SAndroid Build Coastguard Worker }
147*6777b538SAndroid Build Coastguard Worker 
TEST_F(UDPSocketPerfTest,Write)148*6777b538SAndroid Build Coastguard Worker TEST_F(UDPSocketPerfTest, Write) {
149*6777b538SAndroid Build Coastguard Worker   WriteBenchmark(false);
150*6777b538SAndroid Build Coastguard Worker }
151*6777b538SAndroid Build Coastguard Worker 
TEST_F(UDPSocketPerfTest,WriteNonBlocking)152*6777b538SAndroid Build Coastguard Worker TEST_F(UDPSocketPerfTest, WriteNonBlocking) {
153*6777b538SAndroid Build Coastguard Worker   WriteBenchmark(true);
154*6777b538SAndroid Build Coastguard Worker }
155*6777b538SAndroid Build Coastguard Worker 
156*6777b538SAndroid Build Coastguard Worker }  // namespace
157*6777b538SAndroid Build Coastguard Worker 
158*6777b538SAndroid Build Coastguard Worker }  // namespace net
159