Created
November 26, 2024 04:15
-
-
Save Matt54/03a2665088939ec2c4dd979ae1f9b438 to your computer and use it in GitHub Desktop.
RealityView with LowLevelTexture and Gaussian Blur (separate compute shader passes)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// Don't forget to include this bridging header | |
#ifndef GaussianBlurParams_h | |
#define GaussianBlurParams_h | |
struct GaussianBlurParams { | |
int32_t kernelSize; | |
float intensity; | |
}; | |
#endif /* GaussianBlurParams_h */ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import SwiftUI | |
import RealityKit | |
struct GaussianBlurTextureView: View { | |
@State var isBlurred: Bool = false | |
@State var texture: LowLevelTexture? | |
let commandQueue: MTLCommandQueue | |
let texturePipeline: MTLComputePipelineState | |
let gaussianBlurPipeline: MTLComputePipelineState | |
var intermediateTexture: MTLTexture! | |
init() { | |
let device = MTLCreateSystemDefaultDevice()! | |
self.commandQueue = device.makeCommandQueue()! | |
let library = device.makeDefaultLibrary()! | |
let updateFunction = library.makeFunction(name: "diagonalLineTexture")! | |
self.texturePipeline = try! device.makeComputePipelineState(function: updateFunction) | |
let antiAliasFunction = library.makeFunction(name: "gaussianBlurTextureProcessing")! | |
gaussianBlurPipeline = try! device.makeComputePipelineState(function: antiAliasFunction) | |
// Create intermediate texture | |
let intermediateDesc = MTLTextureDescriptor() | |
intermediateDesc.width = textureDescriptor.width | |
intermediateDesc.height = textureDescriptor.height | |
intermediateDesc.pixelFormat = .rgba16Float | |
intermediateDesc.usage = [.shaderRead, .shaderWrite] | |
intermediateTexture = device.makeTexture(descriptor: intermediateDesc)! | |
} | |
var body: some View { | |
RealityView { content in | |
let entity = try! getEntity() | |
content.add(entity) | |
updateTexture() | |
} | |
.gesture( | |
SpatialTapGesture() | |
.targetedToAnyEntity() | |
.onEnded { value in | |
isBlurred.toggle() | |
updateTexture() | |
} | |
) | |
} | |
func getEntity() throws -> Entity { | |
let width: Float = 1.0 | |
let height: Float = 1.0 | |
let mesh = MeshResource.generatePlane(width: width, height: height) | |
let texture = try LowLevelTexture(descriptor: textureDescriptor) | |
let resource = try TextureResource(from: texture) | |
var material = UnlitMaterial() | |
material.color.texture = .init(resource) | |
let modelComponent = ModelComponent(mesh: mesh, materials: [material]) | |
let entity = ModelEntity() | |
entity.components.set([ | |
InputTargetComponent(), | |
CollisionComponent(shapes: [.generateBox(width: width, height: height, depth: 1)]), | |
]) | |
entity.components.set(modelComponent) | |
entity.scale *= 5.0 | |
self.texture = texture | |
return entity | |
} | |
func updateTexture() { | |
guard let texture else { return } | |
guard let commandBuffer = commandQueue.makeCommandBuffer(), | |
let computeEncoder = commandBuffer.makeComputeCommandEncoder() | |
else { return } | |
commandBuffer.enqueue() | |
let targetTexture = texture.replace(using: commandBuffer) | |
computeEncoder.setComputePipelineState(texturePipeline) | |
computeEncoder.setTexture(isBlurred ? intermediateTexture : targetTexture, index: 0) | |
let w = texturePipeline.threadExecutionWidth | |
let h = texturePipeline.maxTotalThreadsPerThreadgroup / w | |
let threadGroupSize = MTLSizeMake(w, h, 1) | |
let threadGroupCount = MTLSizeMake( | |
(textureDescriptor.width + threadGroupSize.width - 1) / threadGroupSize.width, | |
(textureDescriptor.height + threadGroupSize.height - 1) / threadGroupSize.height, | |
1) | |
computeEncoder.dispatchThreadgroups(threadGroupCount, threadsPerThreadgroup: threadGroupSize) | |
if isBlurred { | |
applyGaussianBlur(computeEncoder: computeEncoder, | |
inputTexture: intermediateTexture, | |
outputTexture: targetTexture, | |
pipelineState: gaussianBlurPipeline | |
) | |
} | |
computeEncoder.endEncoding() | |
commandBuffer.commit() | |
} | |
func applyGaussianBlur(computeEncoder: MTLComputeCommandEncoder, | |
inputTexture: MTLTexture, | |
outputTexture: MTLTexture, | |
pipelineState: MTLComputePipelineState) { | |
computeEncoder.setComputePipelineState(pipelineState) | |
computeEncoder.setTexture(inputTexture, index: 0) | |
computeEncoder.setTexture(outputTexture, index: 1) | |
var params = GaussianBlurParams(kernelSize: Int32(3), | |
intensity: 1.0) | |
computeEncoder.setBytes(¶ms, | |
length: MemoryLayout<GaussianBlurParams>.size, | |
index: 0) | |
let w = pipelineState.threadExecutionWidth | |
let h = pipelineState.maxTotalThreadsPerThreadgroup / w | |
let threadGroupSize = MTLSizeMake(w, h, 1) | |
let threadGroupCount = MTLSizeMake( | |
(outputTexture.width + threadGroupSize.width - 1) / threadGroupSize.width, | |
(outputTexture.height + threadGroupSize.height - 1) / threadGroupSize.height, | |
1) | |
computeEncoder.dispatchThreadgroups(threadGroupCount, | |
threadsPerThreadgroup: threadGroupSize) | |
} | |
var textureDescriptor: LowLevelTexture.Descriptor { | |
var desc = LowLevelTexture.Descriptor() | |
desc.textureType = .type2D | |
desc.arrayLength = 1 | |
desc.width = 2048 | |
desc.height = 2048 | |
desc.depth = 1 | |
desc.mipmapLevelCount = 1 | |
desc.pixelFormat = .bgra8Unorm | |
desc.textureUsage = [.shaderRead, .shaderWrite] | |
desc.swizzle = .init(red: .red, green: .green, blue: .blue, alpha: .alpha) | |
return desc | |
} | |
} | |
#Preview { | |
GaussianBlurTextureView() | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment