You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
278 lines
12 KiB
C++
278 lines
12 KiB
C++
/*
|
|
* Copyright (c) Contributors to the Open 3D Engine Project.
|
|
* For complete copyright and license terms please see the LICENSE at the root of this distribution.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0 OR MIT
|
|
*
|
|
*/
|
|
|
|
|
|
#include <Processing/ImageAssetProducer.h>
|
|
#include <Processing/ImageObjectImpl.h>
|
|
#include <Processing/PixelFormatInfo.h>
|
|
#include <Processing/Utils.h>
|
|
#include <Processing/ImageFlags.h>
|
|
|
|
#include <Atom/RHI.Reflect/Format.h>
|
|
#include <Atom/RHI.Reflect/ImageSubresource.h>
|
|
|
|
#include <Atom/RPI.Reflect/Image/StreamingImageAssetCreator.h>
|
|
#include <Atom/RPI.Reflect/Image/ImageMipChainAssetCreator.h>
|
|
#include <Atom/RPI.Reflect/Image/ImageAsset.h>
|
|
|
|
#include <AzCore/Asset/AssetManager.h>
|
|
|
|
#include <AzToolsFramework/API/EditorAssetSystemAPI.h>
|
|
|
|
namespace ImageProcessingAtom
|
|
{
|
|
using namespace AZ;
|
|
|
|
namespace
|
|
{
|
|
// Trying to fit as many as mips to 64k memory for one mip chain
|
|
const uint32_t s_mininumMipBlockSize = 64 * 1024;
|
|
}
|
|
|
|
ImageAssetProducer::ImageAssetProducer(
|
|
const IImageObjectPtr imageObject,
|
|
AZStd::string_view saveFolder,
|
|
const Data::AssetId& sourceAssetId,
|
|
AZStd::string_view fileName,
|
|
uint8_t numResidentMips,
|
|
uint32_t subId)
|
|
: m_imageObject(imageObject)
|
|
, m_productFolder(saveFolder)
|
|
, m_sourceAssetId(sourceAssetId)
|
|
, m_fileName(fileName)
|
|
, m_numResidentMips(numResidentMips)
|
|
, m_subId(subId)
|
|
{
|
|
AZ_Assert(imageObject, "Input imageObject can't be empty");
|
|
AZ_Assert(sourceAssetId.IsValid(), "The source asset Id is not valid");
|
|
AZ_Assert(saveFolder.size() != 0, "saveFolder shouldn't be empty");
|
|
AZ_Assert(fileName.size() != 0, "fileName shouldn't be empty");
|
|
}
|
|
|
|
AZStd::string ImageAssetProducer::GenerateAssetFullPath(ImageAssetType assetType, uint32_t assetSubId)
|
|
{
|
|
// Note: m_fileName is the filename contains original extension to avoid file name collision if there are files with same name but with different extension.
|
|
// For example, test.jpg's image asset full path will be outputFolder/test.jpg.streamingimage.
|
|
// And test.png's image asset full path will be outputFolder/test.png.streamingimage.
|
|
AZStd::string fileName;
|
|
if (assetType == ImageAssetType::MipChain)
|
|
{
|
|
fileName = AZStd::string::format("%s.%d.%s", m_fileName.c_str(), assetSubId, RPI::ImageMipChainAsset::Extension);
|
|
}
|
|
else if (assetType == ImageAssetType::Image)
|
|
{
|
|
fileName = AZStd::string::format("%s.%s", m_fileName.c_str(), RPI::StreamingImageAsset::Extension);
|
|
}
|
|
else
|
|
{
|
|
AZ_Assert(false, "Unknown image asset type");
|
|
}
|
|
AZStd::string outPath;
|
|
AzFramework::StringFunc::Path::Join(m_productFolder.c_str(), fileName.c_str(), outPath, true, true);
|
|
return outPath;
|
|
}
|
|
|
|
const AZStd::vector<AssetBuilderSDK::JobProduct>& ImageAssetProducer::GetJobProducts() const
|
|
{
|
|
return m_jobProducts;
|
|
}
|
|
|
|
bool ImageAssetProducer::BuildImageAssets()
|
|
{
|
|
Data::AssetId imageAssetId(m_sourceAssetId.m_guid, m_subId);
|
|
|
|
AssetBuilderSDK::JobProduct product;
|
|
RPI::StreamingImageAssetCreator builder;
|
|
builder.Begin(imageAssetId);
|
|
|
|
int32_t arraySize = m_imageObject->HasImageFlags(EIF_Cubemap) ? 6 : 1;
|
|
uint32_t imageWidth = m_imageObject->GetWidth(0);
|
|
// The current cubemap faces are vertically aligned in the same buffer of image object. So the height should be divided by the array size.
|
|
uint32_t imageHeight = m_imageObject->GetHeight(0) / arraySize;
|
|
RHI::Format format = Utils::PixelFormatToRHIFormat(m_imageObject->GetPixelFormat(), m_imageObject->HasImageFlags(EIF_SRGBRead));
|
|
RHI::ImageBindFlags bindFlag = RHI::ImageBindFlags::ShaderRead;
|
|
|
|
RHI::ImageDescriptor imageDesc = RHI::ImageDescriptor::Create2DArray(bindFlag, imageWidth, imageHeight, aznumeric_cast<uint16_t>(arraySize), format);
|
|
imageDesc.m_mipLevels = aznumeric_cast<uint16_t>(m_imageObject->GetMipCount());
|
|
if (m_imageObject->HasImageFlags(EIF_Cubemap))
|
|
{
|
|
imageDesc.m_isCubemap = true;
|
|
}
|
|
|
|
builder.SetImageDescriptor(imageDesc);
|
|
|
|
// Set ImageViewDescriptor for cubemap. The regular 2d images can use the default one.
|
|
if (m_imageObject->HasImageFlags(EIF_Cubemap))
|
|
{
|
|
builder.SetImageViewDescriptor(RHI::ImageViewDescriptor::CreateCubemap());
|
|
}
|
|
|
|
// Build mip chain assets.
|
|
// Start from smallest mips so the mip chain asset for the lowest resolutions may contain more high level mips
|
|
uint32_t lastMip = imageDesc.m_mipLevels;
|
|
uint32_t totalSize = 0;
|
|
AZStd::vector<Data::Asset<RPI::ImageMipChainAsset>> mipChains;
|
|
uint32_t subid = m_subId + 1;
|
|
bool saveProduct = false; // used to skip exporting the tail mip chain to job product
|
|
|
|
// Merge m_residentMipCount amount of mips into a single mip chain, and add it to the StreamingImageAsset's tail mip chain
|
|
if (m_numResidentMips != 0)
|
|
{
|
|
const uint32_t boundedResidentMipCount = AZStd::min(static_cast<uint32_t>(m_numResidentMips), lastMip);
|
|
const uint32_t residentStartMip = lastMip - boundedResidentMipCount;
|
|
|
|
// Build the StreamingImageAsset's tail mip chain
|
|
Data::Asset<RPI::ImageMipChainAsset> chainAsset;
|
|
const bool isSuccess = BuildMipChainAsset(Data::AssetId(m_sourceAssetId.m_guid, subid), residentStartMip, boundedResidentMipCount, chainAsset, saveProduct);
|
|
if (!isSuccess)
|
|
{
|
|
AZ_Warning("Image Processing", false, "Failed to build the StreamingImageAsset's m_tailMipChain");
|
|
return false;
|
|
}
|
|
|
|
mipChains.emplace_back(chainAsset);
|
|
|
|
// Set the state for the remaining mips
|
|
saveProduct = true;
|
|
lastMip = residentStartMip;
|
|
subid++;
|
|
}
|
|
|
|
// Create MipChainAssets for the remaining mips
|
|
for (int32_t mip = lastMip - 1; mip >= 0; mip--)
|
|
{
|
|
totalSize += m_imageObject->GetMipBufSize(mip);
|
|
|
|
// Trying to fit as many as mips to 64k memory for one mip chain
|
|
if (mip == 0 || totalSize + m_imageObject->GetMipBufSize(mip - 1) > s_mininumMipBlockSize)
|
|
{
|
|
// Build a chain
|
|
Data::Asset<RPI::ImageMipChainAsset> chainAsset;
|
|
bool isSuccess = BuildMipChainAsset(Data::AssetId(m_sourceAssetId.m_guid, subid), mip, lastMip - mip, chainAsset, saveProduct);
|
|
saveProduct = true;
|
|
if (!isSuccess)
|
|
{
|
|
AZ_Warning("Image Processing", false, "Failed to build mip chain asset");
|
|
return false;
|
|
}
|
|
|
|
mipChains.emplace_back(chainAsset);
|
|
|
|
subid++;
|
|
lastMip = mip;
|
|
totalSize = 0;
|
|
}
|
|
}
|
|
|
|
// Set the builder's state to non-streaming if it only has a single mip chain, which will be stored in the
|
|
// StreamingImage's tail mip chain
|
|
if (mipChains.size() == 1)
|
|
{
|
|
builder.SetFlags(AZ::RPI::StreamingImageFlags::NotStreamable);
|
|
}
|
|
|
|
// Add mip chains to the builder from mip level 0 to highest
|
|
for (auto it = mipChains.rbegin(); it != mipChains.rend(); it++)
|
|
{
|
|
builder.AddMipChainAsset(*it->GetAs<RPI::ImageMipChainAsset>());
|
|
|
|
// Add all the mip chain assets as dependencies except the tail mip chain since its embedded in the StreamingImageAsset
|
|
if (it->Get() != mipChains.begin()->Get())
|
|
{
|
|
// Use PreLoad for mipchain assets for now.
|
|
// [GFX TODO] [ATOM-14467] Remove unnecessary code in StreamingImage::OnAssetReloaded when runtime switching dependency load behavior is supported
|
|
product.m_dependencies.push_back(AssetBuilderSDK::ProductDependency(it->GetId(), AZ::Data::ProductDependencyInfo::CreateFlags(AZ::Data::AssetLoadBehavior::PreLoad)));
|
|
}
|
|
}
|
|
|
|
product.m_dependenciesHandled = true; // We've output the dependencies immediately above so it's OK to tell the AP we've handled dependencies
|
|
|
|
bool result = false;
|
|
|
|
Data::Asset<RPI::StreamingImageAsset> imageAsset;
|
|
if (builder.End(imageAsset))
|
|
{
|
|
AZStd::string destPath = GenerateAssetFullPath(ImageAssetType::Image, imageAsset.GetId().m_subId);
|
|
result = AZ::Utils::SaveObjectToFile(destPath, AZ::DataStream::ST_BINARY, imageAsset.GetData(), imageAsset.GetData()->GetType(), nullptr);
|
|
if (!result)
|
|
{
|
|
AZ_Warning("Image Processing", false, "Failed to save image asset to file %s", destPath.c_str());
|
|
}
|
|
else
|
|
{
|
|
product.m_productAssetType = imageAsset.GetData()->GetType();
|
|
product.m_productSubID = imageAsset.GetId().m_subId;
|
|
product.m_productFileName = destPath;
|
|
|
|
// The StreamingImageAsset is added to end of product list in purpose.
|
|
// This is in case a new mip chain file is generated, for example when updating the original image's resolution,
|
|
// the mip chain asset hasn't be registered by the AssetCatalog when processing the asset change notification for
|
|
// StreamingImageAsset reload and it leads to an unknown asset error.
|
|
// The Asset system can be modified to solve the problem so the order doesn't matter.
|
|
// The task is tracked in ATOM-242
|
|
m_jobProducts.push_back(AZStd::move(product));
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
bool ImageAssetProducer::BuildMipChainAsset(const Data::AssetId& chainAssetId, uint32_t startMip, uint32_t mipLevels,
|
|
Data::Asset<RPI::ImageMipChainAsset>& outAsset, bool saveAsProduct)
|
|
{
|
|
RPI::ImageMipChainAssetCreator builder;
|
|
uint32_t arraySize = m_imageObject->HasImageFlags(EIF_Cubemap) ? 6 : 1;
|
|
builder.Begin(chainAssetId, aznumeric_cast<uint16_t>(mipLevels), aznumeric_cast<uint16_t>(arraySize));
|
|
|
|
for (uint32_t mip = startMip; mip < startMip + mipLevels; mip++)
|
|
{
|
|
uint8_t* mipBuffer;
|
|
uint32_t pitch;
|
|
m_imageObject->GetImagePointer(mip, mipBuffer, pitch);
|
|
RHI::Format format = Utils::PixelFormatToRHIFormat(m_imageObject->GetPixelFormat(), m_imageObject->HasImageFlags(EIF_SRGBRead));
|
|
|
|
RHI::ImageSubresourceLayout layout = RHI::GetImageSubresourceLayout(RHI::Size(m_imageObject->GetWidth(mip), m_imageObject->GetHeight(mip) / arraySize, 1), format);
|
|
builder.BeginMip(layout);
|
|
|
|
for (uint32_t arrayIndex = 0; arrayIndex < arraySize; ++arrayIndex)
|
|
{
|
|
builder.AddSubImage(mipBuffer + arrayIndex * layout.m_bytesPerImage, layout.m_bytesPerImage);
|
|
}
|
|
|
|
builder.EndMip();
|
|
}
|
|
|
|
bool result = false;
|
|
if (builder.End(outAsset))
|
|
{
|
|
AZStd::string destPath = GenerateAssetFullPath(ImageAssetType::MipChain, outAsset.GetId().m_subId);
|
|
|
|
if (saveAsProduct)
|
|
{
|
|
result = AZ::Utils::SaveObjectToFile(destPath, AZ::DataStream::ST_BINARY, outAsset.GetData(), outAsset.GetData()->GetType(), nullptr);
|
|
|
|
if (result)
|
|
{
|
|
// Save job product
|
|
AssetBuilderSDK::JobProduct product;
|
|
product.m_productAssetType = outAsset.GetData()->GetType();
|
|
product.m_productSubID = outAsset.GetId().m_subId;
|
|
product.m_productFileName = destPath;
|
|
m_jobProducts.emplace_back(AZStd::move(product));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
result = true;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
} // namespace ImageProcessingAtom
|