ArmNN  NotReleased
SendCounterPacket Class Reference

#include <SendCounterPacket.hpp>

Inheritance diagram for SendCounterPacket:
ISendCounterPacket SendCounterPacketTest

Public Types

using CategoryRecord = std::vector< uint32_t >
 
using DeviceRecord = std::vector< uint32_t >
 
using CounterSetRecord = std::vector< uint32_t >
 
using EventRecord = std::vector< uint32_t >
 
using IndexValuePairsVector = std::vector< CounterValue >
 
- Public Types inherited from ISendCounterPacket
using IndexValuePairsVector = std::vector< CounterValue >
 

Public Member Functions

 SendCounterPacket (IBufferManager &buffer)
 
void SendStreamMetaDataPacket () override
 Create and write a StreamMetaDataPacket in the buffer. More...
 
void SendCounterDirectoryPacket (const ICounterDirectory &counterDirectory) override
 Create and write a CounterDirectoryPacket from the parameters to the buffer. More...
 
void SendPeriodicCounterCapturePacket (uint64_t timestamp, const IndexValuePairsVector &values) override
 Create and write a PeriodicCounterCapturePacket from the parameters to the buffer. More...
 
void SendPeriodicCounterSelectionPacket (uint32_t capturePeriod, const std::vector< uint16_t > &selectedCounterIds) override
 Create and write a PeriodicCounterSelectionPacket from the parameters to the buffer. More...
 
- Public Member Functions inherited from ISendCounterPacket
virtual ~ISendCounterPacket ()
 

Static Public Attributes

static const unsigned int PIPE_MAGIC = 0x45495434
 

Protected Member Functions

bool CreateCategoryRecord (const CategoryPtr &category, const Counters &counters, CategoryRecord &categoryRecord, std::string &errorMessage)
 
bool CreateDeviceRecord (const DevicePtr &device, DeviceRecord &deviceRecord, std::string &errorMessage)
 
bool CreateCounterSetRecord (const CounterSetPtr &counterSet, CounterSetRecord &counterSetRecord, std::string &errorMessage)
 
bool CreateEventRecord (const CounterPtr &counter, EventRecord &eventRecord, std::string &errorMessage)
 

Detailed Description

Definition at line 21 of file SendCounterPacket.hpp.

Member Typedef Documentation

◆ CategoryRecord

using CategoryRecord = std::vector<uint32_t>

Definition at line 24 of file SendCounterPacket.hpp.

◆ CounterSetRecord

using CounterSetRecord = std::vector<uint32_t>

Definition at line 26 of file SendCounterPacket.hpp.

◆ DeviceRecord

using DeviceRecord = std::vector<uint32_t>

Definition at line 25 of file SendCounterPacket.hpp.

◆ EventRecord

using EventRecord = std::vector<uint32_t>

Definition at line 27 of file SendCounterPacket.hpp.

◆ IndexValuePairsVector

using IndexValuePairsVector = std::vector<CounterValue>

Definition at line 28 of file SendCounterPacket.hpp.

Constructor & Destructor Documentation

◆ SendCounterPacket()

Member Function Documentation

◆ CreateCategoryRecord()

bool CreateCategoryRecord ( const CategoryPtr category,
const Counters counters,
CategoryRecord categoryRecord,
std::string &  errorMessage 
)
protected

Definition at line 174 of file SendCounterPacket.cpp.

References ARMNN_NO_CONVERSION_WARN_BEGIN, ARMNN_NO_CONVERSION_WARN_END, SendCounterPacket::CreateEventRecord(), and armnn::profiling::uint32_t_size.

Referenced by SendCounterPacketTest::CreateCategoryRecordTest(), and SendCounterPacket::SendCounterDirectoryPacket().

178 {
179  using namespace boost::numeric;
180 
181  BOOST_ASSERT(category);
182 
183  const std::string& categoryName = category->m_Name;
184  const std::vector<uint16_t> categoryCounters = category->m_Counters;
185  uint16_t deviceUid = category->m_DeviceUid;
186  uint16_t counterSetUid = category->m_CounterSetUid;
187 
188  BOOST_ASSERT(!categoryName.empty());
189 
190  // Utils
191  size_t uint32_t_size = sizeof(uint32_t);
192 
193  // Category record word 0:
194  // 16:31 [16] device: the uid of a device element which identifies some hardware device that
195  // the category belongs to
196  // 0:15 [16] counter_set: the uid of a counter_set the category is associated with
197  uint32_t categoryRecordWord0 = (static_cast<uint32_t>(deviceUid) << 16) |
198  (static_cast<uint32_t>(counterSetUid));
199 
200  // Category record word 1:
201  // 16:31 [16] event_count: number of events belonging to this category
202  // 0:15 [16] reserved: all zeros
203  uint32_t categoryRecordWord1 = static_cast<uint32_t>(categoryCounters.size()) << 16;
204 
205  // Category record word 2:
206  // 0:31 [32] event_pointer_table_offset: offset from the beginning of the category data pool to
207  // the event_pointer_table
208  uint32_t categoryRecordWord2 = 0; // The offset is always zero here, as the event pointer table field is always
209  // the first item in the pool
210 
211  // Convert the device name into a SWTrace namestring
212  std::vector<uint32_t> categoryNameBuffer;
213  if (!StringToSwTraceString<SwTraceNameCharPolicy>(categoryName, categoryNameBuffer))
214  {
215  errorMessage = boost::str(boost::format("Cannot convert the name of category \"%1%\" to an SWTrace namestring")
216  % categoryName);
217  return false;
218  }
219 
220  // Process the event records
221  size_t counterCount = categoryCounters.size();
222  std::vector<EventRecord> eventRecords(counterCount);
223  std::vector<uint32_t> eventRecordOffsets(counterCount, 0);
224  size_t eventRecordsSize = 0;
225  uint32_t eventRecordsOffset =
226  numeric_cast<uint32_t>((eventRecords.size() + categoryNameBuffer.size()) * uint32_t_size);
227  for (size_t counterIndex = 0, eventRecordIndex = 0, eventRecordOffsetIndex = 0;
228  counterIndex < counterCount;
229  counterIndex++, eventRecordIndex++, eventRecordOffsetIndex++)
230  {
231  uint16_t counterUid = categoryCounters.at(counterIndex);
232  auto it = counters.find(counterUid);
233  BOOST_ASSERT(it != counters.end());
234  const CounterPtr& counter = it->second;
235 
236  EventRecord& eventRecord = eventRecords.at(eventRecordIndex);
237  if (!CreateEventRecord(counter, eventRecord, errorMessage))
238  {
239  return false;
240  }
241 
242  // Update the total size in words of the event records
243  eventRecordsSize += eventRecord.size();
244 
245  // Add the event record offset to the event pointer table offset field
246  eventRecordOffsets[eventRecordOffsetIndex] = eventRecordsOffset;
247  eventRecordsOffset += numeric_cast<uint32_t>(eventRecord.size() * uint32_t_size);
248  }
249 
250  // Category record word 3:
251  // 0:31 [32] name_offset (offset from the beginning of the category data pool to the name field)
252  uint32_t categoryRecordWord3 = numeric_cast<uint32_t>(eventRecordOffsets.size() * uint32_t_size);
253 
254  // Calculate the size in words of the category record
255  size_t categoryRecordSize = 4u + // The size of the fixed part (device + counter_set + event_count + reserved +
256  // event_pointer_table_offset + name_offset)
257  eventRecordOffsets.size() + // The size of the variable part (the event pointer table +
258  categoryNameBuffer.size() + // and the category name including the null-terminator +
259  eventRecordsSize; // the event records)
260 
261  // Allocate the necessary space for the category record
262  categoryRecord.resize(categoryRecordSize);
263 
265  // Create the category record
266  categoryRecord[0] = categoryRecordWord0; // device + counter_set
267  categoryRecord[1] = categoryRecordWord1; // event_count + reserved
268  categoryRecord[2] = categoryRecordWord2; // event_pointer_table_offset
269  categoryRecord[3] = categoryRecordWord3; // name_offset
270  auto offset = categoryRecord.begin() + 4u;
271  std::copy(eventRecordOffsets.begin(), eventRecordOffsets.end(), offset); // event_pointer_table
272  offset += eventRecordOffsets.size();
273  std::copy(categoryNameBuffer.begin(), categoryNameBuffer.end(), offset); // name
274  offset += categoryNameBuffer.size();
275  for (const EventRecord& eventRecord : eventRecords)
276  {
277  std::copy(eventRecord.begin(), eventRecord.end(), offset); // event_record
278  offset += eventRecord.size();
279  }
281 
282  return true;
283 }
#define ARMNN_NO_CONVERSION_WARN_BEGIN
Definition: Conversion.hpp:36
bool CreateEventRecord(const CounterPtr &counter, EventRecord &eventRecord, std::string &errorMessage)
std::shared_ptr< Counter > CounterPtr
#define ARMNN_NO_CONVERSION_WARN_END
Definition: Conversion.hpp:37

◆ CreateCounterSetRecord()

bool CreateCounterSetRecord ( const CounterSetPtr counterSet,
CounterSetRecord counterSetRecord,
std::string &  errorMessage 
)
protected

Definition at line 335 of file SendCounterPacket.cpp.

Referenced by SendCounterPacketTest::CreateCounterSetRecordTest(), and SendCounterPacket::SendCounterDirectoryPacket().

338 {
339  BOOST_ASSERT(counterSet);
340 
341  uint16_t counterSetUid = counterSet->m_Uid;
342  const std::string& counterSetName = counterSet->m_Name;
343  uint16_t counterSetCount = counterSet->m_Count;
344 
345  BOOST_ASSERT(!counterSetName.empty());
346 
347  // Counter set record word 0:
348  // 16:31 [16] uid: the unique identifier for the counter_set
349  // 0:15 [16] count: the number of counters which can be active in this set at any one time
350  uint32_t counterSetRecordWord0 = (static_cast<uint32_t>(counterSetUid) << 16) |
351  (static_cast<uint32_t>(counterSetCount));
352 
353  // Counter set record word 1:
354  // 0:31 [32] name_offset: offset from the beginning of the counter set pool to the name field
355  uint32_t counterSetRecordWord1 = 0; // The offset is always zero here, as the name field is always
356  // the first (and only) item in the pool
357 
358  // Convert the device name into a SWTrace namestring
359  std::vector<uint32_t> counterSetNameBuffer;
360  if (!StringToSwTraceString<SwTraceNameCharPolicy>(counterSet->m_Name, counterSetNameBuffer))
361  {
362  errorMessage = boost::str(boost::format("Cannot convert the name of counter set %1% (\"%2%\") to "
363  "an SWTrace namestring")
364  % counterSetUid
365  % counterSetName);
366  return false;
367  }
368 
369  // Calculate the size in words of the counter set record
370  size_t counterSetRecordSize = 2u + // The size of the fixed part (uid + cores + name_offset)
371  counterSetNameBuffer.size(); // The size of the variable part (the counter set name
372  // including the null-terminator)
373 
374  // Allocate the space for the counter set record
375  counterSetRecord.resize(counterSetRecordSize);
376 
377  // Create the counter set record
378  counterSetRecord[0] = counterSetRecordWord0; // uid + core
379  counterSetRecord[1] = counterSetRecordWord1; // name_offset
380  auto offset = counterSetRecord.begin() + 2u;
381  std::copy(counterSetNameBuffer.begin(), counterSetNameBuffer.end(), offset); // name
382 
383  return true;
384 }

◆ CreateDeviceRecord()

bool CreateDeviceRecord ( const DevicePtr device,
DeviceRecord deviceRecord,
std::string &  errorMessage 
)
protected

Definition at line 285 of file SendCounterPacket.cpp.

Referenced by SendCounterPacketTest::CreateDeviceRecordTest(), and SendCounterPacket::SendCounterDirectoryPacket().

288 {
289  BOOST_ASSERT(device);
290 
291  uint16_t deviceUid = device->m_Uid;
292  const std::string& deviceName = device->m_Name;
293  uint16_t deviceCores = device->m_Cores;
294 
295  BOOST_ASSERT(!deviceName.empty());
296 
297  // Device record word 0:
298  // 16:31 [16] uid: the unique identifier for the device
299  // 0:15 [16] cores: the number of individual streams of counters for one or more cores of some device
300  uint32_t deviceRecordWord0 = (static_cast<uint32_t>(deviceUid) << 16) |
301  (static_cast<uint32_t>(deviceCores));
302 
303  // Device record word 1:
304  // 0:31 [32] name_offset: offset from the beginning of the device record pool to the name field
305  uint32_t deviceRecordWord1 = 0; // The offset is always zero here, as the name field is always
306  // the first (and only) item in the pool
307 
308  // Convert the device name into a SWTrace string
309  std::vector<uint32_t> deviceNameBuffer;
310  if (!StringToSwTraceString<SwTraceCharPolicy>(deviceName, deviceNameBuffer))
311  {
312  errorMessage = boost::str(boost::format("Cannot convert the name of device %1% (\"%2%\") to an SWTrace string")
313  % deviceUid
314  % deviceName);
315  return false;
316  }
317 
318  // Calculate the size in words of the device record
319  size_t deviceRecordSize = 2u + // The size of the fixed part (uid + cores + name_offset)
320  deviceNameBuffer.size(); // The size of the variable part (the device name including
321  // the null-terminator)
322 
323  // Allocate the necessary space for the device record
324  deviceRecord.resize(deviceRecordSize);
325 
326  // Create the device record
327  deviceRecord[0] = deviceRecordWord0; // uid + core
328  deviceRecord[1] = deviceRecordWord1; // name_offset
329  auto offset = deviceRecord.begin() + 2u;
330  std::copy(deviceNameBuffer.begin(), deviceNameBuffer.end(), offset); // name
331 
332  return true;
333 }

◆ CreateEventRecord()

bool CreateEventRecord ( const CounterPtr counter,
EventRecord eventRecord,
std::string &  errorMessage 
)
protected

Definition at line 386 of file SendCounterPacket.cpp.

References ARMNN_NO_CONVERSION_WARN_BEGIN, ARMNN_NO_CONVERSION_WARN_END, and armnn::profiling::uint32_t_size.

Referenced by SendCounterPacket::CreateCategoryRecord(), and SendCounterPacketTest::CreateEventRecordTest().

389 {
390  using namespace boost::numeric;
391 
392  BOOST_ASSERT(counter);
393 
394  uint16_t counterUid = counter->m_Uid;
395  uint16_t maxCounterUid = counter->m_MaxCounterUid;
396  uint16_t deviceUid = counter->m_DeviceUid;
397  uint16_t counterSetUid = counter->m_CounterSetUid;
398  uint16_t counterClass = counter->m_Class;
399  uint16_t counterInterpolation = counter->m_Interpolation;
400  double counterMultiplier = counter->m_Multiplier;
401  const std::string& counterName = counter->m_Name;
402  const std::string& counterDescription = counter->m_Description;
403  const std::string& counterUnits = counter->m_Units;
404 
405  BOOST_ASSERT(counterClass == 0 || counterClass == 1);
406  BOOST_ASSERT(counterInterpolation == 0 || counterInterpolation == 1);
407  BOOST_ASSERT(counterMultiplier);
408 
409  // Utils
410  size_t uint32_t_size = sizeof(uint32_t);
411 
412  // Event record word 0:
413  // 16:31 [16] max_counter_uid: if the device this event is associated with has more than one core and there
414  // is one of these counters per core this value will be set to
415  // (counter_uid + cores (from device_record)) - 1.
416  // If there is only a single core then this value will be the same as
417  // the counter_uid value
418  // 0:15 [16] count_uid: unique ID for the counter. Must be unique across all counters in all categories
419  uint32_t eventRecordWord0 = (static_cast<uint32_t>(maxCounterUid) << 16) |
420  (static_cast<uint32_t>(counterUid));
421 
422  // Event record word 1:
423  // 16:31 [16] device: UID of the device this event is associated with. Set to zero if the event is NOT
424  // associated with a device
425  // 0:15 [16] counter_set: UID of the counter_set this event is associated with. Set to zero if the event
426  // is NOT associated with a counter_set
427  uint32_t eventRecordWord1 = (static_cast<uint32_t>(deviceUid) << 16) |
428  (static_cast<uint32_t>(counterSetUid));
429 
430  // Event record word 2:
431  // 16:31 [16] class: type describing how to treat each data point in a stream of data points
432  // 0:15 [16] interpolation: type describing how to interpolate each data point in a stream of data points
433  uint32_t eventRecordWord2 = (static_cast<uint32_t>(counterClass) << 16) |
434  (static_cast<uint32_t>(counterInterpolation));
435 
436  // Event record word 3-4:
437  // 0:63 [64] multiplier: internal data stream is represented as integer values, this allows scaling of
438  // those values as if they are fixed point numbers. Zero is not a valid value
439  uint32_t multiplier[2] = { 0u, 0u };
440  BOOST_ASSERT(sizeof(counterMultiplier) == sizeof(multiplier));
441  std::memcpy(multiplier, &counterMultiplier, sizeof(multiplier));
442  uint32_t eventRecordWord3 = multiplier[0];
443  uint32_t eventRecordWord4 = multiplier[1];
444 
445  // Event record word 5:
446  // 0:31 [32] name_offset: offset from the beginning of the event record pool to the name field
447  uint32_t eventRecordWord5 = 0; // The offset is always zero here, as the name field is always
448  // the first item in the pool
449 
450  // Convert the counter name into a SWTrace string
451  std::vector<uint32_t> counterNameBuffer;
452  if (!StringToSwTraceString<SwTraceCharPolicy>(counterName, counterNameBuffer))
453  {
454  errorMessage = boost::str(boost::format("Cannot convert the name of counter %1% (name: \"%2%\") "
455  "to an SWTrace string")
456  % counterUid
457  % counterName);
458  return false;
459  }
460 
461  // Event record word 6:
462  // 0:31 [32] description_offset: offset from the beginning of the event record pool to the description field
463  // The size of the name buffer in bytes
464  uint32_t eventRecordWord6 = numeric_cast<uint32_t>(counterNameBuffer.size() * uint32_t_size);
465 
466  // Convert the counter description into a SWTrace string
467  std::vector<uint32_t> counterDescriptionBuffer;
468  if (!StringToSwTraceString<SwTraceCharPolicy>(counterDescription, counterDescriptionBuffer))
469  {
470  errorMessage = boost::str(boost::format("Cannot convert the description of counter %1% (description: \"%2%\") "
471  "to an SWTrace string")
472  % counterUid
473  % counterName);
474  return false;
475  }
476 
477  // Event record word 7:
478  // 0:31 [32] units_offset: (optional) offset from the beginning of the event record pool to the units field.
479  // An offset value of zero indicates this field is not provided
480  bool includeUnits = !counterUnits.empty();
481  // The size of the description buffer in bytes
482  uint32_t eventRecordWord7 = includeUnits ?
483  eventRecordWord6 +
484  numeric_cast<uint32_t>(counterDescriptionBuffer.size() * uint32_t_size) :
485  0;
486 
487  // Convert the counter units into a SWTrace namestring (optional)
488  std::vector<uint32_t> counterUnitsBuffer;
489  if (includeUnits)
490  {
491  // Convert the counter units into a SWTrace namestring
492  if (!StringToSwTraceString<SwTraceNameCharPolicy>(counterUnits, counterUnitsBuffer))
493  {
494  errorMessage = boost::str(boost::format("Cannot convert the units of counter %1% (units: \"%2%\") "
495  "to an SWTrace string")
496  % counterUid
497  % counterName);
498  return false;
499  }
500  }
501 
502  // Calculate the size in words of the event record
503  size_t eventRecordSize = 8u + // The size of the fixed part (counter_uid + max_counter_uid + device +
504  // counter_set + class + interpolation +
505  // multiplier + name_offset + description_offset +
506  // units_offset)
507  counterNameBuffer.size() + // The size of the variable part (the counter name,
508  counterDescriptionBuffer.size() + // description and units including the null-terminator)
509  counterUnitsBuffer.size();
510 
511  // Allocate the space for the event record
512  eventRecord.resize(eventRecordSize);
513 
515  // Create the event record
516  eventRecord[0] = eventRecordWord0; // max_counter_uid + counter_uid
517  eventRecord[1] = eventRecordWord1; // device + counter_set
518  eventRecord[2] = eventRecordWord2; // class + interpolation
519  eventRecord[3] = eventRecordWord3; // multiplier
520  eventRecord[4] = eventRecordWord4; // multiplier
521  eventRecord[5] = eventRecordWord5; // name_offset
522  eventRecord[6] = eventRecordWord6; // description_offset
523  eventRecord[7] = eventRecordWord7; // units_offset
524  auto offset = eventRecord.begin() + 8u;
525  std::copy(counterNameBuffer.begin(), counterNameBuffer.end(), offset); // name
526  offset += counterNameBuffer.size();
527  std::copy(counterDescriptionBuffer.begin(), counterDescriptionBuffer.end(), offset); // description
528  if (includeUnits)
529  {
530  offset += counterDescriptionBuffer.size();
531  std::copy(counterUnitsBuffer.begin(), counterUnitsBuffer.end(), offset); // units
532  }
534 
535  return true;
536 }
#define ARMNN_NO_CONVERSION_WARN_BEGIN
Definition: Conversion.hpp:36
#define ARMNN_NO_CONVERSION_WARN_END
Definition: Conversion.hpp:37

◆ SendCounterDirectoryPacket()

void SendCounterDirectoryPacket ( const ICounterDirectory counterDirectory)
overridevirtual

Create and write a CounterDirectoryPacket from the parameters to the buffer.

Implements ISendCounterPacket.

Definition at line 538 of file SendCounterPacket.cpp.

References ARMNN_NO_CONVERSION_WARN_BEGIN, ARMNN_NO_CONVERSION_WARN_END, SendCounterPacket::CreateCategoryRecord(), SendCounterPacket::CreateCounterSetRecord(), SendCounterPacket::CreateDeviceRecord(), ICounterDirectory::GetCategories(), ICounterDirectory::GetCategoryCount(), ICounterDirectory::GetCounters(), ICounterDirectory::GetCounterSetCount(), ICounterDirectory::GetCounterSets(), ICounterDirectory::GetDeviceCount(), ICounterDirectory::GetDevices(), armnn::profiling::uint32_t_size, and armnn::profiling::WriteUint32().

Referenced by BOOST_AUTO_TEST_CASE(), and SendCounterPacket::SendCounterPacket().

539 {
540  using namespace boost::numeric;
541 
542  // Get the amount of data that needs to be put into the packet
543  uint16_t categoryCount = counterDirectory.GetCategoryCount();
544  uint16_t deviceCount = counterDirectory.GetDeviceCount();
545  uint16_t counterSetCount = counterDirectory.GetCounterSetCount();
546 
547  // Utils
548  size_t uint32_t_size = sizeof(uint32_t);
549  size_t packetHeaderSize = 2u;
550  size_t bodyHeaderSize = 6u;
551 
552  // Initialize the offset for the pointer tables
553  uint32_t pointerTableOffset = 0;
554 
555  // --------------
556  // Device records
557  // --------------
558 
559  // Process device records
560  std::vector<DeviceRecord> deviceRecords(deviceCount);
561  const Devices& devices = counterDirectory.GetDevices();
562  std::vector<uint32_t> deviceRecordOffsets(deviceCount, 0); // device_records_pointer_table
563  size_t deviceRecordsSize = 0;
564  size_t deviceIndex = 0;
565  size_t deviceRecordOffsetIndex = 0;
566  for (auto it = devices.begin(); it != devices.end(); it++)
567  {
568  const DevicePtr& device = it->second;
569  DeviceRecord& deviceRecord = deviceRecords.at(deviceIndex);
570 
571  std::string errorMessage;
572  if (!CreateDeviceRecord(device, deviceRecord, errorMessage))
573  {
574  CancelOperationAndThrow<RuntimeException>(errorMessage);
575  }
576 
577  // Update the total size in words of the device records
578  deviceRecordsSize += deviceRecord.size();
579 
580  // Add the device record offset to the device records pointer table offset field
581  deviceRecordOffsets[deviceRecordOffsetIndex] = pointerTableOffset;
582  pointerTableOffset += numeric_cast<uint32_t>(deviceRecord.size() * uint32_t_size);
583 
584  deviceIndex++;
585  deviceRecordOffsetIndex++;
586  }
587 
588  // -------------------
589  // Counter set records
590  // -------------------
591 
592  // Process counter set records
593  std::vector<CounterSetRecord> counterSetRecords(counterSetCount);
594  const CounterSets& counterSets = counterDirectory.GetCounterSets();
595  std::vector<uint32_t> counterSetRecordOffsets(counterSetCount, 0); // counter_set_records_pointer_table
596  size_t counterSetRecordsSize = 0;
597  size_t counterSetIndex = 0;
598  size_t counterSetRecordOffsetIndex = 0;
599  for (auto it = counterSets.begin(); it != counterSets.end(); it++)
600  {
601  const CounterSetPtr& counterSet = it->second;
602  CounterSetRecord& counterSetRecord = counterSetRecords.at(counterSetIndex);
603 
604  std::string errorMessage;
605  if (!CreateCounterSetRecord(counterSet, counterSetRecord, errorMessage))
606  {
607  CancelOperationAndThrow<RuntimeException>(errorMessage);
608  }
609 
610  // Update the total size in words of the counter set records
611  counterSetRecordsSize += counterSetRecord.size();
612 
613  // Add the counter set record offset to the counter set records pointer table offset field
614  counterSetRecordOffsets[counterSetRecordOffsetIndex] = pointerTableOffset;
615  pointerTableOffset += numeric_cast<uint32_t>(counterSetRecord.size() * uint32_t_size);
616 
617  counterSetIndex++;
618  counterSetRecordOffsetIndex++;
619  }
620 
621  // ----------------
622  // Category records
623  // ----------------
624 
625  // Process category records
626  std::vector<CategoryRecord> categoryRecords(categoryCount);
627  const Categories& categories = counterDirectory.GetCategories();
628  std::vector<uint32_t> categoryRecordOffsets(categoryCount, 0); // category_records_pointer_table
629  size_t categoryRecordsSize = 0;
630  size_t categoryIndex = 0;
631  size_t categoryRecordOffsetIndex = 0;
632  for (auto it = categories.begin(); it != categories.end(); it++)
633  {
634  const CategoryPtr& category = *it;
635  CategoryRecord& categoryRecord = categoryRecords.at(categoryIndex);
636 
637  std::string errorMessage;
638  if (!CreateCategoryRecord(category, counterDirectory.GetCounters(), categoryRecord, errorMessage))
639  {
640  CancelOperationAndThrow<RuntimeException>(errorMessage);
641  }
642 
643  // Update the total size in words of the category records
644  categoryRecordsSize += categoryRecord.size();
645 
646  // Add the category record offset to the category records pointer table offset field
647  categoryRecordOffsets[categoryRecordOffsetIndex] = pointerTableOffset;
648  pointerTableOffset += numeric_cast<uint32_t>(categoryRecord.size() * uint32_t_size);
649 
650  categoryIndex++;
651  categoryRecordOffsetIndex++;
652  }
653 
654 
655 
656  // Calculate the length in words of the counter directory packet's data (excludes the packet header size)
657  size_t counterDirectoryPacketDataLength =
658  bodyHeaderSize + // The size of the body header
659  deviceRecordOffsets.size() + // The size of the device records pointer table
660  counterSetRecordOffsets.size() + // The size of counter set pointer table
661  categoryRecordOffsets.size() + // The size of category records pointer table
662  deviceRecordsSize + // The total size of the device records
663  counterSetRecordsSize + // The total size of the counter set records
664  categoryRecordsSize; // The total size of the category records
665 
666  // Calculate the size in words of the counter directory packet (the data length plus the packet header size)
667  size_t counterDirectoryPacketSize = packetHeaderSize + // The size of the packet header
668  counterDirectoryPacketDataLength; // The data length
669 
670 
671  // Allocate the necessary space for the counter directory packet
672  std::vector<uint32_t> counterDirectoryPacket(counterDirectoryPacketSize, 0);
673 
674  // -------------
675  // Packet header
676  // -------------
677 
678  // Packet header word 0:
679  // 26:31 [6] packet_family: control Packet Family
680  // 16:25 [10] packet_id: packet identifier
681  // 8:15 [8] reserved: all zeros
682  // 0:7 [8] reserved: all zeros
683  uint32_t packetFamily = 0;
684  uint32_t packetId = 2;
685  uint32_t packetHeaderWord0 = ((packetFamily & 0x3F) << 26) | ((packetId & 0x3FF) << 16);
686 
687  // Packet header word 1:
688  // 0:31 [32] data_length: length of data, in bytes
689  uint32_t packetHeaderWord1 = numeric_cast<uint32_t>(counterDirectoryPacketDataLength * uint32_t_size);
690 
691  // Create the packet header
692  uint32_t packetHeader[2]
693  {
694  packetHeaderWord0, // packet_family + packet_id + reserved + reserved
695  packetHeaderWord1 // data_length
696  };
697 
698  // -----------
699  // Body header
700  // -----------
701 
702  // Body header word 0:
703  // 16:31 [16] device_records_count: number of entries in the device_records_pointer_table
704  // 0:15 [16] reserved: all zeros
705  uint32_t bodyHeaderWord0 = static_cast<uint32_t>(deviceCount) << 16;
706 
707  // Body header word 1:
708  // 0:31 [32] device_records_pointer_table_offset: offset to the device_records_pointer_table
709  uint32_t bodyHeaderWord1 = 0; // The offset is always zero here, as the device record pointer table field is always
710  // the first item in the pool
711 
712  // Body header word 2:
713  // 16:31 [16] counter_set_count: number of entries in the counter_set_pointer_table
714  // 0:15 [16] reserved: all zeros
715  uint32_t bodyHeaderWord2 = static_cast<uint32_t>(counterSetCount) << 16;
716 
717  // Body header word 3:
718  // 0:31 [32] counter_set_pointer_table_offset: offset to the counter_set_pointer_table
719  uint32_t bodyHeaderWord3 =
720  numeric_cast<uint32_t>(deviceRecordOffsets.size() * uint32_t_size); // The size of the device records
721  // pointer table
722 
723 
724  // Body header word 4:
725  // 16:31 [16] categories_count: number of entries in the categories_pointer_table
726  // 0:15 [16] reserved: all zeros
727  uint32_t bodyHeaderWord4 = static_cast<uint32_t>(categoryCount) << 16;
728 
729  // Body header word 3:
730  // 0:31 [32] categories_pointer_table_offset: offset to the categories_pointer_table
731  uint32_t bodyHeaderWord5 =
732  numeric_cast<uint32_t>(deviceRecordOffsets.size() * uint32_t_size + // The size of the device records
733  counterSetRecordOffsets.size() * uint32_t_size); // pointer table, plus the size of
734  // the counter set pointer table
735 
736  // Create the body header
737  uint32_t bodyHeader[6]
738  {
739  bodyHeaderWord0, // device_records_count + reserved
740  bodyHeaderWord1, // device_records_pointer_table_offset
741  bodyHeaderWord2, // counter_set_count + reserved
742  bodyHeaderWord3, // counter_set_pointer_table_offset
743  bodyHeaderWord4, // categories_count + reserved
744  bodyHeaderWord5 // categories_pointer_table_offset
745  };
746 
748  // Create the counter directory packet
749  auto counterDirectoryPacketOffset = counterDirectoryPacket.begin();
750  // packet_header
751  std::copy(packetHeader, packetHeader + packetHeaderSize, counterDirectoryPacketOffset);
752  counterDirectoryPacketOffset += packetHeaderSize;
753  // body_header
754  std::copy(bodyHeader, bodyHeader + bodyHeaderSize, counterDirectoryPacketOffset);
755  counterDirectoryPacketOffset += bodyHeaderSize;
756  // device_records_pointer_table
757  std::copy(deviceRecordOffsets.begin(), deviceRecordOffsets.end(), counterDirectoryPacketOffset);
758  counterDirectoryPacketOffset += deviceRecordOffsets.size();
759  // counter_set_pointer_table
760  std::copy(counterSetRecordOffsets.begin(), counterSetRecordOffsets.end(), counterDirectoryPacketOffset);
761  counterDirectoryPacketOffset += counterSetRecordOffsets.size();
762  // category_pointer_table
763  std::copy(categoryRecordOffsets.begin(), categoryRecordOffsets.end(), counterDirectoryPacketOffset);
764  counterDirectoryPacketOffset += categoryRecordOffsets.size();
765  // device_records
766  for (const DeviceRecord& deviceRecord : deviceRecords)
767  {
768  std::copy(deviceRecord.begin(), deviceRecord.end(), counterDirectoryPacketOffset); // device_record
769  counterDirectoryPacketOffset += deviceRecord.size();
770  }
771  // counter_set_records
772  for (const CounterSetRecord& counterSetRecord : counterSetRecords)
773  {
774  std::copy(counterSetRecord.begin(), counterSetRecord.end(), counterDirectoryPacketOffset); // counter_set_record
775  counterDirectoryPacketOffset += counterSetRecord.size();
776  }
777  // category_records
778  for (const CategoryRecord& categoryRecord : categoryRecords)
779  {
780  std::copy(categoryRecord.begin(), categoryRecord.end(), counterDirectoryPacketOffset); // category_record
781  counterDirectoryPacketOffset += categoryRecord.size();
782  }
784 
785  // Calculate the total size in bytes of the counter directory packet
786  uint32_t totalSize = numeric_cast<uint32_t>(counterDirectoryPacketSize * uint32_t_size);
787 
788  // Reserve space in the buffer for the packet
789  uint32_t reserved = 0;
790  IPacketBufferPtr writeBuffer = m_BufferManager.Reserve(totalSize, reserved);
791 
792  if (writeBuffer == nullptr || reserved < totalSize)
793  {
794  CancelOperationAndThrow<BufferExhaustion>(
795  writeBuffer,
796  boost::str(boost::format("No space left in buffer. Unable to reserve (%1%) bytes.") % totalSize));
797  }
798 
799  // Offset for writing to the buffer
800  uint32_t offset = 0;
801 
802  // Write the counter directory packet to the buffer
803  for (uint32_t counterDirectoryPacketWord : counterDirectoryPacket)
804  {
805  WriteUint32(writeBuffer, offset, counterDirectoryPacketWord);
806  offset += numeric_cast<uint32_t>(uint32_t_size);
807  }
808 
809  m_BufferManager.Commit(writeBuffer, totalSize);
810 }
std::unordered_set< CategoryPtr > Categories
#define ARMNN_NO_CONVERSION_WARN_BEGIN
Definition: Conversion.hpp:36
std::unordered_map< uint16_t, DevicePtr > Devices
std::unique_ptr< CounterSet > CounterSetPtr
std::unordered_map< uint16_t, CounterSetPtr > CounterSets
std::unique_ptr< IPacketBuffer > IPacketBufferPtr
bool CreateDeviceRecord(const DevicePtr &device, DeviceRecord &deviceRecord, std::string &errorMessage)
std::unique_ptr< Category > CategoryPtr
std::vector< uint32_t > CounterSetRecord
std::unique_ptr< Device > DevicePtr
std::vector< uint32_t > CategoryRecord
#define ARMNN_NO_CONVERSION_WARN_END
Definition: Conversion.hpp:37
bool CreateCounterSetRecord(const CounterSetPtr &counterSet, CounterSetRecord &counterSetRecord, std::string &errorMessage)
void WriteUint32(const IPacketBufferPtr &packetBuffer, unsigned int offset, uint32_t value)
bool CreateCategoryRecord(const CategoryPtr &category, const Counters &counters, CategoryRecord &categoryRecord, std::string &errorMessage)

◆ SendPeriodicCounterCapturePacket()

void SendPeriodicCounterCapturePacket ( uint64_t  timestamp,
const IndexValuePairsVector values 
)
overridevirtual

Create and write a PeriodicCounterCapturePacket from the parameters to the buffer.

Implements ISendCounterPacket.

Definition at line 812 of file SendCounterPacket.cpp.

References armnn::profiling::uint16_t_size, armnn::profiling::uint32_t_size, armnn::profiling::WriteUint16(), armnn::profiling::WriteUint32(), and armnn::profiling::WriteUint64().

Referenced by BOOST_AUTO_TEST_CASE(), and SendCounterPacket::SendCounterPacket().

813 {
814  uint32_t uint16_t_size = sizeof(uint16_t);
815  uint32_t uint32_t_size = sizeof(uint32_t);
816  uint32_t uint64_t_size = sizeof(uint64_t);
817 
818  uint32_t packetFamily = 3;
819  uint32_t packetClass = 0;
820  uint32_t packetType = 0;
821  uint32_t headerSize = 2 * uint32_t_size;
822  uint32_t bodySize = uint64_t_size + numeric_cast<uint32_t>(values.size()) * (uint16_t_size + uint32_t_size);
823  uint32_t totalSize = headerSize + bodySize;
824  uint32_t offset = 0;
825  uint32_t reserved = 0;
826 
827  IPacketBufferPtr writeBuffer = m_BufferManager.Reserve(totalSize, reserved);
828 
829  if (writeBuffer == nullptr || reserved < totalSize)
830  {
831  CancelOperationAndThrow<BufferExhaustion>(
832  writeBuffer,
833  boost::str(boost::format("No space left in buffer. Unable to reserve (%1%) bytes.") % totalSize));
834  }
835 
836  // Create header.
837  WriteUint32(writeBuffer,
838  offset,
839  ((packetFamily & 0x0000003F) << 26) |
840  ((packetClass & 0x0000007F) << 19) |
841  ((packetType & 0x00000007) << 16));
842  offset += uint32_t_size;
843  WriteUint32(writeBuffer, offset, bodySize);
844 
845  // Copy captured Timestamp.
846  offset += uint32_t_size;
847  WriteUint64(writeBuffer, offset, timestamp);
848 
849  // Copy selectedCounterIds.
850  offset += uint64_t_size;
851  for (const auto& pair: values)
852  {
853  WriteUint16(writeBuffer, offset, pair.counterId);
854  offset += uint16_t_size;
855  WriteUint32(writeBuffer, offset, pair.counterValue);
856  offset += uint32_t_size;
857  }
858 
859  m_BufferManager.Commit(writeBuffer, totalSize);
860 }
void WriteUint64(const std::unique_ptr< IPacketBuffer > &packetBuffer, unsigned int offset, uint64_t value)
void WriteUint16(const IPacketBufferPtr &packetBuffer, unsigned int offset, uint16_t value)
std::unique_ptr< IPacketBuffer > IPacketBufferPtr
virtual void Commit(IPacketBufferPtr &packetBuffer, unsigned int size, bool notifyConsumer=true)=0
virtual IPacketBufferPtr Reserve(unsigned int requestedSize, unsigned int &reservedSize)=0
void WriteUint32(const IPacketBufferPtr &packetBuffer, unsigned int offset, uint32_t value)

◆ SendPeriodicCounterSelectionPacket()

void SendPeriodicCounterSelectionPacket ( uint32_t  capturePeriod,
const std::vector< uint16_t > &  selectedCounterIds 
)
overridevirtual

Create and write a PeriodicCounterSelectionPacket from the parameters to the buffer.

Implements ISendCounterPacket.

Definition at line 862 of file SendCounterPacket.cpp.

References armnn::profiling::uint16_t_size, armnn::profiling::uint32_t_size, armnn::profiling::WriteUint16(), and armnn::profiling::WriteUint32().

Referenced by BOOST_AUTO_TEST_CASE(), and SendCounterPacket::SendCounterPacket().

864 {
865  uint32_t uint16_t_size = sizeof(uint16_t);
866  uint32_t uint32_t_size = sizeof(uint32_t);
867 
868  uint32_t packetFamily = 0;
869  uint32_t packetId = 4;
870  uint32_t headerSize = 2 * uint32_t_size;
871  uint32_t bodySize = uint32_t_size + numeric_cast<uint32_t>(selectedCounterIds.size()) * uint16_t_size;
872  uint32_t totalSize = headerSize + bodySize;
873  uint32_t offset = 0;
874  uint32_t reserved = 0;
875 
876  IPacketBufferPtr writeBuffer = m_BufferManager.Reserve(totalSize, reserved);
877 
878  if (writeBuffer == nullptr || reserved < totalSize)
879  {
880  CancelOperationAndThrow<BufferExhaustion>(
881  writeBuffer,
882  boost::str(boost::format("No space left in buffer. Unable to reserve (%1%) bytes.") % totalSize));
883  }
884 
885  // Create header.
886  WriteUint32(writeBuffer, offset, ((packetFamily & 0x3F) << 26) | ((packetId & 0x3FF) << 16));
887  offset += uint32_t_size;
888  WriteUint32(writeBuffer, offset, bodySize);
889 
890  // Copy capturePeriod.
891  offset += uint32_t_size;
892  WriteUint32(writeBuffer, offset, capturePeriod);
893 
894  // Copy selectedCounterIds.
895  offset += uint32_t_size;
896  for(const uint16_t& id: selectedCounterIds)
897  {
898  WriteUint16(writeBuffer, offset, id);
899  offset += uint16_t_size;
900  }
901 
902  m_BufferManager.Commit(writeBuffer, totalSize);
903 }
void WriteUint16(const IPacketBufferPtr &packetBuffer, unsigned int offset, uint16_t value)
std::unique_ptr< IPacketBuffer > IPacketBufferPtr
virtual void Commit(IPacketBufferPtr &packetBuffer, unsigned int size, bool notifyConsumer=true)=0
virtual IPacketBufferPtr Reserve(unsigned int requestedSize, unsigned int &reservedSize)=0
void WriteUint32(const IPacketBufferPtr &packetBuffer, unsigned int offset, uint32_t value)

◆ SendStreamMetaDataPacket()

void SendStreamMetaDataPacket ( )
overridevirtual

Create and write a StreamMetaDataPacket in the buffer.

Implements ISendCounterPacket.

Definition at line 29 of file SendCounterPacket.cpp.

References IBufferManager::Commit(), armnn::profiling::EncodeVersion(), armnnUtils::Processes::GetCurrentId(), armnn::profiling::GetHardwareVersion(), armnn::profiling::GetProcessName(), armnn::profiling::GetSoftwareInfo(), armnn::profiling::GetSoftwareVersion(), armnn::info, MAX_METADATA_PACKET_LENGTH, SendCounterPacket::PIPE_MAGIC, IBufferManager::Reserve(), and armnn::profiling::WriteUint32().

Referenced by BOOST_AUTO_TEST_CASE(), and SendCounterPacket::SendCounterPacket().

30 {
31  std::string info(GetSoftwareInfo());
32  std::string hardwareVersion(GetHardwareVersion());
33  std::string softwareVersion(GetSoftwareVersion());
34  std::string processName = GetProcessName().substr(0, 60);
35 
36  uint32_t infoSize = numeric_cast<uint32_t>(info.size()) > 0 ? numeric_cast<uint32_t>(info.size()) + 1 : 0;
37  uint32_t hardwareVersionSize = numeric_cast<uint32_t>(hardwareVersion.size()) > 0 ?
38  numeric_cast<uint32_t>(hardwareVersion.size()) + 1 : 0;
39  uint32_t softwareVersionSize = numeric_cast<uint32_t>(softwareVersion.size()) > 0 ?
40  numeric_cast<uint32_t>(softwareVersion.size()) + 1 : 0;
41  uint32_t processNameSize = numeric_cast<uint32_t>(processName.size()) > 0 ?
42  numeric_cast<uint32_t>(processName.size()) + 1 : 0;
43 
44  uint32_t sizeUint32 = numeric_cast<uint32_t>(sizeof(uint32_t));
45 
46  uint32_t headerSize = 2 * sizeUint32;
47  uint32_t bodySize = 10 * sizeUint32;
48  uint32_t packetVersionCountSize = sizeUint32;
49 
50  // Supported Packets
51  // Stream metadata packet (packet family=0; packet id=0)
52  // Connection Acknowledged packet (packet family=0, packet id=1)
53  // Counter Directory packet (packet family=0; packet id=2)
54  // Request Counter Directory packet (packet family=0, packet id=3)
55  // Periodic Counter Selection packet (packet family=0, packet id=4)
56  // Periodic Counter Capture packet (packet family=1, packet class=0, type=0)
57  uint32_t packetVersionEntries = 6;
58 
59  uint32_t payloadSize = numeric_cast<uint32_t>(infoSize + hardwareVersionSize + softwareVersionSize +
60  processNameSize + packetVersionCountSize +
61  (packetVersionEntries * 2 * sizeUint32));
62 
63  uint32_t totalSize = headerSize + bodySize + payloadSize;
64  uint32_t offset = 0;
65  uint32_t reserved = 0;
66 
67  IPacketBufferPtr writeBuffer = m_BufferManager.Reserve(totalSize, reserved);
68 
69  if (writeBuffer == nullptr || reserved < totalSize)
70  {
71  CancelOperationAndThrow<BufferExhaustion>(
72  writeBuffer,
73  boost::str(boost::format("No space left in buffer. Unable to reserve (%1%) bytes.") % totalSize));
74  }
75 
76  try
77  {
78  // Create header
79 
80  WriteUint32(writeBuffer, offset, 0);
81  offset += sizeUint32;
82  WriteUint32(writeBuffer, offset, totalSize - headerSize);
83 
84  // Packet body
85 
86  offset += sizeUint32;
87  WriteUint32(writeBuffer, offset, PIPE_MAGIC); // pipe_magic
88  offset += sizeUint32;
89  WriteUint32(writeBuffer, offset, EncodeVersion(1, 0, 0)); // stream_metadata_version
90  offset += sizeUint32;
91  WriteUint32(writeBuffer, offset, MAX_METADATA_PACKET_LENGTH); // max_data_length
92  offset += sizeUint32;
94  WriteUint32(writeBuffer, offset, numeric_cast<uint32_t>(pid)); // pid
95  offset += sizeUint32;
96  uint32_t poolOffset = bodySize;
97  WriteUint32(writeBuffer, offset, infoSize ? poolOffset : 0); // offset_info
98  offset += sizeUint32;
99  poolOffset += infoSize;
100  WriteUint32(writeBuffer, offset, hardwareVersionSize ? poolOffset : 0); // offset_hw_version
101  offset += sizeUint32;
102  poolOffset += hardwareVersionSize;
103  WriteUint32(writeBuffer, offset, softwareVersionSize ? poolOffset : 0); // offset_sw_version
104  offset += sizeUint32;
105  poolOffset += softwareVersionSize;
106  WriteUint32(writeBuffer, offset, processNameSize ? poolOffset : 0); // offset_process_name
107  offset += sizeUint32;
108  poolOffset += processNameSize;
109  WriteUint32(writeBuffer, offset, packetVersionEntries ? poolOffset : 0); // offset_packet_version_table
110  offset += sizeUint32;
111  WriteUint32(writeBuffer, offset, 0); // reserved
112  offset += sizeUint32;
113 
114  // Pool
115 
116  if (infoSize)
117  {
118  memcpy(&writeBuffer->GetWritableData()[offset], info.c_str(), infoSize);
119  offset += infoSize;
120  }
121 
122  if (hardwareVersionSize)
123  {
124  memcpy(&writeBuffer->GetWritableData()[offset], hardwareVersion.c_str(), hardwareVersionSize);
125  offset += hardwareVersionSize;
126  }
127 
128  if (softwareVersionSize)
129  {
130  memcpy(&writeBuffer->GetWritableData()[offset], softwareVersion.c_str(), softwareVersionSize);
131  offset += softwareVersionSize;
132  }
133 
134  if (processNameSize)
135  {
136  memcpy(&writeBuffer->GetWritableData()[offset], processName.c_str(), processNameSize);
137  offset += processNameSize;
138  }
139 
140  if (packetVersionEntries)
141  {
142  // Packet Version Count
143  WriteUint32(writeBuffer, offset, packetVersionEntries << 16);
144 
145  // Packet Version Entries
146  uint32_t packetFamily = 0;
147  uint32_t packetId = 0;
148 
149  offset += sizeUint32;
150  for (uint32_t i = 0; i < packetVersionEntries - 1; ++i)
151  {
152  WriteUint32(writeBuffer, offset, ((packetFamily & 0x3F) << 26) | ((packetId++ & 0x3FF) << 16));
153  offset += sizeUint32;
154  WriteUint32(writeBuffer, offset, EncodeVersion(1, 0, 0));
155  offset += sizeUint32;
156  }
157 
158  packetFamily = 1;
159  packetId = 0;
160 
161  WriteUint32(writeBuffer, offset, ((packetFamily & 0x3F) << 26) | ((packetId & 0x3FF) << 16));
162  offset += sizeUint32;
163  WriteUint32(writeBuffer, offset, EncodeVersion(1, 0, 0));
164  }
165  }
166  catch(...)
167  {
168  CancelOperationAndThrow<RuntimeException>(writeBuffer, "Error processing packet.");
169  }
170 
171  m_BufferManager.Commit(writeBuffer, totalSize, false);
172 }
std::string GetSoftwareVersion()
std::unique_ptr< IPacketBuffer > IPacketBufferPtr
std::string GetSoftwareInfo()
static const unsigned int PIPE_MAGIC
std::string GetProcessName()
std::string GetHardwareVersion()
constexpr uint32_t EncodeVersion(uint32_t major, uint32_t minor, uint32_t patch)
virtual void Commit(IPacketBufferPtr &packetBuffer, unsigned int size, bool notifyConsumer=true)=0
virtual IPacketBufferPtr Reserve(unsigned int requestedSize, unsigned int &reservedSize)=0
#define MAX_METADATA_PACKET_LENGTH
void WriteUint32(const IPacketBufferPtr &packetBuffer, unsigned int offset, uint32_t value)

Member Data Documentation

◆ PIPE_MAGIC

const unsigned int PIPE_MAGIC = 0x45495434
static

The documentation for this class was generated from the following files: