From 0f6c619e54687ea34774705de09859c921eb9255 Mon Sep 17 00:00:00 2001 From: Jovan Ristic Date: Wed, 14 Feb 2024 22:26:12 -0800 Subject: [PATCH] D3D12 Pixel History fix format conversion. * DecodeFormattedComponents converted everything to floats which were then reinterpreted as int or uint via the unioned fields of PixelValue, resulting in erroneous value display in PixelHistoryView. * I've made a pixel history specific version of the format conversion which treats the values the way the viewer expects. --- renderdoc/driver/d3d12/d3d12_pixelhistory.cpp | 311 +++++++++++++++++- 1 file changed, 305 insertions(+), 6 deletions(-) diff --git a/renderdoc/driver/d3d12/d3d12_pixelhistory.cpp b/renderdoc/driver/d3d12/d3d12_pixelhistory.cpp index d237afc15c..31a0ae1b43 100644 --- a/renderdoc/driver/d3d12/d3d12_pixelhistory.cpp +++ b/renderdoc/driver/d3d12/d3d12_pixelhistory.cpp @@ -2690,16 +2690,315 @@ bool IsDirectWrite(ResourceUsage usage) usage == ResourceUsage::GenMips); } -void FillInColor(ResourceFormat fmt, const D3D12PixelHistoryValue &value, ModificationValue &mod) +// NOTE: This function is quite similar to formatpacking.cpp::DecodeFormattedComponents, +// but it doesn't convert everything to a float since the pixel history viewer will expect +// them to be in the target's format. +void PixelHistoryDecode(const ResourceFormat &fmt, const byte *data, PixelValue &out) { - FloatVector v4 = DecodeFormattedComponents(fmt, value.color); - memcpy(mod.col.floatValue.data(), &v4, sizeof(v4)); + if(fmt.compType == CompType::UInt || fmt.compType == CompType::SInt || fmt.compCount == 4) + out.floatValue[3] = 0.0f; + + if(fmt.type == ResourceFormatType::R10G10B10A2) + { + if(fmt.compType == CompType::SNorm) + { + Vec4f v = ConvertFromR10G10B10A2SNorm(*(const uint32_t *)data); + out.floatValue[0] = v.x; + out.floatValue[1] = v.y; + out.floatValue[2] = v.z; + out.floatValue[3] = v.w; + } + else if(fmt.compType == CompType::UNorm) + { + Vec4f v = ConvertFromR10G10B10A2(*(const uint32_t *)data); + out.floatValue[0] = v.x; + out.floatValue[1] = v.y; + out.floatValue[2] = v.z; + out.floatValue[3] = v.w; + } + else if(fmt.compType == CompType::UInt) + { + Vec4u v = ConvertFromR10G10B10A2UInt(*(const uint32_t *)data); + out.uintValue[0] = v.x; + out.uintValue[1] = v.y; + out.uintValue[2] = v.z; + out.uintValue[3] = v.w; + } + + // the different types are a union so we can ignore format and just treat it as a data swap + if(fmt.BGRAOrder()) + std::swap(out.uintValue[0], out.uintValue[2]); + } + else if(fmt.type == ResourceFormatType::R11G11B10) + { + Vec3f v = ConvertFromR11G11B10(*(const uint32_t *)data); + out.floatValue[0] = v.x; + out.floatValue[1] = v.y; + out.floatValue[2] = v.z; + } + else if(fmt.type == ResourceFormatType::R5G5B5A1) + { + Vec4f v = ConvertFromB5G5R5A1(*(const uint16_t *)data); + out.floatValue[0] = v.x; + out.floatValue[1] = v.y; + out.floatValue[2] = v.z; + out.floatValue[3] = v.w; + + // conversely we *expect* BGRA order for this format and the above conversion implicitly flips + // when bit-unpacking. So if the format wasn't BGRA order, flip it back + if(!fmt.BGRAOrder()) + std::swap(out.floatValue[0], out.floatValue[2]); + } + else if(fmt.type == ResourceFormatType::R5G6B5) + { + Vec3f v = ConvertFromB5G6R5(*(const uint16_t *)data); + out.floatValue[0] = v.x; + out.floatValue[1] = v.y; + out.floatValue[2] = v.z; + + // conversely we *expect* BGRA order for this format and the above conversion implicitly flips + // when bit-unpacking. So if the format wasn't BGRA order, flip it back + if(!fmt.BGRAOrder()) + std::swap(out.floatValue[0], out.floatValue[2]); + } + else if(fmt.type == ResourceFormatType::R4G4B4A4) + { + Vec4f v = ConvertFromB4G4R4A4(*(const uint16_t *)data); + out.floatValue[0] = v.x; + out.floatValue[1] = v.y; + out.floatValue[2] = v.z; + out.floatValue[3] = v.w; + + // conversely we *expect* BGRA order for this format and the above conversion implicitly flips + // when bit-unpacking. So if the format wasn't BGRA order, flip it back + if(!fmt.BGRAOrder()) + std::swap(out.floatValue[0], out.floatValue[2]); + } + else if(fmt.type == ResourceFormatType::R4G4) + { + Vec4f v = ConvertFromR4G4(*(const uint8_t *)data); + out.floatValue[0] = v.x; + out.floatValue[1] = v.y; + } + else if(fmt.type == ResourceFormatType::R9G9B9E5) + { + Vec3f v = ConvertFromR9G9B9E5(*(const uint32_t *)data); + out.floatValue[0] = v.x; + out.floatValue[1] = v.y; + out.floatValue[2] = v.z; + } + else if(fmt.type == ResourceFormatType::D16S8) + { + uint32_t val = *(const uint32_t *)data; + out.floatValue[0] = float(val & 0x00ffff) / 65535.0f; + out.floatValue[1] = float((val & 0xff0000) >> 16) / 255.0f; + out.floatValue[2] = 0.0f; + } + else if(fmt.type == ResourceFormatType::D24S8) + { + uint32_t val = *(const uint32_t *)data; + out.floatValue[0] = float(val & 0x00ffffff) / 16777215.0f; + out.floatValue[1] = float((val & 0xff000000) >> 24) / 255.0f; + out.floatValue[2] = 0.0f; + } + else if(fmt.type == ResourceFormatType::D32S8) + { + struct ds + { + float f; + uint32_t s; + } val; + val = *(const ds *)data; + out.floatValue[0] = val.f; + out.floatValue[1] = float(val.s) / 255.0f; + out.floatValue[2] = 0.0f; + } + else if(fmt.type == ResourceFormatType::Regular || fmt.type == ResourceFormatType::A8 || + fmt.type == ResourceFormatType::S8) + { + CompType compType = fmt.compType; + for(size_t c = 0; c < fmt.compCount; c++) + { + // alpha is never interpreted as sRGB + if(compType == CompType::UNormSRGB && c == 3) + compType = CompType::UNorm; + + if(fmt.compByteWidth == 8) + { + // we just downcast + const uint64_t *u64 = (const uint64_t *)data; + const int64_t *i64 = (const int64_t *)data; + + if(compType == CompType::Float) + { + out.floatValue[c] = float(*(const double *)u64); + } + else if(compType == CompType::UInt) + { + out.uintValue[c] = uint32_t(*u64); + } + else if(compType == CompType::UScaled) + { + out.floatValue[c] = float(*u64); + } + else if(compType == CompType::SInt) + { + out.intValue[c] = int32_t(*i64); + } + else if(compType == CompType::SScaled) + { + out.floatValue[c] = float(*i64); + } + } + else if(fmt.compByteWidth == 4) + { + const uint32_t *u32 = (const uint32_t *)data; + const int32_t *i32 = (const int32_t *)data; + + if(compType == CompType::Float || compType == CompType::Depth) + { + out.floatValue[c] = *(const float *)u32; + } + else if(compType == CompType::UInt) + { + out.uintValue[c] = uint32_t(*u32); + } + else if(compType == CompType::UScaled) + { + out.floatValue[c] = float(*u32); + } + else if(compType == CompType::SInt) + { + out.intValue[c] = int32_t(*i32); + } + else if(compType == CompType::SScaled) + { + out.floatValue[c] = float(*i32); + } + } + else if(fmt.compByteWidth == 3 && compType == CompType::Depth) + { + // 24-bit depth is a weird edge case we need to assemble it by hand + const uint8_t *u8 = (const uint8_t *)data; + + uint32_t depth = 0; + depth |= uint32_t(u8[0]); + depth |= uint32_t(u8[1]) << 8; + depth |= uint32_t(u8[2]) << 16; + + out.floatValue[c] = float(depth) / float(16777215.0f); + } + else if(fmt.compByteWidth == 2) + { + const uint16_t *u16 = (const uint16_t *)data; + const int16_t *i16 = (const int16_t *)data; + + if(compType == CompType::Float) + { + out.floatValue[c] = ConvertFromHalf(*u16); + } + else if(compType == CompType::UInt) + { + out.uintValue[c] = uint32_t(*u16); + } + else if(compType == CompType::UScaled) + { + out.floatValue[c] = float(*u16); + } + else if(compType == CompType::SInt) + { + out.intValue[c] = int32_t(*i16); + } + else if(compType == CompType::SScaled) + { + out.floatValue[c] = float(*i16); + } + // 16-bit depth is UNORM + else if(compType == CompType::UNorm || compType == CompType::Depth) + { + out.floatValue[c] = float(*u16) / 65535.0f; + } + else if(compType == CompType::SNorm) + { + float f = -1.0f; + + if(*i16 == -32768) + f = -1.0f; + else + f = ((float)*i16) / 32767.0f; + + out.floatValue[c] = f; + } + } + else if(fmt.compByteWidth == 1) + { + const uint8_t *u8 = (const uint8_t *)data; + const int8_t *i8 = (const int8_t *)data; + + if(compType == CompType::UInt) + { + out.uintValue[c] = uint32_t(*u8); + } + else if(compType == CompType::UScaled) + { + out.floatValue[c] = float(*u8); + } + else if(compType == CompType::SInt) + { + out.intValue[c] = int32_t(*i8); + } + else if(compType == CompType::SScaled) + { + out.floatValue[c] = float(*i8); + } + else if(compType == CompType::UNormSRGB) + { + out.floatValue[c] = ConvertFromSRGB8(*u8); + } + else if(compType == CompType::UNorm) + { + out.floatValue[c] = float(*u8) / 255.0f; + } + else if(compType == CompType::SNorm) + { + float f = -1.0f; + + if(*i8 == -128) + f = -1.0f; + else + f = ((float)*i8) / 127.0f; + + out.floatValue[c] = f; + } + } + else + { + RDCERR("Unexpected format to convert from %u %u", fmt.compByteWidth, compType); + } + + data += fmt.compByteWidth; + } + + if(fmt.type == ResourceFormatType::A8) + { + out.floatValue[2] = out.floatValue[0]; + out.floatValue[0] = 0.0f; + } + else if(fmt.type == ResourceFormatType::S8) + { + out.uintValue[1] = out.uintValue[0]; + out.uintValue[0] = 0; + } + + // the different types are a union so we can ignore format and just treat it as a data swap + if(fmt.BGRAOrder()) + std::swap(out.uintValue[0], out.uintValue[2]); + } } -void FillInColor(ResourceFormat fmt, const byte *pValue, ModificationValue &mod) +void FillInColor(ResourceFormat fmt, const D3D12PixelHistoryValue &value, ModificationValue &mod) { - FloatVector v4 = DecodeFormattedComponents(fmt, pValue); - memcpy(mod.col.floatValue.data(), &v4, sizeof(v4)); + PixelHistoryDecode(fmt, value.color, mod.col); } void ConvertAndFillInColor(ResourceFormat srcFmt, ResourceFormat outFmt,