Metal框架详细解析(四十九) —— 将项目从OpenGL转化到Metal(二)

版本记录

版本号 时间
V1.0 2019.01.18 星期五

前言

很多做视频和图像的,相信对这个框架都不是很陌生,它渲染高级3D图形,并使用GPU执行数据并行计算。接下来的几篇我们就详细的解析这个框架。感兴趣的看下面几篇文章。
1. Metal框架详细解析(一)—— 基本概览
2. Metal框架详细解析(二) —— 器件和命令(一)
3. Metal框架详细解析(三) —— 渲染简单的2D三角形(一)
4. Metal框架详细解析(四) —— 关于GPU Family 4(一)
5. Metal框架详细解析(五) —— 关于GPU Family 4之关于Imageblocks(二)
6. Metal框架详细解析(六) —— 关于GPU Family 4之关于Tile Shading(三)
7. Metal框架详细解析(七) —— 关于GPU Family 4之关于光栅顺序组(四)
8. Metal框架详细解析(八) —— 关于GPU Family 4之关于增强的MSAA和Imageblock采样覆盖控制(五)
9. Metal框架详细解析(九) —— 关于GPU Family 4之关于线程组共享(六)
10. Metal框架详细解析(十) —— 基本组件(一)
11. Metal框架详细解析(十一) —— 基本组件之器件选择 - 图形渲染的器件选择(二)
12. Metal框架详细解析(十二) —— 基本组件之器件选择 - 计算处理的设备选择(三)
13. Metal框架详细解析(十三) —— 计算处理(一)
14. Metal框架详细解析(十四) —— 计算处理之你好,计算(二)
15. Metal框架详细解析(十五) —— 计算处理之关于线程和线程组(三)
16. Metal框架详细解析(十六) —— 计算处理之计算线程组和网格大小(四)
17. Metal框架详细解析(十七) —— 工具、分析和调试(一)
18. Metal框架详细解析(十八) —— 工具、分析和调试之Metal GPU Capture(二)
19. Metal框架详细解析(十九) —— 工具、分析和调试之GPU活动监视器(三)
20. Metal框架详细解析(二十) —— 工具、分析和调试之关于Metal着色语言文件名扩展名、使用Metal的命令行工具构建库和标记Metal对象和命令(四)
21. Metal框架详细解析(二十一) —— 基本课程之基本缓冲区(一)
22. Metal框架详细解析(二十二) —— 基本课程之基本纹理(二)
23. Metal框架详细解析(二十三) —— 基本课程之CPU和GPU同步(三)
24. Metal框架详细解析(二十四) —— 基本课程之参数缓冲 - 基本参数缓冲(四)
25. Metal框架详细解析(二十五) —— 基本课程之参数缓冲 - 带有数组和资源堆的参数缓冲区(五)
26. Metal框架详细解析(二十六) —— 基本课程之参数缓冲 - 具有GPU编码的参数缓冲区(六)
27. Metal框架详细解析(二十七) —— 高级技术之图层选择的反射(一)
28. Metal框架详细解析(二十八) —— 高级技术之使用专用函数的LOD(一)
29. Metal框架详细解析(二十九) —— 高级技术之具有参数缓冲区的动态地形(一)
30. Metal框架详细解析(三十) —— 延迟照明(一)
31. Metal框架详细解析(三十一) —— 在视图中混合Metal和OpenGL渲染(一)
32. Metal框架详细解析(三十二) —— Metal渲染管道教程(一)
33. Metal框架详细解析(三十三) —— Metal渲染管道教程(二)
34. Metal框架详细解析(三十四) —— Hello Metal! 一个简单的三角形的实现(一)
35. Metal框架详细解析(三十五) —— Hello Metal! 一个简单的三角形的实现(二)
36. Metal框架详细解析(三十六) —— Metal编程指南之概览(一)
37. Metal框架详细解析(三十七) —— Metal编程指南之基本Metal概念(二)
38. Metal框架详细解析(三十八) —— Metal编程指南之命令组织和执行模型(三)
39. Metal框架详细解析(三十九) —— Metal编程指南之资源对象:缓冲区和纹理(四)
40. Metal框架详细解析(四十) —— Metal编程指南之函数和库(五)
41. Metal框架详细解析(四十一) —— Metal编程指南之图形渲染:渲染命令编码器之Part 1(六)
42. Metal框架详细解析(四十二) —— Metal编程指南之图形渲染:渲染命令编码器之Part 2(七)
43. Metal框架详细解析(四十三) —— Metal编程指南之数据并行计算处理:计算命令编码器(八)
44. Metal框架详细解析(四十四) —— Metal编程指南之缓冲和纹理操作:Blit命令编码器(九)
45. Metal框架详细解析(四十五) —— Metal编程指南之Metal工具(十)
46. Metal框架详细解析(四十六) —— Metal编程指南之Tessellation(十一)
47. Metal框架详细解析(四十七) —— Metal编程指南之资源堆(十二)
48. Metal框架详细解析(四十八) —— 将项目从OpenGL转化到Metal(一)

Adding Shaders

创建一个新文件。 单击File ▸ New ▸ File…,选择iOS ▸ Source ▸ Metal File。 点击下一步。 将其命名为Shaders.metal并将其保存在您想要的任何位置。

Metal使用C ++编写着色器。 在大多数情况下,它类似于用于OpenGLGLSL

将其添加到文件的底部:

struct VertexIn {
    packed_float3 position;
    packed_float4 color;
};

struct VertexOut {
    float4 computedPosition [[position]];
    float4 color;
};

您将使用这些结构作为顶点着色器的输入和输出数据。

1. Writing a Vertex Shader

顶点着色器是为您绘制的每个顶点运行的函数。

在结构下面,添加以下代码:

vertex VertexOut basic_vertex( // 1
  const device VertexIn* vertex_array [[ buffer(0) ]], // 2
  unsigned int vid [[ vertex_id ]]) {                  // 3
    // 4
    VertexIn v = vertex_array[vid];

    // 5
    VertexOut outVertex = VertexOut();
    outVertex.computedPosition = float4(v.position, 1.0);
    outVertex.color = v.color;
    return outVertex;
}
  • 1) vertex表示这是一个顶点着色器函数。 此着色器的返回类型是VertexOut
  • 2) 在这里,您将获得传递给命令编码器的顶点缓冲区。
  • 3) 此参数是为此着色器调用的vertex id
  • 4) 抓取当前vertex id的输入顶点。
  • 5) 在这里,您创建一个VertexOut并从当前VertexIn传递数据。 这只是使用与输入相同的位置和颜色。

此时,顶点着色器的工作已完成。

2. Writing a Fragment Shader

顶点着色器完成后,片段着色器将针对每个可能的像素运行。

在顶点着色器下方,添加以下代码:

fragment float4 basic_fragment(VertexOut interpolated [[stage_in]]) { 
  return float4(interpolated.color);              
}

片段着色器接收顶点着色器的输出 - VertexOut。 然后,片段着色器返回当前片段的颜色。

3. Hooking up the Shaders to the Pipeline

着色器已就位,但您还没有将它们连接到您的管道上。 为此,请返回ViewController.swift,并将此属性添加到类中:

private var pipelineState: MTLRenderPipelineState!

此属性将包含着色器数据。

现在,找到setupMetal()。 在方法的底部添加以下内容:

// 1
let defaultLibrary = metalDevice.makeDefaultLibrary()!
let fragmentProgram = defaultLibrary.makeFunction(name: "basic_fragment")
let vertexProgram = defaultLibrary.makeFunction(name: "basic_vertex")

// 2
let pipelineStateDescriptor = MTLRenderPipelineDescriptor()
pipelineStateDescriptor.vertexFunction = vertexProgram
pipelineStateDescriptor.fragmentFunction = fragmentProgram
pipelineStateDescriptor.colorAttachments[0].pixelFormat = .bgra8Unorm

// 3
pipelineState = try! metalDevice
  .makeRenderPipelineState(descriptor: pipelineStateDescriptor)

这就是代码的作用:

  • 1) 在所有.metal文件中按名称查找顶点和片段着色器。
  • 2) 使用顶点着色器和片段着色器创建描述符对象。 像素格式设置为8位无符号的标准BGRA(Blue Green Red Alpha)
  • 3) 让GPU将所有内容编译到GPU优化(GPU-optimized)对象中。

现在,管道状态已准备就绪。 是时候使用它了。 为此,在draw(in:)中,下面代码的前面:

renderEncoder.drawIndexedPrimitives(
  type: .triangle, 
  indexCount: Indices.count, 
  indexType: .uint32, 
  indexBuffer: indicesBuffer, 
  indexBufferOffset: 0)

添加

renderEncoder.setRenderPipelineState(pipelineState)

构建并运行,你会看到彩色的屏幕。


Matrices

为了操纵场景,您需要将投影和模型视图矩阵(projection and model-view matrices)传递给GPU。 投影矩阵允许您操纵场景的感知,使更近的物体看起来比更远的物体更大。 模型视图矩阵允许您操纵对象或整个场景的位置,旋转和缩放。

为了使用这些矩阵,您将创建一个新的结构。 打开Vertex.swift。 在文件的顶部,添加:

struct SceneMatrices {
    var projectionMatrix: GLKMatrix4 = GLKMatrix4Identity
    var modelviewMatrix: GLKMatrix4 = GLKMatrix4Identity
}

请注意,您仍将使用GLKMatrix4。 这一部分不使用GLKit,因此您可以将它用于Metal中的矩阵。

现在,打开ViewController.swift,并添加两个新属性:

private var sceneMatrices = SceneMatrices()
private var uniformBuffer: MTLBuffer!

到方法draw(in:)中,在下面之前

renderEncoder.setRenderPipelineState(pipelineState)

添加

// 1
let modelViewMatrix = GLKMatrix4MakeTranslation(0.0, 0.0, -6.0)
sceneMatrices.modelviewMatrix = modelViewMatrix

// 2
let uniformBufferSize = MemoryLayout.size(ofValue: sceneMatrices)
uniformBuffer = metalDevice.makeBuffer(
  bytes: &sceneMatrices, 
  length: uniformBufferSize, 
  options: .storageModeShared)

// 3
renderEncoder.setVertexBuffer(uniformBuffer, offset: 0, index: 1)

以下是上面代码的作用:

  • 1) 创建一个矩阵,将对象向后移动6个单位,使其看起来更小。
  • 2) 像之前使用顶点缓冲区一样创建一个统一的缓冲区,但是使用矩阵数据。
  • 3) 将统一缓冲区连接到管道并将其标识符设置为1。

1. Projection Matrix

当你仍然在ViewController.swift中,在mtkView(_:drawableSizeWillChange :)中,添加以下内容:

let aspect = fabsf(Float(size.width) / Float(size.height))  
let projectionMatrix = GLKMatrix4MakePerspective(
  GLKMathDegreesToRadians(65.0), 
  aspect, 
  4.0, 
  10.0)
sceneMatrices.projectionMatrix = projectionMatrix 

此代码基于视图的纵横比创建投影矩阵。 然后,它将它分配给场景的投影矩阵。

有了这个,您的方块现在看起来方形,而不是伸展到整个屏幕。

2. Matrices in Shaders

接下来,您需要在着色器中接收矩阵数据。 打开Shaders.metal。 在最顶层,添加一个新结构:

struct SceneMatrices {
    float4x4 projectionMatrix;
    float4x4 viewModelMatrix;
};

现在,用以下内容替换basic_vertex函数:

vertex VertexOut basic_vertex(
  const device VertexIn* vertex_array [[ buffer(0) ]],
  const device SceneMatrices& scene_matrices [[ buffer(1) ]], // 1
  unsigned int vid [[ vertex_id ]]) {
    // 2
    float4x4 viewModelMatrix = scene_matrices.viewModelMatrix;
    float4x4 projectionMatrix = scene_matrices.projectionMatrix;
    
    VertexIn v = vertex_array[vid];

    // 3
    VertexOut outVertex = VertexOut();
    outVertex
      .computedPosition = projectionMatrix * viewModelMatrix * float4(v.position, 1.0);
    outVertex.color = v.color;
    return outVertex;
}

下面就是代码做的事情:

  • 1) 接收矩阵作为顶点着色器内的参数。
  • 2) 提取视图模型和投影矩阵。
  • 3) 投影和视图模型矩阵乘以位置。

构建并运行应用程序。 你应该看到这个:

这就是一个正方形了。


Making it Spin

在OpenGL实现中,GLViewController提供了lastUpdateDate,它将告诉您何时执行最后一次渲染。 在Metal中,你必须自己创建它。

首先,在ViewController中添加一个新属性:

private var lastUpdateDate = Date()

在方法draw(in: )里,下面之前:

commandBuffer.present(drawable)

添加下面代码

commandBuffer.addCompletedHandler { _ in
  self.lastUpdateDate = Date()
}

有了这个,当一帧绘制完成时,它会将lastUpdateDate更新为当前日期和时间。

现在,是时候旋转了! 在draw(in :)中,替换:

let modelViewMatrix = GLKMatrix4Translate(GLKMatrix4Identity, 0, 0, -6.0)

使用下面的内容

var modelViewMatrix = GLKMatrix4MakeTranslation(0.0, 0.0, -6.0)
let timeSinceLastUpdate = lastUpdateDate.timeIntervalSince(Date()) 

// 1
rotation += 90 * Float(timeSinceLastUpdate)

// 2
modelViewMatrix = GLKMatrix4Rotate(
  modelViewMatrix, 
  GLKMathDegreesToRadians(rotation), 0, 0, 1)

这会将rotation属性增加一个与最后一次渲染和此渲染之间的时间成比例的量。 然后,它将围绕Z轴的旋转应用于模型视图矩阵。

构建并运行应用程序。 你会看到立方体旋转。 成功!

恭喜! 你已经学到了很多关于Metal API的知识! 现在,您了解了Metal中一些最重要的概念,例如着色器,设备,命令缓冲区和管道(shaders, devices, command buffers, and pipelines),并且您对OpenGL和Metal之间的差异进行深入的了解。

更多信息,请务必查看Apple的这些优秀资源:

后记

本篇主要讲述了将项目从OpenGL转化到Metal,感兴趣的给个赞或者关注~~~

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容