原文地址 http://fabiensanglard.net/doom3/
2011年11月23日,id软件保持传统,并发布了其以前引擎的源代码。这一次是轮到了idTech4,使用它,创造出Prey,quake4,当然是Doom 3.在几个小时之内,GitHub上的源代码被下载了400多次,人们开始看游戏内部机制/在其他平台上引导引擎。我也跳上它,并及时完成了Mac OS X的Intel版本 约翰·卡马克。
在清晰度和评论方面,这是 Doom iPhone代码库(这是更新的,因此更好的评论)的id软件最好的开源代码。我强烈建议大家阅读,编译和实验。
下面是我的笔记就我的理解。像往常一样,我已经清理了它:我希望它会节省一个人几个小时,我也希望它会激励我们中的一些人阅读更多的代码,并成为更好的程序员。
第1部分:概述
第2部分:Dmap
第3部分:渲染器
第4部分:剖析
第5部分:脚本
第6部分:访谈(包括与约翰·卡马克的问答)
我注意到我正在使用越来越多的绘图和越来越少的文本来解释代码库。到目前为止,我已经使用gliffy绘制,但这个工具有一些令人沮丧的限制(如缺乏alpha通道)。我正在考虑编写一个专门用于使用SVG和Javascript绘制3D引擎的工具。我想知道这样的事情是否已经存在?无论如何,回到代码...
掌握这样一个突破性引擎的源代码是令人兴奋的。Doom III于2004年发布,为实时引擎设定了新的视觉和音频标准,最为显着的是“统一照明与阴影”。这项技术首次允许艺术家以好莱坞的规模表达自己。即使8年后,在Delta-Labs-4中与HellKnight的第一次遭遇仍然看起来非常棒:
源代码现在通过Github进行分发,这是一件好事,因为来自id Software的FTP服务器几乎总是关闭或重载。
来自TTimo的原始版本与Visual Studio 2010 Professional编译良好。不幸的是Visual Studio 2010“Express”缺少MFC,因此无法使用。这是令人失望的释放,但有些人已经删除了依赖。
Windows 7的 : =========== git clone https://github.com/TTimo/doom3.gpl.git
对于代码阅读和探索,我更喜欢在Mac OS X上使用XCode 4.0:SpotLight的搜索速度,变量亮点和“命令点击”达到定义使得体验优于Visual Studio。XCode项目在发布时被破坏,但是很容易解决几个步骤,现在有一个Github存储库由“坏扇区”,在Mac OS X Lion上运行良好。
MacOS X: ========= git克隆https://github.com/badsector/Doom3-for-MacOSX-
注意:安装 Visual Studio 2010生产力电动工具后,Visual Studio 2010中也可以看到“变量hightlights”和“Control-Click”。我不明白为什么这不是香草安装的一部分。
两个代码库现在都处于最佳状态:一次点击可执行文件!
琐事:为了运行游戏,您将需要base
包含Doom 3游戏的文件夹。因为我不想浪费时间从Doom 3 CD中提取它们并更新它们:我下载了Steam版本。似乎id软件团队做的一样,因为Visual Studio项目发布仍然包含"+set fs_basepath C:\Program Files (x86)\Steam\steamapps\common\doom 3"
在调试设置!
琐事:引擎是用Visual Studio .NET开发的。但代码不具有单行的C#,发布的版本需要Visual Studio 2010 Professional才能编译。
琐事: Id软件团队似乎是Matrix电影系列(黑客帝国)的粉丝:Quake III的工作题目是“Trinity”,Doom III的工作题目是“Neo”。(都出自黑客帝国电影中)
该解决方案分为反映引擎整体架构的项目:
项目 | 构建 | 意见 | |
视窗 | MacO SX | ||
游戏 | gamex86.dll | gamex86.so | Doom3游戏 |
游戏d3xp | gamex86.dll | gamex86.so | Doom3 eXPension(Ressurection)游戏 |
MayaImport | MayaImport.dll | - | 资产创建工具链的一部分:在运行时加载,以打开Maya文件并导入怪物,摄像头路径和地图。 |
毁灭战士 | Doom3.exe | Doom3.app | Doom 3引擎 |
所属类别 | TypeInfo.exe | - | 内部RTTI帮助器:生成GameTypeInfo.h :具有每个成员大小的所有Doom3类类型的映射。这允许通过TypeInfo类进行内存调试。 |
CurlLib | CurlLib.lib | - | HTTP客户端用于下载文件(Staticaly链接到gamex86.dll和doom3.exe)。 |
伊德利卜 | idLib.lib | idLib.a | id软件库。包括解析器,词法分析器,字典...(Staticaly链接到gamex86.dll和doom3.exe)。 |
像idTech2的每个引擎一样,我们找到一个封闭的源代码二进制(doom.exe)和一个开放源代码的动态库(gamex86.dll):
自2004年10月以来,大多数代码库已经通过Doom3 SDK访问:只有Doom3可执行源代码失踪。模式能够构建idlib.a
,gamex86.dll
但发动机的核心仍然是封闭源。
注意:引擎不使用标准C ++库:所有容器(映射,链接列表...)都被重新实现,但libc
被广泛使用。
注意:在游戏模块中,每个类都扩展了idClass。这允许引擎执行内部RTTI,并通过类名实例化类。
琐事:如果你看图纸,你会看到几个基本框架(如Filesystem
)在Doom3.exe项目中。这是一个问题,因为gamex86.dll也需要加载资源。这些子系统由doom3.exe中的gamex86.dll动态加载(这是图中箭头实现的)。如果我们用PE Explorer中的DLL我们可以看到,gamex86.dll导出一个方法:GetGameAPI
:
一切正常完全相同的方式Quake2中加载的渲染和游戏的DDL:交换对象的指针:
当Doom3.exe启动它:exe(这是图中箭头实现的)。如果我们用PE Explorer中的DLL我们可以看到,gamex86.dll导出一个方法: :一切正常完全相同的方式Quake2中加载的渲染和游戏的DDL:交换对象的指针:当Doom3.exe启动它:exe(这是图中箭头实现的)。如果我们用PE Explorer中的DLL我们可以看到,gamex86.dll导出一个方法: :一切正常完全相同的方式Quake2中加载的渲染和游戏的DDL:交换对象的指针:当Doom3.exe启动它:
LoadLibrary
。GetGameAPI
使用win32来获取dll中的地址GetProcAddress
。GetGameAPI
。
gameExport_t * GetGameAPI_t(gameImport_t * import);
在“握手”结束时,Doom3.exe具有指向idGame
对象的指针,Game.dll具有指向gameImport_t
包含对所有缺少子系统的其他引用的对象的指针,例如idFileSystem
。
Gamex86对Doom 3可执行对象的看法:
typedef struct { int 版本; // API版本 idSys * sys; //非便携式系统服务 idCommon * common; // common idCmdSystem * cmdSystem // console command system idCVarSystem * cvarSystem; //控制台变量系统 idFileSystem * fileSystem; //文件系统 idNetworkSystem * networkSystem; //网络系统 idRenderSystem * renderSystem; // render system idSoundSystem * soundSystem; // sound system idRenderModelManager * renderModelManager; //渲染模型管理器 idUserInterfaceManager * uiManager; //用户界面管理器 idDeclManager * declManager; //声明管理器 idAASFileManager * AASFileManager; // AAS文件管理器 idCollisionModelManager * collisionModelManager; //碰撞模型经理 //渲染模型管理器 idUserInterfaceManager * uiManager; //用户界面管理器 idDeclManager * declManager; //声明管理器 idAASFileManager * AASFileManager; // AAS文件管理器 idCollisionModelManager * collisionModelManager; //碰撞模型经理 //渲染模型管理器 idUserInterfaceManager * uiManager; //用户界面管理器 idDeclManager * declManager; //声明管理器 idAASFileManager * AASFileManager; // AAS文件管理器 idCollisionModelManager * collisionModelManager; //碰撞模型经理 // AAS文件管理器 idCollisionModelManager * collisionModelManager; //碰撞模型经理 // AAS文件管理器 idCollisionModelManager * collisionModelManager; //碰撞模型经理 } gameImport_t;
游戏/ Modd对象中的Doom 3的视图:
typedef结构 { int 版本; // API版本 idGame *game; //界面运行游戏 idGameEdit * gameEdit; //界面进行游戏内编辑 } gameExport_t;
注意:了解更好的每个子系统的一个很好的资源是Doom3 SDK文档页面(该页面现在需要人机验证后打开):它似乎是在2004年深入了解代码的人写的(所以可能是开发团队的成员)。
在挖掘之前,一些统计数据来自cloc
:
./cloc-1.56.pl新 2180个文本文件。 2002独特文件。 626个文件被忽略。 http://cloc.sourceforge.net v 1.56 T = 19.0 s(77.9 files / s,47576.6 lines / s) -------------------------------------------------- ----------------------------- 语言文件空白评论代码 -------------------------------------------------- ----------------------------- C ++ 517 87078 113107 366433 C / C ++标头617 29833 27176 111105 C 171 11408 15566 53540 Bourne Shell 29 5399 6516 39966 使43 1196 874 9121 m4 10 1079 232 9025 HTML 55 391 76 4142 Objective C ++ 6 709 656 2606 Perl 10 523 411 2380 yacc 1 95 97 912 Python 10 108 182 895 目标C 1 145 20 768 DOS批次5 0 0 61 Teamcenter def 4 3 0 51 Lisp 1 5 20 25 awk 1 2 1 17 -------------------------------------------------- ----------------------------- SUM:1481 137974 164934 601047 -------------------------------------------------- -----------------------------
代码行的数量通常不是一个很好的指标,但是在这里,为了评估引擎的理解力度可能非常有帮助。与Quake III相比,601,047行代码使引擎两倍“难”。关于id的历史的几个统计软件引擎#代码行:
#代码线 | 厄运 | idTech1 | idTech2 | idTech3 | idTech4 |
发动机 | 39079 | 143855 | 135788 | 239398 | 601032 |
工具 | 341 | 11155 | 28140 | 128417 | - |
总 | 39420 | 155010 | 163928 | 367815 | 601032 |
注意:对于工具来说,idTech3的巨大增长来自lcc
代码库(用于生成QVM字节码的C编译器)。
注意:由于Doom3被集成到引擎代码库中,所以没有任何工具被归结为Doom3。
从高层来看,这里有几个有趣的事实:
idMath::InvSqrt
和空间本地化优化,但大多数代码只是尝试在可用时使用这些工具(GPU着色器,OpenGL VBO,SIMD,Altivec,SMP,L2优化(R_AddModelSurfaces
每个模型处理)...) 。 查看由John Carmack定义的idTech4编码标准(镜像)也很有意思(我特别赞赏关于const
布局的评论)。
这是主要循环展开引擎最重要的部分:
idCommonLocal commonLocal; // OS专用对象 idCommon * common =&commonLocal; //接口指针(因为Init是依赖于操作系统的,它是一种抽象的方法 int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,LPSTR lpCmdLine,int nCmdShow) { Sys_SetPhysicalWorkMemory(192 << 20,1024 << 20); //最小= 201,326,592最大= 1,073,741,824 Sys_CreateConsole(); //由于引擎是多线程的互斥体在这里初始化:每个“关键”(并发执行)代码的一个互斥体。 for(int i = 0; iInitializeCriticalSection(&win32.criticalSections [i]); } common-> Init(0,NULL,lpCmdLine); //评估VRAM有多少(不是通过OpenGL完成但OS调用) Sys_StartAsyncThread(){ //下一个查找运行是一个单独的线程。 而(1){ usleep(16666); //运行在60Hz common-> Async(); //做工作 Sys_TriggerEvent(TRIGGER_EVENT_ONE); //解锁其他线程等待输入 pthread_testcancel(); //检查主线程是否被取消(在关机时)。 } } Sys_ShowConsole 而(1){ Win_Frame(); //显示或隐藏控制台 共>框架(){ session-> Frame() //游戏逻辑 { for(int i = 0; i RunGameTic(){ game-> RunFrame(&cmd); //从这一点起,执行跳转到GameX86.dll的地址空间。 for(ent = activeEntities.Next(); ent!= NULL; ent = ent-> activeNode.Next()) ent-> GetPhysics() - > UpdateTime(time); //让实体思考 } } session-> UpdateScreen(false); //正常的,按序屏幕更新 { renderSystem-> BeginFrame idGame :: Draw // Renderer前端。实际上并没有与GPU沟通! renderSystem-> EndFrame R_IssueRenderCommands //渲染器后端。将GPU优化的命令发布到GPU。 } } } }
有关详细信息,请参阅阅读代码时作为地图使用的完全展开循环。
它是id软件引擎的标准主循环。除了Sys_StartAsyncThread
表示Doom3是多线程的。此线程的目标是处理引擎不希望限制帧速率的时间关键功能:
琐事: idTech4高级对象都是具有虚拟方法的抽象类。这通常会涉及性能问题,因为每个虚拟方法地址在运行时调用之前都必须在vtable中查找。但是有一个“伎俩”来避免这种情况。所有对象都静态实例化:
idCommonLocal commonLocal; //实现 idCommon * common =&commonLocal; //指针for gamex86.dll
由于在数据段中静态分配的对象具有已知类型,编译器可以commonLocal
在调用方法时优化远程执行vtable查找。接口指针在握手期间使用,因此doom3.exe
可以交换对象引用,gamex86.dll
但在这种情况下,vtable的成本未被优化。
琐事:从id软件中读取大多数引擎,我发现一些方法名称自从doom1引擎以来就没有改变:负责抽取鼠标和操纵杆输入的方法仍然被称为:IN_frame()
。
两个重要部分:
dmap
完全脱离了传统的bsp构建器。我在一个专门的页面上深入审查了它。
我使用Xcode的仪器来检查CPU周期在哪里。结果和分析在这里。
在每个idTech产品中,VM和脚本语言从以前的版本都完全改变了,他们再次做到了: 细节在这里。
在阅读代码时,几个新奇使我感到困惑,所以我写信给约翰·卡马克,他很高兴回复深入的解释:
我还编辑了关于idTech4的所有视频和新闻采访。这些都在采访页面。
夏天来了,并不总是很容易集中
...但总的来说,这是一个大部分的阅读。由于idTech5源代码将不会很快被发布(如果有的话),这让我与idTech3(Quake III)尚未被审查。也许如果有足够的人有兴趣,我会写一些关于它的内容。