Step 1 搭建一个简单的渲染框架

news2025/1/15 20:54:20

Step 1 搭建一个简单的渲染框架

万事开头难。从萌生到自己到处看源码手抄一个mini engine出来的想法,到真正敲键盘去抄,转眼过去了很久的时间。这次大概的确是抱着认真的想法,打开VS从零开始抄代码。不知道能坚持多久呢。。。

本次的主题是搭一个简单的渲染框架。这里我们先假定要使用的底层图形API为DX12,当然代码设计上渲染和API层面肯定是要解耦的,如果有一天去抄OpenGL或是Vulcan的代码的时候,总不可能要重写已有的逻辑。另外,由于DX12的初始化逻辑中需要HWND类型的窗口句柄,所以这里也就需要引入原生的Windows API来绘制窗口了。当然同样的道理,窗口系统与绘制的API也没有什么耦合关系。

那么先从Main函数写起,它是最简单的,只做初始化和运行两件事情:

int main()
{
	EngineLoop loop(hInstance);
	if (!loop.Initialize())
		return 0;

	return loop.Run();
}

EngineLoop类里目前包含我们这里需要的窗口系统和图形系统。初始化的逻辑是有顺序的,这里我们先初始化窗口系统,再初始化图形系统:

bool EngineLoop::Initialize()
{
    m_pWindowSystem = MakeShared<WindowSystem>();
    if (!m_pWindowSystem->Initialize(windowSystemCfg))
    {
        return false;
    }

    m_pGraphicsSystem = MakeShared<GraphicsSystem>();
    if (!m_pGraphicsSystem->Initialize(graphicsSystemCfg))
    {
        return false;
    }
    return true;
}

我们用智能指针shared_ptr管理这些System。这样它们的生命周期就和EngineLoop保持一致,不用担心析构的时候忘记释放它们。

为了把创建绘制窗口的API与窗口本身的逻辑分开,窗口系统持有一个LowLevelWindow抽象类的指针。通过这个指针去真正地创建窗口,绘制窗口,以及响应窗口的事件等等:

bool WindowSystem::Initialize(const WindowSystemCfg& cfg)
{
    m_pWindow = MakeShared<WindowsAPIWindow>();

    if (!m_pWindow->Initialize(windowCfg))
    {
        return false;
    }
    return true;
}

同样图形系统也是,我们使用RHI抽象类的指针来真正跟图形硬件打交道:

bool GraphicsSystem::Initialize(const GraphicsSystemCfg& cfg)
{
    m_pRHI = MakeShared<D3D12RHI>();

    if (!m_pRHI->Initialize(rhiCfg))
    {
        return false;
    }
    return true;
}

对于DX12来说,那么就有一个D3D12RHI的子类啦。目前我们先不考虑渲染任何东西,只是把初始化的工作做掉,那需要哪些东西呢?

首先IDXGIFactory和ID3D12Device这两货肯定是需要的,如果没有它们,整个初始化逻辑就没法跑;IDXGISwapChain也是必要的,不然连back buffer都没有;然后我们需要使用绘制指令来进行各种底层操作,那就需要ID3D12CommandQueue,ID3D12CommandAllocator和ID3D12GraphicsCommandList这三剑客了。ID3D12CommandQueue是指令的执行者,它可以包含多个command list;command allocator是存储指令的数据结构,command list里记录的指令实际上是保存到这里。另外,由于GPU指令的执行对CPU来说是异步的,因此还需要一个ID3D12Fence用于同步。我们使用ComPtr来管理这些类,ComPtr对象当引用计数为0时,会自动调用Release接口,从而避免内存泄漏。

class D3D12RHI : public RHI
{
private:

    ComPtr<ID3D12Device> m_pDevice;
    ComPtr<IDXGIFactory4> m_pDxGiFactory;
    ComPtr<ID3D12Fence> m_pFence;
    ComPtr<ID3D12CommandQueue> m_pCommandQueue;
    ComPtr<ID3D12CommandAllocator> m_pDirectCmdListAlloc;
    ComPtr<ID3D12GraphicsCommandList> m_pCommandList;
    ComPtr<IDXGISwapChain> m_pSwapChain;
};

DX12的资源和view是分开的,一个资源可以对应多种view,这里的view以D3D12_CPU_DESCRIPTOR_HANDLE来区分。一个资源每使用一个view,就需要往ID3D12DescriptorHeap申请一个空闲的handle。那么我们可以把这个过程抽象一下,封装一个D3D12DescriptorHeap类,它负责分配空闲的handle给申请者:

class D3D12DescriptorHeap
{
public:
    D3D12DescriptorHeap(ID3D12Device* pDevice, D3D12_DESCRIPTOR_HEAP_TYPE type, UInt32 numDescriptors);
    void Initialize();
    D3D12_CPU_DESCRIPTOR_HANDLE Allocate();

private:
    ID3D12Device* m_pDevice = nullptr;
    D3D12_DESCRIPTOR_HEAP_TYPE m_type;
    UInt32 m_numDescriptors = 0;
    UInt32 m_descriptorSize = 0;
    UInt32 m_remainingFreeHandles;
    CD3DX12_CPU_DESCRIPTOR_HANDLE m_cpuHandle;
    ComPtr<ID3D12DescriptorHeap> m_pDescriptorHeap;
};

初始化过程中我们需要创建若干back buffer和一个depth buffer,这两个buffer的创建方式有区别,但它们本质上都属于资源,因此给它们各自一个类,然后共同继承D3D12Resource这个类,这个类包含一些对资源的通用操作。

class D3D12Resource : public D3D12RHIChild
{
public:
    D3D12Resource(D3D12RHI* pRHI) : D3D12RHIChild(pRHI), m_state(D3D12_RESOURCE_STATE_COMMON) {}
    void SetState(D3D12_RESOURCE_STATES state);

protected:
    D3D12_RESOURCE_STATES m_state;
    ComPtr<ID3D12Resource> m_pResource;
};

class D3D12BackBuffer : public D3D12Resource
{
public:
    D3D12BackBuffer(D3D12RHI* pRHI) : D3D12Resource(pRHI), m_rtvHandle() {}
    void Initialize(Int32 index);
    void Clear(const Color& color);
    D3D12_CPU_DESCRIPTOR_HANDLE GetRtvHandle() const { return m_rtvHandle; }
private:
    D3D12_CPU_DESCRIPTOR_HANDLE m_rtvHandle;
};

class D3D12DepthBuffer : public D3D12Resource
{
public:
    D3D12DepthBuffer(D3D12RHI* pRHI) : D3D12Resource(pRHI), m_dsvHandle() {}
    void Initialize(Int32 clientWidth, Int32 clientHeight, DXGI_FORMAT format);
    void Clear(D3D12_CLEAR_FLAGS clearFlags, Float depth, UInt8 stencil);
    D3D12_CPU_DESCRIPTOR_HANDLE GetDsvHandle() const { return m_dsvHandle; }

private:
    D3D12_CPU_DESCRIPTOR_HANDLE m_dsvHandle;
};

初始化流程完毕之后,我们就要准备update了。现阶段我们啥也不做,就准备一下渲染环境吧。我们在RHI类中添加了PrepareRender和FinishRender两个抽象接口,分别表示准备渲染以及完成渲染提交显示的逻辑。对于DX12来说,在渲染前/后要准备哪些事情呢?

渲染前,首先是清空command,让command相关的数据结构保证可用;然后获取当前要渲染的back buffer,设置其状态为D3D12_RESOURCE_STATE_RENDER_TARGET;然后对back buffer和depth buffer执行clear操作,最后提交给硬件。在渲染结束之后,同样我们要把当前back buffer的状态切回渲染前的,如果是多缓冲要切到下一个可用的back buffer,执行掉中间产生的所有渲染指令,显示到屏幕上。

自此,一个最简单的渲染框架就搭好了,运行起来也就是一个填充满clear color的窗口,并没有什么稀奇,然而背后的代码量却有数百行了。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1091952.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

一文让你玩转Linux多进程开发

Linux多进程开发 主要介绍多进程开发时的要点 进程状态转换 进程反应了进程执行的变化。 进程的状态分为三种 ,运行态,阻塞态,就绪态 在五态模型中分为以下几种,新建态&#xff0c;就绪态&#xff0c;运行态&#xff0c;阻塞态,终止态。 运行态&#xff1a;进程占用处理器正在运…

CSS结构选择器的使用

结构选择器 style>ul li:first-child {//选出第一个孩子进行变色background-color: blue;}</style><ul><li>我是第1个孩子</li><li>我是第2个孩子</li><li>我是第3个孩子</li><li>我是第4个孩子</li><li>…

基于linux的基础线程知识大总结

文章目录 1.线程的基础概念认知1.1什么是线程1.2线程的优缺点1.3一些页表知识的额外补充1.4进程和线程的对比 2.线程的基本控制2.1POSIX线程库2.2创建一个新的线程2.3有关线程id的解释和线程栈区的地址空间布局2.4线程终止2.5线程等待2.6线程分离 3.线程间互斥3.1基本概念3.2互…

多城镇信息发布付费置顶公众号开源版开发

多城镇信息发布付费置顶公众号开源版开发 以下是多城镇信息发布付费置顶公众号的功能列表&#xff1a; 信息发布&#xff1a;用户可以在公众号上发布各类信息&#xff0c;如房屋租售、二手物品交易、招聘信息等。 信息置顶&#xff1a;用户可以选择付费将自己的信息置顶在公众…

《PyTorch深度学习实践》第三讲 梯度下降算法

《PyTorch深度学习实践》第三讲 梯度下降算法 问题描述梯度下降问题分析编程实现代码实现效果 随机梯度下降问题分析编程实现代码实现效果 参考资料 问题描述 梯度下降 问题分析 编程实现 代码 import matplotlib.pyplot as plt# 训练集数据 x_data [1.0, 2.0, 3.0] y_data …

C++算法 —— 贪心(1)介绍

文章目录 1、什么是贪心算法2、特点3、学习方向 1、什么是贪心算法 贪心应当是一个策略&#xff0c;通过局部找到最优&#xff0c;来找到全局最优。它把解决问题的过程分为若干步&#xff0c;解决每一步的时候&#xff0c;都选择当前看起来最优的解法&#xff0c;通过这样做希…

Python数据分析实战-实现F检验(附源码和实现效果)

实现功能 F 检验&#xff08;F-test&#xff09;是一种常用的统计方法&#xff0c;用于比较两个或多个样本方差是否存在显著差异。它可以应用于多种场景&#xff0c;其中一些常见的应用场景包括&#xff1a; 方差分析&#xff08;ANOVA&#xff09;&#xff1a;F 检验在方差分…

【软考-中级】系统集成项目管理工程师-进度管理历年案例

持续更新。。。。。。。。。。。。。。。 进度管理历年案例和解析 2023 上 试题二(20分)2023 上 试题二(20分) 问题1(5分) 结合案例: (1)请写出项目关键路径,并计算项目工期。 答案:项目关键路径 ACEFGJKN,项目工期 80 解析(2)如果活动L工期拖延10天,对整个工期是否有影响…

C语言-程序环境和预处理(1)编译、连接介绍以及预处理函数,预处理符号详解及使用说明。

前言 本篇文章讲述了程序的翻译环境和执行环境&#xff0c;编译、连接&#xff0c;预定义符号&#xff0c;#define&#xff0c;#符号和##符号的相关知识。 文章目录 前言1.程序的翻译环境和执行环境2.编译链接2.1 翻译环境2.2 运行环境 3.预处理详解&#xff08;各预处理符号使…

Java之SPI

Java的SPI&#xff08;Service Provider Interface&#xff09;是一种面向接口编程的机制&#xff0c;用于实现组件之间的解耦和扩展。通过SPI机制&#xff0c;我们可以定义接口&#xff0c;并允许第三方提供不同的实现&#xff0c;从而实现可插拔、可扩展的架构。 SPI讲解 它…

Studio One6.5最新中文版安装步骤

在唱歌效果调试当中&#xff0c;我们经常给客户安装的几款音频工作站。第一&#xff0c;Studio One 6是PreSonus公司开发的一款功能强大的音频工作平台&#xff0c;具有丰富的音频处理功能和灵活的工作流程。以下是Studio One6的一些主要特点&#xff1a; 1.多轨录音和编辑&…

ezEIP信息泄露

漏洞描述 ezEIP存在信息泄露漏洞&#xff0c;通过遍历Cookie中的参数值获取敏感信息 漏洞复现 漏洞Url为 /label/member/getinfo.aspx访问时添加Cookie&#xff08;通过遍历获取用户的登录名电话邮箱等信息&#xff09; WHIR_USERINFORwhir_mem_member_pid1;漏洞证明&…

同比增长29.89%,长城汽车9月销售新车超12万辆

10月8日&#xff0c;长城汽车股份有限公司&#xff08;股票代码601633.SH、02333.HK、82333.HK&#xff1b;以下简称“长城汽车”&#xff09;发布2023年9月产销数据。今年9月&#xff0c;长城汽车销售新车121,632辆&#xff0c;同比增长29.89%&#xff0c;1-9月累计销售864,04…

安捷伦E9321A射频传感器

安捷伦E9321A射频传感器 E9321A 是 Agilent 使用的 6 GHz 0.1 瓦射频传感器。电子测试设备传感器测量波形的功率&#xff0c;例如多音和调制射频 (RF) 波形。传感器使用二极管检测器收集高度精确的调制测量值。50 MHz 至 6 GHz 300 kHz 视频带宽 功率范围&#xff1a;-65 至 20…

Android JNI调用流程

文章目录 前言一、JNI是什么二、JNI的优劣三、JNI的开发流程Java调用C函数1、创建声明native方法的Java工程&#xff0c;加载native函数的动态库&#xff0c;生成.h文件2、创建实现C函数的C工程&#xff0c;将本地代码编译成动态库C函数和Java本地方法的隐式映射&#xff08;相…

压缩炸弹,Java怎么防止

一、什么是压缩炸弹&#xff0c;会有什么危害 1.1 什么是压缩炸弹 压缩炸弹(ZIP)&#xff1a;一个压缩包只有几十KB&#xff0c;但是解压缩后有几十GB&#xff0c;甚至可以去到几百TB&#xff0c;直接撑爆硬盘&#xff0c;或者是在解压过程中CPU飙到100%造成服务器宕机。虽然…

11-网络篇-DNS步骤

1.URL URL就是我们常说的网址 https://www.baidu.com/?from1086k https是协议 m.baidu.com是服务器域名 ?from1086k是路径 2.域名 比如https://www.baidu.com 顶级域名.com 二级域名baidu 三级域名www 3.域名解析DNS DNS就是将域名转换成IP的过程 根域名服务器&#xff1a…

python2 paramiko 各种报错解决方案

一、介绍 paramiko是一个基于SSHv2协议的python库&#xff0c;支持以加密和认证的方式进行远程服务器的连接&#xff0c;用于实现远程文件的上传、下载或通过ssh远程执行命令。 paramiko支持Python&#xff08;2.7&#xff0c;3.4&#xff09;版本 paramiko库可直接使用pip …

谈谈C++中模板分离式编译出现的一些问题

什么是分离式编译 通俗的来讲就是将声明和定义分离在不同文件中 一个程序由若干个源文件共同实现&#xff0c;而每个源文件单独编译生成目标文件&#xff0c;最后将所有 目标文件链接起来形成单一的可执行文件的过程称为分离编译模式。 正常函数与模板分离式编译 看代码&…

生物制剂\化工\化妆品等质检损耗、制造误差处理作业流程图(ODOO15/16)

生物制剂、化工、化妆品等行业&#xff0c;因为产品为液体&#xff0c;产品形态和质量容易在各个业务环节发生变化&#xff0c;常常导致实物和账面数据不一致&#xff0c;如果企业业务流程不清晰&#xff0c;会导致系统大量的库存差异&#xff0c;以及财务难以核算的问题&#…