0adf3505ec
This doesn't cause an API break in the C API but we don't actually make an attempt to verify that it's safe to access any of the device contexts.
233 lines
7.9 KiB
Rust
233 lines
7.9 KiB
Rust
use crate::filter_chain::VulkanObjects;
|
|
use crate::memory::{VulkanBuffer, VulkanImageMemory};
|
|
use crate::texture::{InputImage, VulkanImage};
|
|
use crate::{error, util};
|
|
use ash::vk;
|
|
use librashader_presets::TextureConfig;
|
|
use librashader_runtime::image::{Image, BGRA8};
|
|
use librashader_runtime::scaling::MipmapSize;
|
|
|
|
pub(crate) struct LutTexture {
|
|
_memory: VulkanImageMemory,
|
|
_staging: VulkanBuffer,
|
|
pub image: InputImage,
|
|
}
|
|
|
|
impl LutTexture {
|
|
pub fn new(
|
|
vulkan: &VulkanObjects,
|
|
cmd: vk::CommandBuffer,
|
|
image: Image<BGRA8>,
|
|
config: &TextureConfig,
|
|
) -> error::Result<LutTexture> {
|
|
let image_info = vk::ImageCreateInfo::builder()
|
|
.image_type(vk::ImageType::TYPE_2D)
|
|
.format(vk::Format::B8G8R8A8_UNORM)
|
|
.extent(image.size.into())
|
|
.mip_levels(if config.mipmap {
|
|
image.size.calculate_miplevels()
|
|
} else {
|
|
1
|
|
})
|
|
.array_layers(1)
|
|
.samples(vk::SampleCountFlags::TYPE_1)
|
|
.tiling(vk::ImageTiling::OPTIMAL)
|
|
.usage(
|
|
vk::ImageUsageFlags::SAMPLED
|
|
| vk::ImageUsageFlags::TRANSFER_SRC
|
|
| vk::ImageUsageFlags::TRANSFER_DST,
|
|
)
|
|
.initial_layout(vk::ImageLayout::UNDEFINED);
|
|
|
|
let texture = unsafe { vulkan.device.create_image(&image_info, None)? };
|
|
|
|
let memory = unsafe {
|
|
let mem_reqs = vulkan.device.get_image_memory_requirements(texture);
|
|
VulkanImageMemory::new(&vulkan.device, &vulkan.alloc, mem_reqs, &texture)?
|
|
};
|
|
|
|
let image_subresource = vk::ImageSubresourceRange::builder()
|
|
.level_count(image_info.mip_levels)
|
|
.layer_count(1)
|
|
.aspect_mask(vk::ImageAspectFlags::COLOR);
|
|
|
|
let swizzle_components = vk::ComponentMapping::builder()
|
|
.r(vk::ComponentSwizzle::R)
|
|
.g(vk::ComponentSwizzle::G)
|
|
.b(vk::ComponentSwizzle::B)
|
|
.a(vk::ComponentSwizzle::A);
|
|
|
|
let view_info = vk::ImageViewCreateInfo::builder()
|
|
.view_type(vk::ImageViewType::TYPE_2D)
|
|
.format(vk::Format::B8G8R8A8_UNORM)
|
|
.image(texture)
|
|
.subresource_range(*image_subresource)
|
|
.components(*swizzle_components);
|
|
|
|
let texture_view = unsafe { vulkan.device.create_image_view(&view_info, None)? };
|
|
|
|
let mut staging = VulkanBuffer::new(
|
|
&vulkan.device,
|
|
&vulkan.alloc,
|
|
vk::BufferUsageFlags::TRANSFER_SRC,
|
|
image.bytes.len(),
|
|
)?;
|
|
|
|
staging.as_mut_slice()?.copy_from_slice(&image.bytes);
|
|
|
|
unsafe {
|
|
util::vulkan_image_layout_transition_levels(
|
|
&vulkan.device,
|
|
cmd,
|
|
texture,
|
|
vk::REMAINING_MIP_LEVELS,
|
|
vk::ImageLayout::UNDEFINED,
|
|
if config.mipmap {
|
|
vk::ImageLayout::GENERAL
|
|
} else {
|
|
vk::ImageLayout::TRANSFER_DST_OPTIMAL
|
|
},
|
|
vk::AccessFlags::empty(),
|
|
vk::AccessFlags::TRANSFER_WRITE,
|
|
vk::PipelineStageFlags::TOP_OF_PIPE,
|
|
vk::PipelineStageFlags::TRANSFER,
|
|
vk::QUEUE_FAMILY_IGNORED,
|
|
vk::QUEUE_FAMILY_IGNORED,
|
|
);
|
|
|
|
let builder = vk::BufferImageCopy::builder()
|
|
.image_subresource(
|
|
*vk::ImageSubresourceLayers::builder()
|
|
.aspect_mask(vk::ImageAspectFlags::COLOR)
|
|
.mip_level(0)
|
|
.base_array_layer(0)
|
|
.layer_count(1),
|
|
)
|
|
.image_extent(image.size.into());
|
|
|
|
vulkan.device.cmd_copy_buffer_to_image(
|
|
cmd,
|
|
staging.handle,
|
|
texture,
|
|
if config.mipmap {
|
|
vk::ImageLayout::GENERAL
|
|
} else {
|
|
vk::ImageLayout::TRANSFER_DST_OPTIMAL
|
|
},
|
|
&[*builder],
|
|
)
|
|
}
|
|
|
|
// generate mipmaps
|
|
for level in 1..image_info.mip_levels {
|
|
let source_size = image.size.scale_mipmap(level - 1);
|
|
let target_size = image.size.scale_mipmap(level);
|
|
|
|
let src_offsets = [
|
|
vk::Offset3D { x: 0, y: 0, z: 0 },
|
|
vk::Offset3D {
|
|
x: source_size.width as i32,
|
|
y: source_size.height as i32,
|
|
z: 1,
|
|
},
|
|
];
|
|
|
|
let dst_offsets = [
|
|
vk::Offset3D { x: 0, y: 0, z: 0 },
|
|
vk::Offset3D {
|
|
x: target_size.width as i32,
|
|
y: target_size.height as i32,
|
|
z: 1,
|
|
},
|
|
];
|
|
let src_subresource = vk::ImageSubresourceLayers::builder()
|
|
.aspect_mask(vk::ImageAspectFlags::COLOR)
|
|
.mip_level(level - 1)
|
|
.base_array_layer(0)
|
|
.layer_count(1);
|
|
|
|
let dst_subresource = vk::ImageSubresourceLayers::builder()
|
|
.aspect_mask(vk::ImageAspectFlags::COLOR)
|
|
.mip_level(level)
|
|
.base_array_layer(0)
|
|
.layer_count(1);
|
|
|
|
let image_blit = vk::ImageBlit::builder()
|
|
.src_subresource(*src_subresource)
|
|
.src_offsets(src_offsets)
|
|
.dst_subresource(*dst_subresource)
|
|
.dst_offsets(dst_offsets);
|
|
|
|
unsafe {
|
|
util::vulkan_image_layout_transition_levels(
|
|
&vulkan.device,
|
|
cmd,
|
|
texture,
|
|
vk::REMAINING_MIP_LEVELS,
|
|
vk::ImageLayout::GENERAL,
|
|
vk::ImageLayout::GENERAL,
|
|
vk::AccessFlags::TRANSFER_WRITE,
|
|
vk::AccessFlags::TRANSFER_READ,
|
|
vk::PipelineStageFlags::TRANSFER,
|
|
vk::PipelineStageFlags::TRANSFER,
|
|
vk::QUEUE_FAMILY_IGNORED,
|
|
vk::QUEUE_FAMILY_IGNORED,
|
|
);
|
|
|
|
// todo: respect mipmap filter?
|
|
vulkan.device.cmd_blit_image(
|
|
cmd,
|
|
texture,
|
|
vk::ImageLayout::GENERAL,
|
|
texture,
|
|
vk::ImageLayout::GENERAL,
|
|
&[*image_blit],
|
|
config.filter_mode.into(),
|
|
);
|
|
}
|
|
}
|
|
|
|
unsafe {
|
|
util::vulkan_image_layout_transition_levels(
|
|
&vulkan.device,
|
|
cmd,
|
|
texture,
|
|
vk::REMAINING_MIP_LEVELS,
|
|
if config.mipmap {
|
|
vk::ImageLayout::GENERAL
|
|
} else {
|
|
vk::ImageLayout::TRANSFER_DST_OPTIMAL
|
|
},
|
|
vk::ImageLayout::SHADER_READ_ONLY_OPTIMAL,
|
|
vk::AccessFlags::TRANSFER_WRITE,
|
|
vk::AccessFlags::SHADER_READ,
|
|
vk::PipelineStageFlags::TRANSFER,
|
|
vk::PipelineStageFlags::FRAGMENT_SHADER,
|
|
vk::QUEUE_FAMILY_IGNORED,
|
|
vk::QUEUE_FAMILY_IGNORED,
|
|
);
|
|
}
|
|
|
|
Ok(LutTexture {
|
|
_memory: memory,
|
|
_staging: staging,
|
|
image: InputImage {
|
|
image_view: texture_view,
|
|
image: VulkanImage {
|
|
size: image.size,
|
|
image: texture,
|
|
format: vk::Format::B8G8R8A8_UNORM,
|
|
},
|
|
filter_mode: config.filter_mode,
|
|
wrap_mode: config.wrap_mode,
|
|
mip_filter: config.filter_mode,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
|
|
impl AsRef<InputImage> for LutTexture {
|
|
fn as_ref(&self) -> &InputImage {
|
|
&self.image
|
|
}
|
|
}
|