2024-08-05 05:44:51 +00:00
|
|
|
import Foundation
|
|
|
|
import Metal
|
|
|
|
import QuartzCore.CAMetalLayer
|
|
|
|
import simd
|
2024-08-05 07:19:49 +00:00
|
|
|
import ShaderTypes
|
2024-08-05 05:44:51 +00:00
|
|
|
|
2024-08-05 10:09:33 +00:00
|
|
|
fileprivate let cubeVertices: [ShaderVertex] = [
|
|
|
|
.init(position: .init(-1, -1, 1, 1), normal: .init( 0, 0, 1, 0), texCoord: .init(0, 0)),
|
|
|
|
.init(position: .init( 1, -1, 1, 1), normal: .init( 0, 0, 1, 0), texCoord: .init(1, 0)),
|
|
|
|
.init(position: .init(-1, 1, 1, 1), normal: .init( 0, 0, 1, 0), texCoord: .init(0, 1)),
|
|
|
|
.init(position: .init( 1, 1, 1, 1), normal: .init( 0, 0, 1, 0), texCoord: .init(1, 1)),
|
|
|
|
.init(position: .init( 1, -1, 1, 1), normal: .init( 1, 0, 0, 0), texCoord: .init(0, 0)),
|
|
|
|
.init(position: .init( 1, -1, -1, 1), normal: .init( 1, 0, 0, 0), texCoord: .init(1, 0)),
|
|
|
|
.init(position: .init( 1, 1, 1, 1), normal: .init( 1, 0, 0, 0), texCoord: .init(0, 1)),
|
|
|
|
.init(position: .init( 1, 1, -1, 1), normal: .init( 1, 0, 0, 0), texCoord: .init(1, 1)),
|
|
|
|
.init(position: .init( 1, -1, -1, 1), normal: .init( 0, 0, -1, 0), texCoord: .init(0, 0)),
|
|
|
|
.init(position: .init(-1, -1, -1, 1), normal: .init( 0, 0, -1, 0), texCoord: .init(1, 0)),
|
|
|
|
.init(position: .init( 1, 1, -1, 1), normal: .init( 0, 0, -1, 0), texCoord: .init(0, 1)),
|
|
|
|
.init(position: .init(-1, 1, -1, 1), normal: .init( 0, 0, -1, 0), texCoord: .init(1, 1)),
|
|
|
|
.init(position: .init(-1, -1, -1, 1), normal: .init(-1, 0, 0, 0), texCoord: .init(0, 0)),
|
|
|
|
.init(position: .init(-1, -1, 1, 1), normal: .init(-1, 0, 0, 0), texCoord: .init(1, 0)),
|
|
|
|
.init(position: .init(-1, 1, -1, 1), normal: .init(-1, 0, 0, 0), texCoord: .init(0, 1)),
|
|
|
|
.init(position: .init(-1, 1, 1, 1), normal: .init(-1, 0, 0, 0), texCoord: .init(1, 1)),
|
|
|
|
.init(position: .init(-1, -1, -1, 1), normal: .init( 0, -1, 0, 0), texCoord: .init(0, 0)),
|
|
|
|
.init(position: .init( 1, -1, -1, 1), normal: .init( 0, -1, 0, 0), texCoord: .init(1, 0)),
|
|
|
|
.init(position: .init(-1, -1, 1, 1), normal: .init( 0, -1, 0, 0), texCoord: .init(0, 1)),
|
|
|
|
.init(position: .init( 1, -1, 1, 1), normal: .init( 0, -1, 0, 0), texCoord: .init(1, 1)),
|
|
|
|
.init(position: .init(-1, 1, 1, 1), normal: .init( 0, 1, 0, 0), texCoord: .init(0, 0)),
|
|
|
|
.init(position: .init( 1, 1, 1, 1), normal: .init( 0, 1, 0, 0), texCoord: .init(1, 0)),
|
|
|
|
.init(position: .init(-1, 1, -1, 1), normal: .init( 0, 1, 0, 0), texCoord: .init(0, 1)),
|
|
|
|
.init(position: .init( 1, 1, -1, 1), normal: .init( 0, 1, 0, 0), texCoord: .init(1, 1)),
|
|
|
|
]
|
|
|
|
|
|
|
|
fileprivate let cubeIndices: [UInt16] = [
|
|
|
|
0, 1, 2, 2, 1, 3,
|
|
|
|
4, 5, 6, 6, 5, 7,
|
|
|
|
8, 9, 10, 10, 9, 11,
|
|
|
|
12, 13, 14, 14, 13, 15,
|
|
|
|
16, 17, 18, 18, 17, 19,
|
|
|
|
20, 21, 22, 22, 21, 23
|
|
|
|
]
|
2024-08-05 05:44:51 +00:00
|
|
|
|
2024-08-05 10:09:33 +00:00
|
|
|
class Renderer {
|
2024-08-07 06:36:23 +00:00
|
|
|
private let depthFormat: MTLPixelFormat = .depth16Unorm
|
|
|
|
|
2024-08-05 05:44:51 +00:00
|
|
|
private var device: MTLDevice
|
|
|
|
private var layer: CAMetalLayer
|
2024-08-09 10:53:30 +00:00
|
|
|
private var viewport: MTLViewport
|
2024-08-07 06:36:23 +00:00
|
|
|
private var aspectRatio: Float
|
2024-08-05 05:44:51 +00:00
|
|
|
private var queue: MTLCommandQueue
|
|
|
|
private var lib: MTLLibrary
|
|
|
|
private let passDescription = MTLRenderPassDescriptor()
|
|
|
|
private var pso: MTLRenderPipelineState
|
2024-08-07 06:36:23 +00:00
|
|
|
private var depthStencilState: MTLDepthStencilState
|
|
|
|
private var depthStencilTexture: MTLTexture
|
2024-08-05 05:44:51 +00:00
|
|
|
|
2024-08-05 10:09:33 +00:00
|
|
|
private var vtxBuffer: MTLBuffer, idxBuffer: MTLBuffer
|
|
|
|
private var defaultTexture: MTLTexture
|
|
|
|
private var cubeTexture: MTLTexture? = nil
|
2024-08-05 05:44:51 +00:00
|
|
|
|
|
|
|
fileprivate static func createMetalDevice() -> MTLDevice? {
|
|
|
|
MTLCopyAllDevices().reduce(nil, { best, dev in
|
|
|
|
if best == nil { dev }
|
|
|
|
else if !best!.isLowPower || dev.isLowPower { best }
|
|
|
|
else if best!.supportsRaytracing || !dev.supportsRaytracing { best }
|
|
|
|
else { dev }
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-08-07 06:36:23 +00:00
|
|
|
init(layer metalLayer: CAMetalLayer, size: SIMD2<Int>) throws {
|
2024-08-05 05:44:51 +00:00
|
|
|
self.layer = metalLayer
|
|
|
|
|
|
|
|
// Select best Metal device
|
|
|
|
guard let device = Self.createMetalDevice() else {
|
|
|
|
throw RendererError.initFailure("Failed to create Metal device")
|
|
|
|
}
|
|
|
|
self.device = device
|
|
|
|
|
|
|
|
layer.device = device
|
|
|
|
layer.pixelFormat = MTLPixelFormat.bgra8Unorm
|
|
|
|
|
|
|
|
// Setup command queue
|
|
|
|
guard let queue = device.makeCommandQueue() else {
|
|
|
|
throw RendererError.initFailure("Failed to create command queue")
|
|
|
|
}
|
|
|
|
self.queue = queue
|
2024-08-07 06:36:23 +00:00
|
|
|
|
2024-08-09 10:53:30 +00:00
|
|
|
self.viewport = Self.makeViewport(size: size)
|
2024-08-07 06:36:23 +00:00
|
|
|
self.aspectRatio = Float(size.x) / Float(size.y)
|
|
|
|
|
|
|
|
passDescription.colorAttachments[0].loadAction = .clear
|
|
|
|
passDescription.colorAttachments[0].storeAction = .store
|
2024-08-05 05:44:51 +00:00
|
|
|
passDescription.colorAttachments[0].clearColor = MTLClearColorMake(0.1, 0.1, 0.1, 1.0)
|
2024-08-07 06:36:23 +00:00
|
|
|
passDescription.depthAttachment.loadAction = .clear
|
|
|
|
passDescription.depthAttachment.storeAction = .dontCare
|
|
|
|
passDescription.depthAttachment.clearDepth = 1.0
|
|
|
|
|
|
|
|
guard let depthStencilTexture = Self.createDepthTexture(device, size, format: depthFormat) else {
|
|
|
|
throw RendererError.initFailure("Failed to create depth buffer")
|
|
|
|
}
|
|
|
|
self.depthStencilTexture = depthStencilTexture
|
|
|
|
passDescription.depthAttachment.texture = self.depthStencilTexture
|
|
|
|
|
|
|
|
let stencilDepthDescription = MTLDepthStencilDescriptor()
|
|
|
|
stencilDepthDescription.depthCompareFunction = .less // OpenGL default
|
|
|
|
stencilDepthDescription.isDepthWriteEnabled = true
|
|
|
|
guard let depthStencilState = device.makeDepthStencilState(descriptor: stencilDepthDescription) else {
|
|
|
|
throw RendererError.initFailure("Failed to create depth stencil state")
|
|
|
|
}
|
|
|
|
self.depthStencilState = depthStencilState
|
2024-08-05 05:44:51 +00:00
|
|
|
|
|
|
|
// Create shader library & grab functions
|
|
|
|
do {
|
2024-08-05 07:08:16 +00:00
|
|
|
self.lib = try device.makeDefaultLibrary(bundle: Bundle.main)
|
|
|
|
} catch {
|
2024-08-05 05:44:51 +00:00
|
|
|
throw RendererError.initFailure("Metal shader compilation failed:\n\(error.localizedDescription)")
|
|
|
|
}
|
|
|
|
let vertexProgram = lib.makeFunction(name: "vertexMain")
|
|
|
|
let fragmentProgram = lib.makeFunction(name: "fragmentMain")
|
|
|
|
|
|
|
|
// Set up pipeline state
|
|
|
|
let pipeDescription = MTLRenderPipelineDescriptor()
|
|
|
|
pipeDescription.vertexFunction = vertexProgram
|
|
|
|
pipeDescription.fragmentFunction = fragmentProgram
|
|
|
|
pipeDescription.colorAttachments[0].pixelFormat = layer.pixelFormat
|
2024-08-07 06:36:23 +00:00
|
|
|
pipeDescription.depthAttachmentPixelFormat = depthFormat
|
2024-08-05 05:44:51 +00:00
|
|
|
do {
|
|
|
|
self.pso = try device.makeRenderPipelineState(descriptor: pipeDescription)
|
2024-08-05 07:08:16 +00:00
|
|
|
} catch {
|
2024-08-05 05:44:51 +00:00
|
|
|
throw RendererError.initFailure("Failed to create pipeline state: \(error.localizedDescription)")
|
|
|
|
}
|
|
|
|
|
2024-08-05 10:09:33 +00:00
|
|
|
// Create cube mesh buffers
|
2024-08-05 05:44:51 +00:00
|
|
|
guard let vtxBuffer = device.makeBuffer(
|
2024-08-05 10:09:33 +00:00
|
|
|
bytes: cubeVertices,
|
|
|
|
length: cubeVertices.count * MemoryLayout<ShaderVertex>.stride,
|
2024-08-05 05:44:51 +00:00
|
|
|
options: .storageModeManaged)
|
|
|
|
else {
|
|
|
|
throw RendererError.initFailure("Failed to create vertex buffer")
|
|
|
|
}
|
|
|
|
self.vtxBuffer = vtxBuffer
|
2024-08-05 10:09:33 +00:00
|
|
|
guard let idxBuffer = device.makeBuffer(
|
|
|
|
bytes: cubeIndices,
|
|
|
|
length: cubeIndices.count * MemoryLayout<UInt16>.stride,
|
|
|
|
options: .storageModeManaged)
|
|
|
|
else {
|
|
|
|
throw RendererError.initFailure("Failed to create index buffer")
|
|
|
|
}
|
|
|
|
self.idxBuffer = idxBuffer
|
|
|
|
|
2024-08-05 10:56:06 +00:00
|
|
|
// Create a default texture
|
2024-08-05 10:09:33 +00:00
|
|
|
do {
|
2024-08-06 06:34:51 +00:00
|
|
|
self.defaultTexture = try Self.loadTexture(device, queue, image2D: Image2D(Data([
|
2024-08-05 10:09:33 +00:00
|
|
|
0xFF, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0xFF,
|
|
|
|
0x00, 0x00, 0x00, 0xFF, 0xFF, 0x00, 0xFF, 0xFF
|
|
|
|
]), format: .abgr8888, width: 2, height: 2, stride: 2 * 4))
|
|
|
|
} catch {
|
|
|
|
throw RendererError.initFailure("Failed to create default texture")
|
|
|
|
}
|
|
|
|
|
2024-08-05 10:56:06 +00:00
|
|
|
// Load texture from a file in the bundle
|
2024-08-05 10:09:33 +00:00
|
|
|
do {
|
2024-08-06 06:34:51 +00:00
|
|
|
self.cubeTexture = try Self.loadTexture(device, queue, resourcePath: "test.png")
|
2024-08-05 10:09:33 +00:00
|
|
|
} catch RendererError.loadFailure(let message) {
|
2024-08-05 10:56:06 +00:00
|
|
|
printErr("Failed to load texture image: \(message)")
|
2024-08-05 10:09:33 +00:00
|
|
|
} catch {
|
2024-08-05 10:56:06 +00:00
|
|
|
printErr("Failed to load texture image: unknown error")
|
2024-08-05 10:09:33 +00:00
|
|
|
}
|
2024-08-05 05:44:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
deinit {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2024-08-06 06:34:51 +00:00
|
|
|
static func loadTexture(_ device: MTLDevice, _ queue: MTLCommandQueue, resourcePath path: String) throws -> MTLTexture {
|
2024-08-05 10:09:33 +00:00
|
|
|
do {
|
2024-08-06 06:34:51 +00:00
|
|
|
return try loadTexture(device, queue, url: Bundle.main.getResource(path))
|
2024-08-05 10:09:33 +00:00
|
|
|
} catch ContentError.resourceNotFound(let message) {
|
|
|
|
throw RendererError.loadFailure(message)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-06 06:34:51 +00:00
|
|
|
static func loadTexture(_ device: MTLDevice, _ queue: MTLCommandQueue, url imageUrl: URL) throws -> MTLTexture {
|
2024-08-05 10:09:33 +00:00
|
|
|
do {
|
2024-08-06 06:34:51 +00:00
|
|
|
return try loadTexture(device, queue, image2D: try NSImageLoader.open(url: imageUrl))
|
2024-08-05 10:09:33 +00:00
|
|
|
} catch ImageLoaderError.openFailed(let message) {
|
|
|
|
throw RendererError.loadFailure(message)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-06 06:34:51 +00:00
|
|
|
static func loadTexture(_ device: MTLDevice, _ queue: MTLCommandQueue, image2D image: Image2D) throws -> MTLTexture {
|
2024-08-05 10:09:33 +00:00
|
|
|
let texDesc = MTLTextureDescriptor()
|
|
|
|
texDesc.width = image.width
|
|
|
|
texDesc.height = image.height
|
|
|
|
texDesc.pixelFormat = .rgba8Unorm_srgb
|
|
|
|
texDesc.textureType = .type2D
|
2024-08-06 06:34:51 +00:00
|
|
|
texDesc.storageMode = .private
|
2024-08-05 10:09:33 +00:00
|
|
|
texDesc.usage = .shaderRead
|
|
|
|
guard let newTexture = device.makeTexture(descriptor: texDesc) else {
|
|
|
|
throw RendererError.loadFailure("Failed to create texture descriptor")
|
|
|
|
}
|
2024-08-06 06:34:51 +00:00
|
|
|
|
|
|
|
guard let texData = image.data.withUnsafeBytes({ bytes in
|
|
|
|
device.makeBuffer(bytes: bytes.baseAddress!, length: bytes.count, options: [ .storageModeShared ])
|
|
|
|
}) else {
|
|
|
|
throw RendererError.loadFailure("Failed to create shared texture data buffer")
|
|
|
|
}
|
|
|
|
|
|
|
|
guard let cmdBuffer = queue.makeCommandBuffer(),
|
|
|
|
let blitEncoder = cmdBuffer.makeBlitCommandEncoder()
|
|
|
|
else {
|
|
|
|
throw RendererError.loadFailure("Failed to create blit command encoder")
|
2024-08-05 10:09:33 +00:00
|
|
|
}
|
2024-08-06 06:34:51 +00:00
|
|
|
|
|
|
|
blitEncoder.copy(
|
|
|
|
from: texData,
|
|
|
|
sourceOffset: 0,
|
|
|
|
sourceBytesPerRow: image.stride,
|
|
|
|
sourceBytesPerImage: image.stride * image.height,
|
|
|
|
sourceSize: .init(width: image.width, height: image.height, depth: 1),
|
|
|
|
|
|
|
|
to: newTexture,
|
|
|
|
destinationSlice: 0,
|
|
|
|
destinationLevel: 0,
|
|
|
|
destinationOrigin: .init(x: 0, y: 0, z: 0))
|
|
|
|
blitEncoder.endEncoding()
|
|
|
|
|
|
|
|
cmdBuffer.addCompletedHandler { _ in
|
|
|
|
//FIXME: look into if this needs to be synchronised
|
|
|
|
//printErr("Texture was added?")
|
|
|
|
}
|
|
|
|
cmdBuffer.commit()
|
|
|
|
|
2024-08-05 10:09:33 +00:00
|
|
|
return newTexture
|
|
|
|
}
|
|
|
|
|
2024-08-07 06:36:23 +00:00
|
|
|
private static func createDepthTexture(_ device: MTLDevice, _ size: SIMD2<Int>, format: MTLPixelFormat
|
|
|
|
) -> MTLTexture? {
|
|
|
|
let texDescriptor = MTLTextureDescriptor.texture2DDescriptor(
|
|
|
|
pixelFormat: format,
|
|
|
|
width: size.x,
|
|
|
|
height: size.y,
|
|
|
|
mipmapped: false)
|
|
|
|
texDescriptor.depth = 1
|
|
|
|
texDescriptor.sampleCount = 1
|
|
|
|
texDescriptor.usage = [ .renderTarget, .shaderRead ]
|
|
|
|
#if !NDEBUG
|
|
|
|
texDescriptor.storageMode = .private
|
|
|
|
#else
|
|
|
|
texDescriptor.storageMode = .memoryless
|
|
|
|
#endif
|
|
|
|
|
|
|
|
guard let depthStencilTexture = device.makeTexture(descriptor: texDescriptor) else { return nil }
|
|
|
|
depthStencilTexture.label = "Depth buffer"
|
|
|
|
|
|
|
|
return depthStencilTexture
|
|
|
|
}
|
|
|
|
|
2024-08-09 10:53:30 +00:00
|
|
|
static func makeViewport(size: SIMD2<Int>) -> MTLViewport {
|
|
|
|
MTLViewport(
|
|
|
|
originX: 0.0, originY: 0.0,
|
|
|
|
width: Double(size.x),
|
|
|
|
height: Double(size.y),
|
|
|
|
znear: 0, zfar: 1)
|
|
|
|
}
|
|
|
|
|
2024-08-05 05:44:51 +00:00
|
|
|
func resize(size: SIMD2<Int>) {
|
2024-08-07 06:36:23 +00:00
|
|
|
if Int(self.viewport.width) != size.x || Int(self.viewport.height) != size.y {
|
|
|
|
if let depthStencilTexture = Self.createDepthTexture(device, size, format: depthFormat) {
|
|
|
|
self.depthStencilTexture = depthStencilTexture
|
|
|
|
passDescription.depthAttachment.texture = self.depthStencilTexture
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
self.aspectRatio = Float(size.x) / Float(size.y)
|
2024-08-09 10:53:30 +00:00
|
|
|
self.viewport = Self.makeViewport(size: size)
|
2024-08-05 05:44:51 +00:00
|
|
|
}
|
|
|
|
|
2024-08-08 06:05:30 +00:00
|
|
|
//FIXME: temp
|
|
|
|
var camera = Camera()
|
|
|
|
var time: Float = 0
|
2024-08-06 07:35:38 +00:00
|
|
|
|
2024-08-05 05:44:51 +00:00
|
|
|
func paint() throws {
|
2024-08-08 06:05:30 +00:00
|
|
|
camera.update(deltaTime: 0.025)
|
2024-08-07 06:36:23 +00:00
|
|
|
#if true
|
2024-08-06 07:35:38 +00:00
|
|
|
let projection = matrix_float4x4.perspective(
|
2024-08-08 06:05:30 +00:00
|
|
|
verticalFov: Float(60.0).radians,
|
2024-08-07 06:36:23 +00:00
|
|
|
aspect: aspectRatio,
|
|
|
|
near: 0.003,
|
2024-08-08 06:05:30 +00:00
|
|
|
far: 100)
|
2024-08-07 06:36:23 +00:00
|
|
|
#else
|
|
|
|
let projection = matrix_float4x4.orthographic(
|
|
|
|
left: -aspectRatio, right: aspectRatio,
|
|
|
|
bottom: -1, top: 1,
|
|
|
|
near: 0, far: -4)
|
|
|
|
#endif
|
2024-08-08 06:05:30 +00:00
|
|
|
let view = camera.view
|
2024-08-06 07:35:38 +00:00
|
|
|
let model: matrix_float4x4 =
|
2024-08-08 06:05:30 +00:00
|
|
|
.translate(.init(0, -1, 0)) * .scale(.init(10, 0.1, 10))
|
|
|
|
//.translate(.init(0, sin(time * 0.5) * 0.75, -2)) *
|
|
|
|
//.scale(0.5) *
|
|
|
|
//.rotate(y: time)
|
2024-08-06 07:35:38 +00:00
|
|
|
|
|
|
|
time += 0.025
|
|
|
|
|
|
|
|
var uniforms = ShaderUniforms(model: model, projView: projection * view)
|
2024-08-06 06:51:29 +00:00
|
|
|
|
2024-08-05 05:44:51 +00:00
|
|
|
guard let rt = layer.nextDrawable() else {
|
|
|
|
throw RendererError.drawFailure("Failed to get next drawable render target")
|
|
|
|
}
|
|
|
|
|
|
|
|
passDescription.colorAttachments[0].texture = rt.texture
|
|
|
|
|
|
|
|
guard let commandBuf: MTLCommandBuffer = queue.makeCommandBuffer() else {
|
|
|
|
throw RendererError.drawFailure("Failed to make command buffer from queue")
|
|
|
|
}
|
|
|
|
guard let encoder = commandBuf.makeRenderCommandEncoder(descriptor: passDescription) else {
|
|
|
|
throw RendererError.drawFailure("Failed to make render encoder from command buffer")
|
|
|
|
}
|
|
|
|
|
2024-08-09 10:53:30 +00:00
|
|
|
encoder.setCullMode(.none)
|
2024-08-07 06:36:23 +00:00
|
|
|
encoder.setFrontFacing(.counterClockwise) // OpenGL default
|
2024-08-05 05:44:51 +00:00
|
|
|
encoder.setViewport(viewport)
|
|
|
|
encoder.setRenderPipelineState(pso)
|
2024-08-07 06:36:23 +00:00
|
|
|
encoder.setDepthStencilState(depthStencilState)
|
2024-08-05 05:44:51 +00:00
|
|
|
|
2024-08-05 10:09:33 +00:00
|
|
|
encoder.setFragmentTexture(cubeTexture ?? defaultTexture, index: 0)
|
|
|
|
encoder.setVertexBuffer(vtxBuffer,
|
|
|
|
offset: 0,
|
|
|
|
index: ShaderInputIdx.vertices.rawValue)
|
2024-08-06 06:51:29 +00:00
|
|
|
// Ideal as long as our uniforms total 4 KB or less
|
|
|
|
encoder.setVertexBytes(&uniforms,
|
|
|
|
length: MemoryLayout<ShaderUniforms>.stride,
|
|
|
|
index: ShaderInputIdx.uniforms.rawValue)
|
2024-08-05 10:09:33 +00:00
|
|
|
encoder.drawIndexedPrimitives(
|
|
|
|
type: .triangle,
|
|
|
|
indexCount: cubeIndices.count,
|
|
|
|
indexType: .uint16,
|
|
|
|
indexBuffer: idxBuffer,
|
|
|
|
indexBufferOffset: 0)
|
2024-08-05 05:44:51 +00:00
|
|
|
|
|
|
|
encoder.endEncoding()
|
|
|
|
commandBuf.present(rt)
|
|
|
|
commandBuf.commit()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum RendererError: Error {
|
|
|
|
case initFailure(_ message: String)
|
2024-08-05 10:09:33 +00:00
|
|
|
case loadFailure(_ message: String)
|
2024-08-05 05:44:51 +00:00
|
|
|
case drawFailure(_ message: String)
|
|
|
|
}
|