use core::ops::Range;
use objc2::{Message, extern_protocol, msg_send, rc::Retained, runtime::ProtocolObject};
use objc2_foundation::{NSObjectProtocol, NSRange, NSString};
use crate::util::ref_slice_as_ptr;
use crate::*;
extern_protocol!(
pub unsafe trait MTL4CommandBuffer: NSObjectProtocol {
#[unsafe(method(device))]
#[unsafe(method_family = none)]
fn device(&self) -> Retained<ProtocolObject<dyn MTLDevice>>;
#[unsafe(method(beginCommandBufferWithAllocator:))]
#[unsafe(method_family = none)]
fn begin_command_buffer_with_allocator(
&self,
allocator: &ProtocolObject<dyn MTL4CommandAllocator>,
);
#[unsafe(method(beginCommandBufferWithAllocator:options:))]
#[unsafe(method_family = none)]
fn begin_command_buffer_with_allocator_options(
&self,
allocator: &ProtocolObject<dyn MTL4CommandAllocator>,
options: &MTL4CommandBufferOptions,
);
#[unsafe(method(endCommandBuffer))]
#[unsafe(method_family = none)]
fn end_command_buffer(&self);
#[unsafe(method(renderCommandEncoderWithDescriptor:))]
#[unsafe(method_family = none)]
fn render_command_encoder_with_descriptor(
&self,
descriptor: &MTL4RenderPassDescriptor,
) -> Option<Retained<ProtocolObject<dyn MTL4RenderCommandEncoder>>>;
#[unsafe(method(renderCommandEncoderWithDescriptor:options:))]
#[unsafe(method_family = none)]
fn render_command_encoder_with_descriptor_options(
&self,
descriptor: &MTL4RenderPassDescriptor,
options: MTL4RenderEncoderOptions,
) -> Option<Retained<ProtocolObject<dyn MTL4RenderCommandEncoder>>>;
#[unsafe(method(computeCommandEncoder))]
#[unsafe(method_family = none)]
fn compute_command_encoder(&self) -> Option<Retained<ProtocolObject<dyn MTL4ComputeCommandEncoder>>>;
#[unsafe(method(machineLearningCommandEncoder))]
#[unsafe(method_family = none)]
fn machine_learning_command_encoder(
&self
) -> Option<Retained<ProtocolObject<dyn MTL4MachineLearningCommandEncoder>>>;
#[unsafe(method(useResidencySet:))]
#[unsafe(method_family = none)]
fn use_residency_set(
&self,
residency_set: &ProtocolObject<dyn MTLResidencySet>,
);
#[unsafe(method(popDebugGroup))]
#[unsafe(method_family = none)]
fn pop_debug_group(&self);
#[unsafe(method(writeTimestampIntoHeap:atIndex:))]
#[unsafe(method_family = none)]
fn write_timestamp_into_heap_at_index(
&self,
counter_heap: &ProtocolObject<dyn MTL4CounterHeap>,
index: usize,
);
}
);
#[allow(unused)]
pub trait MTL4CommandBufferExt: MTL4CommandBuffer + Message {
fn label(&self) -> Option<String>;
fn set_label(
&self,
label: Option<&str>,
);
fn push_debug_group(
&self,
label: &str,
);
fn write_timestamp_into_heap_at_index_usize(
&self,
counter_heap: &ProtocolObject<dyn MTL4CounterHeap>,
index: usize,
);
fn resolve_counter_heap_into_buffer_with_range(
&self,
counter_heap: &ProtocolObject<dyn MTL4CounterHeap>,
range: Range<usize>,
buffer_range: MTL4BufferRange,
fence_to_wait: Option<&ProtocolObject<dyn MTLFence>>,
fence_to_update: Option<&ProtocolObject<dyn MTLFence>>,
);
fn use_residency_sets(&self, residency_sets: &[&ProtocolObject<dyn MTLResidencySet>]);
}
impl MTL4CommandBufferExt for ProtocolObject<dyn MTL4CommandBuffer> {
fn label(&self) -> Option<String> {
let s: Option<Retained<NSString>> = unsafe { msg_send![self, label] };
s.map(|v| v.to_string())
}
fn set_label(
&self,
label: Option<&str>,
) {
unsafe {
let _: () = msg_send![self, setLabel: label.map(NSString::from_str).as_deref()];
}
}
fn push_debug_group(
&self,
label: &str,
) {
unsafe {
let _: () = msg_send![self, pushDebugGroup: &*NSString::from_str(label)];
}
}
fn write_timestamp_into_heap_at_index_usize(
&self,
counter_heap: &ProtocolObject<dyn MTL4CounterHeap>,
index: usize,
) {
unsafe {
let _: () = msg_send![self, writeTimestampIntoHeap: counter_heap, atIndex: index];
}
}
fn resolve_counter_heap_into_buffer_with_range(
&self,
counter_heap: &ProtocolObject<dyn MTL4CounterHeap>,
range: Range<usize>,
buffer_range: MTL4BufferRange,
fence_to_wait: Option<&ProtocolObject<dyn MTLFence>>,
fence_to_update: Option<&ProtocolObject<dyn MTLFence>>,
) {
let ns_range = NSRange {
location: range.start,
length: range.end.saturating_sub(range.start),
};
unsafe {
let _: () = msg_send![
self,
resolveCounterHeap: counter_heap,
withRange: ns_range,
intoBuffer: buffer_range,
waitFence: fence_to_wait,
updateFence: fence_to_update
];
}
}
fn use_residency_sets(&self, residency_sets: &[&ProtocolObject<dyn MTLResidencySet>]) {
let ptr = ref_slice_as_ptr(residency_sets);
unsafe {
let _: () = msg_send![self, useResidencySets: ptr, count: residency_sets.len()];
}
}
}