xref: /aosp_15_r20/external/armnn/profiling/client/src/SendCounterPacket.cpp (revision 89c4ff92f2867872bb9e2354d150bf0c8c502810)
1*89c4ff92SAndroid Build Coastguard Worker //
2*89c4ff92SAndroid Build Coastguard Worker // Copyright © 2017 Arm Ltd and Contributors. All rights reserved.
3*89c4ff92SAndroid Build Coastguard Worker // SPDX-License-Identifier: MIT
4*89c4ff92SAndroid Build Coastguard Worker //
5*89c4ff92SAndroid Build Coastguard Worker 
6*89c4ff92SAndroid Build Coastguard Worker #include "SendCounterPacket.hpp"
7*89c4ff92SAndroid Build Coastguard Worker 
8*89c4ff92SAndroid Build Coastguard Worker #include <common/include/Assert.hpp>
9*89c4ff92SAndroid Build Coastguard Worker #include <common/include/Conversion.hpp>
10*89c4ff92SAndroid Build Coastguard Worker #include <common/include/Constants.hpp>
11*89c4ff92SAndroid Build Coastguard Worker #include <common/include/EncodeVersion.hpp>
12*89c4ff92SAndroid Build Coastguard Worker #include <common/include/Processes.hpp>
13*89c4ff92SAndroid Build Coastguard Worker #include <common/include/ProfilingException.hpp>
14*89c4ff92SAndroid Build Coastguard Worker #include <common/include/SwTrace.hpp>
15*89c4ff92SAndroid Build Coastguard Worker 
16*89c4ff92SAndroid Build Coastguard Worker #include <fmt/format.h>
17*89c4ff92SAndroid Build Coastguard Worker 
18*89c4ff92SAndroid Build Coastguard Worker #include <cstring>
19*89c4ff92SAndroid Build Coastguard Worker 
20*89c4ff92SAndroid Build Coastguard Worker namespace arm
21*89c4ff92SAndroid Build Coastguard Worker {
22*89c4ff92SAndroid Build Coastguard Worker 
23*89c4ff92SAndroid Build Coastguard Worker namespace pipe
24*89c4ff92SAndroid Build Coastguard Worker {
25*89c4ff92SAndroid Build Coastguard Worker 
SendStreamMetaDataPacket()26*89c4ff92SAndroid Build Coastguard Worker void SendCounterPacket::SendStreamMetaDataPacket()
27*89c4ff92SAndroid Build Coastguard Worker {
28*89c4ff92SAndroid Build Coastguard Worker     const std::string info(m_SoftwareInfo);
29*89c4ff92SAndroid Build Coastguard Worker     const std::string hardwareVersion(m_HardwareVersion);
30*89c4ff92SAndroid Build Coastguard Worker     const std::string softwareVersion(m_SoftwareVersion);
31*89c4ff92SAndroid Build Coastguard Worker     const std::string processName = GetProcessName().substr(0, 60);
32*89c4ff92SAndroid Build Coastguard Worker 
33*89c4ff92SAndroid Build Coastguard Worker     const uint32_t infoSize =            arm::pipe::numeric_cast<uint32_t>(info.size()) + 1;
34*89c4ff92SAndroid Build Coastguard Worker     const uint32_t hardwareVersionSize = arm::pipe::numeric_cast<uint32_t>(hardwareVersion.size()) + 1;
35*89c4ff92SAndroid Build Coastguard Worker     const uint32_t softwareVersionSize = arm::pipe::numeric_cast<uint32_t>(softwareVersion.size()) + 1;
36*89c4ff92SAndroid Build Coastguard Worker     const uint32_t processNameSize =     arm::pipe::numeric_cast<uint32_t>(processName.size()) + 1;
37*89c4ff92SAndroid Build Coastguard Worker 
38*89c4ff92SAndroid Build Coastguard Worker     const uint32_t sizeUint32 = sizeof(uint32_t);
39*89c4ff92SAndroid Build Coastguard Worker 
40*89c4ff92SAndroid Build Coastguard Worker     const uint32_t headerSize = 2 * sizeUint32;
41*89c4ff92SAndroid Build Coastguard Worker     const uint32_t bodySize = 10 * sizeUint32;
42*89c4ff92SAndroid Build Coastguard Worker     const uint32_t packetVersionCountSize = sizeUint32;
43*89c4ff92SAndroid Build Coastguard Worker 
44*89c4ff92SAndroid Build Coastguard Worker     // Supported Packets
45*89c4ff92SAndroid Build Coastguard Worker     // Packet Encoding version 1.0.0
46*89c4ff92SAndroid Build Coastguard Worker     // Control packet family
47*89c4ff92SAndroid Build Coastguard Worker     //   Stream metadata packet (packet family=0; packet id=0)
48*89c4ff92SAndroid Build Coastguard Worker     //   Connection Acknowledged packet ( packet family=0, packet id=1) Version 1.0.0
49*89c4ff92SAndroid Build Coastguard Worker     //   Counter Directory packet (packet family=0; packet id=2) Version 1.0.0
50*89c4ff92SAndroid Build Coastguard Worker     //   Request Counter Directory packet ( packet family=0, packet id=3) Version 1.0.0
51*89c4ff92SAndroid Build Coastguard Worker     //   Periodic Counter Selection packet ( packet family=0, packet id=4) Version 1.0.0
52*89c4ff92SAndroid Build Coastguard Worker     //   Per Job Counter Selection packet ( packet family=0, packet id=5) Version 1.0.0
53*89c4ff92SAndroid Build Coastguard Worker     //   Activate Timeline Reporting (packet family = 0, packet id = 6) Version 1.0.0
54*89c4ff92SAndroid Build Coastguard Worker     //   Deactivate Timeline Reporting (packet family = 0, packet id = 7) Version 1.0.0
55*89c4ff92SAndroid Build Coastguard Worker     // Counter Packet Family
56*89c4ff92SAndroid Build Coastguard Worker     //   Periodic Counter Capture (packet_family = 3, packet_class = 0, packet_type = 0) Version 1.0.0
57*89c4ff92SAndroid Build Coastguard Worker     //   Per-Job Counter Capture (packet_family = 3, packet_class = 1, packet_type = 0,1) Version  1.0.0
58*89c4ff92SAndroid Build Coastguard Worker     // Timeline Packet Family
59*89c4ff92SAndroid Build Coastguard Worker     //   Timeline Message Directory (packet_family = 1, packet_class = 0, packet_type = 0) Version 1.0.0
60*89c4ff92SAndroid Build Coastguard Worker     //   Timeline Message (packet_family = 1, packet_class = 0, packet_type = 1) Version 1.0.0
61*89c4ff92SAndroid Build Coastguard Worker     std::vector<std::pair<uint32_t, uint32_t>> packetVersions;
62*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(0, 0), arm::pipe::EncodeVersion(1, 0, 0)));
63*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(0, 1), arm::pipe::EncodeVersion(1, 0, 0)));
64*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(0, 2), arm::pipe::EncodeVersion(1, 0, 0)));
65*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(0, 3), arm::pipe::EncodeVersion(1, 0, 0)));
66*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(0, 4), arm::pipe::EncodeVersion(1, 0, 0)));
67*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(0, 5), arm::pipe::EncodeVersion(1, 0, 0)));
68*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(0, 6), arm::pipe::EncodeVersion(1, 0, 0)));
69*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(0, 7), arm::pipe::EncodeVersion(1, 0, 0)));
70*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(3, 0, 0), arm::pipe::EncodeVersion(1, 0, 0)));
71*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(3, 1, 0), arm::pipe::EncodeVersion(1, 0, 0)));
72*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(3, 1, 1), arm::pipe::EncodeVersion(1, 0, 0)));
73*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(1, 0, 0), arm::pipe::EncodeVersion(1, 0, 0)));
74*89c4ff92SAndroid Build Coastguard Worker     packetVersions.push_back(std::make_pair(ConstructHeader(1, 0, 1), arm::pipe::EncodeVersion(1, 0, 0)));
75*89c4ff92SAndroid Build Coastguard Worker     uint32_t numberOfVersions = arm::pipe::numeric_cast<uint32_t>(packetVersions.size());
76*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetVersionSize = arm::pipe::numeric_cast<uint32_t>(numberOfVersions * 2 * sizeUint32);
77*89c4ff92SAndroid Build Coastguard Worker 
78*89c4ff92SAndroid Build Coastguard Worker     const uint32_t payloadSize = arm::pipe::numeric_cast<uint32_t>(infoSize + hardwareVersionSize +
79*89c4ff92SAndroid Build Coastguard Worker                                                                softwareVersionSize + processNameSize +
80*89c4ff92SAndroid Build Coastguard Worker                                                                packetVersionCountSize + packetVersionSize);
81*89c4ff92SAndroid Build Coastguard Worker 
82*89c4ff92SAndroid Build Coastguard Worker     const uint32_t totalSize = headerSize + bodySize + payloadSize;
83*89c4ff92SAndroid Build Coastguard Worker     uint32_t offset = 0;
84*89c4ff92SAndroid Build Coastguard Worker     uint32_t reserved = 0;
85*89c4ff92SAndroid Build Coastguard Worker 
86*89c4ff92SAndroid Build Coastguard Worker     IPacketBufferPtr writeBuffer = m_BufferManager.Reserve(totalSize, reserved);
87*89c4ff92SAndroid Build Coastguard Worker 
88*89c4ff92SAndroid Build Coastguard Worker     if (writeBuffer == nullptr || reserved < totalSize)
89*89c4ff92SAndroid Build Coastguard Worker     {
90*89c4ff92SAndroid Build Coastguard Worker         CancelOperationAndThrow<arm::pipe::BufferExhaustion>(
91*89c4ff92SAndroid Build Coastguard Worker             writeBuffer,
92*89c4ff92SAndroid Build Coastguard Worker             fmt::format("No space left in buffer. Unable to reserve ({}) bytes.", totalSize));
93*89c4ff92SAndroid Build Coastguard Worker     }
94*89c4ff92SAndroid Build Coastguard Worker 
95*89c4ff92SAndroid Build Coastguard Worker     try
96*89c4ff92SAndroid Build Coastguard Worker     {
97*89c4ff92SAndroid Build Coastguard Worker         // Create header
98*89c4ff92SAndroid Build Coastguard Worker 
99*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, 0);
100*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
101*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, totalSize - headerSize);
102*89c4ff92SAndroid Build Coastguard Worker 
103*89c4ff92SAndroid Build Coastguard Worker         // Packet body
104*89c4ff92SAndroid Build Coastguard Worker 
105*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
106*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, arm::pipe::PIPE_MAGIC); // pipe_magic
107*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
108*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, arm::pipe::EncodeVersion(1, 0, 0)); // stream_metadata_version
109*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
110*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, MAX_METADATA_PACKET_LENGTH); // max_data_length
111*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
112*89c4ff92SAndroid Build Coastguard Worker         int pid = arm::pipe::GetCurrentProcessId();
113*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, arm::pipe::numeric_cast<uint32_t>(pid)); // pid
114*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
115*89c4ff92SAndroid Build Coastguard Worker         uint32_t poolOffset = bodySize;
116*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, poolOffset); // offset_info
117*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
118*89c4ff92SAndroid Build Coastguard Worker         poolOffset += infoSize;
119*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, poolOffset); // offset_hw_version
120*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
121*89c4ff92SAndroid Build Coastguard Worker         poolOffset += hardwareVersionSize;
122*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, poolOffset); // offset_sw_version
123*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
124*89c4ff92SAndroid Build Coastguard Worker         poolOffset += softwareVersionSize;
125*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, poolOffset); // offset_process_name
126*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
127*89c4ff92SAndroid Build Coastguard Worker         poolOffset += processNameSize;
128*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, poolOffset); // offset_packet_version_table
129*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
130*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, 0); // reserved
131*89c4ff92SAndroid Build Coastguard Worker         offset += sizeUint32;
132*89c4ff92SAndroid Build Coastguard Worker 
133*89c4ff92SAndroid Build Coastguard Worker         // Pool
134*89c4ff92SAndroid Build Coastguard Worker 
135*89c4ff92SAndroid Build Coastguard Worker         if (infoSize)
136*89c4ff92SAndroid Build Coastguard Worker         {
137*89c4ff92SAndroid Build Coastguard Worker             memcpy(&writeBuffer->GetWritableData()[offset], info.c_str(), infoSize);
138*89c4ff92SAndroid Build Coastguard Worker             offset += infoSize;
139*89c4ff92SAndroid Build Coastguard Worker         }
140*89c4ff92SAndroid Build Coastguard Worker 
141*89c4ff92SAndroid Build Coastguard Worker         memcpy(&writeBuffer->GetWritableData()[offset], hardwareVersion.c_str(), hardwareVersionSize);
142*89c4ff92SAndroid Build Coastguard Worker         offset += hardwareVersionSize;
143*89c4ff92SAndroid Build Coastguard Worker         memcpy(&writeBuffer->GetWritableData()[offset], softwareVersion.c_str(), softwareVersionSize);
144*89c4ff92SAndroid Build Coastguard Worker         offset += softwareVersionSize;
145*89c4ff92SAndroid Build Coastguard Worker         memcpy(&writeBuffer->GetWritableData()[offset], processName.c_str(), processNameSize);
146*89c4ff92SAndroid Build Coastguard Worker         offset += processNameSize;
147*89c4ff92SAndroid Build Coastguard Worker 
148*89c4ff92SAndroid Build Coastguard Worker         if (!packetVersions.empty())
149*89c4ff92SAndroid Build Coastguard Worker         {
150*89c4ff92SAndroid Build Coastguard Worker             // Packet Version Count
151*89c4ff92SAndroid Build Coastguard Worker             WriteUint32(writeBuffer, offset, numberOfVersions << 16);
152*89c4ff92SAndroid Build Coastguard Worker             offset += sizeUint32;
153*89c4ff92SAndroid Build Coastguard Worker 
154*89c4ff92SAndroid Build Coastguard Worker             // Packet Version Entries
155*89c4ff92SAndroid Build Coastguard Worker             for (std::pair<uint32_t, uint32_t>& packetVersion : packetVersions)
156*89c4ff92SAndroid Build Coastguard Worker             {
157*89c4ff92SAndroid Build Coastguard Worker                 WriteUint32(writeBuffer, offset, packetVersion.first);
158*89c4ff92SAndroid Build Coastguard Worker                 offset += sizeUint32;
159*89c4ff92SAndroid Build Coastguard Worker                 WriteUint32(writeBuffer, offset, packetVersion.second);
160*89c4ff92SAndroid Build Coastguard Worker                 offset += sizeUint32;
161*89c4ff92SAndroid Build Coastguard Worker             }
162*89c4ff92SAndroid Build Coastguard Worker         }
163*89c4ff92SAndroid Build Coastguard Worker     }
164*89c4ff92SAndroid Build Coastguard Worker     catch(...)
165*89c4ff92SAndroid Build Coastguard Worker     {
166*89c4ff92SAndroid Build Coastguard Worker         CancelOperationAndThrow<arm::pipe::ProfilingException>(writeBuffer, "Error processing packet.");
167*89c4ff92SAndroid Build Coastguard Worker     }
168*89c4ff92SAndroid Build Coastguard Worker 
169*89c4ff92SAndroid Build Coastguard Worker     m_BufferManager.Commit(writeBuffer, totalSize, false);
170*89c4ff92SAndroid Build Coastguard Worker }
171*89c4ff92SAndroid Build Coastguard Worker 
CreateCategoryRecord(const CategoryPtr & category,const Counters & counters,CategoryRecord & categoryRecord,std::string & errorMessage)172*89c4ff92SAndroid Build Coastguard Worker bool SendCounterPacket::CreateCategoryRecord(const CategoryPtr& category,
173*89c4ff92SAndroid Build Coastguard Worker                                              const Counters& counters,
174*89c4ff92SAndroid Build Coastguard Worker                                              CategoryRecord& categoryRecord,
175*89c4ff92SAndroid Build Coastguard Worker                                              std::string& errorMessage)
176*89c4ff92SAndroid Build Coastguard Worker {
177*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(category);
178*89c4ff92SAndroid Build Coastguard Worker 
179*89c4ff92SAndroid Build Coastguard Worker     const std::string& categoryName = category->m_Name;
180*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(!categoryName.empty());
181*89c4ff92SAndroid Build Coastguard Worker 
182*89c4ff92SAndroid Build Coastguard Worker     // Remove any duplicate counters
183*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint16_t> categoryCounters;
184*89c4ff92SAndroid Build Coastguard Worker     for (size_t counterIndex = 0; counterIndex < category->m_Counters.size(); ++counterIndex)
185*89c4ff92SAndroid Build Coastguard Worker     {
186*89c4ff92SAndroid Build Coastguard Worker         uint16_t counterUid = category->m_Counters.at(counterIndex);
187*89c4ff92SAndroid Build Coastguard Worker         auto it = counters.find(counterUid);
188*89c4ff92SAndroid Build Coastguard Worker         if (it == counters.end())
189*89c4ff92SAndroid Build Coastguard Worker         {
190*89c4ff92SAndroid Build Coastguard Worker             errorMessage = fmt::format("Counter ({}) not found in category ({})",
191*89c4ff92SAndroid Build Coastguard Worker                                        counterUid,
192*89c4ff92SAndroid Build Coastguard Worker                                        category->m_Name );
193*89c4ff92SAndroid Build Coastguard Worker             return false;
194*89c4ff92SAndroid Build Coastguard Worker         }
195*89c4ff92SAndroid Build Coastguard Worker 
196*89c4ff92SAndroid Build Coastguard Worker         const CounterPtr& counter = it->second;
197*89c4ff92SAndroid Build Coastguard Worker 
198*89c4ff92SAndroid Build Coastguard Worker         if (counterUid == counter->m_MaxCounterUid)
199*89c4ff92SAndroid Build Coastguard Worker         {
200*89c4ff92SAndroid Build Coastguard Worker             categoryCounters.emplace_back(counterUid);
201*89c4ff92SAndroid Build Coastguard Worker         }
202*89c4ff92SAndroid Build Coastguard Worker     }
203*89c4ff92SAndroid Build Coastguard Worker     if (categoryCounters.empty())
204*89c4ff92SAndroid Build Coastguard Worker     {
205*89c4ff92SAndroid Build Coastguard Worker         errorMessage = fmt::format("No valid counters found in category ({})", categoryName);
206*89c4ff92SAndroid Build Coastguard Worker         return false;
207*89c4ff92SAndroid Build Coastguard Worker     }
208*89c4ff92SAndroid Build Coastguard Worker 
209*89c4ff92SAndroid Build Coastguard Worker     // Utils
210*89c4ff92SAndroid Build Coastguard Worker     const size_t uint32_t_size = sizeof(uint32_t);
211*89c4ff92SAndroid Build Coastguard Worker 
212*89c4ff92SAndroid Build Coastguard Worker     // Convert the device name into a SWTrace namestring
213*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> categoryNameBuffer;
214*89c4ff92SAndroid Build Coastguard Worker     if (!arm::pipe::StringToSwTraceString<arm::pipe::SwTraceNameCharPolicy>(categoryName, categoryNameBuffer))
215*89c4ff92SAndroid Build Coastguard Worker     {
216*89c4ff92SAndroid Build Coastguard Worker         errorMessage = fmt::format("Cannot convert the name of category ({}) to an SWTrace namestring",
217*89c4ff92SAndroid Build Coastguard Worker                                    categoryName);
218*89c4ff92SAndroid Build Coastguard Worker         return false;
219*89c4ff92SAndroid Build Coastguard Worker     }
220*89c4ff92SAndroid Build Coastguard Worker 
221*89c4ff92SAndroid Build Coastguard Worker     // Category record word 1:
222*89c4ff92SAndroid Build Coastguard Worker     // 16:31 [16] event_count: number of events belonging to this category
223*89c4ff92SAndroid Build Coastguard Worker     // 0:15  [16] reserved: all zeros
224*89c4ff92SAndroid Build Coastguard Worker     const uint32_t categoryRecordWord1 = static_cast<uint32_t>(categoryCounters.size()) << 16;
225*89c4ff92SAndroid Build Coastguard Worker 
226*89c4ff92SAndroid Build Coastguard Worker     // Category record word 2:
227*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] event_pointer_table_offset: offset from the beginning of the category data pool to
228*89c4ff92SAndroid Build Coastguard Worker     //                                       the event_pointer_table
229*89c4ff92SAndroid Build Coastguard Worker     const uint32_t categoryRecordWord2 = static_cast<uint32_t>(3u * uint32_t_size);
230*89c4ff92SAndroid Build Coastguard Worker 
231*89c4ff92SAndroid Build Coastguard Worker     // Process the event records
232*89c4ff92SAndroid Build Coastguard Worker     const size_t counterCount = categoryCounters.size();
233*89c4ff92SAndroid Build Coastguard Worker     std::vector<EventRecord> eventRecords(counterCount);
234*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> eventRecordOffsets(counterCount, 0);
235*89c4ff92SAndroid Build Coastguard Worker     size_t eventRecordsSize = 0;
236*89c4ff92SAndroid Build Coastguard Worker     uint32_t eventRecordsOffset = arm::pipe::numeric_cast<uint32_t>(
237*89c4ff92SAndroid Build Coastguard Worker                     (eventRecords.size() + categoryNameBuffer.size()) * uint32_t_size);
238*89c4ff92SAndroid Build Coastguard Worker     for (size_t counterIndex = 0, eventRecordIndex = 0, eventRecordOffsetIndex = 0;
239*89c4ff92SAndroid Build Coastguard Worker          counterIndex < counterCount;
240*89c4ff92SAndroid Build Coastguard Worker          counterIndex++, eventRecordIndex++, eventRecordOffsetIndex++)
241*89c4ff92SAndroid Build Coastguard Worker     {
242*89c4ff92SAndroid Build Coastguard Worker         uint16_t counterUid = categoryCounters.at(counterIndex);
243*89c4ff92SAndroid Build Coastguard Worker         auto it = counters.find(counterUid);
244*89c4ff92SAndroid Build Coastguard Worker         const CounterPtr& counter = it->second;
245*89c4ff92SAndroid Build Coastguard Worker 
246*89c4ff92SAndroid Build Coastguard Worker         EventRecord& eventRecord = eventRecords.at(eventRecordIndex);
247*89c4ff92SAndroid Build Coastguard Worker         if (!CreateEventRecord(counter, eventRecord, errorMessage))
248*89c4ff92SAndroid Build Coastguard Worker         {
249*89c4ff92SAndroid Build Coastguard Worker             return false;
250*89c4ff92SAndroid Build Coastguard Worker         }
251*89c4ff92SAndroid Build Coastguard Worker 
252*89c4ff92SAndroid Build Coastguard Worker         // Update the total size in words of the event records
253*89c4ff92SAndroid Build Coastguard Worker         eventRecordsSize += eventRecord.size();
254*89c4ff92SAndroid Build Coastguard Worker 
255*89c4ff92SAndroid Build Coastguard Worker         // Add the event record offset to the event pointer table offset field
256*89c4ff92SAndroid Build Coastguard Worker         eventRecordOffsets[eventRecordOffsetIndex] = eventRecordsOffset;
257*89c4ff92SAndroid Build Coastguard Worker         eventRecordsOffset += arm::pipe::numeric_cast<uint32_t>(eventRecord.size() * uint32_t_size);
258*89c4ff92SAndroid Build Coastguard Worker     }
259*89c4ff92SAndroid Build Coastguard Worker 
260*89c4ff92SAndroid Build Coastguard Worker     // Category record word 3:
261*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] name_offset (offset from the beginning of the category data pool to the name field)
262*89c4ff92SAndroid Build Coastguard Worker     const uint32_t categoryRecordWord3 = arm::pipe::numeric_cast<uint32_t>(
263*89c4ff92SAndroid Build Coastguard Worker             (3u + eventRecordOffsets.size()) * uint32_t_size);
264*89c4ff92SAndroid Build Coastguard Worker 
265*89c4ff92SAndroid Build Coastguard Worker     // Calculate the size in words of the category record
266*89c4ff92SAndroid Build Coastguard Worker     const size_t categoryRecordSize = 3u +// The size of the fixed part (device + counter_set + event_count +
267*89c4ff92SAndroid Build Coastguard Worker                                           // reserved + event_pointer_table_offset + name_offset)
268*89c4ff92SAndroid Build Coastguard Worker                                       eventRecordOffsets.size() + // The size of the variable part (
269*89c4ff92SAndroid Build Coastguard Worker                                       categoryNameBuffer.size() + // the event pointer table + the category name
270*89c4ff92SAndroid Build Coastguard Worker                                       eventRecordsSize;           // including the null-terminator + the event records)
271*89c4ff92SAndroid Build Coastguard Worker 
272*89c4ff92SAndroid Build Coastguard Worker     // Allocate the necessary space for the category record
273*89c4ff92SAndroid Build Coastguard Worker     categoryRecord.resize(categoryRecordSize);
274*89c4ff92SAndroid Build Coastguard Worker 
275*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_NO_CONVERSION_WARN_BEGIN
276*89c4ff92SAndroid Build Coastguard Worker     // Create the category record
277*89c4ff92SAndroid Build Coastguard Worker     categoryRecord[0] = categoryRecordWord1; // event_count + reserved
278*89c4ff92SAndroid Build Coastguard Worker     categoryRecord[1] = categoryRecordWord2; // event_pointer_table_offset
279*89c4ff92SAndroid Build Coastguard Worker     categoryRecord[2] = categoryRecordWord3; // name_offset
280*89c4ff92SAndroid Build Coastguard Worker     auto offset = categoryRecord.begin() + 3u;
281*89c4ff92SAndroid Build Coastguard Worker     std::copy(eventRecordOffsets.begin(), eventRecordOffsets.end(), offset); // event_pointer_table
282*89c4ff92SAndroid Build Coastguard Worker     offset += eventRecordOffsets.size();
283*89c4ff92SAndroid Build Coastguard Worker     std::copy(categoryNameBuffer.begin(), categoryNameBuffer.end(), offset); // name
284*89c4ff92SAndroid Build Coastguard Worker     offset += categoryNameBuffer.size();
285*89c4ff92SAndroid Build Coastguard Worker     for (const EventRecord& eventRecord : eventRecords)
286*89c4ff92SAndroid Build Coastguard Worker     {
287*89c4ff92SAndroid Build Coastguard Worker         std::copy(eventRecord.begin(), eventRecord.end(), offset); // event_record
288*89c4ff92SAndroid Build Coastguard Worker         offset += eventRecord.size();
289*89c4ff92SAndroid Build Coastguard Worker     }
290*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_NO_CONVERSION_WARN_END
291*89c4ff92SAndroid Build Coastguard Worker 
292*89c4ff92SAndroid Build Coastguard Worker     return true;
293*89c4ff92SAndroid Build Coastguard Worker }
294*89c4ff92SAndroid Build Coastguard Worker 
CreateDeviceRecord(const DevicePtr & device,DeviceRecord & deviceRecord,std::string & errorMessage)295*89c4ff92SAndroid Build Coastguard Worker bool SendCounterPacket::CreateDeviceRecord(const DevicePtr& device,
296*89c4ff92SAndroid Build Coastguard Worker                                            DeviceRecord& deviceRecord,
297*89c4ff92SAndroid Build Coastguard Worker                                            std::string& errorMessage)
298*89c4ff92SAndroid Build Coastguard Worker {
299*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(device);
300*89c4ff92SAndroid Build Coastguard Worker 
301*89c4ff92SAndroid Build Coastguard Worker     uint16_t deviceUid = device->m_Uid;
302*89c4ff92SAndroid Build Coastguard Worker     const std::string& deviceName = device->m_Name;
303*89c4ff92SAndroid Build Coastguard Worker     uint16_t deviceCores = device->m_Cores;
304*89c4ff92SAndroid Build Coastguard Worker 
305*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(!deviceName.empty());
306*89c4ff92SAndroid Build Coastguard Worker 
307*89c4ff92SAndroid Build Coastguard Worker     // Device record word 0:
308*89c4ff92SAndroid Build Coastguard Worker     // 16:31 [16] uid: the unique identifier for the device
309*89c4ff92SAndroid Build Coastguard Worker     // 0:15  [16] cores: the number of individual streams of counters for one or more cores of some device
310*89c4ff92SAndroid Build Coastguard Worker     const uint32_t deviceRecordWord0 = (static_cast<uint32_t>(deviceUid) << 16) |
311*89c4ff92SAndroid Build Coastguard Worker                                  (static_cast<uint32_t>(deviceCores));
312*89c4ff92SAndroid Build Coastguard Worker 
313*89c4ff92SAndroid Build Coastguard Worker     // Device record word 1:
314*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] name_offset: offset from the beginning of the device record pool to the name field
315*89c4ff92SAndroid Build Coastguard Worker     const uint32_t deviceRecordWord1 = 8u; // The offset is always eight here, as the name field is always
316*89c4ff92SAndroid Build Coastguard Worker                                            // the first (and only) item in the pool and there are two device words
317*89c4ff92SAndroid Build Coastguard Worker 
318*89c4ff92SAndroid Build Coastguard Worker     // Convert the device name into a SWTrace string
319*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> deviceNameBuffer;
320*89c4ff92SAndroid Build Coastguard Worker     if (!arm::pipe::StringToSwTraceString<arm::pipe::SwTraceCharPolicy>(deviceName, deviceNameBuffer))
321*89c4ff92SAndroid Build Coastguard Worker     {
322*89c4ff92SAndroid Build Coastguard Worker         errorMessage = fmt::format("Cannot convert the name of device {} ({}) to an SWTrace string",
323*89c4ff92SAndroid Build Coastguard Worker                                    deviceUid,
324*89c4ff92SAndroid Build Coastguard Worker                                    deviceName);
325*89c4ff92SAndroid Build Coastguard Worker         return false;
326*89c4ff92SAndroid Build Coastguard Worker     }
327*89c4ff92SAndroid Build Coastguard Worker 
328*89c4ff92SAndroid Build Coastguard Worker     // Calculate the size in words of the device record
329*89c4ff92SAndroid Build Coastguard Worker     const size_t deviceRecordSize = 2u + // The size of the fixed part (uid + cores + name_offset)
330*89c4ff92SAndroid Build Coastguard Worker                               deviceNameBuffer.size(); // The size of the variable part (the device name including
331*89c4ff92SAndroid Build Coastguard Worker                                                        // the null-terminator)
332*89c4ff92SAndroid Build Coastguard Worker 
333*89c4ff92SAndroid Build Coastguard Worker     // Allocate the necessary space for the device record
334*89c4ff92SAndroid Build Coastguard Worker     deviceRecord.resize(deviceRecordSize);
335*89c4ff92SAndroid Build Coastguard Worker 
336*89c4ff92SAndroid Build Coastguard Worker     // Create the device record
337*89c4ff92SAndroid Build Coastguard Worker     deviceRecord[0] = deviceRecordWord0; // uid + core
338*89c4ff92SAndroid Build Coastguard Worker     deviceRecord[1] = deviceRecordWord1; // name_offset
339*89c4ff92SAndroid Build Coastguard Worker     auto offset = deviceRecord.begin() + 2u;
340*89c4ff92SAndroid Build Coastguard Worker     std::copy(deviceNameBuffer.begin(), deviceNameBuffer.end(), offset); // name
341*89c4ff92SAndroid Build Coastguard Worker 
342*89c4ff92SAndroid Build Coastguard Worker     return true;
343*89c4ff92SAndroid Build Coastguard Worker }
344*89c4ff92SAndroid Build Coastguard Worker 
CreateCounterSetRecord(const CounterSetPtr & counterSet,CounterSetRecord & counterSetRecord,std::string & errorMessage)345*89c4ff92SAndroid Build Coastguard Worker bool SendCounterPacket::CreateCounterSetRecord(const CounterSetPtr& counterSet,
346*89c4ff92SAndroid Build Coastguard Worker                                                CounterSetRecord& counterSetRecord,
347*89c4ff92SAndroid Build Coastguard Worker                                                std::string& errorMessage)
348*89c4ff92SAndroid Build Coastguard Worker {
349*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(counterSet);
350*89c4ff92SAndroid Build Coastguard Worker 
351*89c4ff92SAndroid Build Coastguard Worker     uint16_t counterSetUid = counterSet->m_Uid;
352*89c4ff92SAndroid Build Coastguard Worker     const std::string& counterSetName = counterSet->m_Name;
353*89c4ff92SAndroid Build Coastguard Worker     uint16_t counterSetCount = counterSet->m_Count;
354*89c4ff92SAndroid Build Coastguard Worker 
355*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(!counterSetName.empty());
356*89c4ff92SAndroid Build Coastguard Worker 
357*89c4ff92SAndroid Build Coastguard Worker     // Counter set record word 0:
358*89c4ff92SAndroid Build Coastguard Worker     // 16:31 [16] uid: the unique identifier for the counter_set
359*89c4ff92SAndroid Build Coastguard Worker     // 0:15  [16] count: the number of counters which can be active in this set at any one time
360*89c4ff92SAndroid Build Coastguard Worker     const uint32_t counterSetRecordWord0 = (static_cast<uint32_t>(counterSetUid) << 16) |
361*89c4ff92SAndroid Build Coastguard Worker                                            (static_cast<uint32_t>(counterSetCount));
362*89c4ff92SAndroid Build Coastguard Worker 
363*89c4ff92SAndroid Build Coastguard Worker     // Counter set record word 1:
364*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] name_offset: offset from the beginning of the counter set pool to the name field
365*89c4ff92SAndroid Build Coastguard Worker     const uint32_t counterSetRecordWord1 = 8u; // The offset is always eight here, as the name field is always
366*89c4ff92SAndroid Build Coastguard Worker                                                // the first (and only) item in the pool after the two counter set words
367*89c4ff92SAndroid Build Coastguard Worker 
368*89c4ff92SAndroid Build Coastguard Worker     // Convert the device name into a SWTrace namestring
369*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> counterSetNameBuffer;
370*89c4ff92SAndroid Build Coastguard Worker     if (!arm::pipe::StringToSwTraceString<arm::pipe::SwTraceNameCharPolicy>(counterSet->m_Name, counterSetNameBuffer))
371*89c4ff92SAndroid Build Coastguard Worker     {
372*89c4ff92SAndroid Build Coastguard Worker         errorMessage = fmt::format("Cannot convert the name of counter set {} ({}) to an SWTrace namestring",
373*89c4ff92SAndroid Build Coastguard Worker                                    counterSetUid,
374*89c4ff92SAndroid Build Coastguard Worker                                    counterSetName);
375*89c4ff92SAndroid Build Coastguard Worker         return false;
376*89c4ff92SAndroid Build Coastguard Worker     }
377*89c4ff92SAndroid Build Coastguard Worker 
378*89c4ff92SAndroid Build Coastguard Worker     // Calculate the size in words of the counter set record
379*89c4ff92SAndroid Build Coastguard Worker     const size_t counterSetRecordSize = 2u + // The size of the fixed part (uid + cores + name_offset)
380*89c4ff92SAndroid Build Coastguard Worker                                         counterSetNameBuffer.size(); // The size of the variable part (the counter set
381*89c4ff92SAndroid Build Coastguard Worker                                                                      // name including the null-terminator)
382*89c4ff92SAndroid Build Coastguard Worker 
383*89c4ff92SAndroid Build Coastguard Worker     // Allocate the space for the counter set record
384*89c4ff92SAndroid Build Coastguard Worker     counterSetRecord.resize(counterSetRecordSize);
385*89c4ff92SAndroid Build Coastguard Worker 
386*89c4ff92SAndroid Build Coastguard Worker     // Create the counter set record
387*89c4ff92SAndroid Build Coastguard Worker     counterSetRecord[0] = counterSetRecordWord0; // uid + core
388*89c4ff92SAndroid Build Coastguard Worker     counterSetRecord[1] = counterSetRecordWord1; // name_offset
389*89c4ff92SAndroid Build Coastguard Worker     auto offset = counterSetRecord.begin() + 2u;
390*89c4ff92SAndroid Build Coastguard Worker     std::copy(counterSetNameBuffer.begin(), counterSetNameBuffer.end(), offset); // name
391*89c4ff92SAndroid Build Coastguard Worker 
392*89c4ff92SAndroid Build Coastguard Worker     return true;
393*89c4ff92SAndroid Build Coastguard Worker }
394*89c4ff92SAndroid Build Coastguard Worker 
CreateEventRecord(const CounterPtr & counter,EventRecord & eventRecord,std::string & errorMessage)395*89c4ff92SAndroid Build Coastguard Worker bool SendCounterPacket::CreateEventRecord(const CounterPtr& counter,
396*89c4ff92SAndroid Build Coastguard Worker                                           EventRecord& eventRecord,
397*89c4ff92SAndroid Build Coastguard Worker                                           std::string& errorMessage)
398*89c4ff92SAndroid Build Coastguard Worker {
399*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(counter);
400*89c4ff92SAndroid Build Coastguard Worker 
401*89c4ff92SAndroid Build Coastguard Worker     uint16_t           counterUid           = counter->m_Uid;
402*89c4ff92SAndroid Build Coastguard Worker     uint16_t           maxCounterUid        = counter->m_MaxCounterUid;
403*89c4ff92SAndroid Build Coastguard Worker     uint16_t           deviceUid            = counter->m_DeviceUid;
404*89c4ff92SAndroid Build Coastguard Worker     uint16_t           counterSetUid        = counter->m_CounterSetUid;
405*89c4ff92SAndroid Build Coastguard Worker     uint16_t           counterClass         = counter->m_Class;
406*89c4ff92SAndroid Build Coastguard Worker     uint16_t           counterInterpolation = counter->m_Interpolation;
407*89c4ff92SAndroid Build Coastguard Worker     double             counterMultiplier    = counter->m_Multiplier;
408*89c4ff92SAndroid Build Coastguard Worker     const std::string& counterName          = counter->m_Name;
409*89c4ff92SAndroid Build Coastguard Worker     const std::string& counterDescription   = counter->m_Description;
410*89c4ff92SAndroid Build Coastguard Worker     const std::string& counterUnits         = counter->m_Units;
411*89c4ff92SAndroid Build Coastguard Worker 
412*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(counterClass == 0 || counterClass == 1);
413*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(counterInterpolation == 0 || counterInterpolation == 1);
414*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(counterMultiplier);
415*89c4ff92SAndroid Build Coastguard Worker 
416*89c4ff92SAndroid Build Coastguard Worker     // Utils
417*89c4ff92SAndroid Build Coastguard Worker     const size_t uint32_t_size = sizeof(uint32_t);
418*89c4ff92SAndroid Build Coastguard Worker     // eventRecordBlockSize is the size of the fixed part
419*89c4ff92SAndroid Build Coastguard Worker     // (counter_uid + max_counter_uid + device +
420*89c4ff92SAndroid Build Coastguard Worker     // counter_set + class + interpolation +
421*89c4ff92SAndroid Build Coastguard Worker     // multiplier + name_offset + description_offset +
422*89c4ff92SAndroid Build Coastguard Worker     // units_offset)
423*89c4ff92SAndroid Build Coastguard Worker     const size_t eventRecordBlockSize = 8u;
424*89c4ff92SAndroid Build Coastguard Worker 
425*89c4ff92SAndroid Build Coastguard Worker     // Event record word 0:
426*89c4ff92SAndroid Build Coastguard Worker     // 16:31 [16] max_counter_uid: if the device this event is associated with has more than one core and there
427*89c4ff92SAndroid Build Coastguard Worker     //                             is one of these counters per core this value will be set to
428*89c4ff92SAndroid Build Coastguard Worker     //                             (counter_uid + cores (from device_record)) - 1.
429*89c4ff92SAndroid Build Coastguard Worker     //                             If there is only a single core then this value will be the same as
430*89c4ff92SAndroid Build Coastguard Worker     //                             the counter_uid value
431*89c4ff92SAndroid Build Coastguard Worker     // 0:15  [16] count_uid: unique ID for the counter. Must be unique across all counters in all categories
432*89c4ff92SAndroid Build Coastguard Worker     const uint32_t eventRecordWord0 = (static_cast<uint32_t>(maxCounterUid) << 16) |
433*89c4ff92SAndroid Build Coastguard Worker                                       (static_cast<uint32_t>(counterUid));
434*89c4ff92SAndroid Build Coastguard Worker 
435*89c4ff92SAndroid Build Coastguard Worker     // Event record word 1:
436*89c4ff92SAndroid Build Coastguard Worker     // 16:31 [16] device: UID of the device this event is associated with. Set to zero if the event is NOT
437*89c4ff92SAndroid Build Coastguard Worker     //                    associated with a device
438*89c4ff92SAndroid Build Coastguard Worker     // 0:15  [16] counter_set: UID of the counter_set this event is associated with. Set to zero if the event
439*89c4ff92SAndroid Build Coastguard Worker     //                         is NOT associated with a counter_set
440*89c4ff92SAndroid Build Coastguard Worker     const uint32_t eventRecordWord1 = (static_cast<uint32_t>(deviceUid) << 16) |
441*89c4ff92SAndroid Build Coastguard Worker                                       (static_cast<uint32_t>(counterSetUid));
442*89c4ff92SAndroid Build Coastguard Worker 
443*89c4ff92SAndroid Build Coastguard Worker     // Event record word 2:
444*89c4ff92SAndroid Build Coastguard Worker     // 16:31 [16] class: type describing how to treat each data point in a stream of data points
445*89c4ff92SAndroid Build Coastguard Worker     // 0:15  [16] interpolation: type describing how to interpolate each data point in a stream of data points
446*89c4ff92SAndroid Build Coastguard Worker     const uint32_t eventRecordWord2 = (static_cast<uint32_t>(counterClass) << 16) |
447*89c4ff92SAndroid Build Coastguard Worker                                       (static_cast<uint32_t>(counterInterpolation));
448*89c4ff92SAndroid Build Coastguard Worker 
449*89c4ff92SAndroid Build Coastguard Worker     // Event record word 3-4:
450*89c4ff92SAndroid Build Coastguard Worker     // 0:63 [64] multiplier: internal data stream is represented as integer values, this allows scaling of
451*89c4ff92SAndroid Build Coastguard Worker     //                       those values as if they are fixed point numbers. Zero is not a valid value
452*89c4ff92SAndroid Build Coastguard Worker     uint32_t multiplier[2] = { 0u, 0u };
453*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_ASSERT(sizeof(counterMultiplier) == sizeof(multiplier));
454*89c4ff92SAndroid Build Coastguard Worker     std::memcpy(multiplier, &counterMultiplier, sizeof(multiplier));
455*89c4ff92SAndroid Build Coastguard Worker     const uint32_t eventRecordWord3 = multiplier[0];
456*89c4ff92SAndroid Build Coastguard Worker     const uint32_t eventRecordWord4 = multiplier[1];
457*89c4ff92SAndroid Build Coastguard Worker 
458*89c4ff92SAndroid Build Coastguard Worker     // Event record word 5:
459*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] name_offset: offset from the beginning of the event record pool to the name field
460*89c4ff92SAndroid Build Coastguard Worker     const uint32_t eventRecordWord5 = static_cast<uint32_t>(eventRecordBlockSize * uint32_t_size);
461*89c4ff92SAndroid Build Coastguard Worker 
462*89c4ff92SAndroid Build Coastguard Worker     // Convert the counter name into a SWTrace string
463*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> counterNameBuffer;
464*89c4ff92SAndroid Build Coastguard Worker     if (!arm::pipe::StringToSwTraceString<arm::pipe::SwTraceCharPolicy>(counterName, counterNameBuffer))
465*89c4ff92SAndroid Build Coastguard Worker     {
466*89c4ff92SAndroid Build Coastguard Worker         errorMessage = fmt::format("Cannot convert the name of counter {} (name: {}) to an SWTrace string",
467*89c4ff92SAndroid Build Coastguard Worker                                    counterUid,
468*89c4ff92SAndroid Build Coastguard Worker                                    counterName);
469*89c4ff92SAndroid Build Coastguard Worker         return false;
470*89c4ff92SAndroid Build Coastguard Worker     }
471*89c4ff92SAndroid Build Coastguard Worker 
472*89c4ff92SAndroid Build Coastguard Worker     // Event record word 6:
473*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] description_offset: offset from the beginning of the event record pool to the description field
474*89c4ff92SAndroid Build Coastguard Worker     // The size of the name buffer in bytes
475*89c4ff92SAndroid Build Coastguard Worker     uint32_t eventRecordWord6 =
476*89c4ff92SAndroid Build Coastguard Worker             static_cast<uint32_t>((counterNameBuffer.size() + eventRecordBlockSize) * uint32_t_size);
477*89c4ff92SAndroid Build Coastguard Worker 
478*89c4ff92SAndroid Build Coastguard Worker     // Convert the counter description into a SWTrace string
479*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> counterDescriptionBuffer;
480*89c4ff92SAndroid Build Coastguard Worker     if (!arm::pipe::StringToSwTraceString<arm::pipe::SwTraceCharPolicy>(counterDescription, counterDescriptionBuffer))
481*89c4ff92SAndroid Build Coastguard Worker     {
482*89c4ff92SAndroid Build Coastguard Worker         errorMessage = fmt::format("Cannot convert the description of counter {} (description: {}) "
483*89c4ff92SAndroid Build Coastguard Worker                                    "to an SWTrace string",
484*89c4ff92SAndroid Build Coastguard Worker                                    counterUid,
485*89c4ff92SAndroid Build Coastguard Worker                                    counterName);
486*89c4ff92SAndroid Build Coastguard Worker         return false;
487*89c4ff92SAndroid Build Coastguard Worker     }
488*89c4ff92SAndroid Build Coastguard Worker 
489*89c4ff92SAndroid Build Coastguard Worker     // Event record word 7:
490*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] units_offset: (optional) offset from the beginning of the event record pool to the units field.
491*89c4ff92SAndroid Build Coastguard Worker     //                         An offset value of zero indicates this field is not provided
492*89c4ff92SAndroid Build Coastguard Worker     bool includeUnits = !counterUnits.empty();
493*89c4ff92SAndroid Build Coastguard Worker     // The size of the description buffer in bytes
494*89c4ff92SAndroid Build Coastguard Worker     const uint32_t eventRecordWord7 = includeUnits ?
495*89c4ff92SAndroid Build Coastguard Worker                                 eventRecordWord6 +
496*89c4ff92SAndroid Build Coastguard Worker                                 arm::pipe::numeric_cast<uint32_t>(counterDescriptionBuffer.size()
497*89c4ff92SAndroid Build Coastguard Worker                                 * uint32_t_size) :
498*89c4ff92SAndroid Build Coastguard Worker                                 0;
499*89c4ff92SAndroid Build Coastguard Worker 
500*89c4ff92SAndroid Build Coastguard Worker     // Convert the counter units into a SWTrace namestring (optional)
501*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> counterUnitsBuffer;
502*89c4ff92SAndroid Build Coastguard Worker     if (includeUnits)
503*89c4ff92SAndroid Build Coastguard Worker     {
504*89c4ff92SAndroid Build Coastguard Worker         // Convert the counter units into a SWTrace namestring
505*89c4ff92SAndroid Build Coastguard Worker         if (!arm::pipe::StringToSwTraceString<arm::pipe::SwTraceNameCharPolicy>(counterUnits, counterUnitsBuffer))
506*89c4ff92SAndroid Build Coastguard Worker         {
507*89c4ff92SAndroid Build Coastguard Worker             errorMessage = fmt::format("Cannot convert the units of counter {} (units: {}) to an SWTrace string",
508*89c4ff92SAndroid Build Coastguard Worker                                        counterUid,
509*89c4ff92SAndroid Build Coastguard Worker                                        counterName);
510*89c4ff92SAndroid Build Coastguard Worker             return false;
511*89c4ff92SAndroid Build Coastguard Worker         }
512*89c4ff92SAndroid Build Coastguard Worker     }
513*89c4ff92SAndroid Build Coastguard Worker 
514*89c4ff92SAndroid Build Coastguard Worker     // Calculate the size in words of the event record
515*89c4ff92SAndroid Build Coastguard Worker     const size_t eventRecordSize = eventRecordBlockSize +
516*89c4ff92SAndroid Build Coastguard Worker                                    counterNameBuffer.size() +        // The size of the variable part (the counter name,
517*89c4ff92SAndroid Build Coastguard Worker                                    counterDescriptionBuffer.size() + // description and units
518*89c4ff92SAndroid Build Coastguard Worker                                    counterUnitsBuffer.size();        // including the null-terminator)
519*89c4ff92SAndroid Build Coastguard Worker 
520*89c4ff92SAndroid Build Coastguard Worker     // Allocate the space for the event record
521*89c4ff92SAndroid Build Coastguard Worker     eventRecord.resize(eventRecordSize);
522*89c4ff92SAndroid Build Coastguard Worker 
523*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_NO_CONVERSION_WARN_BEGIN
524*89c4ff92SAndroid Build Coastguard Worker     // Create the event record
525*89c4ff92SAndroid Build Coastguard Worker     eventRecord[0] = eventRecordWord0; // max_counter_uid + counter_uid
526*89c4ff92SAndroid Build Coastguard Worker     eventRecord[1] = eventRecordWord1; // device + counter_set
527*89c4ff92SAndroid Build Coastguard Worker     eventRecord[2] = eventRecordWord2; // class + interpolation
528*89c4ff92SAndroid Build Coastguard Worker     eventRecord[3] = eventRecordWord3; // multiplier
529*89c4ff92SAndroid Build Coastguard Worker     eventRecord[4] = eventRecordWord4; // multiplier
530*89c4ff92SAndroid Build Coastguard Worker     eventRecord[5] = eventRecordWord5; // name_offset
531*89c4ff92SAndroid Build Coastguard Worker     eventRecord[6] = eventRecordWord6; // description_offset
532*89c4ff92SAndroid Build Coastguard Worker     eventRecord[7] = eventRecordWord7; // units_offset
533*89c4ff92SAndroid Build Coastguard Worker     auto offset = eventRecord.begin() + 8u;
534*89c4ff92SAndroid Build Coastguard Worker     std::copy(counterNameBuffer.begin(), counterNameBuffer.end(), offset); // name
535*89c4ff92SAndroid Build Coastguard Worker     offset += counterNameBuffer.size();
536*89c4ff92SAndroid Build Coastguard Worker     std::copy(counterDescriptionBuffer.begin(), counterDescriptionBuffer.end(), offset); // description
537*89c4ff92SAndroid Build Coastguard Worker     if (includeUnits)
538*89c4ff92SAndroid Build Coastguard Worker     {
539*89c4ff92SAndroid Build Coastguard Worker         offset += counterDescriptionBuffer.size();
540*89c4ff92SAndroid Build Coastguard Worker         std::copy(counterUnitsBuffer.begin(), counterUnitsBuffer.end(), offset); // units
541*89c4ff92SAndroid Build Coastguard Worker     }
542*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_NO_CONVERSION_WARN_END
543*89c4ff92SAndroid Build Coastguard Worker 
544*89c4ff92SAndroid Build Coastguard Worker     return true;
545*89c4ff92SAndroid Build Coastguard Worker }
546*89c4ff92SAndroid Build Coastguard Worker 
SendCounterDirectoryPacket(const ICounterDirectory & counterDirectory)547*89c4ff92SAndroid Build Coastguard Worker void SendCounterPacket::SendCounterDirectoryPacket(const ICounterDirectory& counterDirectory)
548*89c4ff92SAndroid Build Coastguard Worker {
549*89c4ff92SAndroid Build Coastguard Worker     // Get the amount of data that needs to be put into the packet
550*89c4ff92SAndroid Build Coastguard Worker     const uint16_t categoryCount    = counterDirectory.GetCategoryCount();
551*89c4ff92SAndroid Build Coastguard Worker     const uint16_t deviceCount      = counterDirectory.GetDeviceCount();
552*89c4ff92SAndroid Build Coastguard Worker     const uint16_t counterSetCount  = counterDirectory.GetCounterSetCount();
553*89c4ff92SAndroid Build Coastguard Worker 
554*89c4ff92SAndroid Build Coastguard Worker     // Utils
555*89c4ff92SAndroid Build Coastguard Worker     const size_t uint32_t_size = sizeof(uint32_t);
556*89c4ff92SAndroid Build Coastguard Worker     const size_t packetHeaderSize = 2u;
557*89c4ff92SAndroid Build Coastguard Worker     const size_t bodyHeaderSize = 6u;
558*89c4ff92SAndroid Build Coastguard Worker     const uint32_t bodyHeaderSizeBytes = bodyHeaderSize * uint32_t_size;
559*89c4ff92SAndroid Build Coastguard Worker 
560*89c4ff92SAndroid Build Coastguard Worker     // Initialize the offset for the pointer tables
561*89c4ff92SAndroid Build Coastguard Worker     uint32_t pointerTableOffset = 0;
562*89c4ff92SAndroid Build Coastguard Worker 
563*89c4ff92SAndroid Build Coastguard Worker     // --------------
564*89c4ff92SAndroid Build Coastguard Worker     // Device records
565*89c4ff92SAndroid Build Coastguard Worker     // --------------
566*89c4ff92SAndroid Build Coastguard Worker 
567*89c4ff92SAndroid Build Coastguard Worker     // Process device records
568*89c4ff92SAndroid Build Coastguard Worker     std::vector<DeviceRecord> deviceRecords(deviceCount);
569*89c4ff92SAndroid Build Coastguard Worker     const Devices& devices = counterDirectory.GetDevices();
570*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> deviceRecordOffsets(deviceCount, 0); // device_records_pointer_table
571*89c4ff92SAndroid Build Coastguard Worker     size_t deviceRecordsSize = 0;
572*89c4ff92SAndroid Build Coastguard Worker     size_t deviceIndex = 0;
573*89c4ff92SAndroid Build Coastguard Worker     size_t deviceRecordOffsetIndex = 0;
574*89c4ff92SAndroid Build Coastguard Worker 
575*89c4ff92SAndroid Build Coastguard Worker     pointerTableOffset = arm::pipe::numeric_cast<uint32_t>(deviceCount * uint32_t_size +
576*89c4ff92SAndroid Build Coastguard Worker                                                        counterSetCount * uint32_t_size +
577*89c4ff92SAndroid Build Coastguard Worker                                                        categoryCount   * uint32_t_size);
578*89c4ff92SAndroid Build Coastguard Worker     for (auto it = devices.begin(); it != devices.end(); it++)
579*89c4ff92SAndroid Build Coastguard Worker     {
580*89c4ff92SAndroid Build Coastguard Worker         const DevicePtr& device = it->second;
581*89c4ff92SAndroid Build Coastguard Worker         DeviceRecord& deviceRecord = deviceRecords.at(deviceIndex);
582*89c4ff92SAndroid Build Coastguard Worker 
583*89c4ff92SAndroid Build Coastguard Worker         std::string errorMessage;
584*89c4ff92SAndroid Build Coastguard Worker         if (!CreateDeviceRecord(device, deviceRecord, errorMessage))
585*89c4ff92SAndroid Build Coastguard Worker         {
586*89c4ff92SAndroid Build Coastguard Worker             CancelOperationAndThrow<arm::pipe::ProfilingException>(errorMessage);
587*89c4ff92SAndroid Build Coastguard Worker         }
588*89c4ff92SAndroid Build Coastguard Worker 
589*89c4ff92SAndroid Build Coastguard Worker         // Update the total size in words of the device records
590*89c4ff92SAndroid Build Coastguard Worker         deviceRecordsSize += deviceRecord.size();
591*89c4ff92SAndroid Build Coastguard Worker 
592*89c4ff92SAndroid Build Coastguard Worker         // Add the device record offset to the device records pointer table offset field
593*89c4ff92SAndroid Build Coastguard Worker         deviceRecordOffsets[deviceRecordOffsetIndex] = pointerTableOffset;
594*89c4ff92SAndroid Build Coastguard Worker         pointerTableOffset += arm::pipe::numeric_cast<uint32_t>(deviceRecord.size() * uint32_t_size);
595*89c4ff92SAndroid Build Coastguard Worker 
596*89c4ff92SAndroid Build Coastguard Worker         deviceIndex++;
597*89c4ff92SAndroid Build Coastguard Worker         deviceRecordOffsetIndex++;
598*89c4ff92SAndroid Build Coastguard Worker     }
599*89c4ff92SAndroid Build Coastguard Worker 
600*89c4ff92SAndroid Build Coastguard Worker     // -------------------
601*89c4ff92SAndroid Build Coastguard Worker     // Counter set records
602*89c4ff92SAndroid Build Coastguard Worker     // -------------------
603*89c4ff92SAndroid Build Coastguard Worker 
604*89c4ff92SAndroid Build Coastguard Worker     // Process counter set records
605*89c4ff92SAndroid Build Coastguard Worker     std::vector<CounterSetRecord> counterSetRecords(counterSetCount);
606*89c4ff92SAndroid Build Coastguard Worker     const CounterSets& counterSets = counterDirectory.GetCounterSets();
607*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> counterSetRecordOffsets(counterSetCount, 0); // counter_set_records_pointer_table
608*89c4ff92SAndroid Build Coastguard Worker     size_t counterSetRecordsSize = 0;
609*89c4ff92SAndroid Build Coastguard Worker     size_t counterSetIndex = 0;
610*89c4ff92SAndroid Build Coastguard Worker     size_t counterSetRecordOffsetIndex = 0;
611*89c4ff92SAndroid Build Coastguard Worker 
612*89c4ff92SAndroid Build Coastguard Worker     pointerTableOffset -= arm::pipe::numeric_cast<uint32_t>(deviceCount * uint32_t_size);
613*89c4ff92SAndroid Build Coastguard Worker     for (auto it = counterSets.begin(); it != counterSets.end(); it++)
614*89c4ff92SAndroid Build Coastguard Worker     {
615*89c4ff92SAndroid Build Coastguard Worker         const CounterSetPtr& counterSet = it->second;
616*89c4ff92SAndroid Build Coastguard Worker         CounterSetRecord& counterSetRecord = counterSetRecords.at(counterSetIndex);
617*89c4ff92SAndroid Build Coastguard Worker 
618*89c4ff92SAndroid Build Coastguard Worker         std::string errorMessage;
619*89c4ff92SAndroid Build Coastguard Worker         if (!CreateCounterSetRecord(counterSet, counterSetRecord, errorMessage))
620*89c4ff92SAndroid Build Coastguard Worker         {
621*89c4ff92SAndroid Build Coastguard Worker             CancelOperationAndThrow<arm::pipe::ProfilingException>(errorMessage);
622*89c4ff92SAndroid Build Coastguard Worker         }
623*89c4ff92SAndroid Build Coastguard Worker 
624*89c4ff92SAndroid Build Coastguard Worker         // Update the total size in words of the counter set records
625*89c4ff92SAndroid Build Coastguard Worker         counterSetRecordsSize += counterSetRecord.size();
626*89c4ff92SAndroid Build Coastguard Worker 
627*89c4ff92SAndroid Build Coastguard Worker         // Add the counter set record offset to the counter set records pointer table offset field
628*89c4ff92SAndroid Build Coastguard Worker         counterSetRecordOffsets[counterSetRecordOffsetIndex] = pointerTableOffset;
629*89c4ff92SAndroid Build Coastguard Worker         pointerTableOffset += arm::pipe::numeric_cast<uint32_t>(counterSetRecord.size() * uint32_t_size);
630*89c4ff92SAndroid Build Coastguard Worker 
631*89c4ff92SAndroid Build Coastguard Worker         counterSetIndex++;
632*89c4ff92SAndroid Build Coastguard Worker         counterSetRecordOffsetIndex++;
633*89c4ff92SAndroid Build Coastguard Worker     }
634*89c4ff92SAndroid Build Coastguard Worker 
635*89c4ff92SAndroid Build Coastguard Worker     // ----------------
636*89c4ff92SAndroid Build Coastguard Worker     // Category records
637*89c4ff92SAndroid Build Coastguard Worker     // ----------------
638*89c4ff92SAndroid Build Coastguard Worker 
639*89c4ff92SAndroid Build Coastguard Worker     // Process category records
640*89c4ff92SAndroid Build Coastguard Worker     std::vector<CategoryRecord> categoryRecords(categoryCount);
641*89c4ff92SAndroid Build Coastguard Worker     const Categories& categories = counterDirectory.GetCategories();
642*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> categoryRecordOffsets(categoryCount, 0); // category_records_pointer_table
643*89c4ff92SAndroid Build Coastguard Worker     size_t categoryRecordsSize = 0;
644*89c4ff92SAndroid Build Coastguard Worker     size_t categoryIndex = 0;
645*89c4ff92SAndroid Build Coastguard Worker     size_t categoryRecordOffsetIndex = 0;
646*89c4ff92SAndroid Build Coastguard Worker 
647*89c4ff92SAndroid Build Coastguard Worker     pointerTableOffset -= arm::pipe::numeric_cast<uint32_t>(counterSetCount * uint32_t_size);
648*89c4ff92SAndroid Build Coastguard Worker     for (auto it = categories.begin(); it != categories.end(); it++)
649*89c4ff92SAndroid Build Coastguard Worker     {
650*89c4ff92SAndroid Build Coastguard Worker         const CategoryPtr& category = *it;
651*89c4ff92SAndroid Build Coastguard Worker         CategoryRecord& categoryRecord = categoryRecords.at(categoryIndex);
652*89c4ff92SAndroid Build Coastguard Worker 
653*89c4ff92SAndroid Build Coastguard Worker         std::string errorMessage;
654*89c4ff92SAndroid Build Coastguard Worker         if (!CreateCategoryRecord(category, counterDirectory.GetCounters(), categoryRecord, errorMessage))
655*89c4ff92SAndroid Build Coastguard Worker         {
656*89c4ff92SAndroid Build Coastguard Worker             CancelOperationAndThrow<arm::pipe::ProfilingException>(errorMessage);
657*89c4ff92SAndroid Build Coastguard Worker         }
658*89c4ff92SAndroid Build Coastguard Worker 
659*89c4ff92SAndroid Build Coastguard Worker         // Update the total size in words of the category records
660*89c4ff92SAndroid Build Coastguard Worker         categoryRecordsSize += categoryRecord.size();
661*89c4ff92SAndroid Build Coastguard Worker 
662*89c4ff92SAndroid Build Coastguard Worker         // Add the category record offset to the category records pointer table offset field
663*89c4ff92SAndroid Build Coastguard Worker         categoryRecordOffsets[categoryRecordOffsetIndex] = pointerTableOffset;
664*89c4ff92SAndroid Build Coastguard Worker         pointerTableOffset += arm::pipe::numeric_cast<uint32_t>(categoryRecord.size() * uint32_t_size);
665*89c4ff92SAndroid Build Coastguard Worker 
666*89c4ff92SAndroid Build Coastguard Worker         categoryIndex++;
667*89c4ff92SAndroid Build Coastguard Worker         categoryRecordOffsetIndex++;
668*89c4ff92SAndroid Build Coastguard Worker     }
669*89c4ff92SAndroid Build Coastguard Worker 
670*89c4ff92SAndroid Build Coastguard Worker     // Calculate the length in words of the counter directory packet's data (excludes the packet header size)
671*89c4ff92SAndroid Build Coastguard Worker     const size_t counterDirectoryPacketDataLength =
672*89c4ff92SAndroid Build Coastguard Worker                  bodyHeaderSize +                 // The size of the body header
673*89c4ff92SAndroid Build Coastguard Worker                  deviceRecordOffsets.size() +     // The size of the device records pointer table
674*89c4ff92SAndroid Build Coastguard Worker                  counterSetRecordOffsets.size() + // The size of counter set pointer table
675*89c4ff92SAndroid Build Coastguard Worker                  categoryRecordOffsets.size() +   // The size of category records pointer table
676*89c4ff92SAndroid Build Coastguard Worker                  deviceRecordsSize +              // The total size of the device records
677*89c4ff92SAndroid Build Coastguard Worker                  counterSetRecordsSize +          // The total size of the counter set records
678*89c4ff92SAndroid Build Coastguard Worker                  categoryRecordsSize;             // The total size of the category records
679*89c4ff92SAndroid Build Coastguard Worker 
680*89c4ff92SAndroid Build Coastguard Worker     // Calculate the size in words of the counter directory packet (the data length plus the packet header size)
681*89c4ff92SAndroid Build Coastguard Worker     const size_t counterDirectoryPacketSize = packetHeaderSize +                // The size of the packet header
682*89c4ff92SAndroid Build Coastguard Worker                                               counterDirectoryPacketDataLength; // The data length
683*89c4ff92SAndroid Build Coastguard Worker 
684*89c4ff92SAndroid Build Coastguard Worker     // Allocate the necessary space for the counter directory packet
685*89c4ff92SAndroid Build Coastguard Worker     std::vector<uint32_t> counterDirectoryPacket(counterDirectoryPacketSize, 0);
686*89c4ff92SAndroid Build Coastguard Worker 
687*89c4ff92SAndroid Build Coastguard Worker     // -------------
688*89c4ff92SAndroid Build Coastguard Worker     // Packet header
689*89c4ff92SAndroid Build Coastguard Worker     // -------------
690*89c4ff92SAndroid Build Coastguard Worker 
691*89c4ff92SAndroid Build Coastguard Worker     // Packet header word 0:
692*89c4ff92SAndroid Build Coastguard Worker     // 26:31 [6]  packet_family: control Packet Family
693*89c4ff92SAndroid Build Coastguard Worker     // 16:25 [10] packet_id: packet identifier
694*89c4ff92SAndroid Build Coastguard Worker     // 8:15  [8]  reserved: all zeros
695*89c4ff92SAndroid Build Coastguard Worker     // 0:7   [8]  reserved: all zeros
696*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetFamily = 0;
697*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetId = 2;
698*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetHeaderWord0 = ((packetFamily & 0x3F) << 26) | ((packetId & 0x3FF) << 16);
699*89c4ff92SAndroid Build Coastguard Worker 
700*89c4ff92SAndroid Build Coastguard Worker     // Packet header word 1:
701*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] data_length: length of data, in bytes
702*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetHeaderWord1 = arm::pipe::numeric_cast<uint32_t>(
703*89c4ff92SAndroid Build Coastguard Worker             counterDirectoryPacketDataLength * uint32_t_size);
704*89c4ff92SAndroid Build Coastguard Worker 
705*89c4ff92SAndroid Build Coastguard Worker     // Create the packet header
706*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetHeader[2]
707*89c4ff92SAndroid Build Coastguard Worker     {
708*89c4ff92SAndroid Build Coastguard Worker         packetHeaderWord0, // packet_family + packet_id + reserved + reserved
709*89c4ff92SAndroid Build Coastguard Worker         packetHeaderWord1  // data_length
710*89c4ff92SAndroid Build Coastguard Worker     };
711*89c4ff92SAndroid Build Coastguard Worker 
712*89c4ff92SAndroid Build Coastguard Worker     // -----------
713*89c4ff92SAndroid Build Coastguard Worker     // Body header
714*89c4ff92SAndroid Build Coastguard Worker     // -----------
715*89c4ff92SAndroid Build Coastguard Worker 
716*89c4ff92SAndroid Build Coastguard Worker     // Body header word 0:
717*89c4ff92SAndroid Build Coastguard Worker     // 16:31 [16] device_records_count: number of entries in the device_records_pointer_table
718*89c4ff92SAndroid Build Coastguard Worker     // 0:15  [16] reserved: all zeros
719*89c4ff92SAndroid Build Coastguard Worker     const uint32_t bodyHeaderWord0 = static_cast<uint32_t>(deviceCount) << 16;
720*89c4ff92SAndroid Build Coastguard Worker 
721*89c4ff92SAndroid Build Coastguard Worker     // Body header word 1:
722*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] device_records_pointer_table_offset: offset to the device_records_pointer_table
723*89c4ff92SAndroid Build Coastguard Worker     const uint32_t bodyHeaderWord1 = bodyHeaderSizeBytes; // The offset is always the bodyHeaderSize,
724*89c4ff92SAndroid Build Coastguard Worker                                                           // as the device record pointer table field
725*89c4ff92SAndroid Build Coastguard Worker                                                           // is always the first item in the pool
726*89c4ff92SAndroid Build Coastguard Worker 
727*89c4ff92SAndroid Build Coastguard Worker     // Body header word 2:
728*89c4ff92SAndroid Build Coastguard Worker     // 16:31 [16] counter_set_count: number of entries in the counter_set_pointer_table
729*89c4ff92SAndroid Build Coastguard Worker     // 0:15  [16] reserved: all zeros
730*89c4ff92SAndroid Build Coastguard Worker     const uint32_t bodyHeaderWord2 = static_cast<uint32_t>(counterSetCount) << 16;
731*89c4ff92SAndroid Build Coastguard Worker 
732*89c4ff92SAndroid Build Coastguard Worker     // Body header word 3:
733*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] counter_set_pointer_table_offset: offset to the counter_set_pointer_table
734*89c4ff92SAndroid Build Coastguard Worker     const uint32_t bodyHeaderWord3 = arm::pipe::numeric_cast<uint32_t>(deviceRecordOffsets.size() *
735*89c4ff92SAndroid Build Coastguard Worker                                                                    uint32_t_size +       // The size of the
736*89c4ff92SAndroid Build Coastguard Worker                                                                    bodyHeaderSizeBytes); // device records pointer table
737*89c4ff92SAndroid Build Coastguard Worker 
738*89c4ff92SAndroid Build Coastguard Worker     // Body header word 4:
739*89c4ff92SAndroid Build Coastguard Worker     // 16:31 [16] categories_count: number of entries in the categories_pointer_table
740*89c4ff92SAndroid Build Coastguard Worker     // 0:15  [16] reserved: all zeros
741*89c4ff92SAndroid Build Coastguard Worker     const uint32_t bodyHeaderWord4 = static_cast<uint32_t>(categoryCount) << 16;
742*89c4ff92SAndroid Build Coastguard Worker 
743*89c4ff92SAndroid Build Coastguard Worker     // Body header word 3:
744*89c4ff92SAndroid Build Coastguard Worker     // 0:31 [32] categories_pointer_table_offset: offset to the categories_pointer_table
745*89c4ff92SAndroid Build Coastguard Worker     const uint32_t bodyHeaderWord5 =
746*89c4ff92SAndroid Build Coastguard Worker                    arm::pipe::numeric_cast<uint32_t>(
747*89c4ff92SAndroid Build Coastguard Worker                        deviceRecordOffsets.size() * uint32_t_size +     // The size of the device records
748*89c4ff92SAndroid Build Coastguard Worker                        counterSetRecordOffsets.size() * uint32_t_size   // pointer table, plus the size of
749*89c4ff92SAndroid Build Coastguard Worker                        +  bodyHeaderSizeBytes);                         // the counter set pointer table
750*89c4ff92SAndroid Build Coastguard Worker 
751*89c4ff92SAndroid Build Coastguard Worker     // Create the body header
752*89c4ff92SAndroid Build Coastguard Worker     const uint32_t bodyHeader[bodyHeaderSize]
753*89c4ff92SAndroid Build Coastguard Worker     {
754*89c4ff92SAndroid Build Coastguard Worker         bodyHeaderWord0, // device_records_count + reserved
755*89c4ff92SAndroid Build Coastguard Worker         bodyHeaderWord1, // device_records_pointer_table_offset
756*89c4ff92SAndroid Build Coastguard Worker         bodyHeaderWord2, // counter_set_count + reserved
757*89c4ff92SAndroid Build Coastguard Worker         bodyHeaderWord3, // counter_set_pointer_table_offset
758*89c4ff92SAndroid Build Coastguard Worker         bodyHeaderWord4, // categories_count + reserved
759*89c4ff92SAndroid Build Coastguard Worker         bodyHeaderWord5  // categories_pointer_table_offset
760*89c4ff92SAndroid Build Coastguard Worker     };
761*89c4ff92SAndroid Build Coastguard Worker 
762*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_NO_CONVERSION_WARN_BEGIN
763*89c4ff92SAndroid Build Coastguard Worker     // Create the counter directory packet
764*89c4ff92SAndroid Build Coastguard Worker     auto counterDirectoryPacketOffset = counterDirectoryPacket.begin();
765*89c4ff92SAndroid Build Coastguard Worker     // packet_header
766*89c4ff92SAndroid Build Coastguard Worker     std::copy(packetHeader, packetHeader + packetHeaderSize, counterDirectoryPacketOffset);
767*89c4ff92SAndroid Build Coastguard Worker     counterDirectoryPacketOffset += packetHeaderSize;
768*89c4ff92SAndroid Build Coastguard Worker     // body_header
769*89c4ff92SAndroid Build Coastguard Worker     std::copy(bodyHeader, bodyHeader + bodyHeaderSize, counterDirectoryPacketOffset);
770*89c4ff92SAndroid Build Coastguard Worker     counterDirectoryPacketOffset += bodyHeaderSize;
771*89c4ff92SAndroid Build Coastguard Worker     // device_records_pointer_table
772*89c4ff92SAndroid Build Coastguard Worker     std::copy(deviceRecordOffsets.begin(), deviceRecordOffsets.end(), counterDirectoryPacketOffset);
773*89c4ff92SAndroid Build Coastguard Worker     counterDirectoryPacketOffset += deviceRecordOffsets.size();
774*89c4ff92SAndroid Build Coastguard Worker     // counter_set_pointer_table
775*89c4ff92SAndroid Build Coastguard Worker     std::copy(counterSetRecordOffsets.begin(), counterSetRecordOffsets.end(), counterDirectoryPacketOffset);
776*89c4ff92SAndroid Build Coastguard Worker     counterDirectoryPacketOffset += counterSetRecordOffsets.size();
777*89c4ff92SAndroid Build Coastguard Worker     // category_pointer_table
778*89c4ff92SAndroid Build Coastguard Worker     std::copy(categoryRecordOffsets.begin(), categoryRecordOffsets.end(), counterDirectoryPacketOffset);
779*89c4ff92SAndroid Build Coastguard Worker     counterDirectoryPacketOffset += categoryRecordOffsets.size();
780*89c4ff92SAndroid Build Coastguard Worker     // device_records
781*89c4ff92SAndroid Build Coastguard Worker     for (const DeviceRecord& deviceRecord : deviceRecords)
782*89c4ff92SAndroid Build Coastguard Worker     {
783*89c4ff92SAndroid Build Coastguard Worker         std::copy(deviceRecord.begin(), deviceRecord.end(), counterDirectoryPacketOffset); // device_record
784*89c4ff92SAndroid Build Coastguard Worker         counterDirectoryPacketOffset += deviceRecord.size();
785*89c4ff92SAndroid Build Coastguard Worker     }
786*89c4ff92SAndroid Build Coastguard Worker     // counter_set_records
787*89c4ff92SAndroid Build Coastguard Worker     for (const CounterSetRecord& counterSetRecord : counterSetRecords)
788*89c4ff92SAndroid Build Coastguard Worker     {
789*89c4ff92SAndroid Build Coastguard Worker         std::copy(counterSetRecord.begin(), counterSetRecord.end(), counterDirectoryPacketOffset); // counter_set_record
790*89c4ff92SAndroid Build Coastguard Worker         counterDirectoryPacketOffset += counterSetRecord.size();
791*89c4ff92SAndroid Build Coastguard Worker     }
792*89c4ff92SAndroid Build Coastguard Worker     // category_records
793*89c4ff92SAndroid Build Coastguard Worker     for (const CategoryRecord& categoryRecord : categoryRecords)
794*89c4ff92SAndroid Build Coastguard Worker     {
795*89c4ff92SAndroid Build Coastguard Worker         std::copy(categoryRecord.begin(), categoryRecord.end(), counterDirectoryPacketOffset); // category_record
796*89c4ff92SAndroid Build Coastguard Worker         counterDirectoryPacketOffset += categoryRecord.size();
797*89c4ff92SAndroid Build Coastguard Worker     }
798*89c4ff92SAndroid Build Coastguard Worker     ARM_PIPE_NO_CONVERSION_WARN_END
799*89c4ff92SAndroid Build Coastguard Worker 
800*89c4ff92SAndroid Build Coastguard Worker     // Calculate the total size in bytes of the counter directory packet
801*89c4ff92SAndroid Build Coastguard Worker     uint32_t totalSize = arm::pipe::numeric_cast<uint32_t>(counterDirectoryPacketSize * uint32_t_size);
802*89c4ff92SAndroid Build Coastguard Worker 
803*89c4ff92SAndroid Build Coastguard Worker     // Reserve space in the buffer for the packet
804*89c4ff92SAndroid Build Coastguard Worker     uint32_t reserved = 0;
805*89c4ff92SAndroid Build Coastguard Worker     IPacketBufferPtr writeBuffer = m_BufferManager.Reserve(totalSize, reserved);
806*89c4ff92SAndroid Build Coastguard Worker 
807*89c4ff92SAndroid Build Coastguard Worker     if (writeBuffer == nullptr || reserved < totalSize)
808*89c4ff92SAndroid Build Coastguard Worker     {
809*89c4ff92SAndroid Build Coastguard Worker         CancelOperationAndThrow<arm::pipe::BufferExhaustion>(
810*89c4ff92SAndroid Build Coastguard Worker             writeBuffer,
811*89c4ff92SAndroid Build Coastguard Worker             fmt::format("No space left in buffer. Unable to reserve ({}) bytes.", totalSize));
812*89c4ff92SAndroid Build Coastguard Worker     }
813*89c4ff92SAndroid Build Coastguard Worker 
814*89c4ff92SAndroid Build Coastguard Worker     // Offset for writing to the buffer
815*89c4ff92SAndroid Build Coastguard Worker     uint32_t offset = 0;
816*89c4ff92SAndroid Build Coastguard Worker 
817*89c4ff92SAndroid Build Coastguard Worker     // Write the counter directory packet to the buffer
818*89c4ff92SAndroid Build Coastguard Worker     for (uint32_t counterDirectoryPacketWord : counterDirectoryPacket)
819*89c4ff92SAndroid Build Coastguard Worker     {
820*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, counterDirectoryPacketWord);
821*89c4ff92SAndroid Build Coastguard Worker         offset += arm::pipe::numeric_cast<uint32_t>(uint32_t_size);
822*89c4ff92SAndroid Build Coastguard Worker     }
823*89c4ff92SAndroid Build Coastguard Worker 
824*89c4ff92SAndroid Build Coastguard Worker     m_BufferManager.Commit(writeBuffer, totalSize);
825*89c4ff92SAndroid Build Coastguard Worker }
826*89c4ff92SAndroid Build Coastguard Worker 
SendPeriodicCounterCapturePacket(uint64_t timestamp,const IndexValuePairsVector & values)827*89c4ff92SAndroid Build Coastguard Worker void SendCounterPacket::SendPeriodicCounterCapturePacket(uint64_t timestamp, const IndexValuePairsVector& values)
828*89c4ff92SAndroid Build Coastguard Worker {
829*89c4ff92SAndroid Build Coastguard Worker     uint32_t uint16_t_size = sizeof(uint16_t);
830*89c4ff92SAndroid Build Coastguard Worker     uint32_t uint32_t_size = sizeof(uint32_t);
831*89c4ff92SAndroid Build Coastguard Worker     uint32_t uint64_t_size = sizeof(uint64_t);
832*89c4ff92SAndroid Build Coastguard Worker 
833*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetFamily = 3;
834*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetClass = 0;
835*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetType = 0;
836*89c4ff92SAndroid Build Coastguard Worker     uint32_t headerSize = 2 * uint32_t_size;
837*89c4ff92SAndroid Build Coastguard Worker     uint32_t bodySize = uint64_t_size + arm::pipe::numeric_cast<uint32_t>(
838*89c4ff92SAndroid Build Coastguard Worker         values.size()) * (uint16_t_size + uint32_t_size);
839*89c4ff92SAndroid Build Coastguard Worker     uint32_t totalSize = headerSize + bodySize;
840*89c4ff92SAndroid Build Coastguard Worker     uint32_t offset = 0;
841*89c4ff92SAndroid Build Coastguard Worker     uint32_t reserved = 0;
842*89c4ff92SAndroid Build Coastguard Worker 
843*89c4ff92SAndroid Build Coastguard Worker     IPacketBufferPtr writeBuffer = m_BufferManager.Reserve(totalSize, reserved);
844*89c4ff92SAndroid Build Coastguard Worker 
845*89c4ff92SAndroid Build Coastguard Worker     if (writeBuffer == nullptr || reserved < totalSize)
846*89c4ff92SAndroid Build Coastguard Worker     {
847*89c4ff92SAndroid Build Coastguard Worker         CancelOperationAndThrow<arm::pipe::BufferExhaustion>(
848*89c4ff92SAndroid Build Coastguard Worker             writeBuffer,
849*89c4ff92SAndroid Build Coastguard Worker             fmt::format("No space left in buffer. Unable to reserve ({}) bytes.", totalSize));
850*89c4ff92SAndroid Build Coastguard Worker     }
851*89c4ff92SAndroid Build Coastguard Worker 
852*89c4ff92SAndroid Build Coastguard Worker     // Create header.
853*89c4ff92SAndroid Build Coastguard Worker     WriteUint32(writeBuffer,
854*89c4ff92SAndroid Build Coastguard Worker                 offset,
855*89c4ff92SAndroid Build Coastguard Worker                 ((packetFamily & 0x0000003F) << 26) |
856*89c4ff92SAndroid Build Coastguard Worker                 ((packetClass  & 0x0000007F) << 19) |
857*89c4ff92SAndroid Build Coastguard Worker                 ((packetType   & 0x00000007) << 16));
858*89c4ff92SAndroid Build Coastguard Worker     offset += uint32_t_size;
859*89c4ff92SAndroid Build Coastguard Worker     WriteUint32(writeBuffer, offset, bodySize);
860*89c4ff92SAndroid Build Coastguard Worker 
861*89c4ff92SAndroid Build Coastguard Worker     // Copy captured Timestamp.
862*89c4ff92SAndroid Build Coastguard Worker     offset += uint32_t_size;
863*89c4ff92SAndroid Build Coastguard Worker     WriteUint64(writeBuffer, offset, timestamp);
864*89c4ff92SAndroid Build Coastguard Worker 
865*89c4ff92SAndroid Build Coastguard Worker     // Copy selectedCounterIds.
866*89c4ff92SAndroid Build Coastguard Worker     offset += uint64_t_size;
867*89c4ff92SAndroid Build Coastguard Worker     for (const auto& pair: values)
868*89c4ff92SAndroid Build Coastguard Worker     {
869*89c4ff92SAndroid Build Coastguard Worker         WriteUint16(writeBuffer, offset, pair.counterId);
870*89c4ff92SAndroid Build Coastguard Worker         offset += uint16_t_size;
871*89c4ff92SAndroid Build Coastguard Worker         WriteUint32(writeBuffer, offset, pair.counterValue);
872*89c4ff92SAndroid Build Coastguard Worker         offset += uint32_t_size;
873*89c4ff92SAndroid Build Coastguard Worker     }
874*89c4ff92SAndroid Build Coastguard Worker 
875*89c4ff92SAndroid Build Coastguard Worker     m_BufferManager.Commit(writeBuffer, totalSize);
876*89c4ff92SAndroid Build Coastguard Worker }
877*89c4ff92SAndroid Build Coastguard Worker 
SendPeriodicCounterSelectionPacket(uint32_t capturePeriod,const std::vector<uint16_t> & selectedCounterIds)878*89c4ff92SAndroid Build Coastguard Worker void SendCounterPacket::SendPeriodicCounterSelectionPacket(uint32_t capturePeriod,
879*89c4ff92SAndroid Build Coastguard Worker                                                            const std::vector<uint16_t>& selectedCounterIds)
880*89c4ff92SAndroid Build Coastguard Worker {
881*89c4ff92SAndroid Build Coastguard Worker     uint32_t uint16_t_size = sizeof(uint16_t);
882*89c4ff92SAndroid Build Coastguard Worker     uint32_t uint32_t_size = sizeof(uint32_t);
883*89c4ff92SAndroid Build Coastguard Worker 
884*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetFamily = 0;
885*89c4ff92SAndroid Build Coastguard Worker     uint32_t packetId = 4;
886*89c4ff92SAndroid Build Coastguard Worker     uint32_t headerSize = 2 * uint32_t_size;
887*89c4ff92SAndroid Build Coastguard Worker     uint32_t bodySize = uint32_t_size + arm::pipe::numeric_cast<uint32_t>(selectedCounterIds.size()) * uint16_t_size;
888*89c4ff92SAndroid Build Coastguard Worker     uint32_t totalSize = headerSize + bodySize;
889*89c4ff92SAndroid Build Coastguard Worker     uint32_t offset = 0;
890*89c4ff92SAndroid Build Coastguard Worker     uint32_t reserved = 0;
891*89c4ff92SAndroid Build Coastguard Worker 
892*89c4ff92SAndroid Build Coastguard Worker     IPacketBufferPtr writeBuffer = m_BufferManager.Reserve(totalSize, reserved);
893*89c4ff92SAndroid Build Coastguard Worker 
894*89c4ff92SAndroid Build Coastguard Worker     if (writeBuffer == nullptr || reserved < totalSize)
895*89c4ff92SAndroid Build Coastguard Worker     {
896*89c4ff92SAndroid Build Coastguard Worker         CancelOperationAndThrow<arm::pipe::BufferExhaustion>(
897*89c4ff92SAndroid Build Coastguard Worker             writeBuffer,
898*89c4ff92SAndroid Build Coastguard Worker             fmt::format("No space left in buffer. Unable to reserve ({}) bytes.", totalSize));
899*89c4ff92SAndroid Build Coastguard Worker     }
900*89c4ff92SAndroid Build Coastguard Worker 
901*89c4ff92SAndroid Build Coastguard Worker     // Create header.
902*89c4ff92SAndroid Build Coastguard Worker     WriteUint32(writeBuffer, offset, ((packetFamily & 0x3F) << 26) | ((packetId & 0x3FF) << 16));
903*89c4ff92SAndroid Build Coastguard Worker     offset += uint32_t_size;
904*89c4ff92SAndroid Build Coastguard Worker     WriteUint32(writeBuffer, offset, bodySize);
905*89c4ff92SAndroid Build Coastguard Worker 
906*89c4ff92SAndroid Build Coastguard Worker     // Copy capturePeriod.
907*89c4ff92SAndroid Build Coastguard Worker     offset += uint32_t_size;
908*89c4ff92SAndroid Build Coastguard Worker     WriteUint32(writeBuffer, offset, capturePeriod);
909*89c4ff92SAndroid Build Coastguard Worker 
910*89c4ff92SAndroid Build Coastguard Worker     // Copy selectedCounterIds.
911*89c4ff92SAndroid Build Coastguard Worker     offset += uint32_t_size;
912*89c4ff92SAndroid Build Coastguard Worker     for(const uint16_t& id: selectedCounterIds)
913*89c4ff92SAndroid Build Coastguard Worker     {
914*89c4ff92SAndroid Build Coastguard Worker         WriteUint16(writeBuffer, offset, id);
915*89c4ff92SAndroid Build Coastguard Worker         offset += uint16_t_size;
916*89c4ff92SAndroid Build Coastguard Worker     }
917*89c4ff92SAndroid Build Coastguard Worker 
918*89c4ff92SAndroid Build Coastguard Worker     m_BufferManager.Commit(writeBuffer, totalSize);
919*89c4ff92SAndroid Build Coastguard Worker }
920*89c4ff92SAndroid Build Coastguard Worker 
921*89c4ff92SAndroid Build Coastguard Worker } // namespace pipe
922*89c4ff92SAndroid Build Coastguard Worker 
923*89c4ff92SAndroid Build Coastguard Worker } // namespace arm
924