Apple Metal 2 5.你好,計算
原文https://developer.apple.com/documentation/metal/fundamental_lessons/hello_compute
你好,平行計算
概覽
In the Basic Texturing sample, you learned how to render a 2D image by applying a texture to a single quad.
In this sample, you’ll learn how to execute compute-processing workloads in Metal for image processing. In particular, you’ll learn how to work with the compute processing pipeline and write kernel functions.
GPGPU 原理
Graphics processing units (GPUs) were originally designed to process large amounts of graphics data, such as vertices or fragments, in a very fast and efficient manner. This design is evident in the GPU hardware architecture itself, which has many processing cores that execute workloads in parallel.
Throughout the history of GPU design, the parallel-processing architecture has remained fairly consistent, but the processing cores have become increasingly programmable. This change enabled GPUs to move away from a fixed-function pipeline toward a programmable pipeline, a change that also enabled general-purpose GPU (GPGPU) programming.
In the GPGPU model, the GPU can be used for any kind of processing task and isn’t limited to graphics data. For example, GPUs can be used for cryptography, machine learning, physics, or finance. In Metal, GPGPU workloads are known as compute-processing workloads, or compute.
Graphics and compute workloads are not mutually exclusive; Metal provides a unified framework and language that enables seamless integration of graphics and compute workloads. In fact, this sample demonstrates this integration by:
Using a compute pipeline that converts a color image to a grayscale image
Using a graphics pipeline that renders the grayscale image to a quad surface
建立計算處理管線
The compute processing pipeline is made up of only one stage, a programmable kernel function, that executes a compute pass. The kernel function reads from and writes to resources directly, without passing resource data through various pipeline stages.
A MTLComputePipelineState object represents a compute processing pipeline. Unlike a graphics rendering pipeline, you can create a MTLComputePipelineState object with a single kernel function, without using a pipeline descriptor.
// Create a compute kernel function
id <MTLFunction> kernelFunction = [defaultLibrary newFunctionWithName:@"grayscaleKernel"];
// Create a compute kernel
_computePipelineState = [_device newComputePipelineStateWithFunction:kernelFunction
error:&error];
編寫核函式
This sample loads image data into a texture and then uses a kernel function to convert the texture’s pixels from color to grayscale. The kernel function processes the pixels independently and concurrently.
注意:
An equivalent algorithm can be written for and executed by the CPU. However, a GPU solution is faster because the texture’s pixels don’t need to be processed sequentially.
The kernel function in this sample is called grayscaleKernel and its signature is shown below:
kernel void
grayscaleKernel(texture2d<half, access::read> inTexture [[texture(AAPLTextureIndexInput)]],
texture2d<half, access::write> outTexture [[texture(AAPLTextureIndexOutput)]],
uint2 gid [[thread_position_in_grid]])
The function takes the following resource parameters:
inTexture: A read-only, 2D texture that contains the input color pixels.
outTexture: A write-only, 2D texture that stores the output grayscale pixels.
Textures that specify a read access qualifier can be read from using the read() function. Textures that specify a write access qualifier can be written to using the write() function.
A kernel function executes once per thread, which is analogous to how a vertex function executes once per vertex. Threads are organized into a 3D grid; an encoded compute pass specifies how many threads to process by declaring the size of the grid. Because this sample processes a 2D texture, the threads are arranged in a 2D grid where each thread corresponds to a unique texel.
The kernel function’s gid parameter uses the [[thread_position_in_grid]] attribute qualifier, which locates a thread within the compute grid. Each execution of the kernel function has a unique gid value that enables each thread to work distinctly.
A grayscale pixel has the same value for each of its RGB components. This value can be calculated by simply averaging the RGB components of a color pixel, or by applying certain weights to each component. This sample uses the Rec. 709 luma coefficients for the color-to-grayscale conversion.
half4 inColor = inTexture.read(gid);
half gray = dot(inColor.rgb, kRec709Luma);
outTexture.write(half4(gray, gray, gray, 1.0), gid);
執行計算管道
A MTLComputeCommandEncoder object contains the commands for executing a compute pass, including references to the kernel function and its resources. Unlike a render command encoder, you can create a MTLComputeCommandEncoder without using a pass descriptor.
id <MTLComputeCommandEncoder> computeEncoder = [commandBuffer computeCommandEncoder];
[computeEncoder setComputePipelineState:_computePipelineState];
[computeEncoder setTexture:_inputTexture
atIndex:AAPLTextureIndexInput];
[computeEncoder setTexture:_outputTexture
atIndex:AAPLTextureIndexOutput];
A compute pass must specify the number of times to execute a kernel function. This number corresponds to the grid size, which is defined in terms of threads and threadgroups. A threadgroup is a 3D group of threads that are executed concurrently by a kernel function. In this sample, each thread corresponds to a unique texel, and the grid size must be at least the size of the 2D image. For simplicity, this sample uses a 16 x 16 threadgroup size which is small enough to be used by any GPU. In practice, however, selecting an efficient threadgroup size depends on both the size of the data and the capabilities of a specific device.
// Set the compute kernel's thread group size of 16x16
_threadgroupSize = MTLSizeMake(16, 16, 1);
// Calculate the number of rows and columsn of thread groups given the width of our input image.
// Ensure we cover the entire image (or more) so we process every pixel.
_threadgroupCount.width = (_inputTexture.width + _threadgroupSize.width - 1) / _threadgroupSize.width;
_threadgroupCount.height = (_inputTexture.height + _threadgroupSize.height - 1) / _threadgroupSize.height;
The sample finalizes the compute pass by issuing a dispatch call and ending the encoding of compute commands.
[computeEncoder dispatchThreadgroups:_threadgroupCount
threadsPerThreadgroup:_threadgroupSize];
[computeEncoder endEncoding];
The sample then continues to encode the rendering commands first introduced in the Basic Texturing sample. The commands for the compute pass and the render pass use the same grayscale texture, are appended into the same command buffer, and are submitted to the GPU at the same time. However, the grayscale conversion in the compute pass is always executed before the quad rendering in the render pass.
相關文章
- Metal入門教程(四)灰度計算
- [MetalKit]46-Introduction to compute using Metal 用 Metal 進行計算的簡介
- metal2的四個新features
- metal
- [MetalKit]43-Metal By Tutorials book! Metal教程書籍
- [MetalKit]47-Introducing Metal 3 Metal 3簡介
- Apple Configurator 2 使用入門APP
- [MetalKit]45-Using eGPUs with Metal 在 eGPU上使用 MetalGPU
- 你好,我是程式設計師程式設計師
- 計算機網路概述2計算機網路
- 你好
- WebGPU 計算管線、計算著色器(通用計算)入門案例:2D 物理模擬WebGPU
- 6-2 計算素數和
- nand2tetris_hack計算機NaN計算機
- 展現領導廠商風采新華三計算與儲存新品“要你好看”
- 《設計模式》 - 5. 介面卡模式( Adapter )設計模式APT
- [譯]Metal 渲染管線教程
- 在 SwiftUI 中使用 Metal ShaderSwiftUI
- 你好,promisePromise
- 你好WebAssemblyWeb
- 你好laravelLaravel
- 你好,掘金
- LevelDB,你好~
- 你好,世界
- 你好,Spring!Spring
- 計算機網路 王道論壇2計算機網路
- 計算機網路小白學習-2計算機網路
- 2 簡單的計算機模型MARIE計算機模型
- 9.3.2另一種計算機器2計算機
- day2-計算符和if語句
- Python計算機視覺-第2章Python計算機視覺
- 靠Play Pass反擊Apple Arcade,谷歌有勝算嗎?APP谷歌
- Metal日記:使用步驟指南
- LoadBalancer for bare metal Kubernetes cluster-MetalLB
- [20190823]關於CPU成本計算2.txt
- 2. 計算機怎麼表示數字計算機
- 計算機網路基礎-2-物理層計算機網路
- 計算機組成原理學習 Part 2計算機
- 你好,iLogtail 2.0!AI