2023-06-02 19:43:28 +00:00
|
|
|
// Copyright 2023 Dolphin Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
|
|
|
#include "VideoCommon/Assets/TextureAsset.h"
|
|
|
|
|
2024-06-01 04:31:10 +00:00
|
|
|
#include <optional>
|
|
|
|
|
|
|
|
#include "Common/JsonUtil.h"
|
2023-06-02 19:43:28 +00:00
|
|
|
#include "Common/Logging/Log.h"
|
2023-10-17 00:23:00 +00:00
|
|
|
#include "Common/StringUtil.h"
|
2023-08-24 05:35:31 +00:00
|
|
|
#include "VideoCommon/BPMemory.h"
|
2023-06-02 19:43:28 +00:00
|
|
|
|
|
|
|
namespace VideoCommon
|
|
|
|
{
|
2023-08-24 05:35:31 +00:00
|
|
|
namespace
|
|
|
|
{
|
2024-06-01 04:31:10 +00:00
|
|
|
std::optional<WrapMode> ReadWrapModeFromJSON(const picojson::object& json, const std::string& uv)
|
2023-08-24 05:35:31 +00:00
|
|
|
{
|
2024-06-01 04:31:10 +00:00
|
|
|
auto uv_mode = ReadStringFromJson(json, uv).value_or("");
|
|
|
|
Common::ToLower(&uv_mode);
|
2023-08-24 05:35:31 +00:00
|
|
|
|
2024-06-01 04:31:10 +00:00
|
|
|
if (uv_mode == "clamp")
|
2023-08-24 05:35:31 +00:00
|
|
|
{
|
2024-06-01 04:31:10 +00:00
|
|
|
return WrapMode::Clamp;
|
2023-08-24 05:35:31 +00:00
|
|
|
}
|
2024-06-01 04:31:10 +00:00
|
|
|
else if (uv_mode == "repeat")
|
2023-08-24 05:35:31 +00:00
|
|
|
{
|
2024-06-01 04:31:10 +00:00
|
|
|
return WrapMode::Repeat;
|
2023-08-24 05:35:31 +00:00
|
|
|
}
|
2024-06-01 04:31:10 +00:00
|
|
|
else if (uv_mode == "mirror")
|
2023-08-24 05:35:31 +00:00
|
|
|
{
|
2024-06-01 04:31:10 +00:00
|
|
|
return WrapMode::Mirror;
|
2023-08-24 05:35:31 +00:00
|
|
|
}
|
2024-06-01 04:31:10 +00:00
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<FilterMode> ReadFilterModeFromJSON(const picojson::object& json,
|
|
|
|
const std::string& filter)
|
|
|
|
{
|
|
|
|
auto filter_mode = ReadStringFromJson(json, filter).value_or("");
|
|
|
|
Common::ToLower(&filter_mode);
|
|
|
|
|
|
|
|
if (filter_mode == "linear")
|
2023-08-24 05:35:31 +00:00
|
|
|
{
|
2024-06-01 04:31:10 +00:00
|
|
|
return FilterMode::Linear;
|
2023-08-24 05:35:31 +00:00
|
|
|
}
|
2024-06-01 04:31:10 +00:00
|
|
|
else if (filter_mode == "near")
|
2023-08-24 05:35:31 +00:00
|
|
|
{
|
2024-06-01 04:31:10 +00:00
|
|
|
return FilterMode::Near;
|
2023-08-24 05:35:31 +00:00
|
|
|
}
|
|
|
|
|
2024-06-01 04:31:10 +00:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ParseSampler(const VideoCommon::CustomAssetLibrary::AssetID& asset_id,
|
|
|
|
const picojson::object& json, SamplerState* sampler)
|
|
|
|
{
|
|
|
|
if (!sampler) [[unlikely]]
|
2023-08-24 05:35:31 +00:00
|
|
|
return false;
|
2024-06-01 04:31:10 +00:00
|
|
|
|
|
|
|
*sampler = RenderState::GetLinearSamplerState();
|
|
|
|
|
|
|
|
const auto sampler_state_wrap_iter = json.find("wrap_mode");
|
|
|
|
if (sampler_state_wrap_iter != json.end())
|
2023-08-24 05:35:31 +00:00
|
|
|
{
|
2024-06-01 04:31:10 +00:00
|
|
|
if (!sampler_state_wrap_iter->second.is<picojson::object>())
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO, "Asset '{}' failed to parse json, 'wrap_mode' is not the right type",
|
|
|
|
asset_id);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const auto sampler_state_wrap_obj = sampler_state_wrap_iter->second.get<picojson::object>();
|
|
|
|
|
|
|
|
if (const auto mode = ReadWrapModeFromJSON(sampler_state_wrap_obj, "u"))
|
|
|
|
{
|
|
|
|
sampler->tm0.wrap_u = *mode;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO,
|
|
|
|
"Asset '{}' failed to parse json, 'wrap_mode[u]' has an invalid "
|
|
|
|
"value",
|
|
|
|
asset_id);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (const auto mode = ReadWrapModeFromJSON(sampler_state_wrap_obj, "v"))
|
|
|
|
{
|
|
|
|
sampler->tm0.wrap_v = *mode;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO,
|
|
|
|
"Asset '{}' failed to parse json, 'wrap_mode[v]' has an invalid "
|
|
|
|
"value",
|
|
|
|
asset_id);
|
|
|
|
return false;
|
|
|
|
}
|
2023-08-24 05:35:31 +00:00
|
|
|
}
|
2024-06-01 04:31:10 +00:00
|
|
|
|
|
|
|
const auto sampler_state_filter_iter = json.find("filter_mode");
|
|
|
|
if (sampler_state_filter_iter != json.end())
|
2023-08-24 05:35:31 +00:00
|
|
|
{
|
2024-06-01 04:31:10 +00:00
|
|
|
if (!sampler_state_filter_iter->second.is<picojson::object>())
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO, "Asset '{}' failed to parse json, 'filter_mode' is not the right type",
|
|
|
|
asset_id);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const auto sampler_state_filter_obj = sampler_state_filter_iter->second.get<picojson::object>();
|
|
|
|
|
|
|
|
if (const auto mode = ReadFilterModeFromJSON(sampler_state_filter_obj, "min"))
|
|
|
|
{
|
|
|
|
sampler->tm0.min_filter = *mode;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO,
|
|
|
|
"Asset '{}' failed to parse json, 'filter_mode[min]' has an invalid "
|
|
|
|
"value",
|
|
|
|
asset_id);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (const auto mode = ReadFilterModeFromJSON(sampler_state_filter_obj, "mag"))
|
|
|
|
{
|
|
|
|
sampler->tm0.mag_filter = *mode;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO,
|
|
|
|
"Asset '{}' failed to parse json, 'filter_mode[mag]' has an invalid "
|
|
|
|
"value",
|
|
|
|
asset_id);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (const auto mode = ReadFilterModeFromJSON(sampler_state_filter_obj, "mipmap"))
|
|
|
|
{
|
|
|
|
sampler->tm0.mipmap_filter = *mode;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO,
|
|
|
|
"Asset '{}' failed to parse json, 'filter_mode[mipmap]' has an invalid "
|
|
|
|
"value",
|
|
|
|
asset_id);
|
|
|
|
return false;
|
|
|
|
}
|
2023-08-24 05:35:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
bool TextureData::FromJson(const CustomAssetLibrary::AssetID& asset_id,
|
|
|
|
const picojson::object& json, TextureData* data)
|
|
|
|
{
|
|
|
|
const auto type_iter = json.find("type");
|
|
|
|
if (type_iter == json.end())
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO, "Asset '{}' failed to parse json, property entry 'type' not found",
|
|
|
|
asset_id);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!type_iter->second.is<std::string>())
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO,
|
|
|
|
"Asset '{}' failed to parse json, property entry 'type' is not "
|
|
|
|
"the right json type",
|
|
|
|
asset_id);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
std::string type = type_iter->second.to_str();
|
2023-10-17 00:23:00 +00:00
|
|
|
Common::ToLower(&type);
|
2023-08-24 05:35:31 +00:00
|
|
|
|
|
|
|
if (type == "texture2d")
|
|
|
|
{
|
|
|
|
data->m_type = TextureData::Type::Type_Texture2D;
|
2023-09-06 02:11:19 +00:00
|
|
|
|
|
|
|
if (!ParseSampler(asset_id, json, &data->m_sampler))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (type == "texturecube")
|
|
|
|
{
|
|
|
|
data->m_type = TextureData::Type::Type_TextureCube;
|
2023-08-24 05:35:31 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO,
|
|
|
|
"Asset '{}' failed to parse json, texture type '{}' "
|
|
|
|
"an invalid option",
|
|
|
|
asset_id, type);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-06-02 19:43:28 +00:00
|
|
|
CustomAssetLibrary::LoadInfo GameTextureAsset::LoadImpl(const CustomAssetLibrary::AssetID& asset_id)
|
|
|
|
{
|
2023-09-06 05:16:26 +00:00
|
|
|
auto potential_data = std::make_shared<TextureData>();
|
2023-06-03 04:57:32 +00:00
|
|
|
const auto loaded_info = m_owning_library->LoadGameTexture(asset_id, potential_data.get());
|
2023-06-02 19:43:28 +00:00
|
|
|
if (loaded_info.m_bytes_loaded == 0)
|
|
|
|
return {};
|
2023-06-05 04:01:29 +00:00
|
|
|
{
|
|
|
|
std::lock_guard lk(m_data_lock);
|
|
|
|
m_loaded = true;
|
|
|
|
m_data = std::move(potential_data);
|
|
|
|
}
|
2023-06-02 19:43:28 +00:00
|
|
|
return loaded_info;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GameTextureAsset::Validate(u32 native_width, u32 native_height) const
|
|
|
|
{
|
2023-06-05 04:01:29 +00:00
|
|
|
std::lock_guard lk(m_data_lock);
|
2023-06-02 19:43:28 +00:00
|
|
|
|
|
|
|
if (!m_loaded)
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO,
|
|
|
|
"Game texture can't be validated for asset '{}' because it is not loaded yet.",
|
|
|
|
GetAssetId());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-09-06 05:16:26 +00:00
|
|
|
if (m_data->m_texture.m_slices.empty())
|
2023-06-02 19:43:28 +00:00
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(VIDEO,
|
|
|
|
"Game texture can't be validated for asset '{}' because no data was available.",
|
|
|
|
GetAssetId());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-09-06 05:16:26 +00:00
|
|
|
if (m_data->m_texture.m_slices.size() > 1)
|
2023-08-13 21:09:45 +00:00
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(
|
|
|
|
VIDEO,
|
|
|
|
"Game texture can't be validated for asset '{}' because it has more slices than expected.",
|
|
|
|
GetAssetId());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-09-06 05:16:26 +00:00
|
|
|
const auto& slice = m_data->m_texture.m_slices[0];
|
2023-08-13 21:09:45 +00:00
|
|
|
if (slice.m_levels.empty())
|
|
|
|
{
|
|
|
|
ERROR_LOG_FMT(
|
|
|
|
VIDEO,
|
|
|
|
"Game texture can't be validated for asset '{}' because first slice has no data available.",
|
|
|
|
GetAssetId());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-06-02 19:43:28 +00:00
|
|
|
// Verify that the aspect ratio of the texture hasn't changed, as this could have
|
|
|
|
// side-effects.
|
2023-08-13 21:09:45 +00:00
|
|
|
const VideoCommon::CustomTextureData::ArraySlice::Level& first_mip = slice.m_levels[0];
|
2023-06-02 19:43:28 +00:00
|
|
|
if (first_mip.width * native_height != first_mip.height * native_width)
|
|
|
|
{
|
2023-06-12 02:28:08 +00:00
|
|
|
// Note: this feels like this should return an error but
|
|
|
|
// for legacy reasons this is only a notice that something *could*
|
|
|
|
// go wrong
|
|
|
|
WARN_LOG_FMT(
|
2023-06-02 19:43:28 +00:00
|
|
|
VIDEO,
|
|
|
|
"Invalid custom texture size {}x{} for game texture asset '{}'. The aspect differs "
|
|
|
|
"from the native size {}x{}.",
|
|
|
|
first_mip.width, first_mip.height, GetAssetId(), native_width, native_height);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Same deal if the custom texture isn't a multiple of the native size.
|
|
|
|
if (native_width != 0 && native_height != 0 &&
|
|
|
|
(first_mip.width % native_width || first_mip.height % native_height))
|
|
|
|
{
|
2023-06-12 02:28:08 +00:00
|
|
|
// Note: this feels like this should return an error but
|
|
|
|
// for legacy reasons this is only a notice that something *could*
|
|
|
|
// go wrong
|
|
|
|
WARN_LOG_FMT(
|
2023-06-02 19:43:28 +00:00
|
|
|
VIDEO,
|
|
|
|
"Invalid custom texture size {}x{} for game texture asset '{}'. Please use an integer "
|
|
|
|
"upscaling factor based on the native size {}x{}.",
|
|
|
|
first_mip.width, first_mip.height, GetAssetId(), native_width, native_height);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
} // namespace VideoCommon
|