- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我刚刚开始使用 Metal,在掌握一些基本的东西时遇到了困难。我一直在阅读大量关于 Metal 的网页,并研究了 Apple 的示例等等,但我的理解仍然存在差距。我认为我的主要困惑点是:处理顶点缓冲区的正确方法是什么,我怎么知道何时可以安全地重用它们?这种混淆以多种方式表现出来,正如我将在下面描述的那样,也许我的混淆的这些不同表现需要以不同的方式来解决。
更具体地说,我在 macOS 上的 Objective-C 中使用 MTKView 子类来显示非常简单的 2D 形状:内部具有背景颜色的 View 的整体框架,该整体框架内部具有不同背景颜色的 0+ 个矩形子框架在它们里面,然后是每个子帧内的 0+ 各种颜色的平面阴影方块。我的顶点函数只是一个简单的坐标变换,我的片段函数只是通过它接收到的颜色,基于苹果的三角形演示应用程序。对于带有单个正方形的单个子帧,我可以正常工作。到现在为止还挺好。
有几件事让我感到困惑。
一:我可以设计我的代码,用单个顶点缓冲区和对 drawPrimitives:
的单个调用来渲染整个事物。 ,在一次大爆炸中绘制所有(子)框架和正方形。然而,这不是最佳的,因为它破坏了我的代码的封装,其中每个子帧代表一个对象的状态(包含 0+ 方格的东西);我想让每个对象负责绘制自己的内容。因此,最好让每个对象都设置一个顶点缓冲区并创建自己的 drawPrimitives:
称呼。但是由于对象将按顺序绘制(这是一个单线程应用程序),我想在所有这些绘制操作中重用相同的顶点缓冲区,而不是让每个对象必须分配和拥有一个单独的顶点缓冲区。但我能做到吗?我打电话后drawPrimitives:
,我猜必须将顶点缓冲区的内容复制到 GPU,并且我假设 (?) 这不是同步完成的,因此立即开始修改下一个对象绘图的顶点缓冲区是不安全的。那么:我怎么知道 Metal 何时完成了缓冲区,我可以再次开始修改它?
二:即使#1 有一个明确定义的答案,这样我就可以阻塞直到 Metal 完成缓冲区,然后开始为下一个 drawPrimitives:
修改它打电话,这样的设计合理吗?我想这意味着我的 CPU 线程会反复阻塞以等待内存传输,这不是很好。那么这是否会促使我进行每个对象都有自己的顶点缓冲区的设计?
三:好的,假设每个对象都有自己的顶点缓冲区,或者我用一个大顶点缓冲区对整个事物进行一次“大爆炸”渲染(我认为这个问题适用于这两种设计)。我打电话后presentDrawable:
然后 commit
在我的命令缓冲区上,我的应用程序将关闭并做一些工作,然后将尝试更新显示,因此我的绘图代码现在再次执行。我想重用我之前分配的顶点缓冲区,覆盖其中的数据以进行新的、更新的显示。但同样:我怎么知道什么时候是安全的?据我了解,事实是commit
返回到我的代码并不意味着 Metal 已经完成将我的顶点缓冲区复制到 GPU,在一般情况下,我必须假设这可能需要任意长的时间,所以当我重新输入我的绘图代码。正确的说法是什么?再说一遍:我应该阻塞等待直到它们可用(但是我应该这样做),还是应该有第二组顶点缓冲区,以防 Metal 仍然忙于第一组? (这似乎只是将问题推到了顶峰,因为当我为第三次更新输入绘图代码时,之前使用的两组缓冲区可能都不可用,对吗?那么我可以添加第三组顶点缓冲区,但是第四次更新……)
四:对于绘制框架和子框架,我只想写一个每个人都可以调用的可重用的“drawFrame”类型的函数,但我对正确的设计有点困惑。使用 OpenGL 这很容易:
- (void)drawViewFrameInBounds:(NSRect)bounds
{
int ox = (int)bounds.origin.x, oy = (int)bounds.origin.y;
glColor3f(0.77f, 0.77f, 0.77f);
glRecti(ox, oy, ox + 1, oy + (int)bounds.size.height);
glRecti(ox + 1, oy, ox + (int)bounds.size.width - 1, oy + 1);
glRecti(ox + (int)bounds.size.width - 1, oy, ox + (int)bounds.size.width, oy + (int)bounds.size.height);
glRecti(ox + 1, oy + (int)bounds.size.height - 1, ox + (int)bounds.size.width - 1, oy + (int)bounds.size.height);
}
drawPrimitives:
,因为如果它被连续调用两次,当第二次调用想要修改缓冲区时,Metal 可能还没有从第一次调用中复制顶点数据。我显然不想在每次调用函数时都分配一个新的顶点缓冲区。我可以让调用者传入一个顶点缓冲区供函数使用,但这只会将问题推到一个级别;那么调用者应该如何处理这种情况呢?也许我可以让函数将新顶点附加到调用者提供的缓冲区中不断增长的顶点列表的末尾;但这似乎迫使整个渲染完全预先计划(以便我可以预先分配一个合适大小的大缓冲区以适合每个人将绘制的所有顶点 - 这需要顶级绘图代码以某种方式知道如何每个对象最终会生成许多顶点,这违反了封装),或者做一个设计,我有一个扩展的顶点缓冲区,当它的容量证明不足时,它会根据需要重新分配。我知道如何做这些事情;但他们都感觉不对。我正在为什么是正确的设计而苦苦挣扎,因为我认为我不太了解 Metal 的内存模型。有什么建议吗?对很长的多部分问题表示歉意,但我认为所有这些都归结为同样的基本缺乏理解。
最佳答案
对您潜在问题的简短回答是:在该命令缓冲区完成之前,您不应覆盖添加到命令缓冲区的命令所使用的资源。确定这一点的最佳方法是添加完成处理程序。您也可以投票 status
命令缓冲区的属性,但这并不好。
首先,在提交命令缓冲区之前,不会将任何内容复制到 GPU。此外,正如您所指出的,即使在您提交命令缓冲区之后,您也不能假设数据已完全复制到 GPU。
其次,在简单的情况下,您应该将帧的所有绘图放入单个命令缓冲区。创建和提交大量命令缓冲区(例如为每个绘制的对象创建一个缓冲区)会增加开销。
这两点结合在一起意味着您通常不能在同一帧中重用资源。基本上,您将不得不双重或三重缓冲才能同时获得正确性和良好的性能。
一种典型的技术是创建一个由信号量保护的小型缓冲区池。信号量计数最初是池中缓冲区的数量。需要缓冲区的代码在信号量上等待,当成功时,从池中取出缓冲区。它还应该向命令缓冲区添加一个完成处理程序,将缓冲区放回池中并向信号量发出信号。
您可以使用动态缓冲区池。如果代码需要一个缓冲区并且池是空的,它会创建一个缓冲区而不是阻塞。然后,当它完成时,它将缓冲区添加到池中,有效地增加了池的大小。但是,这样做通常没有意义。如果 CPU 的运行速度远远领先于 GPU,则您只需要三个以上的缓冲区,这并没有真正的好处。
至于您想让每个对象自行绘制的愿望,那当然可以做到。我会使用一个大的顶点缓冲区以及一些关于到目前为止已经使用了多少的元数据。每个需要绘制的对象都将其顶点数据附加到缓冲区,并根据该顶点数据对其绘制命令进行编码。您将使用 vertexStart
参数使绘图命令引用顶点缓冲区中的正确位置。
您还应该考虑使用图元重新启动值进行索引绘图,因此只有一个绘制命令可以绘制所有图元。每个对象将其图元添加到共享顶点数据和索引缓冲区,然后一些高级 Controller 将进行绘制。
关于objective-c - 什么时候重写并重用 MTLBuffer 或其他 Metal 顶点缓冲区是安全的?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60020817/
我习惯于使用 Apache 服务器,所以当启用 mod_rewrite 时,我可以创建一个 htaccess 文件并使用 URL 重写。 这是我的 htaccess 文件: RewriteEngine
我正在尝试编写一个 mixin 来修改输出的父选择器。这个想法是,在调用 mixin 的情况下,父选择器需要对其进行字符串替换。我有大部分工作,但我不知道如何吞下 & . .test { @inc
我有一个本地目录(上传)和一个 S3 桶设置。 当用户上传图片时,文件存储在本地目录:/uploads/member_id/image_name30 分钟后,系统将文件上传到 S3 使用相同的路径:s
我正在尝试使用以下内容重写代理页面的正文链接: sub_filter http://proxied.page.come http://local.page.com; sub_filte
关闭。这个问题需要更多focused .它目前不接受答案。 想改善这个问题吗?更新问题,使其仅关注一个问题 editing this post . 1年前关闭。 Improve this questi
我尝试在我的 JSF 应用程序中使用“重写”(http://ocpsoft.org/rewrite/)。 一切都很好,我已经创建了规则: .addRule(Join.path("/profile/{p
我可以在 AEM 中大致看到两种 URL 重写方法: /etc/map/http(s)下的Sling映射(sling:Mapping) 使用链接重写器/TransformerFactory 重写 UR
我有一个 onclick 函数,我想将 anchor 添加到 href 值。我不想更改 URL,因为我需要该网站仍然可以为没有 javascript 的人/出于 SEO 目的而运行。所以这是我尝试使用
我必须在 UILabel 中显示货币和价格。在一个标签中,但使用不同的字体大小。现在看起来像这样: ...我这样做是重写drawTextInRect:,如下所示: - (void)drawTextIn
我正在尝试使用以下内容进行重定向: RewriteRule ^reviews/area/Santa-Barbara%2F$"/reviews/area/santa-barbara" [R=301,NC
我使用 FOSUserBundle 并且我想覆盖他的 registerAction Controller 。我阅读了与覆盖 FOSUserBundle Controller 相关的文档,但它不起作用。
我正在尝试让 URL 重写在我的网站上运行。这是我的 .htaccess 的内容: RewriteEngine On RewriteRule ^blog/?$ index.php?page=blog
好吧,这让我发疯了......我正在尝试像这样重写我的网址: Now: http://www.somedomain.com/Somepage.aspx http://www.somedomain.co
final方法不能在子类中重写。但凭借 Scala 的魔力,这似乎是可能的。 考虑以下示例: trait Test { final def doIt(s: String): String = s
我有一个类似下面的查询: Select ser.key From dbo.Enrlmt ser Where ser.wd >= @FromDate AND ser.wd ser.wd
我是 nginx 的新手,只是想做一些我认为应该很简单的事情。如果我这样做:- curl http://localhost:8008/12345678 我希望返回 index.html 页面。但是我得
我们的一位客户创建了一个二维码,其中 url 中包含一个空格。 我将如何编写处理此问题的 nginx 重定向? 在字符串中使用诸如“%20”之类的东西的几次尝试似乎会导致 nginx 出错或使 con
我正在尝试覆盖 appendChild 方法,以便我可以控制动态创建的元素并在插入页面之前根据需要修改它们。我尝试了这个示例代码,只是为了看看它是否可以完成: var f = Element.prot
我目前正在使用以下功能,当用户单击某处以确定是否隐藏下拉菜单(在 react 中)。一切正常,但当我单击正文时,它会记录以下内容。 我尝试重写它几次,但我找不到解决这个问题的方法。 Uncaught
我正在开发一个 Spring Integration/Boot 应用程序。我使用多文档 application.yml (src/main/resources/application.yml) 来设置
我是一名优秀的程序员,十分优秀!