本文主要是介绍Metal 笔记,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
Metal的大坑
1 顶点buffer,顶点是4个float对齐的,index数据没发现对齐要求
2 顶点shader,你输出的顶点(projMatrix*modelviewMatrix*pos) 范围:xy是[-1,1],z是[0,1]
3 片段shader,片段shader给你输入的点:in.position,xy的范围是[0,渲染目标的宽高],比如渲染到1024*1024的纹理,xy范围就是[0,1024]
z的范围还是[0,1]
device:
代表渲染环境,整个程序只有一个,是个单例对象,可以从view获得,也可以单独获得
1 创建commandQueue
2 创建纹理
3 创建 library(shader)
4 创建 renderPipleLine( 相当于OpenGL里面shader链接后的program)
5 创建 buffer (vbo)
library:
就是shader:
MTL::Library* pLibrary = _pDevice->newLibrary( NS::String::string(shaderSrc, UTF8StringEncoding), nullptr, &pError );
//通过shader里面的函数名字拿到函数
MTL::Function* pVertexFn = pLibrary->newFunction( NS::String::string("vertexMain", UTF8StringEncoding) );
MTL::Function* pFragFn = pLibrary->newFunction( NS::String::string("fragmentMain", UTF8StringEncoding) );
buffer:
相当于vbo:
MTL::Buffer* pVertexPositionsBuffer = _pDevice->newBuffer( positionsDataSize, MTL::ResourceStorageModeManaged );
MTL::Buffer* pVertexColorsBuffer = _pDevice->newBuffer( colorDataSize, MTL::ResourceStorageModeManaged );
//拷贝数据
memcpy( pVertexPositionsBuffer->contents(), positions, positionsDataSize );
memcpy( pVertexColorsBuffer->contents(), colors, colorDataSize );
//上传到显卡
pVertexPositionsBuffer->didModifyRange( NS::Range::Make( 0, _pVertexPositionsBuffer->length() ) );
pVertexColorsBuffer->didModifyRange( NS::Range::Make( 0, _pVertexColorsBuffer->length() ) );
纹理:
使用纹理描述符来创建,
纹理描述符对象,直接new
MTLTextureDescriptor *texDescriptor = [MTLTextureDescriptor new];
texDescriptor.textureType = MTLTextureType2D;
texDescriptor.width = 512;
texDescriptor.height = 512;
texDescriptor.pixelFormat = MTLPixelFormatRGBA8Unorm;
texDescriptor.usage = MTLTextureUsageRenderTarget |
MTLTextureUsageShaderRead;
_renderTargetTexture = [_device newTextureWithDescriptor:texDescriptor];
RenderPipeline(RenderPipelineState):
RenderPipelineDescriptor desp
desp.xxxx 设置。。。
就是指定shader,开启混合,深度测试,如渲染目标的像素格式,深度buffer的像素格式
MTL::RenderPipelineState* _pIntersectProgram = _pDevice->newRenderPipelineState( pDesc, &pError );
RenderCommandEncoder(创建的时候就要指定渲染目标)就叫renderpass吧:
可以渲染一个mesh
1 指定colorAttachments:纹理或者view
需要用到 RenderPassDescriptor
渲染到纹理需:
renderPassDescriptor.colorAttachments[0].texture = _renderTargetTexture;
renderPassDescriptor.colorAttachments[0].loadAction = MTLLoadActionClear;
renderPassDescriptor.colorAttachments[0].clearColor = MTLClearColorMake(1, 1, 1, 1);
renderPassDescriptor.colorAttachments[0].storeAction = MTLStoreActionStore;
//用commandBuffer来创建
RenderCommandEncoder renderEncoder = commandBuffer.renderCommandEncoderWithDescriptor
渲染到窗口:
renderPassDescriptor = view.currentRenderPassDescriptor;
RenderCommandEncoder renderEncoder = commandBuffer.renderCommandEncoderWithDescriptor
2 指定用什么程序来渲染pipeline
renderEncoder.setRenderPipelineState(renderPipeline);
3 指定shader用到的各种数据
//设置渲染方式
renderEncoder.drawPrimitives(MTLPrimitiveTypeTriangle..)
//设置纹理
renderEncoder setFragmentTexture:_renderTargetTexture atIndex:AAPLTextureInputIndexColor
//设置mesh的顶点数据,可以通过buffer来指定,这样才效率高。
renderEncoder->setVertexBuffer( _pVertexPositionsBuffer, 0, 0 );
renderEncoder->setVertexBuffer( _pVertexColorsBuffer, 0, 1 );
renderEncoder->setVertexBytes(data,len)//低效
renderEncoder.endEncoding()//这之后这个对象就可以销毁了,东西已经保存在commandQueue里面了
commandBuffer:
创建多个renderEncoder,提交渲染,显示渲染结果到view
commandBuffer presentDrawable:view.currentDrawable
commandBuffer commit
CommandQueue:
创建commandBuffer
[_commandQueue commandBuffer];
onDraw(){
CommandQueue 创建一个 commandBuffer
commandBuffer 创建若干个RenderCommandEncoder 有几个mesh 就创建几个
}
这篇关于Metal 笔记的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!