14.7 OpenGL图元装配和光栅化:早期各片段测试

2024-02-14 00:44

本文主要是介绍14.7 OpenGL图元装配和光栅化:早期各片段测试,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

早期各片段测试 Early Per-Fragment Tests

layout(early_fragment_tests) in;
如果片段着色器指定了 Early_fragment_tests 布局限定符,则将在片段着色器执行之前执行本节中描述的每个片段测试。 否则,它们将在片段着色器执行后执行。

在光栅化阶段生成片段之后,会在片段着色器执行之前对每个片段进行一系列的每片段操作。如果在这些操作中的任何一处片段被丢弃,则该片段将不会被后续任何阶段处理,包括片段着色器的执行。

按照以下顺序,对每个片段执行三个基本的片段操作以及可选的另外三个操作:

  1. 像素所有权测试:确定当前片段是否属于当前视口范围内的有效像素。
  2. 剪裁测试:检查片段是否位于剪裁区域内,即窗口坐标系下的可见区域。
  3. 多采样片段操作:当启用多重采样时,会处理每个片段对应的所有样本点,并根据它们的结果来决定最终片段的颜色和深度值。

若启用了早期每片段操作,还会进行以下测试:

  1. 模板测试:根据模板缓冲区的内容判断片段是否应该被绘制或剔除。
  2. 深度缓冲测试:对比片段的深度值与深度缓冲区中的现有值,以确定片段是否遮挡了已有的片段或者应该被遮挡。
  3. 遮挡查询样本计数:在进行性能分析或优化时,用于统计通过所有测试并可能影响屏幕输出的片段数量。

像素所有权测试 Pixel Ownership Test

第一个测试是确定帧缓冲中位置(xw,yw)处的像素当前是否由GL(更准确地说,由此GL上下文)所拥有。如果不是,则窗口系统决定了传入片段的命运。可能的结果是丢弃片段,或者将一些后续逐片段操作应用于片段的某个子集。此测试允许窗口系统控制GL的行为,例如,当GL窗口被遮挡时。

如果绘制帧缓冲是帧缓冲对象(参见第17.4.1节),则像素所有权测试始终通过,因为帧缓冲对象的像素由GL拥有,而不是窗口系统。如果绘制帧缓冲是默认帧缓冲,则窗口系统控制像素所有权。

剪裁测试 Scissor Test

剪切测试(Scissor Test)是一种图形渲染过程中的功能,它用于判断像素坐标(xw, yw)是否位于每个视口所定义的剪切矩形区域内。在OpenGL等图形API中,通过设置剪切矩形可以限制帧缓冲区中某个指定区域进行绘制或清除操作。

具体来说,剪切矩形由四个值确定:左边界(left)、下边界(bottom)、宽度(width)和高度(height)。这些值可以通过调用相应的API函数如glScissor()来设置,并且针对每个视口都可以独立定义一个剪切矩形。

当光栅化过程中产生的片段(fragment)的窗口坐标(xw, yw)满足以下条件时,该片段通过剪切测试:

  • 左边界小于等于xw并且xw小于左边界加上宽度,即 left ≤ xw < left + width
  • 下边界小于等于yw并且yw小于下边界加上高度,即 bottom ≤ yw < bottom + height

若片段坐标不在剪切矩形范围内,则该片段将被丢弃,不会进行后续的绘制操作。这对于限制渲染到屏幕特定区域、优化性能以及实现一些特殊效果非常有用。

void glScissorArrayv( uint first, sizei count, const int *v );
void glScissorIndexed( uint index, int left, int bottom, sizei width, sizei height );
void glScissorIndexedv( uint index, int *v );
void glScissor( int left, int bottom, sizei width, sizei height );glEnable(GL_SCISSOR_TEST);
glDisable(GL_SCISSOR_TEST);void glEnablei( enum target, uint index );
void glDisablei( enum target, uint index );
boolean glIsEnabledi( enum target, uint index );
设置一组裁剪矩形,每个矩形应用于相应的视口
void glScissorArrayv( uint first, sizei count, const int *v );
  • first:要修改的第一个裁剪矩形的索引。
  • count:裁剪矩形的数量。
  • v:一个整数数组的地址,依次包含裁剪矩形的左边、底部、宽度和高度。
int scissors[8] = { 0,    0,  300, 200,  // 视口 0 的 左下宽高350, 50,  200, 200   // 视口 1 的 左下宽高
};
glScissorArrayv(0, 2, scissors); // 使用ScissorArrayv设置剪切矩形
glEnable(GL_SCISSOR_TEST); // 确保剪切测试已启用// 对于第一个视口,其剪切矩形范围是[  0,  0]到[300, 200];
// 对于第二个视口,其剪切矩形范围是[350, 50]到[550, 250]。
void glScissorIndexed( uint index, int left, int bottom, sizei width, sizei height );// 等价于int v[] = { lef t, bottom, width, height };
glScissorArrayv(index, 1, v);
void glScissor( int left, int bottom, sizei width, sizei height );// 等价于for (uint i = 0; i < MAX_VIEWPORTS; i++) {glScissorIndexed(i, left, bottom, width, height);
}
void glScissorIndexedv( uint index, int *v );// 等价于glScissorArrayv(index, 1, v);
// 启用或禁用所有视口的剪切测试
glEnable(GL_SCISSOR_TEST);
glDisable(GL_SCISSOR_TEST);// 针对单个视口独立启用或禁用剪切测试
glEnablei(GL_SCISSOR_TEST, viewportIndex); // 启用指定视口的剪切测试
glDisablei(GL_SCISSOR_TEST, viewportIndex); // 禁用指定视口的剪切测试// 查询特定视口的剪切测试状态
GLboolean isEnabled = glIsEnabledi(GL_SCISSOR_TEST, viewportIndex);// 如果要查询默认视口(即索引为0的视口)的剪切测试状态
GLboolean isEnabledDefaultViewport = glIsEnabled(GL_SCISSOR_TEST);

多采样片段操作 Multisample Fragment Operations

在OpenGL或其他类似的图形渲染管线中,片段覆盖(fragment coverage)的修改步骤是一个关键阶段,它涉及到多个状态变量,包括SAMPLE_COVERAGE、SAMPLE_COVERAGE_VALUE、SAMPLE_COVERAGE_INVERT、SAMPLE_MASK和SAMPLE_MASK_VALUE。这一过程发生在多采样(MULTISAMPLE)功能启用且SAMPLE_BUFFERS值为1的情况下。

首先,片段着色器输出的alpha值仅针对颜色编号0、索引0的组件,并且若着色器未写入该alpha值,则其为未定义。

样本覆盖率相关的操作可以通过Enable和Disable函数控制,对应的目标分别为SAMPLE_COVERAGE和SAMPLE_MASK。当SAMPLE_COVERAGE启用时,会根据SAMPLE_COVERAGE_VALUE生成一个临时覆盖掩码,并与当前片段覆盖进行逻辑AND运算。如果SAMPLE_COVERAGE_INVERT设为TRUE,还会先对这个临时掩码进行反转操作。

进一步,如果SAMPLE_MASK也被启用,片段覆盖还需与SAMPLE_MASK_VALUE做AND运算以更新最终的片段覆盖值。

SAMPLE_COVERAGE_VALUE和是否反转的设置通过SampleCoverage函数完成,其中value参数代表覆盖率数值,invert参数指示是否翻转掩码。而SAMPLE_MASK_VALUE则由SampleMaski函数设定,该函数允许指定特定掩码字中的位掩码。

这些值的转换算法不强制具体实现方式,但通常期望保证一定比例关系和伪随机性,以避免图像出现因采样位置规律性导致的瑕疵。查询上述状态变量的具体值,可以使用相应的查询函数。

void glSampleCoverage( float value, boolean invert );
  • value参数是样本覆盖率的数值(SAMPLE_COVERAGE_VALUE)。它被限制在区间 [0, 1] 内,表示覆盖的样本比例。如果所有的样本都覆盖,value应为1.0;如果没有样本覆盖,value应为0.0。
  • invert参数是否反转样本覆盖率(SAMPLE_COVERAGE_INVERT)。如果设置为true,则覆盖率会被反转。
void glSampleMaski( uint maskNumber, bitfield mask );
  • maskNumber:表示要修改的掩码编号,因为在OpenGL 4.2及更高版本中,可以针对不同的视口或者同一视口中不同的掩码单元进行独立设置。

  • mask:这是一个位域值,每一位对应于一个子样本,如果该位为1,则相应的子样本会被启用;如果为0,则对应的子样本被禁用。掩码长度应与当前激活帧缓冲区的样本数量相匹配。

早期片段测试限定符 The Early Fragment Test Qualifier

在OpenGL等图形渲染管线中,模板测试(stencil test)、深度缓冲区测试(depth buffer test)和遮挡查询样本计数(occlusion query sample counting)的操作执行时机取决于当前激活的片段着色器是否启用了早期片段测试功能(参见第15.2.4节)。当开启早期每个片段操作时,这些测试将在片段着色器执行前进行,并根据测试结果相应地更新模板缓冲区、深度缓冲区的内容以及遮挡查询的样本计数。一旦片段着色器执行完毕后,将不再重复执行这些操作。

如果当前没有活动程序,或者活动程序中没有片段着色器,或者程序链接时关闭了早期片段测试,那么这些操作会在片段着色器执行之后按第17.3节描述的顺序执行。

另外,如果开启了早期片段测试,片段着色器计算出的深度值将不会对最终深度判断产生影响。同时要注意的是,即使片段或样本由于后期片段操作(例如alpha-to-coverage测试)导致被丢弃,它们仍然可能会更新模板缓冲区、深度缓冲区的数据以及遮挡查询的样本计数。这意味着,在决定是否绘制某个像素之前,可能会先完成某些相关的缓冲区更新和统计计算。

这篇关于14.7 OpenGL图元装配和光栅化:早期各片段测试的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/707062

相关文章

如何测试计算机的内存是否存在问题? 判断电脑内存故障的多种方法

《如何测试计算机的内存是否存在问题?判断电脑内存故障的多种方法》内存是电脑中非常重要的组件之一,如果内存出现故障,可能会导致电脑出现各种问题,如蓝屏、死机、程序崩溃等,如何判断内存是否出现故障呢?下... 如果你的电脑是崩溃、冻结还是不稳定,那么它的内存可能有问题。要进行检查,你可以使用Windows 11

性能测试介绍

性能测试是一种测试方法,旨在评估系统、应用程序或组件在现实场景中的性能表现和可靠性。它通常用于衡量系统在不同负载条件下的响应时间、吞吐量、资源利用率、稳定性和可扩展性等关键指标。 为什么要进行性能测试 通过性能测试,可以确定系统是否能够满足预期的性能要求,找出性能瓶颈和潜在的问题,并进行优化和调整。 发现性能瓶颈:性能测试可以帮助发现系统的性能瓶颈,即系统在高负载或高并发情况下可能出现的问题

字节面试 | 如何测试RocketMQ、RocketMQ?

字节面试:RocketMQ是怎么测试的呢? 答: 首先保证消息的消费正确、设计逆向用例,在验证消息内容为空等情况时的消费正确性; 推送大批量MQ,通过Admin控制台查看MQ消费的情况,是否出现消费假死、TPS是否正常等等问题。(上述都是临场发挥,但是RocketMQ真正的测试点,还真的需要探讨) 01 先了解RocketMQ 作为测试也是要简单了解RocketMQ。简单来说,就是一个分

【测试】输入正确用户名和密码,点击登录没有响应的可能性原因

目录 一、前端问题 1. 界面交互问题 2. 输入数据校验问题 二、网络问题 1. 网络连接中断 2. 代理设置问题 三、后端问题 1. 服务器故障 2. 数据库问题 3. 权限问题: 四、其他问题 1. 缓存问题 2. 第三方服务问题 3. 配置问题 一、前端问题 1. 界面交互问题 登录按钮的点击事件未正确绑定,导致点击后无法触发登录操作。 页面可能存在

业务中14个需要进行A/B测试的时刻[信息图]

在本指南中,我们将全面了解有关 A/B测试 的所有内容。 我们将介绍不同类型的A/B测试,如何有效地规划和启动测试,如何评估测试是否成功,您应该关注哪些指标,多年来我们发现的常见错误等等。 什么是A/B测试? A/B测试(有时称为“分割测试”)是一种实验类型,其中您创建两种或多种内容变体——如登录页面、电子邮件或广告——并将它们显示给不同的受众群体,以查看哪一种效果最好。 本质上,A/B测

Verybot之OpenCV应用一:安装与图像采集测试

在Verybot上安装OpenCV是很简单的,只需要执行:         sudo apt-get update         sudo apt-get install libopencv-dev         sudo apt-get install python-opencv         下面就对安装好的OpenCV进行一下测试,编写一个通过USB摄像头采

BIRT 报表的自动化测试

来源:http://www.ibm.com/developerworks/cn/opensource/os-cn-ecl-birttest/如何为 BIRT 报表编写自动化测试用例 BIRT 是一项很受欢迎的报表制作工具,但目前对其的测试还是以人工测试为主。本文介绍了如何对 BIRT 报表进行自动化测试,以及在实际项目中的一些测试实践,从而提高了测试的效率和准确性 -------

可测试,可维护,可移植:上位机软件分层设计的重要性

互联网中,软件工程师岗位会分前端工程师,后端工程师。这是由于互联网软件规模庞大,从业人员众多。前后端分别根据各自需求发展不一样的技术栈。那么上位机软件呢?它规模小,通常一个人就能开发一个项目。它还有必要分前后端吗? 有必要。本文从三个方面论述。分别是可测试,可维护,可移植。 可测试 软件黑盒测试更普遍,但很难覆盖所有应用场景。于是有了接口测试、模块化测试以及单元测试。都是通过降低测试对象

day45-测试平台搭建之前端vue学习-基础4

目录 一、生命周期         1.1.概念         1.2.常用的生命周期钩子         1.3.关于销毁Vue实例         1.4.原理​编辑         1.5.代码 二、非单文件组件         2.1.组件         2.2.使用组件的三大步骤         2.3.注意点         2.4.关于VueComponen

读Spring实战(第四版)概括—装配Bean

很久很久以前读过Spring实战(第三版),因为第三版和第四部差异还是特别明显的,在整体思想上有了比较重大的改变,比如用注解和JavaConfig替换Xml以及现在非常火热的Springboot在书的最后也有提到。OK,开始看书,书本的第一章讲了一下Spring存在的目的(简化Java开发)和Spring的功能,以及Spring3->Spring4增加了哪些功能,那我就从第二章开始概括本书,以给我