• <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>

            永遠(yuǎn)也不完美的程序

            不斷學(xué)習(xí),不斷實踐,不斷的重構(gòu)……

            常用鏈接

            統(tǒng)計

            積分與排名

            好友鏈接

            最新評論

            渲染狀態(tài)管理(轉(zhuǎn))

            來源:http://blog.csdn.net/duduliao/archive/2008/08/25/2827683.aspx
            渲染狀態(tài)管理 出 處:Maple Studio
            [ 2003-06-16 ] 作 者:maple


              目 錄
              1 基本思想
              2 實際問題
              3 渲染腳本


            --------------------------------------------------------------------------------

              提高3D圖形程序的性能是個很大的課題。圖形程序的優(yōu)化大致可以分成兩大任務(wù),一是要有好的場景管理程序,能快速剔除不可見多邊形,并根據(jù)對象距相機(jī)遠(yuǎn)近選擇合適的細(xì)節(jié)(LOD);二是要有好的渲染程序,能快速渲染送入渲染管線的可見多邊形。
              我們知道,使用OpenGL或Direct3D渲染圖形時,首先要設(shè)置渲染狀態(tài),渲染狀態(tài)用于控制渲染器的渲染行為。應(yīng)用程序可以通過改變渲染狀態(tài)來控制OpenGL或Direct3D的渲染行為。比如設(shè)置Vertex/Fragment Program、綁定紋理、打開深度測試、設(shè)置霧效等。
              改變渲染狀態(tài)對于顯卡而言是比較耗時的操作,而如果能合理管理渲染狀態(tài),避免多余的狀態(tài)切換,將明顯提升圖形程序性能。這篇文章將討論渲染狀態(tài)的管理。

            1、基本思想
              我們考慮一個典型的游戲場景,包含人、動物、植物、建筑、交通工具、武器等。稍微分析一下就會發(fā)現(xiàn),實際上場景里很多對象的渲染狀態(tài)是一樣的,比如所有的人和動物的渲染狀態(tài)一般都一樣,所有的植物渲染狀態(tài)也一樣,同樣建筑、交通工具、武器也是如此。我們可以把具有相同的渲染狀態(tài)的對象歸為一組,然后分組渲染,對每組對象只需要在渲染前設(shè)置一次渲染狀態(tài),并且還可以保存當(dāng)前的渲染狀態(tài),設(shè)置渲染狀態(tài)時只需改變和當(dāng)前狀態(tài)不一樣的狀態(tài)。這樣可以大大減少多余的狀態(tài)切換。下面的代碼段演示了這種方法:

              // 渲染狀態(tài)組鏈表,由場景管理程序填充
              RenderStateGroupList groupList;
              // 當(dāng)前渲染狀態(tài)
              RenderState curState;

              ……

              // 遍歷鏈表中的每個組
              RenderStateGroup *group = groupList.GetFirst();
              while ( group != NULL )
              {
                // 設(shè)置該組的渲染狀態(tài)
                RenderState *state = group->GetRenderState();
                state->ApplyRenderState( curState );

                // 該渲染狀態(tài)組的對象鏈表
                RenderableObjectList *objList = group->GetRenderableObjectList();
                // 遍歷對象鏈表的每個對象
                RenderableObject *obj = objList->GetFirst();
                while ( obj != NULL )
                {
                  // 渲染對象
                  obj->Render();
                  obj = objList->GetNext();
                }

                group = groupList.GetNext();
              }

              其中RenderState類的ApplyRenderState方法形如:

              void RenderState::ApplyRenderState( RenderState &curState )
              {
                // 深度測試
                if ( depthTest != curState.depthTest )
                {
                  SetDepthTest( depthTest );
                  curState.depthTest = depthTest;
                }

                // Alpha測試
                if ( alphaTest != curState.alphaTest )
                {
                  SetAlphaTest( alphaTest );
                  curState.alphaTest = alphaTest;
                }

                // 其它渲染狀態(tài)
                ……
              }

              這些分組的渲染狀態(tài)一般被稱為Material或Shader。這里Material不同于OpenGL和Direct3D里面用于光照的材質(zhì), Shader也不同于OpenGL里面的Vertex/Fragment Program和Direct3D里面的Vertex/Pixel Shader。而是指封裝了的顯卡渲染圖形需要的狀態(tài)(也包括了OpenGL和Direct3D原來的Material和Shader)。
              從字面上看,Material(材質(zhì))更側(cè)重于對象表面外觀屬性的描述,而Shader(這個詞實在不好用中文表示)則有用程序控制對象表面外觀的含義。由于顯卡可編程管線的引入,渲染狀態(tài)中包含了Vertex/Fragment Program,這些小程序可以控制物體的渲染,所以我覺得將封裝的渲染狀態(tài)稱為Shader更合適。這篇文章也將稱之為Shader。
              上面的代碼段只是簡單的演示了渲染狀態(tài)管理的基本思路,實際上渲染狀態(tài)的管理需要考慮很多問題。

            2、實際問題

              2.1 消耗時間問題
              改變渲染狀態(tài)時,不同的狀態(tài)消耗的時間并不一樣,甚至在不同條件下改變渲染狀態(tài)消耗的時間也不一樣。比如綁定紋理是一個很耗時的操作,而當(dāng)紋理已經(jīng)在顯卡的紋理緩存中時,速度就會非常快。而且隨著硬件和軟件的發(fā)展,一些很耗時的渲染狀態(tài)的消耗時間可能會有減少。因此并沒有一個準(zhǔn)確的消耗時間的數(shù)據(jù)。
              雖然消耗時間無法量化,情況不同消耗的時間也不一樣,但一般來說下面這些狀態(tài)切換是比較消耗時間的:

            Vertex/Fragment Program模式和固定管線模式的切換(FF,F(xiàn)ixed Function Pipeline)

            Vertex/Fragment Program本身程序的切換。
            改變Vertex/Fragment Program常量。
            紋理切換。
            頂點和索引緩存(Vertex & Index Buffers)切換。
              有時需要根據(jù)消耗時間的多少來做折衷,下面將會遇到這種情況。

              2.2 渲染狀態(tài)分類
              實際場景中,往往會出現(xiàn)這樣的情況,一類對象其它渲染狀態(tài)都一樣,只是紋理和頂點、索引數(shù)據(jù)不同。比如場景中的人,只是身材、長相、服裝等不同,也就是說只有紋理、頂點、索引數(shù)據(jù)不同,而其它如Vertex/Fragment Program、深度測試等渲染狀態(tài)都一樣。相反,一般不會存在紋理和頂點、索引數(shù)據(jù)相同,而其他渲染狀態(tài)不同的情況。我們可以把紋理、頂點、索引數(shù)據(jù)不歸入到Shader中,這樣場景中所有的人都可以用一個Shader來渲染,然后在這個Shader下對紋理進(jìn)行分組排序,相同紋理的人放在一起渲染。

              2.3 多道渲染(Multipass Rendering)
              有些比較復(fù)雜的圖形效果,在低檔顯卡上需要渲染多次,每次渲染一種效果,然后用GL_BLEND合成為最終效果。這種方法叫多道渲染Multipass Rendering,渲染一次就是一個pass。比如做逐像素凹凸光照,需要計算環(huán)境光、漫射光凹凸效果、高光凹凸效果,在NV20顯卡上只需要1個 pass,而在NV10顯卡上則需要3個pass。Shader應(yīng)該支持多道渲染,即一個Shader應(yīng)該分別包含每個pass的渲染狀態(tài)。
              不同的pass往往渲染狀態(tài)和紋理都不同,而頂點、索引數(shù)據(jù)是一樣的。這帶來一個問題:是以對象為單位渲染,一次渲染一個對象的所有pass,然后渲染下一個對象;還是以pass為單位渲染,第一次渲染所有對象的第一個pass,第二次渲染所有對象的第二個pass。下面的程序段演示了這兩種方式:

              2.3.1 以對象為單位渲染

              // 渲染狀態(tài)組鏈表,由場景管理程序填充
              ShaderGroupList groupList;

              ……

              // 遍歷鏈表中的每個組
              ShaderGroup *group = groupList.GetFirst();
              while ( group != NULL )
              {
                Shader *shader = group->GetShader();
                RenderableObjectList *objList = group->GetRenderableObjectList();

                // 遍歷相同Shader的每個對象
                RenderableObject *obj = objList->GetFirst();
                while ( obj != NULL )
                {
                  // 獲取shader的pass數(shù)
                  int iNumPasses = shader->GetPassNum();
                  for ( int i = 0; i < iNumPasses; i++ )
                  {
                    // 設(shè)置shader第i個pass的渲染狀態(tài)
                    shader->ApplyPass( i );
                    // 渲染對象
                    obj->Render();
                  }
                  obj = objList->GetNext();
                }
                group = groupList->GetNext();
              }

              2.3.2 以pass為單位渲染

              // 渲染狀態(tài)組鏈表,由場景管理程序填充
              ShaderGroupList groupList;

              ……

              for ( int i = 0; i < MAX_PASSES_NUM; i++ )
              {
                // 遍歷鏈表中的每個組
                ShaderGroup *group = groupList.GetFirst();
                while ( group != NULL )
                {
                  Shader *shader = group->GetShader();
                  int iNumPasses = shader->GetPassNum();

                  // 如果shader的pass數(shù)小于循環(huán)次數(shù),跳過此shader
                  if( i >= iNumPasses )
                  {
                    group = groupList->GetNext();
                    continue;
                  }

                  // 設(shè)置shader第i個pass的渲染狀態(tài)
                  shader->ApplyPass( i );
                  RenderableObjectList *objList = group->GetRenderableObjectList();

                  // 遍歷相同Shader的每個對象
                  RenderableObject *obj = objList->GetFirst();
                  while ( obj != NULL )
                  {
                    obj->Render();
                    obj = objList->GetNext();
                  }
                  group = groupList->GetNext();
                }
              }

              這兩種方式各有什么優(yōu)缺點呢?以對象為單位渲染,渲染一個對象的第一個pass后,馬上緊接著渲染這個對象的第二個pass,而每個pass的頂點和索引數(shù)據(jù)是相同的,因此第一個 pass將頂點和索引數(shù)據(jù)送入顯卡后,顯卡Cache中已經(jīng)有了這個對象頂點和索引數(shù)據(jù),后續(xù)pass不必重新將頂點和索引數(shù)據(jù)拷到顯卡,因此速度會非常快。而問題是每個pass的渲染狀態(tài)都不同,這使得實際上每次渲染都要設(shè)置新的渲染狀態(tài),會產(chǎn)生大量的多余渲染狀態(tài)切換。
              以pass為單位渲染則正好相反,以Shader分組,相同Shader的對象一起渲染,可以只在這組開始時設(shè)置一次渲染狀態(tài),相比以對象為單位,大大減少了渲染狀態(tài)切換。可是每次渲染的對象不同,因此每次都要將對象的頂點和索引數(shù)據(jù)拷貝到顯卡,會消耗不少時間。可見想減少渲染狀態(tài)切換就要頻繁拷貝頂點索引數(shù)據(jù),而想減少拷貝頂點索引數(shù)據(jù)又不得不增加渲染狀態(tài)切換。魚與熊掌不可兼得。由于硬件條件和場景數(shù)據(jù)的情況比較復(fù)雜,具體哪種方法效率較高并沒有定式,兩種方法都有人使用,具體選用那種方法需要在實際環(huán)境測試后才能知道。

              2.3 多光源問題
              待續(xù)……

              2.4 陰影問題
              待續(xù)……

            3、渲染腳本
              現(xiàn)在很多圖形程序都會自己定義一種腳本文件來描述Shader。
              比如較早的OGRE(Object-oriented Graphics Rendering Engine,面向?qū)ο髨D形渲染引擎)的Material腳本,Quake3的Shader腳本,以及剛問世不久的Direct3D的Effect File,nVIDIA的CgFX腳本(文件格式與Direct3D Effect File兼容),ATI RenderMonkey使用的xml格式的腳本。OGRE Material和Quake3 Shader這兩種腳本比較有歷史了,不支持可編程渲染管線。而后面三種比較新的腳本都支持可編程渲染管線。

            腳本 特性 范例
            OGRE Material 封裝各種渲染狀態(tài),不支持可編程渲染管線 >>>>
            Quake3 Shader 封裝渲染狀態(tài),支持一些特效,不支持可編程渲染管線 >>>>
            Direct3D Effect File 封裝渲染狀態(tài),支持multipass,支持可編程渲染管線 >>>>
            nVIDIA CgFX腳本 封裝渲染狀態(tài),支持multipass,支持可編程渲染管線 >>>>
            ATI RenderMonkey腳本 封裝渲染狀態(tài),支持multipass,支持可編程渲染管線 >>>>

              使用腳本來控制渲染有很多好處:

            可以非常方便的修改一個物體的外觀而不需重新編寫或編譯程序。
            可以用外圍工具以所見即所得的方式來創(chuàng)建、修改腳本文件(類似ATI RenderMonkey的工作方式),便于美工、關(guān)卡設(shè)計人員設(shè)定對象外觀,建立外圍工具與圖形引擎的聯(lián)系。
            可以在渲染時將相同外觀屬性及渲染狀態(tài)的對象(也就是Shader相同的對象)歸為一組,然后分組渲染,對每組對象只需要在渲染前設(shè)置一次渲染狀態(tài),大大減少了多余的狀態(tài)切換。


            本文來自CSDN博客,轉(zhuǎn)載請標(biāo)明出處:http://blog.csdn.net/duduliao/archive/2008/08/25/2827683.aspx

            posted on 2009-06-13 11:44 狂爛球 閱讀(509) 評論(0)  編輯 收藏 引用 所屬分類: 圖形編程

            久久97精品久久久久久久不卡| 国产成人综合久久精品尤物| 久久青青草视频| 久久久久av无码免费网| 伊人色综合久久天天人手人婷 | 精品人妻久久久久久888| 久久精品欧美日韩精品| 国产精品天天影视久久综合网| 久久精品国产亚洲精品2020| 精品久久久久中文字| 久久只这里是精品66| 久久国产精品99久久久久久老狼| 日韩十八禁一区二区久久| 99久久国语露脸精品国产| 一级a性色生活片久久无| 99久久国产综合精品五月天喷水| 久久福利资源国产精品999| 99久久亚洲综合精品网站| 亚洲国产精品无码久久一线 | 99久久国产热无码精品免费| 久久久久女教师免费一区| 久久99精品久久久久久久不卡| 欧美国产成人久久精品| 看全色黄大色大片免费久久久| 久久久中文字幕| 99精品久久精品一区二区| 久久夜色精品国产噜噜亚洲AV| 午夜精品久久久内射近拍高清| 国产精品伦理久久久久久| 久久91精品国产91久久麻豆| 久久精品国产亚洲AV香蕉| 偷偷做久久久久网站| 色99久久久久高潮综合影院| 久久精品这里只有精99品| 久久久久国产日韩精品网站| 国产激情久久久久影院小草| 国产成人AV综合久久| 久久精品女人天堂AV麻| 欧美亚洲另类久久综合婷婷| 中文字幕亚洲综合久久菠萝蜜| 日韩亚洲国产综合久久久|