2021-05-22 10:39:28 +10:00
|
|
|
// Copyright 2021 The piet-gpu authors.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
//
|
|
|
|
// Also licensed under MIT license, at your choice.
|
|
|
|
|
2021-11-21 16:52:29 +11:00
|
|
|
mod clear;
|
2022-05-10 12:39:59 +10:00
|
|
|
mod timer;
|
2021-05-29 12:59:04 +10:00
|
|
|
mod util;
|
|
|
|
|
|
|
|
use std::mem;
|
|
|
|
use std::sync::{Arc, Mutex};
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
use block::Block;
|
2021-05-29 12:59:04 +10:00
|
|
|
use cocoa_foundation::base::id;
|
|
|
|
use cocoa_foundation::foundation::{NSInteger, NSUInteger};
|
2022-05-10 12:39:59 +10:00
|
|
|
use foreign_types::ForeignType;
|
2021-05-27 15:07:00 +10:00
|
|
|
use objc::rc::autoreleasepool;
|
2021-05-29 12:59:04 +10:00
|
|
|
use objc::runtime::{Object, BOOL, YES};
|
2021-05-27 15:07:00 +10:00
|
|
|
use objc::{class, msg_send, sel, sel_impl};
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
use metal::{CGFloat, CommandBufferRef, MTLFeatureSet};
|
2021-05-29 12:59:04 +10:00
|
|
|
|
|
|
|
use raw_window_handle::{HasRawWindowHandle, RawWindowHandle};
|
2021-05-27 15:07:00 +10:00
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
use crate::{
|
|
|
|
BufferUsage, ComputePassDescriptor, Error, GpuInfo, ImageFormat, MapMode, WorkgroupLimits,
|
|
|
|
};
|
2021-05-22 10:39:28 +10:00
|
|
|
|
2021-05-29 12:59:04 +10:00
|
|
|
use util::*;
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
use self::timer::{CounterSampleBuffer, CounterSet, TimeCalibration};
|
|
|
|
|
2021-05-27 12:08:56 +10:00
|
|
|
pub struct MtlInstance;
|
2021-05-22 10:39:28 +10:00
|
|
|
|
2021-05-27 12:08:56 +10:00
|
|
|
pub struct MtlDevice {
|
2021-05-22 10:39:28 +10:00
|
|
|
device: metal::Device,
|
2021-05-29 12:59:04 +10:00
|
|
|
cmd_queue: Arc<Mutex<metal::CommandQueue>>,
|
2021-05-27 15:07:00 +10:00
|
|
|
gpu_info: GpuInfo,
|
2021-11-21 16:52:29 +11:00
|
|
|
helpers: Arc<Helpers>,
|
2022-05-10 12:39:59 +10:00
|
|
|
timer_set: Option<CounterSet>,
|
|
|
|
counter_style: CounterStyle,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Type of counter sampling.
|
|
|
|
///
|
|
|
|
/// See https://developer.apple.com/documentation/metal/counter_sampling/sampling_gpu_data_into_counter_sample_buffers
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
|
|
|
|
enum CounterStyle {
|
|
|
|
None,
|
|
|
|
Stage,
|
|
|
|
Command,
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-05-29 12:59:04 +10:00
|
|
|
pub struct MtlSurface {
|
|
|
|
layer: metal::MetalLayer,
|
|
|
|
}
|
2021-05-27 12:08:56 +10:00
|
|
|
|
2021-05-29 12:59:04 +10:00
|
|
|
pub struct MtlSwapchain {
|
|
|
|
layer: metal::MetalLayer,
|
|
|
|
cmd_queue: Arc<Mutex<metal::CommandQueue>>,
|
|
|
|
drawable: Mutex<Option<metal::MetalDrawable>>,
|
|
|
|
n_drawables: usize,
|
|
|
|
drawable_ix: usize,
|
|
|
|
}
|
2021-05-27 12:08:56 +10:00
|
|
|
|
2021-05-27 15:07:00 +10:00
|
|
|
#[derive(Clone)]
|
2021-05-27 12:08:56 +10:00
|
|
|
pub struct Buffer {
|
|
|
|
buffer: metal::Buffer,
|
|
|
|
pub(crate) size: u64,
|
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
2021-05-29 12:59:04 +10:00
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct Image {
|
|
|
|
texture: metal::Texture,
|
|
|
|
width: u32,
|
|
|
|
height: u32,
|
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
2021-05-28 23:38:02 +10:00
|
|
|
// This is the way gfx-hal does it, but a more Vulkan-like strategy would be
|
|
|
|
// to have a semaphore that gets signaled from the command buffer's completion
|
|
|
|
// handler.
|
|
|
|
pub enum Fence {
|
|
|
|
Idle,
|
|
|
|
CmdBufPending(metal::CommandBuffer),
|
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
|
|
|
pub struct Semaphore;
|
|
|
|
|
2021-05-27 15:07:00 +10:00
|
|
|
pub struct CmdBuf {
|
|
|
|
cmd_buf: metal::CommandBuffer,
|
2021-11-21 16:52:29 +11:00
|
|
|
helpers: Arc<Helpers>,
|
2022-05-10 12:39:59 +10:00
|
|
|
cur_encoder: Encoder,
|
|
|
|
time_calibration: Arc<Mutex<TimeCalibration>>,
|
|
|
|
counter_style: CounterStyle,
|
2021-05-27 15:07:00 +10:00
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
enum Encoder {
|
|
|
|
None,
|
|
|
|
Compute(metal::ComputeCommandEncoder, Option<(id, u32)>),
|
|
|
|
Blit(metal::BlitCommandEncoder),
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct QueryPool {
|
|
|
|
counter_sample_buf: Option<CounterSampleBuffer>,
|
|
|
|
calibration: Arc<Mutex<Option<Arc<Mutex<TimeCalibration>>>>>,
|
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
2021-05-27 15:07:00 +10:00
|
|
|
pub struct Pipeline(metal::ComputePipelineState);
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct DescriptorSetBuilder(DescriptorSet);
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct DescriptorSet {
|
|
|
|
buffers: Vec<Buffer>,
|
2021-05-29 12:59:04 +10:00
|
|
|
images: Vec<Image>,
|
2021-05-27 15:07:00 +10:00
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
2021-11-21 16:52:29 +11:00
|
|
|
struct Helpers {
|
|
|
|
clear_pipeline: metal::ComputePipelineState,
|
|
|
|
}
|
|
|
|
|
2021-05-27 12:08:56 +10:00
|
|
|
impl MtlInstance {
|
|
|
|
pub fn new(
|
2021-05-29 12:59:04 +10:00
|
|
|
window_handle: Option<&dyn HasRawWindowHandle>,
|
2021-05-27 12:08:56 +10:00
|
|
|
) -> Result<(MtlInstance, Option<MtlSurface>), Error> {
|
2021-05-29 12:59:04 +10:00
|
|
|
let mut surface = None;
|
|
|
|
if let Some(window_handle) = window_handle {
|
|
|
|
let window_handle = window_handle.raw_window_handle();
|
|
|
|
if let RawWindowHandle::MacOS(w) = window_handle {
|
|
|
|
unsafe {
|
|
|
|
surface = Self::make_surface(w.ns_view as id, w.ns_window as id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok((MtlInstance, surface))
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn make_surface(ns_view: id, ns_window: id) -> Option<MtlSurface> {
|
|
|
|
let ca_ml_class = class!(CAMetalLayer);
|
|
|
|
let is_ca_ml: BOOL = msg_send![ns_view, isKindOfClass: ca_ml_class];
|
|
|
|
if is_ca_ml == YES {
|
|
|
|
todo!("create surface from layer")
|
|
|
|
}
|
|
|
|
let layer: id = msg_send![ns_view, layer];
|
|
|
|
let use_current = !layer.is_null() && {
|
|
|
|
let result: BOOL = msg_send![layer, isKindOfClass: ca_ml_class];
|
|
|
|
result == YES
|
|
|
|
};
|
|
|
|
let metal_layer = if use_current {
|
|
|
|
mem::transmute::<_, &metal::MetalLayerRef>(layer).to_owned()
|
|
|
|
} else {
|
|
|
|
let metal_layer: metal::MetalLayer = msg_send![ca_ml_class, new];
|
|
|
|
let () = msg_send![ns_view, setLayer: metal_layer.as_ref()];
|
|
|
|
let () = msg_send![ns_view, setWantsLayer: YES];
|
|
|
|
let bounds: CGRect = msg_send![ns_view, bounds];
|
|
|
|
let () = msg_send![metal_layer, setFrame: bounds];
|
|
|
|
|
|
|
|
if !ns_window.is_null() {
|
|
|
|
let scale_factor: CGFloat = msg_send![ns_window, backingScaleFactor];
|
|
|
|
let () = msg_send![metal_layer, setContentsScale: scale_factor];
|
|
|
|
}
|
|
|
|
// gfx-hal sets a delegate here
|
|
|
|
metal_layer
|
|
|
|
};
|
|
|
|
let () = msg_send![metal_layer, setContentsGravity: kCAGravityTopLeft];
|
|
|
|
Some(MtlSurface { layer: metal_layer })
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO might do some enumeration of devices
|
|
|
|
|
2021-05-27 15:07:00 +10:00
|
|
|
pub fn device(&self, _surface: Option<&MtlSurface>) -> Result<MtlDevice, Error> {
|
2021-05-22 10:39:28 +10:00
|
|
|
if let Some(device) = metal::Device::system_default() {
|
2021-05-27 15:07:00 +10:00
|
|
|
let cmd_queue = device.new_command_queue();
|
2021-12-26 11:55:13 +11:00
|
|
|
Ok(MtlDevice::new_from_raw_mtl(device, cmd_queue))
|
2021-05-22 10:39:28 +10:00
|
|
|
} else {
|
|
|
|
Err("can't create system default Metal device".into())
|
|
|
|
}
|
|
|
|
}
|
2021-05-27 12:08:56 +10:00
|
|
|
|
|
|
|
pub unsafe fn swapchain(
|
|
|
|
&self,
|
2021-05-29 12:59:04 +10:00
|
|
|
_width: usize,
|
|
|
|
_height: usize,
|
2021-05-27 12:08:56 +10:00
|
|
|
device: &MtlDevice,
|
|
|
|
surface: &MtlSurface,
|
|
|
|
) -> Result<MtlSwapchain, Error> {
|
2021-05-29 12:59:04 +10:00
|
|
|
surface.layer.set_device(&device.device);
|
|
|
|
let n_drawables = surface.layer.maximum_drawable_count() as usize;
|
|
|
|
Ok(MtlSwapchain {
|
|
|
|
layer: surface.layer.to_owned(),
|
|
|
|
cmd_queue: device.cmd_queue.clone(),
|
|
|
|
drawable: Default::default(),
|
|
|
|
n_drawables,
|
|
|
|
drawable_ix: 0,
|
|
|
|
})
|
2021-05-27 12:08:56 +10:00
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-12-26 11:55:13 +11:00
|
|
|
impl MtlDevice {
|
|
|
|
pub fn new_from_raw_mtl(device: metal::Device, cmd_queue: metal::CommandQueue) -> MtlDevice {
|
|
|
|
let is_mac = device.supports_feature_set(MTLFeatureSet::macOS_GPUFamily1_v1);
|
|
|
|
let is_ios = device.supports_feature_set(MTLFeatureSet::iOS_GPUFamily1_v1);
|
|
|
|
let version = NSOperatingSystemVersion::get();
|
|
|
|
|
|
|
|
let use_staging_buffers =
|
|
|
|
if (is_mac && version.at_least(10, 15)) || (is_ios && version.at_least(13, 0)) {
|
|
|
|
!device.has_unified_memory()
|
|
|
|
} else {
|
|
|
|
!device.is_low_power()
|
|
|
|
};
|
|
|
|
// TODO: these are conservative; we need to derive these from
|
|
|
|
// supports_feature_set queries.
|
|
|
|
let gpu_info = GpuInfo {
|
|
|
|
has_descriptor_indexing: false,
|
|
|
|
has_subgroups: false,
|
|
|
|
subgroup_size: None,
|
|
|
|
// The workgroup limits are taken from the minimum of a desktop installation;
|
|
|
|
// we don't support iOS right now, but in case of testing on those devices it might
|
|
|
|
// need to change these (or just queried properly).
|
|
|
|
workgroup_limits: WorkgroupLimits {
|
|
|
|
max_size: [1024, 1024, 64],
|
|
|
|
max_invocations: 1024,
|
|
|
|
},
|
|
|
|
has_memory_model: false,
|
|
|
|
use_staging_buffers,
|
|
|
|
};
|
|
|
|
let helpers = Arc::new(Helpers {
|
|
|
|
clear_pipeline: clear::make_clear_pipeline(&device),
|
|
|
|
});
|
2022-05-10 12:39:59 +10:00
|
|
|
// Timer stuff
|
|
|
|
let timer_set = CounterSet::get_timer_counter_set(&device);
|
|
|
|
let counter_style = if timer_set.is_some() {
|
|
|
|
if device.supports_counter_sampling(metal::MTLCounterSamplingPoint::AtStageBoundary) {
|
|
|
|
CounterStyle::Stage
|
|
|
|
} else if device
|
|
|
|
.supports_counter_sampling(metal::MTLCounterSamplingPoint::AtDispatchBoundary)
|
|
|
|
{
|
|
|
|
CounterStyle::Command
|
|
|
|
} else {
|
|
|
|
CounterStyle::None
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
CounterStyle::None
|
|
|
|
};
|
2021-12-26 11:55:13 +11:00
|
|
|
MtlDevice {
|
|
|
|
device,
|
|
|
|
cmd_queue: Arc::new(Mutex::new(cmd_queue)),
|
|
|
|
gpu_info,
|
|
|
|
helpers,
|
2022-05-10 12:39:59 +10:00
|
|
|
timer_set,
|
|
|
|
counter_style,
|
2021-12-26 11:55:13 +11:00
|
|
|
}
|
|
|
|
}
|
2022-01-19 13:41:28 +11:00
|
|
|
|
|
|
|
pub fn cmd_buf_from_raw_mtl(&self, raw_cmd_buf: metal::CommandBuffer) -> CmdBuf {
|
|
|
|
let cmd_buf = raw_cmd_buf;
|
|
|
|
let helpers = self.helpers.clone();
|
2022-05-10 12:39:59 +10:00
|
|
|
let cur_encoder = Encoder::None;
|
|
|
|
let time_calibration = Default::default();
|
|
|
|
CmdBuf {
|
|
|
|
cmd_buf,
|
|
|
|
helpers,
|
|
|
|
cur_encoder,
|
|
|
|
time_calibration,
|
|
|
|
counter_style: self.counter_style,
|
|
|
|
}
|
2022-01-19 13:41:28 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn image_from_raw_mtl(&self, texture: metal::Texture, width: u32, height: u32) -> Image {
|
|
|
|
Image {
|
|
|
|
texture,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
}
|
|
|
|
}
|
2021-12-26 11:55:13 +11:00
|
|
|
}
|
|
|
|
|
2021-05-29 08:17:36 +10:00
|
|
|
impl crate::backend::Device for MtlDevice {
|
2021-05-22 10:39:28 +10:00
|
|
|
type Buffer = Buffer;
|
|
|
|
|
|
|
|
type Image = Image;
|
|
|
|
|
|
|
|
type Pipeline = Pipeline;
|
|
|
|
|
|
|
|
type DescriptorSet = DescriptorSet;
|
|
|
|
|
|
|
|
type QueryPool = QueryPool;
|
|
|
|
|
|
|
|
type CmdBuf = CmdBuf;
|
|
|
|
|
|
|
|
type Fence = Fence;
|
|
|
|
|
|
|
|
type Semaphore = Semaphore;
|
|
|
|
|
|
|
|
type DescriptorSetBuilder = DescriptorSetBuilder;
|
|
|
|
|
|
|
|
type Sampler = ();
|
|
|
|
|
2021-05-27 12:08:56 +10:00
|
|
|
type ShaderSource = str;
|
|
|
|
|
2021-05-22 10:39:28 +10:00
|
|
|
fn query_gpu_info(&self) -> crate::GpuInfo {
|
2021-05-27 15:07:00 +10:00
|
|
|
self.gpu_info.clone()
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-05-27 12:08:56 +10:00
|
|
|
fn create_buffer(&self, size: u64, usage: BufferUsage) -> Result<Self::Buffer, Error> {
|
|
|
|
let options = if usage.contains(BufferUsage::MAP_READ) {
|
2021-05-27 15:07:00 +10:00
|
|
|
metal::MTLResourceOptions::StorageModeShared
|
|
|
|
| metal::MTLResourceOptions::CPUCacheModeDefaultCache
|
2021-05-27 12:08:56 +10:00
|
|
|
} else if usage.contains(BufferUsage::MAP_WRITE) {
|
2021-05-27 15:07:00 +10:00
|
|
|
metal::MTLResourceOptions::StorageModeShared
|
|
|
|
| metal::MTLResourceOptions::CPUCacheModeWriteCombined
|
2021-05-22 10:39:28 +10:00
|
|
|
} else {
|
|
|
|
metal::MTLResourceOptions::StorageModePrivate
|
|
|
|
};
|
|
|
|
let buffer = self.device.new_buffer(size, options);
|
2021-05-27 12:08:56 +10:00
|
|
|
Ok(Buffer { buffer, size })
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-05-29 12:59:04 +10:00
|
|
|
unsafe fn destroy_buffer(&self, _buffer: &Self::Buffer) -> Result<(), Error> {
|
2021-05-27 15:07:00 +10:00
|
|
|
// This defers dropping until the buffer object is dropped. We probably need
|
|
|
|
// to rethink buffer lifetime if descriptor sets can retain references.
|
|
|
|
Ok(())
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2022-01-20 06:58:01 +11:00
|
|
|
unsafe fn create_image2d(
|
|
|
|
&self,
|
|
|
|
width: u32,
|
|
|
|
height: u32,
|
|
|
|
format: ImageFormat,
|
|
|
|
) -> Result<Self::Image, Error> {
|
2021-05-29 12:59:04 +10:00
|
|
|
let desc = metal::TextureDescriptor::new();
|
|
|
|
desc.set_width(width as u64);
|
|
|
|
desc.set_height(height as u64);
|
|
|
|
// These are defaults so don't need to be explicitly set.
|
|
|
|
//desc.set_depth(1);
|
|
|
|
//desc.set_mipmap_level_count(1);
|
2022-01-20 06:58:01 +11:00
|
|
|
let mtl_format = match format {
|
|
|
|
ImageFormat::A8 => metal::MTLPixelFormat::R8Unorm,
|
2022-05-10 18:09:02 +10:00
|
|
|
ImageFormat::Rgba8 => metal::MTLPixelFormat::BGRA8Unorm,
|
2022-01-20 06:58:01 +11:00
|
|
|
};
|
|
|
|
desc.set_pixel_format(mtl_format);
|
2021-05-29 12:59:04 +10:00
|
|
|
desc.set_usage(metal::MTLTextureUsage::ShaderRead | metal::MTLTextureUsage::ShaderWrite);
|
|
|
|
let texture = self.device.new_texture(&desc);
|
|
|
|
Ok(Image {
|
|
|
|
texture,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn destroy_image(&self, _image: &Self::Image) -> Result<(), Error> {
|
2021-12-04 15:25:42 +11:00
|
|
|
// TODO figure out what we want to do here
|
|
|
|
Ok(())
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-11-11 07:29:40 +11:00
|
|
|
unsafe fn create_compute_pipeline(
|
|
|
|
&self,
|
|
|
|
code: &Self::ShaderSource,
|
|
|
|
_bind_types: &[crate::BindType],
|
|
|
|
) -> Result<Self::Pipeline, Error> {
|
|
|
|
let options = metal::CompileOptions::new();
|
|
|
|
let library = self.device.new_library_with_source(code, &options)?;
|
|
|
|
let function = library.get_function("main0", None)?;
|
|
|
|
let pipeline = self
|
|
|
|
.device
|
|
|
|
.new_compute_pipeline_state_with_function(&function)?;
|
|
|
|
Ok(Pipeline(pipeline))
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn descriptor_set_builder(&self) -> Self::DescriptorSetBuilder {
|
2021-05-27 15:07:00 +10:00
|
|
|
DescriptorSetBuilder::default()
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2022-06-24 05:05:09 +10:00
|
|
|
unsafe fn update_buffer_descriptor(
|
|
|
|
&self,
|
|
|
|
ds: &mut Self::DescriptorSet,
|
|
|
|
index: u32,
|
|
|
|
buf: &Self::Buffer,
|
|
|
|
) {
|
|
|
|
ds.buffers[index as usize] = buf.clone();
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn update_image_descriptor(
|
|
|
|
&self,
|
|
|
|
ds: &mut Self::DescriptorSet,
|
|
|
|
index: u32,
|
|
|
|
image: &Self::Image,
|
|
|
|
) {
|
|
|
|
ds.images[index as usize - ds.buffers.len()] = image.clone();
|
|
|
|
}
|
|
|
|
|
2021-05-22 10:39:28 +10:00
|
|
|
fn create_cmd_buf(&self) -> Result<Self::CmdBuf, Error> {
|
2021-05-29 12:59:04 +10:00
|
|
|
let cmd_queue = self.cmd_queue.lock().unwrap();
|
2022-05-10 12:39:59 +10:00
|
|
|
// A discussion about autorelease pools.
|
|
|
|
//
|
|
|
|
// Autorelease pools are a sore point in Rust/Objective-C interop. Basically,
|
|
|
|
// you can have any two of correctness, ergonomics, and performance. Here we've
|
|
|
|
// chosen the first two, using the pattern of a fine grained autorelease pool
|
|
|
|
// to give the Obj-C object Rust-like lifetime semantics whenever objects are
|
|
|
|
// created as autorelease (by convention, this is any object creation with an
|
|
|
|
// Obj-C method name that doesn't begin with "new" or "alloc").
|
|
|
|
//
|
|
|
|
// To gain back some of the performance, we'd need a way to wrap an autorelease
|
|
|
|
// pool over a chunk of work - that could be one frame of rendering, but for
|
|
|
|
// tests that iterate a number of command buffer submissions, it would need to
|
|
|
|
// be around that. On non-mac platforms, it would be a no-op.
|
|
|
|
//
|
|
|
|
// In any case, this way, the caller doesn't need to worry, and the performance
|
|
|
|
// hit might not be so bad (perhaps we should measure).
|
|
|
|
|
2021-05-27 15:07:00 +10:00
|
|
|
// consider new_command_buffer_with_unretained_references for performance
|
2022-05-10 12:39:59 +10:00
|
|
|
let cmd_buf = autoreleasepool(|| cmd_queue.new_command_buffer().to_owned());
|
2021-11-21 16:52:29 +11:00
|
|
|
let helpers = self.helpers.clone();
|
2022-05-10 12:39:59 +10:00
|
|
|
let cur_encoder = Encoder::None;
|
|
|
|
let time_calibration = Default::default();
|
|
|
|
Ok(CmdBuf {
|
|
|
|
cmd_buf,
|
|
|
|
helpers,
|
|
|
|
cur_encoder,
|
|
|
|
time_calibration,
|
|
|
|
counter_style: self.counter_style,
|
|
|
|
})
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-09-07 03:17:16 +10:00
|
|
|
unsafe fn destroy_cmd_buf(&self, _cmd_buf: Self::CmdBuf) -> Result<(), Error> {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-05-22 10:39:28 +10:00
|
|
|
fn create_query_pool(&self, n_queries: u32) -> Result<Self::QueryPool, Error> {
|
2022-05-10 12:39:59 +10:00
|
|
|
if let Some(timer_set) = &self.timer_set {
|
|
|
|
let pool = CounterSampleBuffer::new(&self.device, n_queries as u64, timer_set)
|
|
|
|
.ok_or("error creating timer query pool")?;
|
|
|
|
return Ok(QueryPool {
|
|
|
|
counter_sample_buf: Some(pool),
|
|
|
|
calibration: Default::default(),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
Ok(QueryPool::default())
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn fetch_query_pool(&self, pool: &Self::QueryPool) -> Result<Vec<f64>, Error> {
|
2022-05-10 12:39:59 +10:00
|
|
|
if let Some(raw) = &pool.counter_sample_buf {
|
|
|
|
let resolved = raw.resolve();
|
|
|
|
let calibration = pool.calibration.lock().unwrap();
|
|
|
|
if let Some(calibration) = &*calibration {
|
|
|
|
let calibration = calibration.lock().unwrap();
|
|
|
|
let result = resolved
|
|
|
|
.iter()
|
|
|
|
.map(|time_ns| calibration.correlate(*time_ns))
|
|
|
|
.collect();
|
|
|
|
return Ok(result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Maybe should return None indicating it wasn't successful? But that might break.
|
2021-05-27 15:07:00 +10:00
|
|
|
Ok(Vec::new())
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-05-27 12:08:56 +10:00
|
|
|
unsafe fn run_cmd_bufs(
|
2021-05-22 10:39:28 +10:00
|
|
|
&self,
|
2021-05-27 12:08:56 +10:00
|
|
|
cmd_bufs: &[&Self::CmdBuf],
|
2021-05-28 23:38:02 +10:00
|
|
|
_wait_semaphores: &[&Self::Semaphore],
|
|
|
|
_signal_semaphores: &[&Self::Semaphore],
|
2021-05-28 09:02:12 +10:00
|
|
|
fence: Option<&mut Self::Fence>,
|
2021-05-22 10:39:28 +10:00
|
|
|
) -> Result<(), Error> {
|
2022-05-10 12:39:59 +10:00
|
|
|
unsafe fn add_scheduled_handler(
|
|
|
|
cmd_buf: &metal::CommandBufferRef,
|
|
|
|
block: &Block<(&CommandBufferRef,), ()>,
|
|
|
|
) {
|
|
|
|
msg_send![cmd_buf, addScheduledHandler: block]
|
|
|
|
}
|
2021-05-28 23:38:02 +10:00
|
|
|
for cmd_buf in cmd_bufs {
|
2022-05-10 12:39:59 +10:00
|
|
|
let time_calibration = cmd_buf.time_calibration.clone();
|
|
|
|
let start_block = block::ConcreteBlock::new(move |buffer: &metal::CommandBufferRef| {
|
|
|
|
let device: id = msg_send![buffer, device];
|
|
|
|
let mut time_calibration = time_calibration.lock().unwrap();
|
|
|
|
let cpu_ts_ptr = &mut time_calibration.cpu_start_ts as *mut _;
|
|
|
|
let gpu_ts_ptr = &mut time_calibration.gpu_start_ts as *mut _;
|
|
|
|
// TODO: only do this if supported.
|
|
|
|
let () = msg_send![device, sampleTimestamps: cpu_ts_ptr gpuTimestamp: gpu_ts_ptr];
|
|
|
|
})
|
|
|
|
.copy();
|
|
|
|
add_scheduled_handler(&cmd_buf.cmd_buf, &start_block);
|
|
|
|
let time_calibration = cmd_buf.time_calibration.clone();
|
|
|
|
let completed_block =
|
|
|
|
block::ConcreteBlock::new(move |buffer: &metal::CommandBufferRef| {
|
|
|
|
let device: id = msg_send![buffer, device];
|
|
|
|
let mut time_calibration = time_calibration.lock().unwrap();
|
|
|
|
let cpu_ts_ptr = &mut time_calibration.cpu_end_ts as *mut _;
|
|
|
|
let gpu_ts_ptr = &mut time_calibration.gpu_end_ts as *mut _;
|
|
|
|
// TODO: only do this if supported.
|
|
|
|
let () =
|
|
|
|
msg_send![device, sampleTimestamps: cpu_ts_ptr gpuTimestamp: gpu_ts_ptr];
|
|
|
|
})
|
|
|
|
.copy();
|
|
|
|
cmd_buf.cmd_buf.add_completed_handler(&completed_block);
|
2021-05-28 23:38:02 +10:00
|
|
|
cmd_buf.cmd_buf.commit();
|
|
|
|
}
|
|
|
|
if let Some(last_cmd_buf) = cmd_bufs.last() {
|
|
|
|
if let Some(fence) = fence {
|
|
|
|
*fence = Fence::CmdBufPending(last_cmd_buf.cmd_buf.to_owned());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-11-26 08:12:25 +11:00
|
|
|
unsafe fn map_buffer(
|
|
|
|
&self,
|
|
|
|
buffer: &Self::Buffer,
|
|
|
|
offset: u64,
|
2021-11-26 16:34:06 +11:00
|
|
|
_size: u64,
|
|
|
|
_mode: MapMode,
|
2021-11-26 08:12:25 +11:00
|
|
|
) -> Result<*mut u8, Error> {
|
|
|
|
let contents_ptr = buffer.buffer.contents();
|
|
|
|
if contents_ptr.is_null() {
|
|
|
|
return Err("probably trying to map private buffer".into());
|
|
|
|
}
|
|
|
|
Ok((contents_ptr as *mut u8).add(offset as usize))
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn unmap_buffer(
|
|
|
|
&self,
|
2021-11-26 16:34:06 +11:00
|
|
|
_buffer: &Self::Buffer,
|
2021-11-26 08:12:25 +11:00
|
|
|
_offset: u64,
|
|
|
|
_size: u64,
|
|
|
|
_mode: MapMode,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-05-22 10:39:28 +10:00
|
|
|
unsafe fn create_semaphore(&self) -> Result<Self::Semaphore, Error> {
|
2021-05-29 12:59:04 +10:00
|
|
|
Ok(Semaphore)
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-05-28 23:38:02 +10:00
|
|
|
unsafe fn create_fence(&self, _signaled: bool) -> Result<Self::Fence, Error> {
|
|
|
|
// Doesn't handle signaled case. Maybe the fences should have more
|
|
|
|
// limited functionality than, say, Vulkan.
|
|
|
|
Ok(Fence::Idle)
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-09-07 03:17:16 +10:00
|
|
|
unsafe fn destroy_fence(&self, _fence: Self::Fence) -> Result<(), Error> {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-05-28 09:11:30 +10:00
|
|
|
unsafe fn wait_and_reset(&self, fences: Vec<&mut Self::Fence>) -> Result<(), Error> {
|
2021-05-28 23:38:02 +10:00
|
|
|
for fence in fences {
|
|
|
|
match fence {
|
|
|
|
Fence::Idle => (),
|
|
|
|
Fence::CmdBufPending(cmd_buf) => {
|
|
|
|
cmd_buf.wait_until_completed();
|
|
|
|
// TODO: this would be a good place to check errors, currently
|
|
|
|
// dropped on the floor.
|
|
|
|
*fence = Fence::Idle;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-05-29 08:17:36 +10:00
|
|
|
unsafe fn get_fence_status(&self, fence: &mut Self::Fence) -> Result<bool, Error> {
|
|
|
|
match fence {
|
|
|
|
Fence::Idle => Ok(true),
|
|
|
|
Fence::CmdBufPending(cmd_buf) => {
|
|
|
|
Ok(cmd_buf.status() == metal::MTLCommandBufferStatus::Completed)
|
|
|
|
}
|
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn create_sampler(&self, params: crate::SamplerParams) -> Result<Self::Sampler, Error> {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-29 08:17:36 +10:00
|
|
|
impl crate::backend::CmdBuf<MtlDevice> for CmdBuf {
|
|
|
|
unsafe fn begin(&mut self) {}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
unsafe fn finish(&mut self) {
|
|
|
|
self.flush_encoder();
|
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
2021-10-22 12:07:46 +11:00
|
|
|
unsafe fn reset(&mut self) -> bool {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
unsafe fn begin_compute_pass(&mut self, desc: &ComputePassDescriptor) {
|
|
|
|
// TODO: we might want to get better about validation but the following
|
|
|
|
// assert is likely to trigger, and also a case can be made that
|
|
|
|
// validation should be done at the hub level, for consistency.
|
|
|
|
//debug_assert!(matches!(self.cur_encoder, Encoder::None));
|
|
|
|
self.flush_encoder();
|
|
|
|
autoreleasepool(|| {
|
|
|
|
let (encoder, end_query) = match (&desc.timer_queries, self.counter_style) {
|
|
|
|
(Some(queries), CounterStyle::Stage) => {
|
|
|
|
let descriptor: id =
|
|
|
|
msg_send![class!(MTLComputePassDescriptor), computePassDescriptor];
|
|
|
|
let attachments: id = msg_send![descriptor, sampleBufferAttachments];
|
|
|
|
let index: NSUInteger = 0;
|
|
|
|
let attachment: id = msg_send![attachments, objectAtIndexedSubscript: index];
|
|
|
|
// Here we break the hub/mux separation a bit, for expedience
|
|
|
|
#[allow(irrefutable_let_patterns)]
|
|
|
|
if let crate::hub::QueryPool::Mtl(query_pool) = queries.0 {
|
|
|
|
if let Some(sample_buf) = &query_pool.counter_sample_buf {
|
|
|
|
let () = msg_send![attachment, setSampleBuffer: sample_buf.id()];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let start_index = queries.1 as NSUInteger;
|
|
|
|
let end_index = queries.2 as NSInteger;
|
|
|
|
let () = msg_send![attachment, setStartOfEncoderSampleIndex: start_index];
|
|
|
|
let () = msg_send![attachment, setEndOfEncoderSampleIndex: end_index];
|
|
|
|
(
|
|
|
|
msg_send![
|
|
|
|
self.cmd_buf,
|
|
|
|
computeCommandEncoderWithDescriptor: descriptor
|
|
|
|
],
|
|
|
|
None,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
(Some(queries), CounterStyle::Command) => {
|
|
|
|
let encoder = self.cmd_buf.new_compute_command_encoder();
|
|
|
|
#[allow(irrefutable_let_patterns)]
|
|
|
|
let end_query = if let crate::hub::QueryPool::Mtl(query_pool) = queries.0 {
|
|
|
|
if let Some(sample_buf) = &query_pool.counter_sample_buf {
|
|
|
|
let sample_index = queries.1 as NSUInteger;
|
|
|
|
let sample_buf = sample_buf.id();
|
|
|
|
let () = msg_send![encoder, sampleCountersInBuffer: sample_buf atSampleIndex: sample_index withBarrier: true];
|
|
|
|
Some((sample_buf, queries.2))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
|
|
|
(encoder, end_query)
|
|
|
|
}
|
|
|
|
_ => (self.cmd_buf.new_compute_command_encoder(), None),
|
|
|
|
};
|
|
|
|
self.cur_encoder = Encoder::Compute(encoder.to_owned(), end_query);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-05-22 10:39:28 +10:00
|
|
|
unsafe fn dispatch(
|
|
|
|
&mut self,
|
|
|
|
pipeline: &Pipeline,
|
|
|
|
descriptor_set: &DescriptorSet,
|
2021-05-29 08:17:36 +10:00
|
|
|
workgroup_count: (u32, u32, u32),
|
|
|
|
workgroup_size: (u32, u32, u32),
|
2021-05-22 10:39:28 +10:00
|
|
|
) {
|
2022-05-10 12:39:59 +10:00
|
|
|
let encoder = self.compute_command_encoder();
|
2021-05-27 15:07:00 +10:00
|
|
|
encoder.set_compute_pipeline_state(&pipeline.0);
|
2021-05-29 12:59:04 +10:00
|
|
|
let mut buf_ix = 0;
|
2021-05-27 15:07:00 +10:00
|
|
|
for buffer in &descriptor_set.buffers {
|
2021-05-29 12:59:04 +10:00
|
|
|
encoder.set_buffer(buf_ix, Some(&buffer.buffer), 0);
|
|
|
|
buf_ix += 1;
|
|
|
|
}
|
2021-12-04 15:25:42 +11:00
|
|
|
let mut img_ix = buf_ix;
|
2021-05-29 12:59:04 +10:00
|
|
|
for image in &descriptor_set.images {
|
|
|
|
encoder.set_texture(img_ix, Some(&image.texture));
|
|
|
|
img_ix += 1;
|
2021-05-27 15:07:00 +10:00
|
|
|
}
|
2021-05-29 08:17:36 +10:00
|
|
|
let workgroup_count = metal::MTLSize {
|
|
|
|
width: workgroup_count.0 as u64,
|
|
|
|
height: workgroup_count.1 as u64,
|
|
|
|
depth: workgroup_count.2 as u64,
|
2021-05-27 15:07:00 +10:00
|
|
|
};
|
2021-05-29 08:17:36 +10:00
|
|
|
let workgroup_size = metal::MTLSize {
|
|
|
|
width: workgroup_size.0 as u64,
|
|
|
|
height: workgroup_size.1 as u64,
|
|
|
|
depth: workgroup_size.2 as u64,
|
2021-05-27 15:07:00 +10:00
|
|
|
};
|
2021-05-29 08:17:36 +10:00
|
|
|
encoder.dispatch_thread_groups(workgroup_count, workgroup_size);
|
2022-05-10 12:39:59 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn end_compute_pass(&mut self) {
|
|
|
|
// TODO: might validate that we are in a compute encoder state
|
|
|
|
self.flush_encoder();
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn memory_barrier(&mut self) {
|
2021-05-28 23:38:02 +10:00
|
|
|
// We'll probably move to explicit barriers, but for now rely on
|
|
|
|
// Metal's own tracking.
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-05-29 08:17:36 +10:00
|
|
|
unsafe fn host_barrier(&mut self) {}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
|
|
|
unsafe fn image_barrier(
|
|
|
|
&mut self,
|
2021-05-29 12:59:04 +10:00
|
|
|
_image: &Image,
|
|
|
|
_src_layout: crate::ImageLayout,
|
|
|
|
_dst_layout: crate::ImageLayout,
|
2021-05-22 10:39:28 +10:00
|
|
|
) {
|
2021-05-29 12:59:04 +10:00
|
|
|
// I think these are being tracked.
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
unsafe fn clear_buffer(&mut self, buffer: &Buffer, size: Option<u64>) {
|
2021-11-21 16:52:29 +11:00
|
|
|
let size = size.unwrap_or(buffer.size);
|
2022-05-10 12:39:59 +10:00
|
|
|
let _ = self.compute_command_encoder();
|
|
|
|
// Getting this directly is a workaround for a borrow checker issue.
|
|
|
|
if let Encoder::Compute(e, _) = &self.cur_encoder {
|
|
|
|
clear::encode_clear(e, &self.helpers.clear_pipeline, &buffer.buffer, size);
|
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
unsafe fn copy_buffer(&mut self, src: &Buffer, dst: &Buffer) {
|
|
|
|
let encoder = self.blit_command_encoder();
|
2021-05-29 12:59:04 +10:00
|
|
|
let size = src.size.min(dst.size);
|
|
|
|
encoder.copy_from_buffer(&src.buffer, 0, &dst.buffer, 0, size);
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
unsafe fn copy_image_to_buffer(&mut self, src: &Image, dst: &Buffer) {
|
|
|
|
let encoder = self.blit_command_encoder();
|
2021-05-29 12:59:04 +10:00
|
|
|
assert_eq!(dst.size, (src.width as u64) * (src.height as u64) * 4);
|
|
|
|
let bytes_per_row = (src.width * 4) as NSUInteger;
|
|
|
|
let src_size = metal::MTLSize {
|
|
|
|
width: src.width as NSUInteger,
|
|
|
|
height: src.height as NSUInteger,
|
|
|
|
depth: 1,
|
|
|
|
};
|
|
|
|
let origin = metal::MTLOrigin { x: 0, y: 0, z: 0 };
|
|
|
|
encoder.copy_from_texture_to_buffer(
|
|
|
|
&src.texture,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
origin,
|
|
|
|
src_size,
|
|
|
|
&dst.buffer,
|
|
|
|
0,
|
|
|
|
bytes_per_row,
|
|
|
|
bytes_per_row * src.height as NSUInteger,
|
|
|
|
metal::MTLBlitOption::empty(),
|
|
|
|
);
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
unsafe fn copy_buffer_to_image(&mut self, src: &Buffer, dst: &Image) {
|
|
|
|
let encoder = self.blit_command_encoder();
|
2021-05-29 12:59:04 +10:00
|
|
|
assert_eq!(src.size, (dst.width as u64) * (dst.height as u64) * 4);
|
|
|
|
let bytes_per_row = (dst.width * 4) as NSUInteger;
|
|
|
|
let src_size = metal::MTLSize {
|
|
|
|
width: dst.width as NSUInteger,
|
|
|
|
height: dst.height as NSUInteger,
|
|
|
|
depth: 1,
|
|
|
|
};
|
|
|
|
let origin = metal::MTLOrigin { x: 0, y: 0, z: 0 };
|
|
|
|
encoder.copy_from_buffer_to_texture(
|
|
|
|
&src.buffer,
|
|
|
|
0,
|
|
|
|
bytes_per_row,
|
|
|
|
bytes_per_row * dst.height as NSUInteger,
|
|
|
|
src_size,
|
|
|
|
&dst.texture,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
origin,
|
|
|
|
metal::MTLBlitOption::empty(),
|
|
|
|
);
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
unsafe fn blit_image(&mut self, src: &Image, dst: &Image) {
|
|
|
|
let encoder = self.blit_command_encoder();
|
2021-05-29 12:59:04 +10:00
|
|
|
let src_size = metal::MTLSize {
|
|
|
|
width: src.width.min(dst.width) as NSUInteger,
|
|
|
|
height: src.width.min(dst.height) as NSUInteger,
|
|
|
|
depth: 1,
|
|
|
|
};
|
|
|
|
let origin = metal::MTLOrigin { x: 0, y: 0, z: 0 };
|
|
|
|
encoder.copy_from_texture(
|
|
|
|
&src.texture,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
origin,
|
|
|
|
src_size,
|
|
|
|
&dst.texture,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
origin,
|
|
|
|
);
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2022-05-10 12:39:59 +10:00
|
|
|
unsafe fn reset_query_pool(&mut self, pool: &QueryPool) {
|
|
|
|
let mut calibration = pool.calibration.lock().unwrap();
|
|
|
|
*calibration = Some(self.time_calibration.clone());
|
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
|
|
|
|
unsafe fn write_timestamp(&mut self, pool: &QueryPool, query: u32) {
|
2022-05-10 12:39:59 +10:00
|
|
|
if let Some(buf) = &pool.counter_sample_buf {
|
|
|
|
if matches!(self.cur_encoder, Encoder::None) {
|
|
|
|
self.cur_encoder =
|
|
|
|
Encoder::Compute(self.cmd_buf.new_compute_command_encoder().to_owned(), None);
|
|
|
|
}
|
|
|
|
let sample_index = query as NSUInteger;
|
|
|
|
if self.counter_style == CounterStyle::Command {
|
|
|
|
match &self.cur_encoder {
|
|
|
|
Encoder::Compute(e, _) => {
|
|
|
|
let () = msg_send![e.as_ptr(), sampleCountersInBuffer: buf.id() atSampleIndex: sample_index withBarrier: true];
|
|
|
|
}
|
|
|
|
Encoder::None => unreachable!(),
|
|
|
|
_ => todo!(),
|
|
|
|
}
|
|
|
|
} else if self.counter_style == CounterStyle::Stage {
|
|
|
|
match &self.cur_encoder {
|
|
|
|
Encoder::Compute(_e, _) => {
|
|
|
|
println!("write_timestamp is not supported for stage-style encoders");
|
|
|
|
}
|
|
|
|
_ => (),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CmdBuf {
|
|
|
|
fn compute_command_encoder(&mut self) -> &metal::ComputeCommandEncoder {
|
|
|
|
if !matches!(self.cur_encoder, Encoder::Compute(..)) {
|
|
|
|
self.flush_encoder();
|
|
|
|
self.cur_encoder =
|
|
|
|
Encoder::Compute(self.cmd_buf.new_compute_command_encoder().to_owned(), None);
|
|
|
|
}
|
|
|
|
if let Encoder::Compute(e, _) = &self.cur_encoder {
|
|
|
|
e
|
|
|
|
} else {
|
|
|
|
unreachable!()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn blit_command_encoder(&mut self) -> &metal::BlitCommandEncoder {
|
|
|
|
if !matches!(self.cur_encoder, Encoder::Blit(_)) {
|
|
|
|
self.flush_encoder();
|
|
|
|
self.cur_encoder = Encoder::Blit(self.cmd_buf.new_blit_command_encoder().to_owned());
|
|
|
|
}
|
|
|
|
if let Encoder::Blit(e) = &self.cur_encoder {
|
|
|
|
e
|
|
|
|
} else {
|
|
|
|
unreachable!()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn flush_encoder(&mut self) {
|
|
|
|
match std::mem::replace(&mut self.cur_encoder, Encoder::None) {
|
|
|
|
Encoder::Compute(e, Some((sample_buf, end_query))) => {
|
|
|
|
let sample_index = end_query as NSUInteger;
|
|
|
|
unsafe {
|
|
|
|
let () = msg_send![e.as_ptr(), sampleCountersInBuffer: sample_buf atSampleIndex: sample_index withBarrier: true];
|
|
|
|
}
|
|
|
|
e.end_encoding();
|
|
|
|
}
|
|
|
|
Encoder::Compute(e, None) => e.end_encoding(),
|
|
|
|
Encoder::Blit(e) => e.end_encoding(),
|
|
|
|
Encoder::None => (),
|
|
|
|
}
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-29 08:17:36 +10:00
|
|
|
impl crate::backend::DescriptorSetBuilder<MtlDevice> for DescriptorSetBuilder {
|
2021-05-22 10:39:28 +10:00
|
|
|
fn add_buffers(&mut self, buffers: &[&Buffer]) {
|
2021-05-27 15:07:00 +10:00
|
|
|
self.0.buffers.extend(buffers.iter().copied().cloned());
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
fn add_images(&mut self, images: &[&Image]) {
|
2021-05-29 12:59:04 +10:00
|
|
|
self.0.images.extend(images.iter().copied().cloned());
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
fn add_textures(&mut self, images: &[&Image]) {
|
2021-05-29 12:59:04 +10:00
|
|
|
self.add_images(images);
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
|
|
|
|
2021-05-29 12:59:04 +10:00
|
|
|
unsafe fn build(
|
|
|
|
self,
|
|
|
|
_device: &MtlDevice,
|
|
|
|
_pipeline: &Pipeline,
|
|
|
|
) -> Result<DescriptorSet, Error> {
|
2021-05-27 15:07:00 +10:00
|
|
|
Ok(self.0)
|
2021-05-22 10:39:28 +10:00
|
|
|
}
|
2021-05-27 12:08:56 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
impl MtlSwapchain {
|
|
|
|
pub unsafe fn next(&mut self) -> Result<(usize, Semaphore), Error> {
|
2021-05-29 12:59:04 +10:00
|
|
|
let drawable_ix = self.drawable_ix;
|
|
|
|
self.drawable_ix = (drawable_ix + 1) % self.n_drawables;
|
|
|
|
Ok((drawable_ix, Semaphore))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn image(&self, _idx: usize) -> Image {
|
|
|
|
let (drawable, texture) = autoreleasepool(|| {
|
|
|
|
let drawable = self.layer.next_drawable().unwrap();
|
|
|
|
(drawable.to_owned(), drawable.texture().to_owned())
|
|
|
|
});
|
|
|
|
*self.drawable.lock().unwrap() = Some(drawable);
|
|
|
|
let size = self.layer.drawable_size();
|
|
|
|
Image {
|
|
|
|
texture,
|
|
|
|
width: size.width.round() as u32,
|
|
|
|
height: size.height.round() as u32,
|
|
|
|
}
|
2021-05-27 12:08:56 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn present(
|
|
|
|
&self,
|
2021-05-29 12:59:04 +10:00
|
|
|
_image_idx: usize,
|
|
|
|
_semaphores: &[&Semaphore],
|
2021-05-27 12:08:56 +10:00
|
|
|
) -> Result<bool, Error> {
|
2021-05-29 12:59:04 +10:00
|
|
|
let drawable = self.drawable.lock().unwrap().take();
|
|
|
|
if let Some(drawable) = drawable {
|
|
|
|
autoreleasepool(|| {
|
|
|
|
let cmd_queue = self.cmd_queue.lock().unwrap();
|
|
|
|
let cmd_buf = cmd_queue.new_command_buffer();
|
|
|
|
cmd_buf.present_drawable(&drawable);
|
|
|
|
cmd_buf.commit();
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
println!("no drawable; present called without acquiring image?");
|
|
|
|
}
|
|
|
|
Ok(false)
|
2021-05-27 15:07:00 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)]
|
|
|
|
struct NSOperatingSystemVersion {
|
|
|
|
major: NSInteger,
|
|
|
|
minor: NSInteger,
|
|
|
|
patch: NSInteger,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl NSOperatingSystemVersion {
|
|
|
|
fn get() -> NSOperatingSystemVersion {
|
|
|
|
unsafe {
|
|
|
|
let process_info: *mut Object = msg_send![class!(NSProcessInfo), processInfo];
|
|
|
|
msg_send![process_info, operatingSystemVersion]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn at_least(&self, major: u32, minor: u32) -> bool {
|
|
|
|
let major = major as NSInteger;
|
|
|
|
let minor = minor as NSInteger;
|
|
|
|
self.major > major || (self.major == major && self.minor >= minor)
|
|
|
|
}
|
|
|
|
}
|