CmMaterial.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708
  1. #include "CmMaterial.h"
  2. #include "CmException.h"
  3. #include "CmShader.h"
  4. #include "CmTechnique.h"
  5. #include "CmPass.h"
  6. #include "CmRenderSystem.h"
  7. #include "CmGpuProgramParams.h"
  8. #include "CmHardwareBufferManager.h"
  9. #include "CmGpuProgram.h"
  10. #include "CmGpuParamDesc.h"
  11. #include "CmMaterialRTTI.h"
  12. #include "CmDebug.h"
  13. namespace CamelotEngine
  14. {
  15. Material::Material()
  16. {
  17. // Material doesn't do anything render thread specific, so we can just initialize right away
  18. initialize_internal();
  19. }
  20. void Material::initialize_internal()
  21. {
  22. Resource::initialize_internal();
  23. }
  24. void Material::setShader(ShaderPtr shader)
  25. {
  26. mShader = shader;
  27. initBestTechnique();
  28. }
  29. void Material::initBestTechnique()
  30. {
  31. mBestTechnique = nullptr;
  32. mParametersPerPass.clear();
  33. if(mShader)
  34. {
  35. mBestTechnique = mShader->getBestTechnique();
  36. if(mBestTechnique == nullptr)
  37. return;
  38. mValidShareableParamBlocks.clear();
  39. mValidParams.clear();
  40. vector<const GpuParamDesc*>::type allParamDescs;
  41. // Make sure all gpu programs are fully loaded
  42. for(UINT32 i = 0; i < mBestTechnique->getNumPasses(); i++)
  43. {
  44. PassPtr curPass = mBestTechnique->getPass(i);
  45. GpuProgramHandle vertProgram = curPass->getVertexProgram();
  46. if(vertProgram)
  47. {
  48. vertProgram.waitUntilLoaded();
  49. allParamDescs.push_back(&vertProgram->getParamDesc());
  50. }
  51. GpuProgramHandle fragProgram = curPass->getFragmentProgram();
  52. if(fragProgram)
  53. {
  54. fragProgram.waitUntilLoaded();
  55. allParamDescs.push_back(&fragProgram->getParamDesc());
  56. }
  57. GpuProgramHandle geomProgram = curPass->getGeometryProgram();
  58. if(geomProgram)
  59. {
  60. geomProgram.waitUntilLoaded();
  61. allParamDescs.push_back(&geomProgram->getParamDesc());
  62. }
  63. GpuProgramHandle hullProgram = curPass->getHullProgram();
  64. if(hullProgram)
  65. {
  66. hullProgram.waitUntilLoaded();
  67. allParamDescs.push_back(&hullProgram->getParamDesc());
  68. }
  69. GpuProgramHandle domainProgram = curPass->getDomainProgram();
  70. if(domainProgram)
  71. {
  72. domainProgram.waitUntilLoaded();
  73. allParamDescs.push_back(&domainProgram->getParamDesc());
  74. }
  75. GpuProgramHandle computeProgram = curPass->getComputeProgram();
  76. if(computeProgram)
  77. {
  78. computeProgram.waitUntilLoaded();
  79. allParamDescs.push_back(&computeProgram->getParamDesc());
  80. }
  81. }
  82. // Fill out various helper structures
  83. set<String>::type validParameters = determineValidParameters(allParamDescs);
  84. set<String>::type validShareableParamBlocks = determineValidShareableParamBlocks(allParamDescs);
  85. map<String, String>::type paramToParamBlockMap = determineParameterToBlockMapping(allParamDescs);
  86. map<String, GpuParamBlockPtr>::type paramBlocks;
  87. // Create param blocks
  88. const map<String, SHADER_PARAM_BLOCK_DESC>::type& shaderDesc = mShader->getParamBlocks();
  89. for(auto iter = validShareableParamBlocks.begin(); iter != validShareableParamBlocks.end(); ++iter)
  90. {
  91. bool isShared = false;
  92. GpuParamBlockUsage usage = GPBU_STATIC;
  93. auto iterFind = shaderDesc.find(*iter);
  94. if(iterFind != shaderDesc.end())
  95. {
  96. isShared = iterFind->second.shared;
  97. usage = iterFind->second.usage;
  98. }
  99. GpuParamBlockDesc blockDesc;
  100. for(auto iter2 = allParamDescs.begin(); iter2 != allParamDescs.end(); ++iter2)
  101. {
  102. auto findParamBlockDesc = (*iter2)->paramBlocks.find(*iter);
  103. if(findParamBlockDesc != (*iter2)->paramBlocks.end())
  104. {
  105. blockDesc = findParamBlockDesc->second;
  106. break;
  107. }
  108. }
  109. GpuParamBlockPtr newParamBlockBuffer;
  110. if(!isShared)
  111. newParamBlockBuffer = HardwareBufferManager::instance().createGpuParamBlock(blockDesc, usage);
  112. paramBlocks[*iter] = newParamBlockBuffer;
  113. mValidShareableParamBlocks.insert(*iter);
  114. }
  115. // Create data param mappings
  116. const map<String, SHADER_DATA_PARAM_DESC>::type& dataParamDesc = mShader->getDataParams();
  117. for(auto iter = dataParamDesc.begin(); iter != dataParamDesc.end(); ++iter)
  118. {
  119. auto findIter = validParameters.find(iter->first);
  120. // Not valid so we skip it
  121. if(findIter == validParameters.end())
  122. continue;
  123. auto findBlockIter = paramToParamBlockMap.find(iter->first);
  124. if(findBlockIter == paramToParamBlockMap.end())
  125. CM_EXCEPT(InternalErrorException, "Parameter doesn't exist in param to param block map but exists in valid param map.");
  126. String& paramBlockName = findBlockIter->second;
  127. mValidParams.insert(iter->first);
  128. }
  129. // Create object param mappings
  130. const map<String, SHADER_OBJECT_PARAM_DESC>::type& objectParamDesc = mShader->getObjectParams();
  131. for(auto iter = objectParamDesc.begin(); iter != objectParamDesc.end(); ++iter)
  132. {
  133. auto findIter = validParameters.find(iter->first);
  134. // Not valid so we skip it
  135. if(findIter == validParameters.end())
  136. continue;
  137. mValidParams.insert(iter->first);
  138. }
  139. for(UINT32 i = 0; i < mBestTechnique->getNumPasses(); i++)
  140. {
  141. PassPtr curPass = mBestTechnique->getPass(i);
  142. PassParametersPtr params = PassParametersPtr(new PassParameters());
  143. GpuProgramHandle vertProgram = curPass->getVertexProgram();
  144. if(vertProgram)
  145. params->mVertParams = vertProgram->createParameters();
  146. GpuProgramHandle fragProgram = curPass->getFragmentProgram();
  147. if(fragProgram)
  148. params->mFragParams = fragProgram->createParameters();
  149. GpuProgramHandle geomProgram = curPass->getGeometryProgram();
  150. if(geomProgram)
  151. params->mGeomParams = geomProgram->createParameters();
  152. GpuProgramHandle hullProgram = curPass->getHullProgram();
  153. if(hullProgram)
  154. params->mHullParams = hullProgram->createParameters();
  155. GpuProgramHandle domainProgram = curPass->getDomainProgram();
  156. if(domainProgram)
  157. params->mDomainParams = domainProgram->createParameters();
  158. GpuProgramHandle computeProgram = curPass->getComputeProgram();
  159. if(computeProgram)
  160. params->mComputeParams = computeProgram->createParameters();
  161. mParametersPerPass.push_back(params);
  162. }
  163. // Assign param block buffers
  164. for(auto iter = mParametersPerPass.begin(); iter != mParametersPerPass.end(); ++iter)
  165. {
  166. PassParametersPtr params = *iter;
  167. for(UINT32 i = 0; i < params->getNumParams(); i++)
  168. {
  169. GpuParamsPtr& paramPtr = params->getParamByIdx(i);
  170. if(paramPtr)
  171. {
  172. // Assign shareable buffers
  173. for(auto iterBlock = mValidShareableParamBlocks.begin(); iterBlock != mValidShareableParamBlocks.end(); ++iterBlock)
  174. {
  175. const String& paramBlockName = *iterBlock;
  176. if(paramPtr->hasParamBlock(paramBlockName))
  177. {
  178. GpuParamBlockPtr blockBuffer = paramBlocks[paramBlockName];
  179. paramPtr->setParamBlock(paramBlockName, blockBuffer);
  180. }
  181. }
  182. // Create non-shareable ones
  183. const GpuParamDesc& desc = paramPtr->getParamDesc();
  184. for(auto iterBlockDesc = desc.paramBlocks.begin(); iterBlockDesc != desc.paramBlocks.end(); ++iterBlockDesc)
  185. {
  186. if(!iterBlockDesc->second.isShareable)
  187. {
  188. GpuParamBlockPtr newParamBlockBuffer = HardwareBufferManager::instance().createGpuParamBlock(iterBlockDesc->second);
  189. paramPtr->setParamBlock(iterBlockDesc->first, newParamBlockBuffer);
  190. }
  191. }
  192. }
  193. }
  194. }
  195. }
  196. }
  197. set<String>::type Material::determineValidParameters(const vector<const GpuParamDesc*>::type& paramDescs) const
  198. {
  199. map<String, const GpuParamDataDesc*>::type foundDataParams;
  200. map<String, const GpuParamObjectDesc*>::type foundObjectParams;
  201. set<String>::type validParameters;
  202. for(auto iter = paramDescs.begin(); iter != paramDescs.end(); ++iter)
  203. {
  204. const GpuParamDesc& curDesc = **iter;
  205. // Check regular data params
  206. for(auto iter2 = curDesc.params.begin(); iter2 != curDesc.params.end(); ++iter2)
  207. {
  208. bool isParameterValid = true;
  209. const GpuParamDataDesc& curParam = iter2->second;
  210. auto objectFindIter = foundObjectParams.find(iter2->first);
  211. if(objectFindIter != foundObjectParams.end())
  212. isParameterValid = false; // Data param but we found another as object param with the same name
  213. if(isParameterValid)
  214. {
  215. auto dataFindIter = foundDataParams.find(iter2->first);
  216. if(dataFindIter == foundDataParams.end())
  217. validParameters.insert(iter2->first);
  218. else
  219. {
  220. const GpuParamDataDesc* otherParam = dataFindIter->second;
  221. if(!areParamsEqual(curParam, *otherParam, true))
  222. isParameterValid = false;
  223. }
  224. }
  225. if(!isParameterValid)
  226. validParameters.erase(iter2->first);
  227. }
  228. // Check sampler params
  229. for(auto iter2 = curDesc.samplers.begin(); iter2 != curDesc.samplers.end(); ++iter2)
  230. {
  231. bool isParameterValid = true;
  232. const GpuParamObjectDesc& curParam = iter2->second;
  233. auto dataFindIter = foundDataParams.find(iter2->first);
  234. if(dataFindIter != foundDataParams.end())
  235. isParameterValid = false; // Object param but we found another as data param with the same name
  236. if(isParameterValid)
  237. {
  238. auto objectFindIter = foundObjectParams.find(iter2->first);
  239. if(objectFindIter == foundObjectParams.end())
  240. validParameters.insert(iter2->first);
  241. else
  242. {
  243. const GpuParamObjectDesc* otherParam = objectFindIter->second;
  244. if(!areParamsEqual(curParam, *otherParam))
  245. isParameterValid = false;
  246. }
  247. }
  248. if(!isParameterValid)
  249. validParameters.erase(iter2->first);
  250. }
  251. // Check texture params
  252. for(auto iter2 = curDesc.textures.begin(); iter2 != curDesc.textures.end(); ++iter2)
  253. {
  254. bool isParameterValid = true;
  255. const GpuParamObjectDesc& curParam = iter2->second;
  256. auto dataFindIter = foundDataParams.find(iter2->first);
  257. if(dataFindIter != foundDataParams.end())
  258. isParameterValid = false; // Object param but we found another as data param with the same name
  259. if(isParameterValid)
  260. {
  261. auto objectFindIter = foundObjectParams.find(iter2->first);
  262. if(objectFindIter == foundObjectParams.end())
  263. validParameters.insert(iter2->first);
  264. else
  265. {
  266. const GpuParamObjectDesc* otherParam = objectFindIter->second;
  267. if(!areParamsEqual(curParam, *otherParam))
  268. isParameterValid = false;
  269. }
  270. }
  271. if(!isParameterValid)
  272. validParameters.erase(iter2->first);
  273. }
  274. // Check buffer params
  275. for(auto iter2 = curDesc.buffers.begin(); iter2 != curDesc.buffers.end(); ++iter2)
  276. {
  277. bool isParameterValid = true;
  278. const GpuParamObjectDesc& curParam = iter2->second;
  279. auto dataFindIter = foundDataParams.find(iter2->first);
  280. if(dataFindIter != foundDataParams.end())
  281. isParameterValid = false; // Object param but we found another as data param with the same name
  282. if(isParameterValid)
  283. {
  284. auto objectFindIter = foundObjectParams.find(iter2->first);
  285. if(objectFindIter == foundObjectParams.end())
  286. validParameters.insert(iter2->first);
  287. else
  288. {
  289. const GpuParamObjectDesc* otherParam = objectFindIter->second;
  290. if(!areParamsEqual(curParam, *otherParam))
  291. isParameterValid = false;
  292. }
  293. }
  294. if(!isParameterValid)
  295. {
  296. auto validParamIter = validParameters.find(iter2->first);
  297. if(validParamIter != validParameters.end()) // Do this check so we only report this error once
  298. {
  299. LOGWRN("Found two parameters with the same name but different contents: " + iter2->first);
  300. validParameters.erase(validParamIter);
  301. }
  302. }
  303. }
  304. }
  305. return validParameters;
  306. }
  307. set<String>::type Material::determineValidShareableParamBlocks(const vector<const GpuParamDesc*>::type& paramDescs) const
  308. {
  309. // Make sure param blocks with the same name actually are the same
  310. map<String, std::pair<String, const GpuParamDesc*>>::type uniqueParamBlocks;
  311. set<String>::type validParamBlocks;
  312. for(auto iter = paramDescs.begin(); iter != paramDescs.end(); ++iter)
  313. {
  314. const GpuParamDesc& curDesc = **iter;
  315. for(auto blockIter = curDesc.paramBlocks.begin(); blockIter != curDesc.paramBlocks.end(); ++blockIter)
  316. {
  317. bool isBlockValid = true;
  318. const GpuParamBlockDesc& curBlock = blockIter->second;
  319. if(!curBlock.isShareable) // Non-shareable buffers are handled differently, they're allowed same names
  320. continue;
  321. auto iterFind = uniqueParamBlocks.find(blockIter->first);
  322. if(iterFind == uniqueParamBlocks.end())
  323. {
  324. uniqueParamBlocks[blockIter->first] = std::make_pair(blockIter->first, *iter);
  325. validParamBlocks.insert(blockIter->first);
  326. continue;
  327. }
  328. String otherBlockName = iterFind->second.first;
  329. const GpuParamDesc* otherDesc = iterFind->second.second;
  330. for(auto myParamIter = curDesc.params.begin(); myParamIter != curDesc.params.end(); ++myParamIter)
  331. {
  332. const GpuParamDataDesc& myParam = myParamIter->second;
  333. if(myParam.paramBlockSlot != curBlock.slot)
  334. continue; // Param is in another block, so we will check it when its time for that block
  335. auto otherParamFind = otherDesc->params.find(myParamIter->first);
  336. // Cannot find other param, blocks aren't equal
  337. if(otherParamFind == otherDesc->params.end())
  338. {
  339. isBlockValid = false;
  340. break;
  341. }
  342. const GpuParamDataDesc& otherParam = otherParamFind->second;
  343. if(!areParamsEqual(myParam, otherParam) || curBlock.name != otherBlockName)
  344. {
  345. isBlockValid = false;
  346. break;
  347. }
  348. }
  349. if(!isBlockValid)
  350. {
  351. auto blockValidIter = validParamBlocks.find(blockIter->first);
  352. if(blockValidIter != validParamBlocks.end()) // Do this check so we only report this error once
  353. {
  354. LOGWRN("Found two param blocks with the same name but different contents: " + blockIter->first);
  355. validParamBlocks.erase(blockValidIter);
  356. }
  357. }
  358. }
  359. }
  360. return validParamBlocks;
  361. }
  362. map<String, String>::type Material::determineParameterToBlockMapping(const vector<const GpuParamDesc*>::type& paramDescs)
  363. {
  364. map<String, String>::type paramToParamBlock;
  365. for(auto iter = paramDescs.begin(); iter != paramDescs.end(); ++iter)
  366. {
  367. const GpuParamDesc& curDesc = **iter;
  368. for(auto iter2 = curDesc.params.begin(); iter2 != curDesc.params.end(); ++iter2)
  369. {
  370. const GpuParamDataDesc& curParam = iter2->second;
  371. auto iterFind = paramToParamBlock.find(curParam.name);
  372. if(iterFind != paramToParamBlock.end())
  373. continue;
  374. for(auto iterBlock = curDesc.paramBlocks.begin(); iterBlock != curDesc.paramBlocks.end(); ++iterBlock)
  375. {
  376. if(iterBlock->second.slot == curParam.paramBlockSlot)
  377. {
  378. paramToParamBlock[curParam.name] = iterBlock->second.name;
  379. break;
  380. }
  381. }
  382. }
  383. }
  384. return paramToParamBlock;
  385. }
  386. bool Material::areParamsEqual(const GpuParamDataDesc& paramA, const GpuParamDataDesc& paramB, bool ignoreBufferOffsets) const
  387. {
  388. bool equal = paramA.arraySize == paramB.arraySize && paramA.elementSize == paramB.elementSize && paramA.type == paramB.type;
  389. if(!ignoreBufferOffsets)
  390. equal &= paramA.cpuMemOffset == paramB.cpuMemOffset && paramA.gpuMemOffset == paramB.gpuMemOffset;
  391. return equal;
  392. }
  393. bool Material::areParamsEqual(const GpuParamObjectDesc& paramA, const GpuParamObjectDesc& paramB) const
  394. {
  395. return paramA.type == paramB.type;
  396. }
  397. void Material::throwIfNotInitialized() const
  398. {
  399. if(mShader == nullptr)
  400. {
  401. CM_EXCEPT(InternalErrorException, "Material does not have shader set.");
  402. }
  403. if(mBestTechnique == nullptr)
  404. {
  405. CM_EXCEPT(InternalErrorException, "Shader does not contain a supported technique.");
  406. }
  407. }
  408. void Material::setTexture(const String& name, TextureHandle& value)
  409. {
  410. throwIfNotInitialized();
  411. auto iterFind = mValidParams.find(name);
  412. if(iterFind == mValidParams.end())
  413. {
  414. LOGWRN("Material doesn't have a parameter named " + name);
  415. return;
  416. }
  417. for(auto iter = mParametersPerPass.begin(); iter != mParametersPerPass.end(); ++iter)
  418. {
  419. PassParametersPtr params = *iter;
  420. for(UINT32 i = 0; i < params->getNumParams(); i++)
  421. {
  422. GpuParamsPtr& paramPtr = params->getParamByIdx(i);
  423. if(paramPtr)
  424. {
  425. if(paramPtr->hasTexture(name))
  426. paramPtr->setTexture(name, value);
  427. }
  428. }
  429. }
  430. }
  431. void Material::setSamplerState(const String& name, SamplerStatePtr samplerState)
  432. {
  433. throwIfNotInitialized();
  434. auto iterFind = mValidParams.find(name);
  435. if(iterFind == mValidParams.end())
  436. {
  437. LOGWRN("Material doesn't have a parameter named " + name);
  438. return;
  439. }
  440. for(auto iter = mParametersPerPass.begin(); iter != mParametersPerPass.end(); ++iter)
  441. {
  442. PassParametersPtr params = *iter;
  443. for(UINT32 i = 0; i < params->getNumParams(); i++)
  444. {
  445. GpuParamsPtr& paramPtr = params->getParamByIdx(i);
  446. if(paramPtr)
  447. {
  448. if(paramPtr->hasSamplerState(name))
  449. paramPtr->setSamplerState(name, samplerState);
  450. }
  451. }
  452. }
  453. }
  454. void Material::setFloat(const String& name, float value)
  455. {
  456. throwIfNotInitialized();
  457. auto iterFind = mValidParams.find(name);
  458. if(iterFind == mValidParams.end())
  459. {
  460. LOGWRN("Material doesn't have a parameter named " + name);
  461. return;
  462. }
  463. setParam(name, value);
  464. }
  465. void Material::setColor(const String& name, const Color& value)
  466. {
  467. throwIfNotInitialized();
  468. auto iterFind = mValidParams.find(name);
  469. if(iterFind == mValidParams.end())
  470. {
  471. LOGWRN("Material doesn't have a parameter named " + name);
  472. return;
  473. }
  474. setParam(name, value);
  475. }
  476. void Material::setVec2(const String& name, const Vector2& value)
  477. {
  478. throwIfNotInitialized();
  479. auto iterFind = mValidParams.find(name);
  480. if(iterFind == mValidParams.end())
  481. {
  482. LOGWRN("Material doesn't have a parameter named " + name);
  483. return;
  484. }
  485. setParam(name, value);
  486. }
  487. void Material::setVec3(const String& name, const Vector3& value)
  488. {
  489. throwIfNotInitialized();
  490. auto iterFind = mValidParams.find(name);
  491. if(iterFind == mValidParams.end())
  492. {
  493. LOGWRN("Material doesn't have a parameter named " + name);
  494. return;
  495. }
  496. setParam(name, value);
  497. }
  498. void Material::setVec4(const String& name, const Vector4& value)
  499. {
  500. throwIfNotInitialized();
  501. auto iterFind = mValidParams.find(name);
  502. if(iterFind == mValidParams.end())
  503. {
  504. LOGWRN("Material doesn't have a parameter named " + name);
  505. return;
  506. }
  507. setParam(name, value);
  508. }
  509. void Material::setMat3(const String& name, const Matrix3& value)
  510. {
  511. throwIfNotInitialized();
  512. auto iterFind = mValidParams.find(name);
  513. if(iterFind == mValidParams.end())
  514. {
  515. LOGWRN("Material doesn't have a parameter named " + name);
  516. return;
  517. }
  518. setParam(name, value);
  519. }
  520. void Material::setMat4(const String& name, const Matrix4& value)
  521. {
  522. throwIfNotInitialized();
  523. auto iterFind = mValidParams.find(name);
  524. if(iterFind == mValidParams.end())
  525. {
  526. LOGWRN("Material doesn't have a parameter named " + name);
  527. return;
  528. }
  529. setParam(name, value);
  530. }
  531. void Material::setParamBlock(const String& name, GpuParamBlockPtr paramBlock)
  532. {
  533. auto iterFind = mValidShareableParamBlocks.find(name);
  534. if(iterFind == mValidShareableParamBlocks.end())
  535. {
  536. LOGWRN("Material doesn't have a parameter block named " + name);
  537. return;
  538. }
  539. for(auto iter = mParametersPerPass.begin(); iter != mParametersPerPass.end(); ++iter)
  540. {
  541. PassParametersPtr params = *iter;
  542. for(UINT32 i = 0; i < params->getNumParams(); i++)
  543. {
  544. GpuParamsPtr& paramPtr = params->getParamByIdx(i);
  545. if(paramPtr)
  546. {
  547. if(paramPtr->hasParamBlock(name))
  548. paramPtr->setParam(name, paramBlock);
  549. }
  550. }
  551. }
  552. }
  553. UINT32 Material::getNumPasses() const
  554. {
  555. throwIfNotInitialized();
  556. return mShader->getBestTechnique()->getNumPasses();
  557. }
  558. PassPtr Material::getPass(UINT32 passIdx) const
  559. {
  560. if(passIdx < 0 || passIdx >= mShader->getBestTechnique()->getNumPasses())
  561. CM_EXCEPT(InvalidParametersException, "Invalid pass index.");
  562. return mShader->getBestTechnique()->getPass(passIdx);
  563. }
  564. PassParametersPtr Material::getPassParameters(UINT32 passIdx) const
  565. {
  566. if(passIdx < 0 || passIdx >= mParametersPerPass.size())
  567. CM_EXCEPT(InvalidParametersException, "Invalid pass index.");
  568. return mParametersPerPass[passIdx];
  569. }
  570. RTTITypeBase* Material::getRTTIStatic()
  571. {
  572. return MaterialRTTI::instance();
  573. }
  574. RTTITypeBase* Material::getRTTI() const
  575. {
  576. return Material::getRTTIStatic();
  577. }
  578. }