Merge pull request #11880 from iwubcode/race_condition_asset
VideoCommon: prevent potential data issue when reloading Asset data
This commit is contained in:
commit
3245786af7
Source/Core/VideoCommon/Assets
|
@ -64,7 +64,11 @@ class CustomLoadableAsset : public CustomAsset
|
||||||
public:
|
public:
|
||||||
using CustomAsset::CustomAsset;
|
using CustomAsset::CustomAsset;
|
||||||
|
|
||||||
const UnderlyingType* GetData() const
|
// Callees should understand that the type returned is
|
||||||
|
// a local copy and 'GetData()' needs to be called
|
||||||
|
// to ensure the latest copy is available if
|
||||||
|
// they want to handle reloads
|
||||||
|
[[nodiscard]] std::shared_ptr<UnderlyingType> GetData() const
|
||||||
{
|
{
|
||||||
std::lock_guard lk(m_lock);
|
std::lock_guard lk(m_lock);
|
||||||
if (m_loaded)
|
if (m_loaded)
|
||||||
|
@ -75,7 +79,7 @@ public:
|
||||||
protected:
|
protected:
|
||||||
bool m_loaded = false;
|
bool m_loaded = false;
|
||||||
mutable std::mutex m_lock;
|
mutable std::mutex m_lock;
|
||||||
UnderlyingType m_data;
|
std::shared_ptr<UnderlyingType> m_data;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace VideoCommon
|
} // namespace VideoCommon
|
||||||
|
|
|
@ -10,20 +10,24 @@ namespace VideoCommon
|
||||||
CustomAssetLibrary::LoadInfo RawTextureAsset::LoadImpl(const CustomAssetLibrary::AssetID& asset_id)
|
CustomAssetLibrary::LoadInfo RawTextureAsset::LoadImpl(const CustomAssetLibrary::AssetID& asset_id)
|
||||||
{
|
{
|
||||||
std::lock_guard lk(m_lock);
|
std::lock_guard lk(m_lock);
|
||||||
const auto loaded_info = m_owning_library->LoadTexture(asset_id, &m_data);
|
auto potential_data = std::make_shared<CustomTextureData>();
|
||||||
|
const auto loaded_info = m_owning_library->LoadTexture(asset_id, potential_data.get());
|
||||||
if (loaded_info.m_bytes_loaded == 0)
|
if (loaded_info.m_bytes_loaded == 0)
|
||||||
return {};
|
return {};
|
||||||
m_loaded = true;
|
m_loaded = true;
|
||||||
|
m_data = std::move(potential_data);
|
||||||
return loaded_info;
|
return loaded_info;
|
||||||
}
|
}
|
||||||
|
|
||||||
CustomAssetLibrary::LoadInfo GameTextureAsset::LoadImpl(const CustomAssetLibrary::AssetID& asset_id)
|
CustomAssetLibrary::LoadInfo GameTextureAsset::LoadImpl(const CustomAssetLibrary::AssetID& asset_id)
|
||||||
{
|
{
|
||||||
std::lock_guard lk(m_lock);
|
std::lock_guard lk(m_lock);
|
||||||
const auto loaded_info = m_owning_library->LoadGameTexture(asset_id, &m_data);
|
auto potential_data = std::make_shared<CustomTextureData>();
|
||||||
|
const auto loaded_info = m_owning_library->LoadGameTexture(asset_id, potential_data.get());
|
||||||
if (loaded_info.m_bytes_loaded == 0)
|
if (loaded_info.m_bytes_loaded == 0)
|
||||||
return {};
|
return {};
|
||||||
m_loaded = true;
|
m_loaded = true;
|
||||||
|
m_data = std::move(potential_data);
|
||||||
return loaded_info;
|
return loaded_info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,7 +43,7 @@ bool GameTextureAsset::Validate(u32 native_width, u32 native_height) const
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (m_data.m_levels.empty())
|
if (m_data->m_levels.empty())
|
||||||
{
|
{
|
||||||
ERROR_LOG_FMT(VIDEO,
|
ERROR_LOG_FMT(VIDEO,
|
||||||
"Game texture can't be validated for asset '{}' because no data was available.",
|
"Game texture can't be validated for asset '{}' because no data was available.",
|
||||||
|
@ -49,7 +53,7 @@ bool GameTextureAsset::Validate(u32 native_width, u32 native_height) const
|
||||||
|
|
||||||
// Verify that the aspect ratio of the texture hasn't changed, as this could have
|
// Verify that the aspect ratio of the texture hasn't changed, as this could have
|
||||||
// side-effects.
|
// side-effects.
|
||||||
const VideoCommon::CustomTextureData::Level& first_mip = m_data.m_levels[0];
|
const VideoCommon::CustomTextureData::Level& first_mip = m_data->m_levels[0];
|
||||||
if (first_mip.width * native_height != first_mip.height * native_width)
|
if (first_mip.width * native_height != first_mip.height * native_width)
|
||||||
{
|
{
|
||||||
ERROR_LOG_FMT(
|
ERROR_LOG_FMT(
|
||||||
|
|
Loading…
Reference in New Issue