mirror of
				https://github.com/encounter/aurora.git
				synced 2025-10-26 11:50:22 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			217 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			217 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "common.hpp"
 | |
| 
 | |
| #include "../internal.hpp"
 | |
| #include "../webgpu/gpu.hpp"
 | |
| #include "aurora/aurora.h"
 | |
| #include "texture.hpp"
 | |
| #include "texture_convert.hpp"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <cstdint>
 | |
| #include <memory>
 | |
| #include <utility>
 | |
| 
 | |
| #include <fmt/format.h>
 | |
| #include <magic_enum.hpp>
 | |
| #include <webgpu/webgpu_cpp.h>
 | |
| 
 | |
| namespace aurora::gfx {
 | |
| using webgpu::g_device;
 | |
| using webgpu::g_queue;
 | |
| 
 | |
| namespace {
 | |
| Module Log("aurora::gfx");
 | |
| 
 | |
| struct TextureFormatInfo {
 | |
|   uint8_t blockWidth;
 | |
|   uint8_t blockHeight;
 | |
|   uint8_t blockSize;
 | |
|   bool compressed;
 | |
| };
 | |
| 
 | |
| TextureFormatInfo format_info(wgpu::TextureFormat format) {
 | |
|   switch (format) {
 | |
|     DEFAULT_FATAL("unimplemented texture format {}", magic_enum::enum_name(format));
 | |
|   case wgpu::TextureFormat::R8Unorm:
 | |
|     return {1, 1, 1, false};
 | |
|   case wgpu::TextureFormat::R16Sint:
 | |
|     return {1, 1, 2, false};
 | |
|   case wgpu::TextureFormat::RGBA8Unorm:
 | |
|   case wgpu::TextureFormat::R32Float:
 | |
|     return {1, 1, 4, false};
 | |
|   case wgpu::TextureFormat::BC1RGBAUnorm:
 | |
|     return {4, 4, 8, true};
 | |
|   }
 | |
| }
 | |
| 
 | |
| wgpu::Extent3D physical_size(wgpu::Extent3D size, TextureFormatInfo info) {
 | |
|   const uint32_t width = ((size.width + info.blockWidth - 1) / info.blockWidth) * info.blockWidth;
 | |
|   const uint32_t height = ((size.height + info.blockHeight - 1) / info.blockHeight) * info.blockHeight;
 | |
|   return {.width = width, .height = height, .depthOrArrayLayers = size.depthOrArrayLayers};
 | |
| }
 | |
| } // namespace
 | |
| 
 | |
| TextureHandle new_static_texture_2d(uint32_t width, uint32_t height, uint32_t mips, u32 format, ArrayRef<uint8_t> data,
 | |
|                                     bool tlut, const char* label) noexcept {
 | |
|   auto handle = new_dynamic_texture_2d(width, height, mips, format, label);
 | |
|   const auto& ref = *handle;
 | |
| 
 | |
|   ByteBuffer buffer;
 | |
|   if (ref.gxFormat != InvalidTextureFormat) {
 | |
|     if (tlut) {
 | |
|       CHECK(ref.size.height == 1, "new_static_texture_2d[{}]: expected tlut height 1, got {}", label, ref.size.height);
 | |
|       CHECK(ref.mipCount == 1, "new_static_texture_2d[{}]: expected tlut mipCount 1, got {}", label, ref.mipCount);
 | |
|       buffer = convert_tlut(ref.gxFormat, ref.size.width, data);
 | |
|     } else {
 | |
|       buffer = convert_texture(ref.gxFormat, ref.size.width, ref.size.height, ref.mipCount, data);
 | |
|     }
 | |
|     if (!buffer.empty()) {
 | |
|       data = {buffer.data(), buffer.size()};
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   uint32_t offset = 0;
 | |
|   for (uint32_t mip = 0; mip < mips; ++mip) {
 | |
|     const wgpu::Extent3D mipSize{
 | |
|         .width = std::max(ref.size.width >> mip, 1u),
 | |
|         .height = std::max(ref.size.height >> mip, 1u),
 | |
|         .depthOrArrayLayers = ref.size.depthOrArrayLayers,
 | |
|     };
 | |
|     const auto info = format_info(ref.format);
 | |
|     const auto physicalSize = physical_size(mipSize, info);
 | |
|     const uint32_t widthBlocks = physicalSize.width / info.blockWidth;
 | |
|     const uint32_t heightBlocks = physicalSize.height / info.blockHeight;
 | |
|     const uint32_t bytesPerRow = widthBlocks * info.blockSize;
 | |
|     const uint32_t dataSize = bytesPerRow * heightBlocks * mipSize.depthOrArrayLayers;
 | |
|     CHECK(offset + dataSize <= data.size(), "new_static_texture_2d[{}]: expected at least {} bytes, got {}", label,
 | |
|           offset + dataSize, data.size());
 | |
|     const wgpu::TexelCopyTextureInfo dstView{
 | |
|         .texture = ref.texture,
 | |
|         .mipLevel = mip,
 | |
|     };
 | |
|     //    const auto range = push_texture_data(data.data() + offset, dataSize, bytesPerRow, heightBlocks);
 | |
|     const wgpu::TexelCopyBufferLayout dataLayout{
 | |
|         //        .offset = range.offset,
 | |
|         .bytesPerRow = bytesPerRow,
 | |
|         .rowsPerImage = heightBlocks,
 | |
|     };
 | |
|     // TODO
 | |
|     //    g_textureUploads.emplace_back(dataLayout, std::move(dstView), physicalSize);
 | |
|     g_queue.WriteTexture(&dstView, data.data() + offset, dataSize, &dataLayout, &physicalSize);
 | |
|     offset += dataSize;
 | |
|   }
 | |
|   if (data.size() != UINT32_MAX && offset < data.size()) {
 | |
|     Log.warn("new_static_texture_2d[{}]: texture used {} bytes, but given {} bytes", label, offset, data.size());
 | |
|   }
 | |
|   return handle;
 | |
| }
 | |
| 
 | |
| TextureHandle new_dynamic_texture_2d(uint32_t width, uint32_t height, uint32_t mips, u32 format,
 | |
|                                      const char* label) noexcept {
 | |
|   const auto wgpuFormat = to_wgpu(format);
 | |
|   const wgpu::Extent3D size{
 | |
|       .width = width,
 | |
|       .height = height,
 | |
|       .depthOrArrayLayers = 1,
 | |
|   };
 | |
|   const wgpu::TextureDescriptor textureDescriptor{
 | |
|       .label = label,
 | |
|       .usage = wgpu::TextureUsage::TextureBinding | wgpu::TextureUsage::CopyDst,
 | |
|       .dimension = wgpu::TextureDimension::e2D,
 | |
|       .size = size,
 | |
|       .format = wgpuFormat,
 | |
|       .mipLevelCount = mips,
 | |
|       .sampleCount = 1,
 | |
|   };
 | |
|   const auto viewLabel = fmt::format("{} view", label);
 | |
|   const wgpu::TextureViewDescriptor textureViewDescriptor{
 | |
|       .label = viewLabel.c_str(),
 | |
|       .format = wgpuFormat,
 | |
|       .dimension = wgpu::TextureViewDimension::e2D,
 | |
|       .mipLevelCount = mips,
 | |
|   };
 | |
|   auto texture = g_device.CreateTexture(&textureDescriptor);
 | |
|   auto textureView = texture.CreateView(&textureViewDescriptor);
 | |
|   return std::make_shared<TextureRef>(std::move(texture), std::move(textureView), size, wgpuFormat, mips, format,
 | |
|                                       false);
 | |
| }
 | |
| 
 | |
| TextureHandle new_render_texture(uint32_t width, uint32_t height, u32 fmt, const char* label) noexcept {
 | |
|   const auto wgpuFormat = webgpu::g_graphicsConfig.surfaceConfiguration.format;
 | |
|   const wgpu::Extent3D size{
 | |
|       .width = width,
 | |
|       .height = height,
 | |
|       .depthOrArrayLayers = 1,
 | |
|   };
 | |
|   const wgpu::TextureDescriptor textureDescriptor{
 | |
|       .label = label,
 | |
|       .usage = wgpu::TextureUsage::TextureBinding | wgpu::TextureUsage::CopyDst,
 | |
|       .dimension = wgpu::TextureDimension::e2D,
 | |
|       .size = size,
 | |
|       .format = wgpuFormat,
 | |
|       .mipLevelCount = 1,
 | |
|       .sampleCount = 1,
 | |
|   };
 | |
|   const auto viewLabel = fmt::format("{} view", label);
 | |
|   const wgpu::TextureViewDescriptor textureViewDescriptor{
 | |
|       .label = viewLabel.c_str(),
 | |
|       .format = wgpuFormat,
 | |
|       .dimension = wgpu::TextureViewDimension::e2D,
 | |
|   };
 | |
|   auto texture = g_device.CreateTexture(&textureDescriptor);
 | |
|   auto textureView = texture.CreateView(&textureViewDescriptor);
 | |
|   return std::make_shared<TextureRef>(std::move(texture), std::move(textureView), size, wgpuFormat, 1, fmt, true);
 | |
| }
 | |
| 
 | |
| void write_texture(const TextureRef& ref, ArrayRef<uint8_t> data) noexcept {
 | |
|   ByteBuffer buffer;
 | |
|   if (ref.gxFormat != InvalidTextureFormat) {
 | |
|     buffer = convert_texture(ref.gxFormat, ref.size.width, ref.size.height, ref.mipCount, data);
 | |
|     if (!buffer.empty()) {
 | |
|       data = {buffer.data(), buffer.size()};
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   uint32_t offset = 0;
 | |
|   for (uint32_t mip = 0; mip < ref.mipCount; ++mip) {
 | |
|     const wgpu::Extent3D mipSize{
 | |
|         .width = std::max(ref.size.width >> mip, 1u),
 | |
|         .height = std::max(ref.size.height >> mip, 1u),
 | |
|         .depthOrArrayLayers = ref.size.depthOrArrayLayers,
 | |
|     };
 | |
|     const auto info = format_info(ref.format);
 | |
|     const auto physicalSize = physical_size(mipSize, info);
 | |
|     const uint32_t widthBlocks = physicalSize.width / info.blockWidth;
 | |
|     const uint32_t heightBlocks = physicalSize.height / info.blockHeight;
 | |
|     const uint32_t bytesPerRow = widthBlocks * info.blockSize;
 | |
|     const uint32_t dataSize = bytesPerRow * heightBlocks * mipSize.depthOrArrayLayers;
 | |
|     CHECK(offset + dataSize <= data.size(), "write_texture: expected at least {} bytes, got {}", offset + dataSize,
 | |
|           data.size());
 | |
|     //    auto dstView = wgpu::ImageCopyTexture{
 | |
|     //        .texture = ref.texture,
 | |
|     //        .mipLevel = mip,
 | |
|     //    };
 | |
|     //    const auto range = push_texture_data(data.data() + offset, dataSize, bytesPerRow, heightBlocks);
 | |
|     //    const auto dataLayout = wgpu::TextureDataLayout{
 | |
|     //        .offset = range.offset,
 | |
|     //        .bytesPerRow = bytesPerRow,
 | |
|     //        .rowsPerImage = heightBlocks,
 | |
|     //    };
 | |
|     //    g_textureUploads.emplace_back(dataLayout, std::move(dstView), physicalSize);
 | |
|     const wgpu::TexelCopyTextureInfo dstView{
 | |
|         .texture = ref.texture,
 | |
|         .mipLevel = mip,
 | |
|     };
 | |
|     const wgpu::TexelCopyBufferLayout dataLayout{
 | |
|         .bytesPerRow = bytesPerRow,
 | |
|         .rowsPerImage = heightBlocks,
 | |
|     };
 | |
|     g_queue.WriteTexture(&dstView, data.data() + offset, dataSize, &dataLayout, &physicalSize);
 | |
|     offset += dataSize;
 | |
|   }
 | |
|   if (data.size() != UINT32_MAX && offset < data.size()) {
 | |
|     Log.warn("write_texture: texture used {} bytes, but given {} bytes", offset, data.size());
 | |
|   }
 | |
| }
 | |
| } // namespace aurora::gfx
 |