mirror of
https://github.com/GayPizzaSpecifications/voxelotl-engine.git
synced 2025-08-03 05:10:57 +00:00
preliminary metal renderer
This commit is contained in:
parent
ca8d4f3a2b
commit
d896f2eaa7
@ -1,37 +1,65 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
import SDL3
|
import SDL3
|
||||||
|
import QuartzCore.CAMetalLayer
|
||||||
|
|
||||||
public class Application {
|
public class Application {
|
||||||
private let cfg: ApplicationConfiguration
|
private let cfg: ApplicationConfiguration
|
||||||
|
|
||||||
private var window: OpaquePointer? = nil
|
private var window: OpaquePointer? = nil
|
||||||
|
private var view: SDL_MetalView? = nil
|
||||||
|
private var renderer: Renderer? = nil
|
||||||
private var lastCounter: UInt64 = 0
|
private var lastCounter: UInt64 = 0
|
||||||
|
|
||||||
|
private var stderr = FileHandle.standardError
|
||||||
|
|
||||||
public init(configuration: ApplicationConfiguration) {
|
public init(configuration: ApplicationConfiguration) {
|
||||||
self.cfg = configuration
|
self.cfg = configuration
|
||||||
}
|
}
|
||||||
|
|
||||||
private func initialize() -> ApplicationExecutionState {
|
private func initialize() -> ApplicationExecutionState {
|
||||||
guard SDL_Init(SDL_INIT_VIDEO) >= 0 else {
|
guard SDL_Init(SDL_INIT_VIDEO) >= 0 else {
|
||||||
print("SDL_Init() error: \(String(cString: SDL_GetError()))")
|
print("SDL_Init() error: \(String(cString: SDL_GetError()))", to: &stderr)
|
||||||
return .exitFailure
|
return .exitFailure
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create SDL window
|
||||||
var windowFlags = SDL_WindowFlags(SDL_WINDOW_HIGH_PIXEL_DENSITY)
|
var windowFlags = SDL_WindowFlags(SDL_WINDOW_HIGH_PIXEL_DENSITY)
|
||||||
if (cfg.flags.contains(.resizable)) {
|
if (cfg.flags.contains(.resizable)) {
|
||||||
windowFlags |= SDL_WindowFlags(SDL_WINDOW_RESIZABLE)
|
windowFlags |= SDL_WindowFlags(SDL_WINDOW_RESIZABLE)
|
||||||
}
|
}
|
||||||
window = SDL_CreateWindow(cfg.title, cfg.width, cfg.height, windowFlags)
|
window = SDL_CreateWindow(cfg.title, cfg.width, cfg.height, windowFlags)
|
||||||
guard window != nil else {
|
guard window != nil else {
|
||||||
print("SDL_CreateWindow() error: \(String(cString: SDL_GetError()))")
|
print("SDL_CreateWindow() error: \(String(cString: SDL_GetError()))", to: &stderr)
|
||||||
return .exitFailure
|
return .exitFailure
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create Metal renderer
|
||||||
|
view = SDL_Metal_CreateView(window)
|
||||||
|
do {
|
||||||
|
let layer = unsafeBitCast(SDL_Metal_GetLayer(view), to: CAMetalLayer.self)
|
||||||
|
self.renderer = try Renderer(layer: layer)
|
||||||
|
} catch RendererError.initFailure(let message) {
|
||||||
|
print("Renderer init error: \(message)", to: &stderr)
|
||||||
|
return .exitFailure
|
||||||
|
} catch {
|
||||||
|
print("Renderer init error: unexpected error", to: &stderr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get window metrics
|
||||||
|
var backBufferWidth: Int32 = 0, backBufferHeight: Int32 = 0
|
||||||
|
guard SDL_GetWindowSizeInPixels(window, &backBufferWidth, &backBufferHeight) >= 0 else {
|
||||||
|
print("SDL_GetWindowSizeInPixels() error: \(String(cString: SDL_GetError()))", to: &stderr)
|
||||||
|
return .exitFailure
|
||||||
|
}
|
||||||
|
renderer!.resize(size: SIMD2<Int>(Int(backBufferWidth), Int(backBufferHeight)))
|
||||||
|
|
||||||
lastCounter = SDL_GetPerformanceCounter()
|
lastCounter = SDL_GetPerformanceCounter()
|
||||||
return .running
|
return .running
|
||||||
}
|
}
|
||||||
|
|
||||||
private func deinitialize() {
|
private func deinitialize() {
|
||||||
|
renderer = nil
|
||||||
|
SDL_Metal_DestroyView(view)
|
||||||
SDL_DestroyWindow(window)
|
SDL_DestroyWindow(window)
|
||||||
SDL_Quit()
|
SDL_Quit()
|
||||||
}
|
}
|
||||||
@ -50,12 +78,24 @@ public class Application {
|
|||||||
}
|
}
|
||||||
return .running
|
return .running
|
||||||
|
|
||||||
|
case SDL_EVENT_WINDOW_PIXEL_SIZE_CHANGED:
|
||||||
|
let backBufferSize = SIMD2(Int(event.window.data1), Int(event.window.data2))
|
||||||
|
renderer!.resize(size: backBufferSize)
|
||||||
|
return .running
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return .running
|
return .running
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private func update(_ deltaTime: Double) -> ApplicationExecutionState {
|
private func update(_ deltaTime: Double) -> ApplicationExecutionState {
|
||||||
|
do {
|
||||||
|
try renderer!.paint()
|
||||||
|
} catch RendererError.drawFailure(let message) {
|
||||||
|
print("Renderer draw error: \(message)", to: &stderr)
|
||||||
|
} catch {
|
||||||
|
print("Renderer draw error: unexpected error", to: &stderr)
|
||||||
|
}
|
||||||
return .running
|
return .running
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -122,3 +162,9 @@ fileprivate enum ApplicationExecutionState {
|
|||||||
case exitSuccess
|
case exitSuccess
|
||||||
case running
|
case running
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extension FileHandle: TextOutputStream {
|
||||||
|
public func write(_ string: String) {
|
||||||
|
self.write(Data(string.utf8))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
add_executable(Voxelotl MACOSX_BUNDLE
|
add_executable(Voxelotl MACOSX_BUNDLE
|
||||||
Assets.xcassets
|
Assets.xcassets
|
||||||
|
Renderer.swift
|
||||||
Application.swift
|
Application.swift
|
||||||
main.swift)
|
main.swift)
|
||||||
|
|
||||||
|
190
Sources/Voxelotl/Renderer.swift
Normal file
190
Sources/Voxelotl/Renderer.swift
Normal file
@ -0,0 +1,190 @@
|
|||||||
|
import Foundation
|
||||||
|
import Metal
|
||||||
|
import QuartzCore.CAMetalLayer
|
||||||
|
import simd
|
||||||
|
|
||||||
|
// Temp:
|
||||||
|
@objc fileprivate enum ShaderInputIdx: NSInteger {
|
||||||
|
case ShaderInputIdxVertices = 0
|
||||||
|
}
|
||||||
|
fileprivate struct ShaderVertex {
|
||||||
|
let position: SIMD4<Float>
|
||||||
|
let color: SIMD4<Float>
|
||||||
|
}
|
||||||
|
|
||||||
|
class Renderer {
|
||||||
|
fileprivate static let shaderSource = """
|
||||||
|
#ifndef SHADERTYPES_H
|
||||||
|
#define SHADERTYPES_H
|
||||||
|
|
||||||
|
#ifdef __METAL_VERSION__
|
||||||
|
# define NS_ENUM(TYPE, NAME) enum NAME : TYPE NAME; enum NAME : TYPE
|
||||||
|
# define NSInteger metal::int32_t
|
||||||
|
#else
|
||||||
|
# import <Foundation/Foundation.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <simd/simd.h>
|
||||||
|
|
||||||
|
typedef NS_ENUM(NSInteger, ShaderInputIdx) {
|
||||||
|
ShaderInputIdxVertices = 0
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
vector_float4 position;
|
||||||
|
vector_float4 color;
|
||||||
|
} ShaderVertex;
|
||||||
|
|
||||||
|
#endif//SHADERTYPES_H
|
||||||
|
|
||||||
|
#include <metal_stdlib>
|
||||||
|
|
||||||
|
using namespace metal;
|
||||||
|
|
||||||
|
struct FragmentInput {
|
||||||
|
float4 position [[position]];
|
||||||
|
float4 color;
|
||||||
|
};
|
||||||
|
|
||||||
|
vertex FragmentInput vertexMain(
|
||||||
|
uint vertexID [[vertex_id]],
|
||||||
|
device const ShaderVertex* vtx [[buffer(ShaderInputIdxVertices)]]
|
||||||
|
){
|
||||||
|
FragmentInput out;
|
||||||
|
out.position = vtx[vertexID].position;
|
||||||
|
out.color = vtx[vertexID].color;
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
fragment float4 fragmentMain(FragmentInput in [[stage_in]]) {
|
||||||
|
return in.color;
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
|
||||||
|
fileprivate static let vertices = [
|
||||||
|
ShaderVertex(position: SIMD4<Float>(-0.5, -0.5, 0.0, 1.0), color: SIMD4<Float>(1.0, 0.0, 0.0, 1.0)),
|
||||||
|
ShaderVertex(position: SIMD4<Float>( 0.0, 0.5, 0.0, 1.0), color: SIMD4<Float>(0.0, 1.0, 0.0, 1.0)),
|
||||||
|
ShaderVertex(position: SIMD4<Float>( 0.5, -0.5, 0.0, 1.0), color: SIMD4<Float>(0.0, 0.0, 1.0, 1.0))
|
||||||
|
]
|
||||||
|
|
||||||
|
private var device: MTLDevice
|
||||||
|
private var layer: CAMetalLayer
|
||||||
|
private var viewport = MTLViewport()
|
||||||
|
private var queue: MTLCommandQueue
|
||||||
|
private var lib: MTLLibrary
|
||||||
|
private let passDescription = MTLRenderPassDescriptor()
|
||||||
|
private var pso: MTLRenderPipelineState
|
||||||
|
|
||||||
|
private var vtxBuffer: MTLBuffer! = nil
|
||||||
|
|
||||||
|
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 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
init(layer metalLayer: CAMetalLayer) throws {
|
||||||
|
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
|
||||||
|
passDescription.colorAttachments[0].loadAction = MTLLoadAction.clear
|
||||||
|
passDescription.colorAttachments[0].storeAction = MTLStoreAction.store
|
||||||
|
passDescription.colorAttachments[0].clearColor = MTLClearColorMake(0.1, 0.1, 0.1, 1.0)
|
||||||
|
|
||||||
|
// Create shader library & grab functions
|
||||||
|
do {
|
||||||
|
//self.lib = try device.makeDefaultLibrary(bundle: Bundle.main)
|
||||||
|
let options = MTLCompileOptions()
|
||||||
|
options.fastMathEnabled = true
|
||||||
|
self.lib = try device.makeLibrary(source: Self.shaderSource, options: options)
|
||||||
|
}
|
||||||
|
catch {
|
||||||
|
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
|
||||||
|
do {
|
||||||
|
self.pso = try device.makeRenderPipelineState(descriptor: pipeDescription)
|
||||||
|
}
|
||||||
|
catch {
|
||||||
|
throw RendererError.initFailure("Failed to create pipeline state: \(error.localizedDescription)")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create vertex buffers
|
||||||
|
guard let vtxBuffer = device.makeBuffer(
|
||||||
|
bytes: Self.vertices,
|
||||||
|
length: Self.vertices.count * MemoryLayout<ShaderVertex>.stride,
|
||||||
|
options: .storageModeManaged)
|
||||||
|
else {
|
||||||
|
throw RendererError.initFailure("Failed to create vertex buffer")
|
||||||
|
}
|
||||||
|
self.vtxBuffer = vtxBuffer
|
||||||
|
}
|
||||||
|
|
||||||
|
deinit {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func resize(size: SIMD2<Int>) {
|
||||||
|
self.viewport = MTLViewport(
|
||||||
|
originX: 0.0,
|
||||||
|
originY: 0.0,
|
||||||
|
width: Double(size.x),
|
||||||
|
height: Double(size.y),
|
||||||
|
znear: 1.0,
|
||||||
|
zfar: -1.0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func paint() throws {
|
||||||
|
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")
|
||||||
|
}
|
||||||
|
|
||||||
|
encoder.setViewport(viewport)
|
||||||
|
encoder.setCullMode(MTLCullMode.none)
|
||||||
|
encoder.setRenderPipelineState(pso)
|
||||||
|
|
||||||
|
encoder.setVertexBuffer(vtxBuffer, offset: 0, index: ShaderInputIdx.ShaderInputIdxVertices.rawValue)
|
||||||
|
encoder.drawPrimitives(type: .triangle, vertexStart: 0, vertexCount: 3)
|
||||||
|
|
||||||
|
encoder.endEncoding()
|
||||||
|
commandBuf.present(rt)
|
||||||
|
commandBuf.commit()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum RendererError: Error {
|
||||||
|
case initFailure(_ message: String)
|
||||||
|
case drawFailure(_ message: String)
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user