ArmNN
 20.05
BufferManager.cpp
Go to the documentation of this file.
1 //
2 // Copyright © 2019 Arm Ltd. All rights reserved.
3 // SPDX-License-Identifier: MIT
4 //
5 
6 #include "BufferManager.hpp"
7 #include "PacketBuffer.hpp"
8 
9 namespace armnn
10 {
11 
12 namespace profiling
13 {
14 
15 BufferManager::BufferManager(unsigned int numberOfBuffers, unsigned int maxPacketSize)
16  : m_MaxBufferSize(maxPacketSize),
17  m_NumberOfBuffers(numberOfBuffers)
18 {
19  Initialize();
20 }
21 
22 IPacketBufferPtr BufferManager::Reserve(unsigned int requestedSize, unsigned int& reservedSize)
23 {
24  reservedSize = 0;
25  std::unique_lock<std::mutex> availableListLock(m_AvailableMutex, std::defer_lock);
26  if (requestedSize > m_MaxBufferSize)
27  {
28  return nullptr;
29  }
30  availableListLock.lock();
31  if (m_AvailableList.empty())
32  {
33  availableListLock.unlock();
34  return nullptr;
35  }
36  IPacketBufferPtr buffer = std::move(m_AvailableList.back());
37  m_AvailableList.pop_back();
38  availableListLock.unlock();
39  reservedSize = requestedSize;
40  return buffer;
41 }
42 
43 void BufferManager::Commit(IPacketBufferPtr& packetBuffer, unsigned int size, bool notifyConsumer)
44 {
45  std::unique_lock<std::mutex> readableListLock(m_ReadableMutex, std::defer_lock);
46  packetBuffer->Commit(size);
47  readableListLock.lock();
48  m_ReadableList.push(std::move(packetBuffer));
49  readableListLock.unlock();
50 
51  if (notifyConsumer)
52  {
53  FlushReadList();
54  }
55 }
56 
57 void BufferManager::Initialize()
58 {
59  m_AvailableList.reserve(m_NumberOfBuffers);
60  for (unsigned int i = 0; i < m_NumberOfBuffers; ++i)
61  {
62  IPacketBufferPtr buffer = std::make_unique<PacketBuffer>(m_MaxBufferSize);
63  m_AvailableList.emplace_back(std::move(buffer));
64  }
65 }
66 
68 {
69  std::unique_lock<std::mutex> availableListLock(m_AvailableMutex, std::defer_lock);
70  packetBuffer->Release();
71  availableListLock.lock();
72  m_AvailableList.push_back(std::move(packetBuffer));
73  availableListLock.unlock();
74 }
75 
77 {
78  //This method should only be called once all threads have been joined
79  std::lock_guard<std::mutex> readableListLock(m_ReadableMutex);
80  std::lock_guard<std::mutex> availableListLock(m_AvailableMutex);
81 
82  m_AvailableList.clear();
83  std::queue<IPacketBufferPtr>().swap(m_ReadableList);
84 
85  Initialize();
86 }
87 
89 {
90  std::unique_lock<std::mutex> readableListLock(m_ReadableMutex);
91  if (!m_ReadableList.empty())
92  {
93  IPacketBufferPtr buffer = std::move(m_ReadableList.front());
94  m_ReadableList.pop();
95  readableListLock.unlock();
96  return buffer;
97  }
98  return nullptr;
99 }
100 
102 {
103  std::unique_lock<std::mutex> availableListLock(m_AvailableMutex, std::defer_lock);
104  packetBuffer->MarkRead();
105  availableListLock.lock();
106  m_AvailableList.push_back(std::move(packetBuffer));
107  availableListLock.unlock();
108 }
109 
111 {
112  m_Consumer = consumer;
113 }
114 
116 {
117  // notify consumer that packet is ready to read
118  if (m_Consumer != nullptr)
119  {
120  m_Consumer->SetReadyToRead();
121  }
122 }
123 
124 } // namespace profiling
125 
126 } // namespace armnn
virtual void SetReadyToRead()=0
Set a "ready to read" flag in the buffer to notify the reading thread to start reading it...
void swap(OriginsDescriptor &first, OriginsDescriptor &second)
void SetConsumer(IConsumer *consumer) override
Set Consumer on the buffer manager to be notified when there is a Commit Can only be one consumer...
Copyright (c) 2020 ARM Limited.
BufferManager(unsigned int numberOfBuffers=5, unsigned int maxPacketSize=4096)
void FlushReadList() override
Notify the Consumer buffer can be read.
void Commit(IPacketBufferPtr &packetBuffer, unsigned int size, bool notifyConsumer=true) override
IPacketBufferPtr Reserve(unsigned int requestedSize, unsigned int &reservedSize) override
IPacketBufferPtr GetReadableBuffer() override
void MarkRead(IPacketBufferPtr &packetBuffer) override
std::unique_ptr< IPacketBuffer > IPacketBufferPtr
void Release(IPacketBufferPtr &packetBuffer) override