blob: 3c4800a36156a37727f4b26172523f373ebdffd5 [file] [log] [blame]
Aron Virginas-Tardfa14772019-09-24 18:24:47 +01001//
Jim Flynnbbfe6032020-07-20 16:57:44 +01002// Copyright © 2019 Arm Ltd and Contributors. All rights reserved.
Aron Virginas-Tardfa14772019-09-24 18:24:47 +01003// SPDX-License-Identifier: MIT
4//
5
6#include "../ProfilingConnectionDumpToFileDecorator.hpp"
Rob Hughes9542f902021-07-14 09:48:54 +01007#include <armnnUtils/Filesystem.hpp>
Keith Davisb10e0812019-10-17 09:52:50 +01008#include <Runtime.hpp>
Jan Eilers8eb25602020-03-09 12:13:48 +00009#include <armnn/utility/IgnoreUnused.hpp>
Matthew Sloyan371b70e2020-09-11 10:14:57 +010010#include <armnn/utility/NumericCast.hpp>
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010011
12#include <fstream>
13#include <sstream>
14
Sadik Armagan1625efc2021-06-10 18:24:34 +010015#include <doctest/doctest.h>
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010016
Cathal Corbett5aa9fd72022-02-25 15:33:28 +000017using namespace arm::pipe;
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010018
19namespace
20{
21
22const std::vector<char> g_Data = { 'd', 'u', 'm', 'm', 'y' };
Matthew Sloyan371b70e2020-09-11 10:14:57 +010023const uint32_t g_DataLength = armnn::numeric_cast<uint32_t>(g_Data.size());
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010024const unsigned char* g_DataPtr = reinterpret_cast<const unsigned char*>(g_Data.data());
25
26class DummyProfilingConnection : public IProfilingConnection
27{
28public:
29 DummyProfilingConnection()
30 : m_Open(true)
Matteo Martincigh67ef2a52019-10-10 13:29:02 +010031 , m_PacketData(std::make_unique<unsigned char[]>(g_DataLength))
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010032 {
33 // populate packet data and construct packet
34 std::memcpy(m_PacketData.get(), g_DataPtr, g_DataLength);
Cathal Corbett5aa9fd72022-02-25 15:33:28 +000035 m_Packet = std::make_unique<Packet>(0u, g_DataLength, m_PacketData);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010036 }
37
38 ~DummyProfilingConnection() = default;
39
Matteo Martincigh54fb9572019-10-02 12:50:57 +010040 bool IsOpen() const override
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010041 {
42 return m_Open;
43 }
44
45 void Close() override
46 {
47 m_Open = false;
48 }
49
50 bool WritePacket(const unsigned char* buffer, uint32_t length) override
51 {
Jan Eilers8eb25602020-03-09 12:13:48 +000052 armnn::IgnoreUnused(buffer);
53 armnn::IgnoreUnused(length);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010054 return true;
55 }
56
Cathal Corbett5aa9fd72022-02-25 15:33:28 +000057 Packet ReadPacket(uint32_t timeout) override
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010058 {
Jan Eilers8eb25602020-03-09 12:13:48 +000059 armnn::IgnoreUnused(timeout);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010060 return std::move(*m_Packet);
61 }
62
63private:
Matteo Martincigh67ef2a52019-10-10 13:29:02 +010064 bool m_Open;
65 std::unique_ptr<unsigned char[]> m_PacketData;
Cathal Corbett5aa9fd72022-02-25 15:33:28 +000066 std::unique_ptr<Packet> m_Packet;
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010067};
68
69std::vector<char> ReadDumpFile(const std::string& dumpFileName)
70{
71 std::ifstream input(dumpFileName, std::ios::binary);
72 return std::vector<char>(std::istreambuf_iterator<char>(input), {});
73}
74
75} // anonymous namespace
76
Sadik Armagan1625efc2021-06-10 18:24:34 +010077TEST_SUITE("ProfilingConnectionDumpToFileDecoratorTests")
78{
79TEST_CASE("DumpIncomingInvalidFile")
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010080{
Jim Flynn4c9ed1d2022-01-23 23:57:20 +000081 ProfilingOptions options;
Keith Davisb10e0812019-10-17 09:52:50 +010082 options.m_IncomingCaptureFile = "/";
83 options.m_OutgoingCaptureFile = "";
84 ProfilingConnectionDumpToFileDecorator decorator(std::make_unique<DummyProfilingConnection>(), options, false);
Sadik Armagan1625efc2021-06-10 18:24:34 +010085 CHECK_THROWS_AS(decorator.ReadPacket(0), armnn::RuntimeException);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010086}
87
Sadik Armagan1625efc2021-06-10 18:24:34 +010088TEST_CASE("DumpIncomingInvalidFileIgnoreErrors")
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010089{
Jim Flynn4c9ed1d2022-01-23 23:57:20 +000090 ProfilingOptions options;
Keith Davisb10e0812019-10-17 09:52:50 +010091 options.m_IncomingCaptureFile = "/";
92 options.m_OutgoingCaptureFile = "";
93 ProfilingConnectionDumpToFileDecorator decorator(std::make_unique<DummyProfilingConnection>(), options, true);
Sadik Armagan1625efc2021-06-10 18:24:34 +010094 CHECK_NOTHROW(decorator.ReadPacket(0));
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010095}
96
Sadik Armagan1625efc2021-06-10 18:24:34 +010097TEST_CASE("DumpIncomingValidFile")
Aron Virginas-Tardfa14772019-09-24 18:24:47 +010098{
Francis Murtagh532a29d2020-06-29 11:50:01 +010099 fs::path fileName = armnnUtils::Filesystem::NamedTempFile("Armnn-DumpIncomingValidFileTest-TempFile");
Aron Virginas-Tar8bf442e2019-11-07 18:41:40 +0000100
Jim Flynn4c9ed1d2022-01-23 23:57:20 +0000101 ProfilingOptions options;
Rob Hughes270233f2019-11-13 11:53:48 +0000102 options.m_IncomingCaptureFile = fileName.string();
Keith Davisb10e0812019-10-17 09:52:50 +0100103 options.m_OutgoingCaptureFile = "";
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100104
Keith Davisb10e0812019-10-17 09:52:50 +0100105 ProfilingConnectionDumpToFileDecorator decorator(std::make_unique<DummyProfilingConnection>(), options, false);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100106
107 // NOTE: unique_ptr is needed here because operator=() is deleted for Packet
Cathal Corbett5aa9fd72022-02-25 15:33:28 +0000108 std::unique_ptr<Packet> packet;
109 CHECK_NOTHROW(packet = std::make_unique<Packet>(decorator.ReadPacket(0)));
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100110
111 decorator.Close();
112
Keith Davisb10e0812019-10-17 09:52:50 +0100113 std::vector<char> data = ReadDumpFile(options.m_IncomingCaptureFile);
Matteo Martincigh67ef2a52019-10-10 13:29:02 +0100114 const char* packetData = reinterpret_cast<const char*>(packet->GetData());
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100115
116 // check if the data read back from the dump file matches the original
117 constexpr unsigned int bytesToSkip = 2u * sizeof(uint32_t); // skip header and packet length
118 int diff = std::strncmp(data.data() + bytesToSkip, packetData, g_DataLength);
Sadik Armagan1625efc2021-06-10 18:24:34 +0100119 CHECK(diff == 0);
Francis Murtagh532a29d2020-06-29 11:50:01 +0100120 fs::remove(fileName);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100121}
122
Sadik Armagan1625efc2021-06-10 18:24:34 +0100123TEST_CASE("DumpOutgoingInvalidFile")
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100124{
Jim Flynn4c9ed1d2022-01-23 23:57:20 +0000125 ProfilingOptions options;
Keith Davisb10e0812019-10-17 09:52:50 +0100126 options.m_IncomingCaptureFile = "";
127 options.m_OutgoingCaptureFile = "/";
128 ProfilingConnectionDumpToFileDecorator decorator(std::make_unique<DummyProfilingConnection>(), options, false);
Sadik Armagan1625efc2021-06-10 18:24:34 +0100129 CHECK_THROWS_AS(decorator.WritePacket(g_DataPtr, g_DataLength), armnn::RuntimeException);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100130}
131
Sadik Armagan1625efc2021-06-10 18:24:34 +0100132TEST_CASE("DumpOutgoingInvalidFileIgnoreErrors")
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100133{
Jim Flynn4c9ed1d2022-01-23 23:57:20 +0000134 ProfilingOptions options;
Keith Davisb10e0812019-10-17 09:52:50 +0100135 options.m_IncomingCaptureFile = "";
136 options.m_OutgoingCaptureFile = "/";
137
138 ProfilingConnectionDumpToFileDecorator decorator(std::make_unique<DummyProfilingConnection>(), options, true);
Sadik Armagan1625efc2021-06-10 18:24:34 +0100139 CHECK_NOTHROW(decorator.WritePacket(g_DataPtr, g_DataLength));
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100140
141 bool success = decorator.WritePacket(g_DataPtr, g_DataLength);
Sadik Armagan1625efc2021-06-10 18:24:34 +0100142 CHECK(!success);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100143}
144
Sadik Armagan1625efc2021-06-10 18:24:34 +0100145TEST_CASE("DumpOutgoingValidFile")
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100146{
Francis Murtagh532a29d2020-06-29 11:50:01 +0100147 fs::path fileName = armnnUtils::Filesystem::NamedTempFile("Armnn-DumpOutgoingValidFileTest-TempFile");
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100148
Jim Flynn4c9ed1d2022-01-23 23:57:20 +0000149 ProfilingOptions options;
Keith Davisb10e0812019-10-17 09:52:50 +0100150 options.m_IncomingCaptureFile = "";
Rob Hughes270233f2019-11-13 11:53:48 +0000151 options.m_OutgoingCaptureFile = fileName.string();
Keith Davisb10e0812019-10-17 09:52:50 +0100152
153 ProfilingConnectionDumpToFileDecorator decorator(std::make_unique<DummyProfilingConnection>(), options, false);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100154
155 bool success = false;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100156 CHECK_NOTHROW(success = decorator.WritePacket(g_DataPtr, g_DataLength));
157 CHECK(success);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100158
159 decorator.Close();
160
Keith Davisb10e0812019-10-17 09:52:50 +0100161 std::vector<char> data = ReadDumpFile(options.m_OutgoingCaptureFile);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100162
163 // check if the data read back from the dump file matches the original
164 int diff = std::strncmp(data.data(), g_Data.data(), g_DataLength);
Sadik Armagan1625efc2021-06-10 18:24:34 +0100165 CHECK(diff == 0);
Francis Murtagh532a29d2020-06-29 11:50:01 +0100166 fs::remove(fileName);
Aron Virginas-Tardfa14772019-09-24 18:24:47 +0100167}
168
Sadik Armagan1625efc2021-06-10 18:24:34 +0100169}