• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            posts - 94, comments - 250, trackbacks - 0, articles - 0
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

            GPU程序在GameByro中的使用

            Posted on 2009-04-22 11:18 Condor 閱讀(1872) 評論(0)  編輯 收藏 引用

            引言:

            GameBryo擁有一套復雜的材質系統,這套材質系統可以根據渲染對象的狀態和屬性生成不同的shader代碼,提高了渲染流程的適應性,可以使你定義一套材質能適應多種渲染對象。同時,GameByro將shader的初始化和使用插件化,方便與美術工具集成,并且實現了平臺無關性。為了實現這些目的,GameByro使用了一套復雜的機制,本文主要解析GameByro如何生成、編譯并使用shader代碼。

            Shader

            GameBryo的shader的接口封裝在NiShader中,頂點數據流聲明,常量表的訪問,渲染狀態的設置都是通過這個類(有點類似于D3Deffect)。在程序運行NiShader是由NiShaderFactory負責管理的,NiShaderFactory通過NiShaderLibrary從文件中創建shader,用全局性的map管理起來。NiShaderLibrary通過解析shader文本創建NiShader對象,并調用3D圖形接口編譯shader代碼,將這個類以dll的形式封裝,就可以作為插件來使用。NiShader類的創建可以通過解析文件來進行,也可以通過C++的類來定制,只需從NiShader上繼承即可。GameByro為PC平臺提供了一個NiD3DXEffectShaderLib庫,這個庫提供了解析shader文件和初始化shader對象的功能。用戶只需按GameByro定義的格式編寫shader代碼的語意和注釋,NiD3DXEffectShaderLibrary就會根據文本來創建NiD3Dshader對象,在應用程序中就可以通過Techinqe的名稱來訪問這個對象。通過這種機制,我們將shader文本文件放在相關美術工具指定的目錄下,在工具中就可以使用這些shader,并且能夠通過shader的語意和注釋為相關參數和變量生成UI,方便美術調試。

            WIN平臺上的整個流程如下:

            1. 應用程序在啟動時會先初始化整個shader系統,接下來導入Shader解析庫和加載庫(dll的形式)。

            2. 接下來應用程序將NiD3DShader的初始化工作委托給NiShaderLibrary來處理,NiShaderLibrary首先通過NiD3DXEffectLoader載入所有的shader文本文件,并通過NiD3DXEffectParser解析文本生成NiD3DXEffectFile對象,同時NiD3DXEffectLoader還負責將shader代碼編譯成二進制形式的GPU程序。

            3. 最后由NiD3DXEffectTechnique負責通過NiD3DXEffectFile上的信息生成NiD3Dshader對象。

            4. 所有的shader對象創建后,NiShaderLibrary的初始化就結束了,最后由NiShaderFactory負責統一管理。

            材質:

            NiMaterial為渲染對象生成和定義Shader,NiMaterialInstance為渲染對象分配 和Cach Shader。NiFragmentMaterial提供了一個Shader Tree框架,在它的繼承類中可以使用這個框架搭建shader tree。這個機制允許NiFragmentMaterial根據對象不同的渲染狀態生成不同的shader代碼,Cach在內存中,并保存到磁盤文件。GameByro描述符的概念大量使用,包括前面提到的Shader解析過程也是通過描述符來傳遞信息。在材質系統中主要使用了NiMaterialDescriptor和NiGPUProgramDescriptor這個兩個類做描述符,這兩個類中保存的信息是兼容的,都是為了描述某種材質在渲染對象的某一特定渲染狀態下所對應的GPU程序的特征。NiFragmentMaterial通過渲染目標的狀態和屬性生成NiMaterialDescriptor,并通過NiMaterialDescriptor查找匹配的shader,如果找不到,則通過shader tree生成相應的shader程序,并保存到磁盤文件中。當下一次應用程序啟動時就可以通過這個文件直接創建NiShader對象??梢哉f通過NiFragmentMaterial生成的shader代碼是為特定的渲染對象在特定的情況下量身打造的。

            整個過程的詳細流程如下:

            1. 在每次渲染一個物體之前,NiMaterialInstance會先判斷這個物體的shader程序是否需要更新,如果不需要更新,就直接返回當前Cach的NiShader;如果需要更新, NiMaterialInstance首先會根據物體的渲染狀態為其生成一個NiMaterialDescriptor,然后將這個NiMaterialDescriptor和當前Cach住的NiShader進行比較,如果匹配仍然返回當前Cach的NiShader,如果不匹配,將獲得shader的工作轉交給NiMaterial進行。

            2. NiMaterial首先通過這個NiShaderFactory 查詢匹配這個NiMaterialDescriptor的NiShader,如果找不到,就通過NiMaterialDescriptor生成NiShader,同時生成一段Shader代碼,并保存到以shader描述符中的特征碼來命名對應的shader文件。

            3. 當獲得相應的NiShader對象后,NiMaterialInstance會調用NiShader的SetupGeometry接口,在這個接口中會進行頂點聲明。

            以下是NiMaterialInstance為Geometry選擇shader的代碼:

            NiShader* NiMaterialInstance::GetCurrentShader(NiRenderObject* pkGeometry,

            const NiPropertyState* pkState,

            const NiDynamicEffectState* pkEffects)

            {

            if (m_spMaterial)

            {

            bool bGetNewShader = m_eNeedsUpdate == DIRTY;

            if (m_eNeedsUpdate == UNKNOWN)

            bGetNewShader = pkGeometry->GetMaterialNeedsUpdateDefault();

            // Check if shader is still current

            if (bGetNewShader && m_spCachedShader)

            {

            bGetNewShader = !m_spMaterial->IsShaderCurrent(m_spCachedShader,

            pkGeometry, pkState, pkEffects, m_uiMaterialExtraData);

            }

            // Get a new shader

            if (bGetNewShader)

            {

            NiShader* pkNewShader = m_spMaterial->GetCurrentShader(

            pkGeometry, pkState, pkEffects, m_uiMaterialExtraData);

            if (pkNewShader)

            {

            NIASSERT(m_spCachedShader != pkNewShader);

            ClearCachedShader();

            m_spCachedShader = pkNewShader;

            if (!pkNewShader->SetupGeometry(pkGeometry, this))

            ClearCachedShader();

            }

            else

            {

            ClearCachedShader();

            }

            }

            m_eNeedsUpdate = UNKNOWN;

            }

            return m_spCachedShader;

            }

            如果想通過NiFragmentMaterial實現自己的shader tree就需要在NiFragmentMaterial提供的接口中實現自己拼裝代碼的邏輯,代碼塊由NiMaterialLibraryNode封裝,NiMaterialLibraryNode既可以直接寫C++代碼來定義,也可以先寫成XML腳本,再由專門的解析工具轉換成C++代碼。

            由NiStandardMaterial生成的shader代碼文件如下圖所示:

            clip_image002

            文件名就是NiMaterialDescriptor的掩碼,用來標識的shader代碼的行為。

            Shader代碼的行為描述如下:

            Shader description:

            APPLYMODE = 1

            WORLDPOSITION = 0

            WORLDNORMAL = 0

            WORLDNBT = 0

            WORLDVIEW = 0

            NORMALMAPTYPE = 0

            PARALLAXMAPCOUNT = 0

            BASEMAPCOUNT = 1

            NORMALMAPCOUNT = 0

            DARKMAPCOUNT = 0

            DETAILMAPCOUNT = 0

            BUMPMAPCOUNT = 0

            GLOSSMAPCOUNT = 0

            GLOWMAPCOUNT = 0

            CUSTOMMAP00COUNT = 0

            CUSTOMMAP01COUNT = 0

            CUSTOMMAP02COUNT = 0

            CUSTOMMAP03COUNT = 0

            CUSTOMMAP04COUNT = 0

            DECALMAPCOUNT = 0

            FOGENABLED = 0

            ENVMAPTYPE = 0

            PROJLIGHTMAPCOUNT = 0

            PROJLIGHTMAPTYPES = 0

            PROJLIGHTMAPCLIPPED = 0

            PROJSHADOWMAPCOUNT = 0

            PROJSHADOWMAPTYPES = 0

            PROJSHADOWMAPCLIPPED = 0

            PERVERTEXLIGHTING = 1

            UVSETFORMAP00 = 0

            UVSETFORMAP01 = 0

            UVSETFORMAP02 = 0

            UVSETFORMAP03 = 0

            UVSETFORMAP04 = 0

            UVSETFORMAP05 = 0

            UVSETFORMAP06 = 0

            UVSETFORMAP07 = 0

            UVSETFORMAP08 = 0

            UVSETFORMAP09 = 0

            UVSETFORMAP10 = 0

            UVSETFORMAP11 = 0

            POINTLIGHTCOUNT = 0

            SPOTLIGHTCOUNT = 0

            DIRLIGHTCOUNT = 0

            SHADOWMAPFORLIGHT = 0

            SPECULAR = 1

            AMBDIFFEMISSIVE = 0

            LIGHTINGMODE = 1

            APPLYAMBIENT = 0

            BASEMAPALPHAONLY = 0

            APPLYEMISSIVE = 0

            SHADOWTECHNIQUE = 0

            ALPHATEST = 0

            NiStanderMaterial就是根據這些掩碼的數據來生成shader代碼,用戶可以通過重載GenerateVertexShadeTree、GeneratePixelShadeTree、CreateShader這些接口來定義自己的shader生成規則。

            增加自己的渲染效果:

            通過前幾節我們可以了解到,想定義自己的材質,一是通過編寫shader代碼完成。在應用程序初始化的時候,這些shader代碼會被初始化成NiShader對象,進一步的通過NiShader對象來初始化NiSingleShaderMaterial對象,并分配給渲染對象。在GameByro默認的渲染流程中,這些步驟都是自動進行的,美術只需在3DMAX插件中為幾何體的材質指定Shader程序,導出到nif文件,應用程序就能正確加載并渲染;二是定義自己的NiMaterialFragment類,在類中定義如何生成shader,在應用程序運行時只要將這個類的實例指派給幾何體,這個類就會自動為幾何體生成shader。這兩種方式對于美術人員來說,主要區別在于,采用第一種方法定義的材質,其渲染數據的設置必須嚴格符合shader代碼中所需的數據,否則就會報錯。(比如說,頂點數據流必須嚴格符合shader程序的定義,必須為shader中每個采樣器提供格式正確的紋理);而采用第二種方法定義的材質,就有很高的容錯和適應性,但是這種容錯性和適應性需要自己寫代碼來完成,GameByro提供的NiStanderMaterial就提供了這套完整的機制。每個貼圖槽內的貼圖如果你設置就會生成相應的貼圖處理流程,如果不設置,就沒有這張貼圖的處理流程。

            為了驗證這個過程,筆者嘗試增加了一個自己的shader特效——SubSurfaceScattering,簡稱3s,其原理是模擬光在半透明物體中散射的效果。由于該效果無須預處理過程,所有的貼圖均來自磁盤文件,所以比較容易融合到GameByro工作流中。

            筆者將在FX COMPOSER中調試通過的fx文件放入SDK中的SDK\Win32\Shaders\Data目錄下,在3DMAX的材質面板選擇GameByroShader,然后就可在顯示shader的組合框中看到文件中定義的Techinqe,選擇點擊apply按鈕,就會出現自定義的參數調整界面。

            clip_image004

            clip_image006

            通過調整參數,最終得到皮膚和玉器的渲染效果如下:

            clip_image007

            皮膚

            clip_image009

            玉器

            總結

            GameByro的這套開發流程非常方便直觀,但是美術僅能為shader程序分配靜態的數據源,比如說光照圖等,CubeMap等;而一些在程序中實時生成的紋理數據則無法整合到美術工具中,比如說陰影圖、折射圖、反射圖等,這些都需要程序寫代碼來實現。調試起來就不大方便了。大部分情況下,我們只需要使用GameByro提供的NiStanderMaterial就可以完成大部分材質的需求,特殊的效果可以自己寫shader或者通過引擎提供的shader庫來完成,只有當我們需要即根據復雜的情況做很多不同的處理時,我們才需要重載NiFragmentMaterial搭建自己的shader tree。不過搭建shader tree的程序一般比較復雜,編寫難度大,雖然引擎允許通過XML文件來編寫材質節點,但是使用起來仍然不方便。GameByro并沒有提供相關的后期處理的開發工具,后期處理的特效并不能所見即所得,這方面還需完善。

            GameByro為幾何體在特定的環境下生成專用的shader代碼,具有一定的靈活性,但是也付出了以下代價:

            l 分析幾何體的屬性和當前狀態,為其生成shader代碼的過程有性能損耗。

            l Shader代碼生成后會保存到磁盤文件中,這個過程如果不使用異步,可能會引起阻塞。

            l 生成的NiShader對象會有內存消耗。由于GameByro默認的實現是將所有的shader文件初始化成NiShader對象,所以當游戲運行的時間久了以后會生成大量的shader文件,這時候內存的消耗可能會很可觀,同時加載的時間也會增加。不過可以自己控制加載的流程,在這里進行性能優化。

            作者:葉起漣漪

            亚洲伊人久久综合中文成人网| 亚洲一区中文字幕久久| 久久人人超碰精品CAOPOREN| 欧美久久天天综合香蕉伊| 丁香色欲久久久久久综合网| 精品综合久久久久久888蜜芽| 天天爽天天爽天天片a久久网| 国产综合成人久久大片91| 四虎国产精品成人免费久久| 久久国产精品99精品国产987| 亚洲精品乱码久久久久久蜜桃| 97热久久免费频精品99| 亚洲欧美国产精品专区久久| 国产精品美女久久久久久2018| 色婷婷噜噜久久国产精品12p| 久久综合给合久久国产免费 | 午夜肉伦伦影院久久精品免费看国产一区二区三区 | 久久久久亚洲av无码专区喷水| 91超碰碰碰碰久久久久久综合| 久久免费看黄a级毛片| 成人亚洲欧美久久久久| 韩国免费A级毛片久久| 99精品久久久久久久婷婷| 国产成人综合久久久久久 | 久久久91精品国产一区二区三区 | 国内精品久久久久久麻豆| 久久天天躁狠狠躁夜夜2020一| 国产亚州精品女人久久久久久 | 久久久久国色AV免费看图片| 久久精品国产精品青草app| 中文字幕乱码久久午夜| 久久精品国产亚洲AV不卡| 伊人久久国产免费观看视频| 久久精品国产亚洲AV不卡| 国产精品成人精品久久久| 国产精品热久久无码av| 国产精品美女久久久免费| 国产激情久久久久影院老熟女免费| a高清免费毛片久久| 18岁日韩内射颜射午夜久久成人 | 久久久无码精品午夜|