mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-03 11:11:14 +00:00
fs: fix locking in FileSystemBufferManager
This commit is contained in:
parent
6cf31486e1
commit
f58eca5f7b
2 changed files with 95 additions and 26 deletions
|
@ -269,13 +269,13 @@ namespace ams::fssystem {
|
||||||
m_cache_handle_table.Finalize();
|
m_cache_handle_table.Finalize();
|
||||||
}
|
}
|
||||||
private:
|
private:
|
||||||
virtual const std::pair<uintptr_t, size_t> DoAllocateBuffer(size_t size, const BufferAttribute &attr) override;
|
virtual const fs::IBufferManager::MemoryRange DoAllocateBuffer(size_t size, const BufferAttribute &attr) override;
|
||||||
|
|
||||||
virtual void DoDeallocateBuffer(uintptr_t address, size_t size) override;
|
virtual void DoDeallocateBuffer(uintptr_t address, size_t size) override;
|
||||||
|
|
||||||
virtual CacheHandle DoRegisterCache(uintptr_t address, size_t size, const BufferAttribute &attr) override;
|
virtual CacheHandle DoRegisterCache(uintptr_t address, size_t size, const BufferAttribute &attr) override;
|
||||||
|
|
||||||
virtual const std::pair<uintptr_t, size_t> DoAcquireCache(CacheHandle handle) override;
|
virtual const fs::IBufferManager::MemoryRange DoAcquireCache(CacheHandle handle) override;
|
||||||
|
|
||||||
virtual size_t DoGetTotalSize() const override;
|
virtual size_t DoGetTotalSize() const override;
|
||||||
|
|
||||||
|
@ -290,6 +290,17 @@ namespace ams::fssystem {
|
||||||
virtual size_t DoGetRetriedCount() const override;
|
virtual size_t DoGetRetriedCount() const override;
|
||||||
|
|
||||||
virtual void DoClearPeak() override;
|
virtual void DoClearPeak() override;
|
||||||
|
private:
|
||||||
|
const fs::IBufferManager::MemoryRange AllocateBufferImpl(size_t size, const BufferAttribute &attr);
|
||||||
|
void DeallocateBufferImpl(uintptr_t address, size_t size);
|
||||||
|
CacheHandle RegisterCacheImpl(uintptr_t address, size_t size, const BufferAttribute &attr);
|
||||||
|
const fs::IBufferManager::MemoryRange AcquireCacheImpl(CacheHandle handle);
|
||||||
|
size_t GetFreeSizeImpl() const;
|
||||||
|
size_t GetTotalAllocatableSizeImpl() const;
|
||||||
|
size_t GetFreeSizePeakImpl() const;
|
||||||
|
size_t GetTotalAllocatableSizePeakImpl() const;
|
||||||
|
size_t GetRetriedCountImpl() const;
|
||||||
|
void ClearPeakImpl();
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -239,21 +239,24 @@ namespace ams::fssystem {
|
||||||
return it != m_attr_list.end() ? std::addressof(*it) : nullptr;
|
return it != m_attr_list.end() ? std::addressof(*it) : nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
const fs::IBufferManager::MemoryRange FileSystemBufferManager::DoAllocateBuffer(size_t size, const BufferAttribute &attr) {
|
const fs::IBufferManager::MemoryRange FileSystemBufferManager::AllocateBufferImpl(size_t size, const BufferAttribute &attr) {
|
||||||
std::scoped_lock lk(m_mutex);
|
/* Get/sanity check the required order. */
|
||||||
|
|
||||||
fs::IBufferManager::MemoryRange range = {};
|
fs::IBufferManager::MemoryRange range = {};
|
||||||
const auto order = m_buddy_heap.GetOrderFromBytes(size);
|
const auto order = m_buddy_heap.GetOrderFromBytes(size);
|
||||||
AMS_ASSERT(order >= 0);
|
AMS_ASSERT(order >= 0);
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
|
/* Try to allocate a buffer at the desired order. */
|
||||||
if (auto address = m_buddy_heap.AllocateByOrder(order); address != 0) {
|
if (auto address = m_buddy_heap.AllocateByOrder(order); address != 0) {
|
||||||
|
/* Check that we allocated enough. */
|
||||||
const auto allocated_size = m_buddy_heap.GetBytesFromOrder(order);
|
const auto allocated_size = m_buddy_heap.GetBytesFromOrder(order);
|
||||||
AMS_ASSERT(size <= allocated_size);
|
AMS_ASSERT(size <= allocated_size);
|
||||||
|
|
||||||
|
/* Set up the range extents. */
|
||||||
range.first = reinterpret_cast<uintptr_t>(address);
|
range.first = reinterpret_cast<uintptr_t>(address);
|
||||||
range.second = allocated_size;
|
range.second = allocated_size;
|
||||||
|
|
||||||
|
/* Update our peak tracking variables. */
|
||||||
const size_t free_size = m_buddy_heap.GetTotalFreeSize();
|
const size_t free_size = m_buddy_heap.GetTotalFreeSize();
|
||||||
m_peak_free_size = std::min(m_peak_free_size, free_size);
|
m_peak_free_size = std::min(m_peak_free_size, free_size);
|
||||||
|
|
||||||
|
@ -262,32 +265,30 @@ namespace ams::fssystem {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* We failed, to we'll need to deallocate something and retry. */
|
||||||
|
++m_retried_count;
|
||||||
|
|
||||||
/* Deallocate a buffer. */
|
/* Deallocate a buffer. */
|
||||||
uintptr_t deallocate_address = 0;
|
uintptr_t deallocate_address = 0;
|
||||||
size_t deallocate_size = 0;
|
size_t deallocate_size = 0;
|
||||||
|
|
||||||
++m_retried_count;
|
|
||||||
if (m_cache_handle_table.UnregisterOldest(std::addressof(deallocate_address), std::addressof(deallocate_size), attr, size)) {
|
if (m_cache_handle_table.UnregisterOldest(std::addressof(deallocate_address), std::addressof(deallocate_size), attr, size)) {
|
||||||
this->DeallocateBuffer(deallocate_address, deallocate_size);
|
this->DeallocateBufferImpl(deallocate_address, deallocate_size);
|
||||||
} else {
|
} else {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Return the range we allocated. */
|
||||||
return range;
|
return range;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FileSystemBufferManager::DoDeallocateBuffer(uintptr_t address, size_t size) {
|
void FileSystemBufferManager::DeallocateBufferImpl(uintptr_t address, size_t size) {
|
||||||
AMS_ASSERT(util::IsPowerOfTwo(size));
|
AMS_ASSERT(util::IsPowerOfTwo(size));
|
||||||
|
|
||||||
std::scoped_lock lk(m_mutex);
|
|
||||||
|
|
||||||
m_buddy_heap.Free(reinterpret_cast<void *>(address), m_buddy_heap.GetOrderFromBytes(size));
|
m_buddy_heap.Free(reinterpret_cast<void *>(address), m_buddy_heap.GetOrderFromBytes(size));
|
||||||
}
|
}
|
||||||
|
|
||||||
FileSystemBufferManager::CacheHandle FileSystemBufferManager::DoRegisterCache(uintptr_t address, size_t size, const BufferAttribute &attr) {
|
FileSystemBufferManager::CacheHandle FileSystemBufferManager::RegisterCacheImpl(uintptr_t address, size_t size, const BufferAttribute &attr) {
|
||||||
std::scoped_lock lk(m_mutex);
|
|
||||||
|
|
||||||
CacheHandle handle = 0;
|
CacheHandle handle = 0;
|
||||||
while (true) {
|
while (true) {
|
||||||
/* Try to register the handle. */
|
/* Try to register the handle. */
|
||||||
|
@ -301,9 +302,9 @@ namespace ams::fssystem {
|
||||||
|
|
||||||
++m_retried_count;
|
++m_retried_count;
|
||||||
if (m_cache_handle_table.UnregisterOldest(std::addressof(deallocate_address), std::addressof(deallocate_size), attr)) {
|
if (m_cache_handle_table.UnregisterOldest(std::addressof(deallocate_address), std::addressof(deallocate_size), attr)) {
|
||||||
this->DeallocateBuffer(deallocate_address, deallocate_size);
|
this->DeallocateBufferImpl(deallocate_address, deallocate_size);
|
||||||
} else {
|
} else {
|
||||||
this->DeallocateBuffer(address, size);
|
this->DeallocateBufferImpl(address, size);
|
||||||
handle = m_cache_handle_table.PublishCacheHandle();
|
handle = m_cache_handle_table.PublishCacheHandle();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -312,9 +313,7 @@ namespace ams::fssystem {
|
||||||
return handle;
|
return handle;
|
||||||
}
|
}
|
||||||
|
|
||||||
const fs::IBufferManager::MemoryRange FileSystemBufferManager::DoAcquireCache(CacheHandle handle) {
|
const fs::IBufferManager::MemoryRange FileSystemBufferManager::AcquireCacheImpl(CacheHandle handle) {
|
||||||
std::scoped_lock lk(m_mutex);
|
|
||||||
|
|
||||||
fs::IBufferManager::MemoryRange range = {};
|
fs::IBufferManager::MemoryRange range = {};
|
||||||
if (m_cache_handle_table.Unregister(std::addressof(range.first), std::addressof(range.second), handle)) {
|
if (m_cache_handle_table.Unregister(std::addressof(range.first), std::addressof(range.second), handle)) {
|
||||||
const size_t total_allocatable_size = m_buddy_heap.GetTotalFreeSize() + m_cache_handle_table.GetTotalCacheSize();
|
const size_t total_allocatable_size = m_buddy_heap.GetTotalFreeSize() + m_cache_handle_table.GetTotalCacheSize();
|
||||||
|
@ -327,6 +326,56 @@ namespace ams::fssystem {
|
||||||
return range;
|
return range;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
size_t FileSystemBufferManager::GetFreeSizeImpl() const {
|
||||||
|
return m_buddy_heap.GetTotalFreeSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t FileSystemBufferManager::GetTotalAllocatableSizeImpl() const {
|
||||||
|
return this->GetFreeSizeImpl() + m_cache_handle_table.GetTotalCacheSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t FileSystemBufferManager::GetFreeSizePeakImpl() const {
|
||||||
|
return m_peak_free_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t FileSystemBufferManager::GetTotalAllocatableSizePeakImpl() const {
|
||||||
|
return m_peak_total_allocatable_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t FileSystemBufferManager::GetRetriedCountImpl() const {
|
||||||
|
return m_retried_count;
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileSystemBufferManager::ClearPeakImpl() {
|
||||||
|
m_peak_free_size = this->GetFreeSizeImpl();
|
||||||
|
m_peak_total_allocatable_size = this->GetTotalAllocatableSizeImpl();
|
||||||
|
m_retried_count = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const fs::IBufferManager::MemoryRange FileSystemBufferManager::DoAllocateBuffer(size_t size, const BufferAttribute &attr) {
|
||||||
|
std::scoped_lock lk(m_mutex);
|
||||||
|
|
||||||
|
return this->AllocateBufferImpl(size, attr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileSystemBufferManager::DoDeallocateBuffer(uintptr_t address, size_t size) {
|
||||||
|
std::scoped_lock lk(m_mutex);
|
||||||
|
|
||||||
|
return this->DeallocateBufferImpl(address, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
FileSystemBufferManager::CacheHandle FileSystemBufferManager::DoRegisterCache(uintptr_t address, size_t size, const BufferAttribute &attr) {
|
||||||
|
std::scoped_lock lk(m_mutex);
|
||||||
|
|
||||||
|
return this->RegisterCacheImpl(address, size, attr);
|
||||||
|
}
|
||||||
|
|
||||||
|
const fs::IBufferManager::MemoryRange FileSystemBufferManager::DoAcquireCache(CacheHandle handle) {
|
||||||
|
std::scoped_lock lk(m_mutex);
|
||||||
|
|
||||||
|
return this->AcquireCacheImpl(handle);
|
||||||
|
}
|
||||||
|
|
||||||
size_t FileSystemBufferManager::DoGetTotalSize() const {
|
size_t FileSystemBufferManager::DoGetTotalSize() const {
|
||||||
return m_total_size;
|
return m_total_size;
|
||||||
}
|
}
|
||||||
|
@ -334,28 +383,37 @@ namespace ams::fssystem {
|
||||||
size_t FileSystemBufferManager::DoGetFreeSize() const {
|
size_t FileSystemBufferManager::DoGetFreeSize() const {
|
||||||
std::scoped_lock lk(m_mutex);
|
std::scoped_lock lk(m_mutex);
|
||||||
|
|
||||||
return m_buddy_heap.GetTotalFreeSize();
|
return this->GetFreeSizeImpl();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t FileSystemBufferManager::DoGetTotalAllocatableSize() const {
|
size_t FileSystemBufferManager::DoGetTotalAllocatableSize() const {
|
||||||
return this->GetFreeSize() + m_cache_handle_table.GetTotalCacheSize();
|
std::scoped_lock lk(m_mutex);
|
||||||
|
|
||||||
|
return this->GetTotalAllocatableSizeImpl();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t FileSystemBufferManager::DoGetFreeSizePeak() const {
|
size_t FileSystemBufferManager::DoGetFreeSizePeak() const {
|
||||||
return m_peak_free_size;
|
std::scoped_lock lk(m_mutex);
|
||||||
|
|
||||||
|
return this->GetFreeSizePeakImpl();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t FileSystemBufferManager::DoGetTotalAllocatableSizePeak() const {
|
size_t FileSystemBufferManager::DoGetTotalAllocatableSizePeak() const {
|
||||||
return m_peak_total_allocatable_size;
|
std::scoped_lock lk(m_mutex);
|
||||||
|
|
||||||
|
return this->GetTotalAllocatableSizePeakImpl();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t FileSystemBufferManager::DoGetRetriedCount() const {
|
size_t FileSystemBufferManager::DoGetRetriedCount() const {
|
||||||
return m_retried_count;
|
std::scoped_lock lk(m_mutex);
|
||||||
|
|
||||||
|
return this->GetRetriedCountImpl();
|
||||||
}
|
}
|
||||||
|
|
||||||
void FileSystemBufferManager::DoClearPeak() {
|
void FileSystemBufferManager::DoClearPeak() {
|
||||||
m_peak_free_size = this->GetFreeSize();
|
std::scoped_lock lk(m_mutex);
|
||||||
m_retried_count = 0;
|
|
||||||
|
return this->ClearPeakImpl();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue