raw memcards: revert last change so flushes are still time-driven.
It turns out the actual slowdown was from memcpy'ing the entire memcard buffer...not synchronization overhead or the flush itself.
This commit is contained in:
parent
674494e472
commit
96d7b642f4
|
@ -12,8 +12,6 @@
|
||||||
#define SIZE_TO_Mb (1024 * 8 * 16)
|
#define SIZE_TO_Mb (1024 * 8 * 16)
|
||||||
#define MC_HDR_SIZE 0xA000
|
#define MC_HDR_SIZE 0xA000
|
||||||
|
|
||||||
const std::chrono::seconds MemoryCard::s_flush_interval{ 15 };
|
|
||||||
|
|
||||||
MemoryCard::MemoryCard(std::string filename, int _card_index, u16 sizeMb)
|
MemoryCard::MemoryCard(std::string filename, int _card_index, u16 sizeMb)
|
||||||
: MemoryCardBase(_card_index, sizeMb)
|
: MemoryCardBase(_card_index, sizeMb)
|
||||||
, m_filename(filename)
|
, m_filename(filename)
|
||||||
|
@ -46,7 +44,6 @@ MemoryCard::MemoryCard(std::string filename, int _card_index, u16 sizeMb)
|
||||||
|
|
||||||
// Class members (including inherited ones) have now been initialized, so
|
// Class members (including inherited ones) have now been initialized, so
|
||||||
// it's safe to startup the flush thread (which reads them).
|
// it's safe to startup the flush thread (which reads them).
|
||||||
m_last_flush = std::chrono::steady_clock::now();
|
|
||||||
m_flush_buffer = std::make_unique<u8[]>(memory_card_size);
|
m_flush_buffer = std::make_unique<u8[]>(memory_card_size);
|
||||||
m_flush_thread = std::thread(&MemoryCard::FlushThread, this);
|
m_flush_thread = std::thread(&MemoryCard::FlushThread, this);
|
||||||
}
|
}
|
||||||
|
@ -55,13 +52,6 @@ MemoryCard::~MemoryCard()
|
||||||
{
|
{
|
||||||
if (m_flush_thread.joinable())
|
if (m_flush_thread.joinable())
|
||||||
{
|
{
|
||||||
// Update the flush buffer one last time, flush, and join.
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> l(m_flush_mutex);
|
|
||||||
memcpy(&m_flush_buffer[0], &m_memcard_data[0], memory_card_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
m_is_exiting.Set();
|
|
||||||
m_flush_trigger.Set();
|
m_flush_trigger.Set();
|
||||||
|
|
||||||
m_flush_thread.join();
|
m_flush_thread.join();
|
||||||
|
@ -78,10 +68,21 @@ void MemoryCard::FlushThread()
|
||||||
Common::SetCurrentThreadName(
|
Common::SetCurrentThreadName(
|
||||||
StringFromFormat("Memcard%x-Flush", card_index).c_str());
|
StringFromFormat("Memcard%x-Flush", card_index).c_str());
|
||||||
|
|
||||||
for (;;)
|
const auto flush_interval = std::chrono::seconds(15);
|
||||||
|
|
||||||
|
while (true)
|
||||||
{
|
{
|
||||||
m_flush_trigger.Wait();
|
// If triggered, we're exiting.
|
||||||
bool do_exit = m_is_exiting.IsSet();
|
// If timed out, check if we need to flush.
|
||||||
|
bool do_exit = m_flush_trigger.WaitFor(flush_interval);
|
||||||
|
if (!do_exit)
|
||||||
|
{
|
||||||
|
bool is_dirty = m_dirty.TestAndClear();
|
||||||
|
if (!is_dirty)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Opening the file is purposefully done each iteration to ensure the
|
// Opening the file is purposefully done each iteration to ensure the
|
||||||
// file doesn't disappear out from under us after the first check.
|
// file doesn't disappear out from under us after the first check.
|
||||||
|
@ -115,8 +116,9 @@ void MemoryCard::FlushThread()
|
||||||
|
|
||||||
{
|
{
|
||||||
std::unique_lock<std::mutex> l(m_flush_mutex);
|
std::unique_lock<std::mutex> l(m_flush_mutex);
|
||||||
pFile.WriteBytes(&m_flush_buffer[0], memory_card_size);
|
memcpy(&m_flush_buffer[0], &m_memcard_data[0], memory_card_size);
|
||||||
}
|
}
|
||||||
|
pFile.WriteBytes(&m_flush_buffer[0], memory_card_size);
|
||||||
|
|
||||||
if (!do_exit)
|
if (!do_exit)
|
||||||
{
|
{
|
||||||
|
@ -132,25 +134,9 @@ void MemoryCard::FlushThread()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Attempt to update the flush buffer and trigger a flush. If we can't get a
|
void MemoryCard::MakeDirty()
|
||||||
// lock in order to update the flush buffer, a write is in progress and a future
|
|
||||||
// write will take care of any changes to the memcard data, so nothing needs to
|
|
||||||
// be done now.
|
|
||||||
void MemoryCard::TryFlush()
|
|
||||||
{
|
{
|
||||||
auto now = std::chrono::steady_clock::now();
|
m_dirty.Set();
|
||||||
if (now - m_last_flush < s_flush_interval)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
m_last_flush = now;
|
|
||||||
|
|
||||||
if (m_flush_mutex.try_lock())
|
|
||||||
{
|
|
||||||
memcpy(&m_flush_buffer[0], &m_memcard_data[0], memory_card_size);
|
|
||||||
m_flush_mutex.unlock();
|
|
||||||
m_flush_trigger.Set();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
s32 MemoryCard::Read(u32 srcaddress, s32 length, u8 *destaddress)
|
s32 MemoryCard::Read(u32 srcaddress, s32 length, u8 *destaddress)
|
||||||
|
@ -175,8 +161,11 @@ s32 MemoryCard::Write(u32 destaddress, s32 length, u8 *srcaddress)
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
std::unique_lock<std::mutex> l(m_flush_mutex);
|
||||||
memcpy(&m_memcard_data[destaddress], srcaddress, length);
|
memcpy(&m_memcard_data[destaddress], srcaddress, length);
|
||||||
TryFlush();
|
}
|
||||||
|
MakeDirty();
|
||||||
return length;
|
return length;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -186,18 +175,23 @@ void MemoryCard::ClearBlock(u32 address)
|
||||||
{
|
{
|
||||||
PanicAlertT("MemoryCard: ClearBlock called on invalid address %x",
|
PanicAlertT("MemoryCard: ClearBlock called on invalid address %x",
|
||||||
address);
|
address);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
std::unique_lock<std::mutex> l(m_flush_mutex);
|
||||||
memset(&m_memcard_data[address], 0xFF, BLOCK_SIZE);
|
memset(&m_memcard_data[address], 0xFF, BLOCK_SIZE);
|
||||||
TryFlush();
|
|
||||||
}
|
}
|
||||||
|
MakeDirty();
|
||||||
}
|
}
|
||||||
|
|
||||||
void MemoryCard::ClearAll()
|
void MemoryCard::ClearAll()
|
||||||
{
|
{
|
||||||
|
{
|
||||||
|
std::unique_lock<std::mutex> l(m_flush_mutex);
|
||||||
memset(&m_memcard_data[0], 0xFF, memory_card_size);
|
memset(&m_memcard_data[0], 0xFF, memory_card_size);
|
||||||
TryFlush();
|
}
|
||||||
|
MakeDirty();
|
||||||
}
|
}
|
||||||
|
|
||||||
void MemoryCard::DoState(PointerWrap &p)
|
void MemoryCard::DoState(PointerWrap &p)
|
||||||
|
|
|
@ -4,7 +4,6 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include "Common/Event.h"
|
#include "Common/Event.h"
|
||||||
#include "Common/Flag.h"
|
#include "Common/Flag.h"
|
||||||
|
@ -19,7 +18,7 @@ public:
|
||||||
MemoryCard(std::string filename, int _card_index, u16 sizeMb = MemCard2043Mb);
|
MemoryCard(std::string filename, int _card_index, u16 sizeMb = MemCard2043Mb);
|
||||||
~MemoryCard();
|
~MemoryCard();
|
||||||
void FlushThread();
|
void FlushThread();
|
||||||
void TryFlush();
|
void MakeDirty();
|
||||||
|
|
||||||
s32 Read(u32 address, s32 length, u8 *destaddress) override;
|
s32 Read(u32 address, s32 length, u8 *destaddress) override;
|
||||||
s32 Write(u32 destaddress, s32 length, u8 *srcaddress) override;
|
s32 Write(u32 destaddress, s32 length, u8 *srcaddress) override;
|
||||||
|
@ -30,11 +29,9 @@ public:
|
||||||
private:
|
private:
|
||||||
std::string m_filename;
|
std::string m_filename;
|
||||||
std::unique_ptr<u8[]> m_memcard_data;
|
std::unique_ptr<u8[]> m_memcard_data;
|
||||||
|
std::unique_ptr<u8[]> m_flush_buffer;
|
||||||
std::thread m_flush_thread;
|
std::thread m_flush_thread;
|
||||||
std::mutex m_flush_mutex;
|
std::mutex m_flush_mutex;
|
||||||
Common::Event m_flush_trigger;
|
Common::Event m_flush_trigger;
|
||||||
Common::Flag m_is_exiting;
|
Common::Flag m_dirty;
|
||||||
std::unique_ptr<u8[]> m_flush_buffer;
|
|
||||||
std::chrono::steady_clock::time_point m_last_flush;
|
|
||||||
static const std::chrono::seconds s_flush_interval;
|
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue