2024-02-12 09:54:13 +11:00
|
|
|
use crate::error::{FilterChainError, Result};
|
2024-02-12 12:38:55 +11:00
|
|
|
use crate::texture::InputTexture;
|
2024-02-14 09:37:48 +11:00
|
|
|
use icrate::Metal::{
|
|
|
|
MTLBlitCommandEncoder, MTLDevice, MTLOrigin, MTLPixelFormatBGRA8Unorm, MTLRegion,
|
|
|
|
MTLResourceStorageModeManaged, MTLResourceStorageModeShared, MTLSize, MTLTexture,
|
|
|
|
MTLTextureDescriptor, MTLTextureUsageShaderRead,
|
|
|
|
};
|
2024-02-12 09:54:13 +11:00
|
|
|
use librashader_presets::TextureConfig;
|
|
|
|
use librashader_runtime::image::{Image, BGRA8};
|
2024-02-13 18:34:26 +11:00
|
|
|
use librashader_runtime::scaling::MipmapSize;
|
2024-02-12 09:54:13 +11:00
|
|
|
use objc2::runtime::ProtocolObject;
|
|
|
|
use std::ffi::c_void;
|
|
|
|
use std::ptr::NonNull;
|
|
|
|
|
2024-02-12 12:38:55 +11:00
|
|
|
pub(crate) struct LutTexture(InputTexture);
|
|
|
|
|
|
|
|
impl AsRef<InputTexture> for LutTexture {
|
|
|
|
fn as_ref(&self) -> &InputTexture {
|
|
|
|
self.0.as_ref()
|
|
|
|
}
|
|
|
|
}
|
2024-02-12 09:54:13 +11:00
|
|
|
|
|
|
|
impl LutTexture {
|
|
|
|
pub fn new(
|
|
|
|
device: &ProtocolObject<dyn MTLDevice>,
|
|
|
|
image: Image<BGRA8>,
|
|
|
|
config: &TextureConfig,
|
2024-02-13 18:34:26 +11:00
|
|
|
mipmapper: &ProtocolObject<dyn MTLBlitCommandEncoder>,
|
2024-02-12 09:54:13 +11:00
|
|
|
) -> Result<Self> {
|
|
|
|
let descriptor = unsafe {
|
|
|
|
let descriptor =
|
|
|
|
MTLTextureDescriptor::texture2DDescriptorWithPixelFormat_width_height_mipmapped(
|
|
|
|
MTLPixelFormatBGRA8Unorm,
|
|
|
|
image.size.width as usize,
|
|
|
|
image.size.height as usize,
|
|
|
|
config.mipmap,
|
|
|
|
);
|
|
|
|
|
|
|
|
descriptor.setSampleCount(1);
|
2024-02-13 18:34:26 +11:00
|
|
|
descriptor.setMipmapLevelCount(if config.mipmap {
|
|
|
|
image.size.calculate_miplevels() as usize
|
|
|
|
} else {
|
|
|
|
1
|
|
|
|
});
|
2024-02-12 09:54:13 +11:00
|
|
|
|
2024-02-14 09:37:48 +11:00
|
|
|
descriptor.setStorageMode(
|
|
|
|
if cfg!(all(target_arch = "aarch64", target_vendor = "apple")) {
|
2024-02-13 18:51:08 +11:00
|
|
|
MTLResourceStorageModeShared
|
|
|
|
} else {
|
|
|
|
MTLResourceStorageModeManaged
|
2024-02-14 09:37:48 +11:00
|
|
|
},
|
|
|
|
);
|
2024-02-13 18:51:08 +11:00
|
|
|
|
2024-02-12 09:54:13 +11:00
|
|
|
descriptor.setUsage(MTLTextureUsageShaderRead);
|
|
|
|
|
|
|
|
descriptor
|
|
|
|
};
|
|
|
|
|
|
|
|
let texture = device
|
|
|
|
.newTextureWithDescriptor(&descriptor)
|
|
|
|
.ok_or(FilterChainError::FailedToCreateTexture)?;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let region = MTLRegion {
|
|
|
|
origin: MTLOrigin { x: 0, y: 0, z: 0 },
|
|
|
|
size: MTLSize {
|
|
|
|
width: image.size.width as usize,
|
|
|
|
height: image.size.height as usize,
|
|
|
|
depth: 1,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
texture.replaceRegion_mipmapLevel_withBytes_bytesPerRow(
|
|
|
|
region,
|
|
|
|
0,
|
|
|
|
// SAFETY: replaceRegion withBytes is const.
|
|
|
|
NonNull::new_unchecked(image.bytes.as_slice().as_ptr() as *mut c_void),
|
|
|
|
4 * image.size.width as usize,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2024-02-16 11:57:34 +11:00
|
|
|
if config.mipmap && texture.mipmapLevelCount() > 1 {
|
2024-02-13 18:34:26 +11:00
|
|
|
mipmapper.generateMipmapsForTexture(&texture);
|
2024-02-12 09:54:13 +11:00
|
|
|
}
|
|
|
|
|
2024-02-12 12:38:55 +11:00
|
|
|
Ok(LutTexture(InputTexture {
|
|
|
|
texture,
|
|
|
|
wrap_mode: config.wrap_mode,
|
|
|
|
filter_mode: config.filter_mode,
|
|
|
|
mip_filter: config.filter_mode,
|
|
|
|
}))
|
2024-02-12 09:54:13 +11:00
|
|
|
}
|
|
|
|
}
|