librashader/librashader-runtime-mtl/src/luts.rs

94 lines
2.9 KiB
Rust
Raw Normal View History

use crate::error::{FilterChainError, Result};
use crate::texture::InputTexture;
use icrate::Metal::{
MTLBlitCommandEncoder, MTLDevice, MTLOrigin, MTLPixelFormatBGRA8Unorm, MTLRegion,
MTLResourceStorageModeManaged, MTLResourceStorageModeShared, MTLSize, MTLTexture,
MTLTextureDescriptor, MTLTextureUsageShaderRead,
};
use librashader_presets::TextureConfig;
use librashader_runtime::image::{Image, BGRA8};
2024-02-13 18:34:26 +11:00
use librashader_runtime::scaling::MipmapSize;
use objc2::runtime::ProtocolObject;
use std::ffi::c_void;
use std::ptr::NonNull;
pub(crate) struct LutTexture(InputTexture);
impl AsRef<InputTexture> for LutTexture {
fn as_ref(&self) -> &InputTexture {
self.0.as_ref()
}
}
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>,
) -> 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
});
descriptor.setStorageMode(
if cfg!(all(target_arch = "aarch64", target_vendor = "apple")) {
MTLResourceStorageModeShared
} else {
MTLResourceStorageModeManaged
},
);
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,
)
}
if config.mipmap && texture.mipmapLevelCount() > 1 {
2024-02-13 18:34:26 +11:00
mipmapper.generateMipmapsForTexture(&texture);
}
Ok(LutTexture(InputTexture {
texture,
wrap_mode: config.wrap_mode,
filter_mode: config.filter_mode,
mip_filter: config.filter_mode,
}))
}
}