HRTFDatabaseLoader.cpp [plain text]
#include "config.h"
#if ENABLE(WEB_AUDIO)
#include "HRTFDatabaseLoader.h"
#include "HRTFDatabase.h"
#include <wtf/MainThread.h>
#include <wtf/NeverDestroyed.h>
namespace WebCore {
static HashMap<double, HRTFDatabaseLoader*>& loaderMap()
{
static NeverDestroyed<HashMap<double, HRTFDatabaseLoader*>> loaderMap;
return loaderMap;
}
PassRefPtr<HRTFDatabaseLoader> HRTFDatabaseLoader::createAndLoadAsynchronouslyIfNecessary(float sampleRate)
{
ASSERT(isMainThread());
RefPtr<HRTFDatabaseLoader> loader = loaderMap().get(sampleRate);
if (loader) {
ASSERT(sampleRate == loader->databaseSampleRate());
return loader;
}
loader = adoptRef(new HRTFDatabaseLoader(sampleRate));
loaderMap().add(sampleRate, loader.get());
loader->loadAsynchronously();
return loader;
}
HRTFDatabaseLoader::HRTFDatabaseLoader(float sampleRate)
: m_databaseLoaderThread(0)
, m_databaseSampleRate(sampleRate)
{
ASSERT(isMainThread());
}
HRTFDatabaseLoader::~HRTFDatabaseLoader()
{
ASSERT(isMainThread());
waitForLoaderThreadCompletion();
m_hrtfDatabase = nullptr;
loaderMap().remove(m_databaseSampleRate);
}
static void databaseLoaderEntry(void* threadData)
{
HRTFDatabaseLoader* loader = reinterpret_cast<HRTFDatabaseLoader*>(threadData);
ASSERT(loader);
loader->load();
}
void HRTFDatabaseLoader::load()
{
ASSERT(!isMainThread());
if (!m_hrtfDatabase.get()) {
m_hrtfDatabase = std::make_unique<HRTFDatabase>(m_databaseSampleRate);
}
}
void HRTFDatabaseLoader::loadAsynchronously()
{
ASSERT(isMainThread());
LockHolder locker(m_threadLock);
if (!m_hrtfDatabase.get() && !m_databaseLoaderThread) {
m_databaseLoaderThread = createThread(databaseLoaderEntry, this, "HRTF database loader");
}
}
bool HRTFDatabaseLoader::isLoaded() const
{
return m_hrtfDatabase.get();
}
void HRTFDatabaseLoader::waitForLoaderThreadCompletion()
{
LockHolder locker(m_threadLock);
if (m_databaseLoaderThread)
waitForThreadCompletion(m_databaseLoaderThread);
m_databaseLoaderThread = 0;
}
}
#endif // ENABLE(WEB_AUDIO)