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

            月下的博客

              C++博客 :: 首頁(yè) :: 聯(lián)系 :: 聚合  :: 管理
              34 Posts :: 0 Stories :: 59 Comments :: 0 Trackbacks

            常用鏈接

            留言簿(5)

            我參與的團(tuán)隊(duì)

            搜索

            •  

            最新評(píng)論

            • 1.?re: 五年感想
            • 5年一轉(zhuǎn)眼,我已經(jīng)進(jìn)入工作的第9個(gè)年頭了,缺少的是思考,一直安于現(xiàn)狀,也該反思了
            • --liquanhai
            • 2.?re: 五年感想
            • 認(rèn)為是設(shè)計(jì)-》現(xiàn)在轉(zhuǎn)變?yōu)槿?!
            • --linda
            • 3.?re: 五年感想
            • 決定游戲成功的因素
            • --linda
            • 4.?re: 五年感想
            • 果真人會(huì)沉淀是真的
            • --張恒
            • 5.?re: 五年感想
            • 樓主 最近股市大漲
              希望你能再接再厲
              再創(chuàng)輝煌
            • --ccsdu2009

            閱讀排行榜

            評(píng)論排行榜

             

            自從以前看了clayman博客后,就很想把自己代碼里dxeffect框架替代掉。如今實(shí)習(xí)結(jié)束正好有短暫的自由時(shí)間,所以就ogrematerial開刀吧。

              山寨代碼也是有學(xué)問的,如果你想搞明白這些代碼的含義的話,最后之前就對(duì)這些已經(jīng)有了一定的使用經(jīng)歷和認(rèn)知,第一步先得自己想一下material系統(tǒng)的組成:
            material
            應(yīng)該包含的:1Material->n Techinque->(n Pass + extra params)->(n TextureUnit + m or 0 shaderUnit + n RenderStates(部分)),Pass里的TextureUnit又包含了這一組texture的相關(guān)信息和設(shè)置,TexcoordSet, AddressMode, FilterSetting等等, ShaderUnit則是對(duì)應(yīng)于vsfs以及新的gs,普通點(diǎn)也就最多兩組。shaderUnit包含了具體的shader和其shaderParams,為何要把shadershaderParams分開呢,不說設(shè)計(jì)優(yōu)雅的考慮,簡(jiǎn)單的說就是方便,很多時(shí)候params可能會(huì)需要共享,譬如shaderA里用到mvpMat,和eyePos diffuse,specular,可能shaderB也只用到這些,但是兩個(gè)shader的內(nèi)部計(jì)算并不相同。到此我們只分析了作為程序中的材質(zhì)的結(jié)構(gòu),還需要考慮如果編寫一個(gè)類似fx格式的material script,這也是另外一個(gè)大頭,簡(jiǎn)單的說就是要做個(gè)腳本解釋,然后給shader參數(shù)加上語義,減少大部分需要手動(dòng)更新的shaderParams。鑒于我如今還沒把這塊扒完,等下次再寫這塊吧。


               當(dāng)你至少能閉著眼睛想透這些了,那就可以開始看ogrematerial system了,先說句題外話,我讀ogre的代碼其實(shí)讀挺久了。一直覺得很難讀,開始覺得是自己水平不夠,但后來實(shí)習(xí)才發(fā)現(xiàn)很多人都有這想法,甚至wolfgang對(duì)此也有批評(píng),而自己曾經(jīng)寫了個(gè)olong引擎,接觸過ipad編程的人應(yīng)該對(duì)這些代碼都比較熟悉~ogre里用了大量的設(shè)計(jì)模式,使得整個(gè)渲染流程從不同類里跳來跳去(SceneMgr這塊看著最累了。。其實(shí)有很多地方看著都很累。。)個(gè)人覺得應(yīng)該把renderSceneMgr里抽離出來做個(gè)RenderMgr,(記住這里的RenderMgr的概念和ogreRenderSystem,后者是提供底層渲染API的接口,而前者是管理renderData并最終遞交給RenderSystem,以前也見有別人博客講過這個(gè)問題)讓SceneMgr專門管理Scene。。
             
            打住打住。。我是來寫material的。。咳咳。。相當(dāng)于渲染來說,ogrematerial system還是相當(dāng)易讀的,也是因?yàn)檫@些模塊相互關(guān)系幾乎都是單向的,我們開始編寫的時(shí)候可以先不考慮Technique,就是直接1 Material->n Pass->...當(dāng)然甚至你都可以把multi pass去掉,就等同于1個(gè)pass,當(dāng)然由于pass都代碼實(shí)際也沒多少,我們還是加上好了。

            原始渲染的偽代碼就是:

             1for each pass
             2{
             3  setRenderStates(like alphe belnd, depth..)
             4  for each texture unit
             5    setTexSettings();
             6  if(hasVertexShader())
             7  {
             8    bindVertexShader();
             9    setShaderParams(vs);
            10  }

            11  if(hasFragmentShader())
            12  {
            13    bindFragmentShader();
            14    setShaderParams(fs);
            15  }

            16  drawPrimitives();
            17}

            加入的RenderQueue也是一樣,只是pass和對(duì)應(yīng)的renderData我們是從renderqueue里取。

            看到這里你會(huì)覺得,嘿,很簡(jiǎn)單嘛,但如果你看了ogre里關(guān)于shader參數(shù)的細(xì)節(jié),你就知道這塊還是比較復(fù)雜的。我們剛只說了一個(gè)運(yùn)行過程,而建立params到constant registers的關(guān)系,以及如何編寫autoConstants的代碼都是魔鬼的細(xì)節(jié)~~對(duì)于dx,我們可以簡(jiǎn)單點(diǎn),直接使用ID3DXConstantTable,它提供了set接口。但ogre則是直接從ID3DXConstantTable中解析參數(shù)。

               設(shè)計(jì)這塊的時(shí)候,腦子里先要對(duì)我整個(gè)過程有個(gè)清晰的認(rèn)識(shí):一方面是從shader里(我們這里假設(shè)是hlsl,asm先不考慮),另一方面則是解析參數(shù)讀入材質(zhì)腳本里定義的參數(shù)(named,autoNamed,或者index),最后將兩者對(duì)應(yīng)上。

            0.類型解釋
            看過OgreGpuProgramParams.cpp/h的人都知道:GpuConstantDefinition,GpuNamedConstants,GpuLogicalIndexUse,GpuSharedParameters,GpuSharedParametersUsage,一堆的struct和class。。其實(shí)按照我們上面這個(gè)思路逐步加斷點(diǎn)分析,代碼也沒那么繁瑣。暫時(shí)不看GpuSharedParameters這塊,大概看下GpuConstantDefinition,重點(diǎn)變成員是physicalIndex和logicalIndex,前者的解釋是buffer中的起始地址,每個(gè)GpuProgramParameters里都包含了vector<int>和vector<float>兩個(gè)buffer,他們存儲(chǔ)了shader變量的值,而physicalIndex就是對(duì)應(yīng)vector的索引。而logicalIndex則是我們后面將hlsl編譯成asm后這些變量所綁定的寄存器id,而由于constant reg的size都是4,即一個(gè)register為4個(gè)float的大小,所以這些logicalIndex則不一定連續(xù),譬如第一個(gè)uniform為float1那么第二個(gè)uniform的logicalIndex雖然是1,但實(shí)際跨過了4個(gè)float,而實(shí)際內(nèi)存中我們的float(int同) buffer里,兩個(gè)變量之間的則就空了3個(gè)float,這就是physicalIndex的作用,所以我們還需要建立一個(gè)從logicalIndex到physicalIndex的map,也就是兩個(gè)GpuLogicalBuffersStructPtr對(duì)象(這兩個(gè)對(duì)象和GpuNamedConstantsPtr在GpuProgram類和GpuProgramParameters是共享的,都是sharedPtr)

            1.解析shader的變量
              
            我們從再具體的讀取代碼來看,假設(shè)我們已經(jīng)從腳本里讀入了對(duì)應(yīng)的shader,經(jīng)過一堆調(diào)用后(略。。)在D3D9HLSLProgram的函數(shù)buildConstantDefinitions->processParamElement將每個(gè)參數(shù)的registerIndex,physicalIndex, type(原子類型,即float, int,),size都寫入GpuLogicalBuffersStructPtrGpuNamedConstantsPtr對(duì)象里(后者當(dāng)然是只有highLevelShader里才有),并會(huì)為這個(gè)定義同樣在插入帶下標(biāo)的一對(duì)鍵值,以便于如果我們傳入的參數(shù)是數(shù)組(譬如float4x4[5]),則在程序里可以通過數(shù)組下標(biāo)訪問對(duì)應(yīng)的數(shù)組變量。再constantDef建立好之后,將GpuNamedConstantsPtr傳入到GpuProgramParameters對(duì)象里,然后向其中的兩個(gè)vector添加上述對(duì)象大小的空間,在將兩個(gè)LogicalToPhysicalMap的指針也傳到shaderParams里,這樣整個(gè)GpuProgramParameters幾乎就完工了。

            2.解析材質(zhì)里定義的params,并將其與前者對(duì)應(yīng)上
              如今就差再把material script定義的param解析出來并與之前shader里的變量進(jìn)行對(duì)應(yīng)。材質(zhì)腳本讀出的每個(gè)變量的信息包含了param類型(namedauto_namedindexauto_index),name,type(float,int..),autoType(auto的才有),初始值等。我們根據(jù)其autoType在全局變量AutoConstantDictionary中查找是否有定義,然后調(diào)用set[Named]AutoConstant,在這里我們?cè)跁?huì)在我們之前從hlsl里解析出的GpuNamedConstantsPtr里查找是否有該名字的變量(畢竟shader里的變量才是最終有用的,甚至由于編譯器優(yōu)化的關(guān)系,由于shader編寫者可能定義了一個(gè)變量而未實(shí)際使用,在編譯shader后,這個(gè)變量實(shí)際是會(huì)被省略的,這樣在GpuNamedConstantsPtr也就找不到了)然后調(diào)用_setRawAutoConstant將這個(gè)變量加入到這個(gè)GpuProgramParameters對(duì)象所持有的autoConstant中。

            (ps:關(guān)于_getFloatConstantLogicalIndexUse的用途我沒看懂,為何需要去在mFloatLogicalToPhysical里去查找這個(gè)logicalIndex是否存在呢,我覺得如果走到這一步肯定shader里肯定就是定義了的,因?yàn)檫@個(gè)logicalIndex是由mNamedConstants里取出,我覺得是沒必要的。。)

            3,更新autoParamsbuffer并最終在DP前進(jìn)行更新

            把上面的過程看懂了,下面就很好理解了。GpuProgramParameters::_updateAutoParams里就是對(duì)GpuProgramParameters對(duì)象所持有的autoConstant值進(jìn)行更新,當(dāng)然這是我們只是對(duì)GpuProgramParameters持有的buffer里的值做更新,最終在文章開頭的偽代碼中的setShaderParamsogre里叫bindGpuProgramParamters而這里的更新就輕而易舉了,我們遍歷logicalToPhysicalMap對(duì)每個(gè)param取出對(duì)應(yīng)的logicalIndexdataPointervector4ofCount,然后調(diào)用SetVertex/PixelShaderConstantF/I即可。



            呼。。終于把這塊寫完了。。上述只是一個(gè)實(shí)現(xiàn)自己的material的基本思路。關(guān)于優(yōu)化的話,我開頭推薦的clayman的文章里有很多敘述。關(guān)于怎么寫ogre script translator,等下一篇再說。。

            posted on 2011-04-18 23:32 月下圓舞曲 閱讀(4110) 評(píng)論(1)  編輯 收藏 引用 所屬分類: 開發(fā)

            Feedback

            # re: 沿著ogre來實(shí)現(xiàn)material system(一) 2012-08-20 09:09 sxx
            09:03:57: OGRE EXCEPTION(2:InvalidParametersException): Parameter called worldVeiwMatrix does not exist. in GpuProgramParameters::_findNamedConstantDefinition at ..\..\..\..\OgreMain\src\OgreGpuProgramParams.cpp (line 1435)  回復(fù)  更多評(píng)論
              

            亚洲国产一成久久精品国产成人综合 | 久久婷婷五月综合国产尤物app | 蜜臀av性久久久久蜜臀aⅴ麻豆| 久久国产乱子伦精品免费强| 亚洲午夜精品久久久久久app| 久久99国产亚洲高清观看首页| 女人高潮久久久叫人喷水| 久久这里只有精品久久| 77777亚洲午夜久久多人| 日本精品久久久久久久久免费| www.久久热.com| 久久综合给合久久狠狠狠97色 | 久久国产乱子精品免费女| 国产毛片欧美毛片久久久| 久久综合九色欧美综合狠狠| 久久精品国产99国产精品澳门| 亚洲色大成网站WWW久久九九| 日本高清无卡码一区二区久久| 91久久精品国产91性色也| 久久福利青草精品资源站免费| 久久久久无码精品国产不卡| 亚洲国产精品无码久久久久久曰| 久久er国产精品免费观看8| 久久久青草青青亚洲国产免观| 精品久久久久久无码专区 | 亚洲а∨天堂久久精品9966| 久久久久久国产精品无码下载| 国产精品青草久久久久福利99| 久久精品国产精品国产精品污| 久久精品9988| 久久精品国产72国产精福利| 久久久精品视频免费观看 | 国产日产久久高清欧美一区| 99久久er这里只有精品18| 九九久久自然熟的香蕉图片| 91精品国产综合久久精品| 国产亚洲婷婷香蕉久久精品| 久久97久久97精品免视看| 久久精品综合一区二区三区| 久久人人爽人人爽人人av东京热| 亚洲午夜久久久久妓女影院|