2
0

ibl.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539
  1. /*
  2. * Copyright 2014 Dario Manesku. All rights reserved.
  3. * License: http://www.opensource.org/licenses/BSD-2-Clause
  4. */
  5. #include <vector>
  6. #include <string>
  7. #include "common.h"
  8. #include "bgfx_utils.h"
  9. #include "imgui/imgui.h"
  10. #include "nanovg/nanovg.h"
  11. #include <bx/readerwriter.h>
  12. #include <bx/string.h>
  13. static float s_texelHalf = 0.0f;
  14. struct Uniforms
  15. {
  16. void init()
  17. {
  18. m_time = 0.0f;
  19. bx::mtxIdentity(m_mtx);
  20. u_mtx = bgfx::createUniform("u_mtx", bgfx::UniformType::Mat4);
  21. u_params = bgfx::createUniform("u_params", bgfx::UniformType::Vec4);
  22. u_flags = bgfx::createUniform("u_flags", bgfx::UniformType::Vec4);
  23. u_camPos = bgfx::createUniform("u_camPos", bgfx::UniformType::Vec4);
  24. u_rgbDiff = bgfx::createUniform("u_rgbDiff", bgfx::UniformType::Vec4);
  25. u_rgbSpec = bgfx::createUniform("u_rgbSpec", bgfx::UniformType::Vec4);
  26. }
  27. // Call this once per frame.
  28. void submitPerFrameUniforms()
  29. {
  30. bgfx::setUniform(u_mtx, m_mtx);
  31. bgfx::setUniform(u_flags, m_flags);
  32. bgfx::setUniform(u_camPos, m_camPosTime);
  33. bgfx::setUniform(u_rgbDiff, m_rgbDiff);
  34. bgfx::setUniform(u_rgbSpec, m_rgbSpec);
  35. }
  36. // Call this before each draw call.
  37. void submitPerDrawUniforms()
  38. {
  39. bgfx::setUniform(u_params, m_params);
  40. }
  41. void destroy()
  42. {
  43. bgfx::destroyUniform(u_rgbSpec);
  44. bgfx::destroyUniform(u_rgbDiff);
  45. bgfx::destroyUniform(u_camPos);
  46. bgfx::destroyUniform(u_flags);
  47. bgfx::destroyUniform(u_params);
  48. bgfx::destroyUniform(u_mtx);
  49. }
  50. union
  51. {
  52. struct
  53. {
  54. float m_glossiness;
  55. float m_exposure;
  56. float m_diffspec;
  57. float m_time;
  58. };
  59. float m_params[4];
  60. };
  61. union
  62. {
  63. struct
  64. {
  65. float m_diffuse;
  66. float m_specular;
  67. float m_diffuseIbl;
  68. float m_specularIbl;
  69. };
  70. float m_flags[4];
  71. };
  72. float m_mtx[16];
  73. float m_camPosTime[4];
  74. float m_rgbDiff[4];
  75. float m_rgbSpec[4];
  76. bgfx::UniformHandle u_mtx;
  77. bgfx::UniformHandle u_params;
  78. bgfx::UniformHandle u_flags;
  79. bgfx::UniformHandle u_camPos;
  80. bgfx::UniformHandle u_rgbDiff;
  81. bgfx::UniformHandle u_rgbSpec;
  82. };
  83. static Uniforms s_uniforms;
  84. struct PosColorTexCoord0Vertex
  85. {
  86. float m_x;
  87. float m_y;
  88. float m_z;
  89. uint32_t m_rgba;
  90. float m_u;
  91. float m_v;
  92. static void init()
  93. {
  94. ms_decl
  95. .begin()
  96. .add(bgfx::Attrib::Position, 3, bgfx::AttribType::Float)
  97. .add(bgfx::Attrib::Color0, 4, bgfx::AttribType::Uint8, true)
  98. .add(bgfx::Attrib::TexCoord0, 2, bgfx::AttribType::Float)
  99. .end();
  100. }
  101. static bgfx::VertexDecl ms_decl;
  102. };
  103. bgfx::VertexDecl PosColorTexCoord0Vertex::ms_decl;
  104. void screenSpaceQuad(float _textureWidth, float _textureHeight, bool _originBottomLeft = false, float _width = 1.0f, float _height = 1.0f)
  105. {
  106. if (bgfx::checkAvailTransientVertexBuffer(3, PosColorTexCoord0Vertex::ms_decl) )
  107. {
  108. bgfx::TransientVertexBuffer vb;
  109. bgfx::allocTransientVertexBuffer(&vb, 3, PosColorTexCoord0Vertex::ms_decl);
  110. PosColorTexCoord0Vertex* vertex = (PosColorTexCoord0Vertex*)vb.data;
  111. const float zz = 0.0f;
  112. const float minx = -_width;
  113. const float maxx = _width;
  114. const float miny = 0.0f;
  115. const float maxy = _height*2.0f;
  116. const float texelHalfW = s_texelHalf/_textureWidth;
  117. const float texelHalfH = s_texelHalf/_textureHeight;
  118. const float minu = -1.0f + texelHalfW;
  119. const float maxu = 1.0f + texelHalfW;
  120. float minv = texelHalfH;
  121. float maxv = 2.0f + texelHalfH;
  122. if (_originBottomLeft)
  123. {
  124. std::swap(minv, maxv);
  125. minv -= 1.0f;
  126. maxv -= 1.0f;
  127. }
  128. vertex[0].m_x = minx;
  129. vertex[0].m_y = miny;
  130. vertex[0].m_z = zz;
  131. vertex[0].m_rgba = 0xffffffff;
  132. vertex[0].m_u = minu;
  133. vertex[0].m_v = minv;
  134. vertex[1].m_x = maxx;
  135. vertex[1].m_y = miny;
  136. vertex[1].m_z = zz;
  137. vertex[1].m_rgba = 0xffffffff;
  138. vertex[1].m_u = maxu;
  139. vertex[1].m_v = minv;
  140. vertex[2].m_x = maxx;
  141. vertex[2].m_y = maxy;
  142. vertex[2].m_z = zz;
  143. vertex[2].m_rgba = 0xffffffff;
  144. vertex[2].m_u = maxu;
  145. vertex[2].m_v = maxv;
  146. bgfx::setVertexBuffer(&vb);
  147. }
  148. }
  149. struct LightProbe
  150. {
  151. enum Enum
  152. {
  153. Wells,
  154. Uffizi,
  155. Pisa,
  156. Ennis,
  157. Grace,
  158. Count
  159. };
  160. void load(const char* _name)
  161. {
  162. char filePath[512];
  163. strcpy(filePath, _name);
  164. strcat(filePath, "_lod.dds");
  165. m_tex = loadTexture(filePath, BGFX_TEXTURE_U_CLAMP|BGFX_TEXTURE_V_CLAMP|BGFX_TEXTURE_W_CLAMP);
  166. strcpy(filePath, _name);
  167. strcat(filePath, "_irr.dds");
  168. m_texIrr = loadTexture(filePath, BGFX_TEXTURE_U_CLAMP|BGFX_TEXTURE_V_CLAMP|BGFX_TEXTURE_W_CLAMP);
  169. }
  170. void destroy()
  171. {
  172. bgfx::destroyTexture(m_tex);
  173. bgfx::destroyTexture(m_texIrr);
  174. }
  175. bgfx::TextureHandle m_tex;
  176. bgfx::TextureHandle m_texIrr;
  177. };
  178. int _main_(int /*_argc*/, char** /*_argv*/)
  179. {
  180. uint32_t width = 1280;
  181. uint32_t height = 720;
  182. uint32_t debug = BGFX_DEBUG_TEXT;
  183. uint32_t reset = BGFX_RESET_VSYNC;
  184. bgfx::init();
  185. bgfx::reset(width, height, reset);
  186. // Enable debug text.
  187. bgfx::setDebug(debug);
  188. // Set views clear state.
  189. bgfx::setViewClear(0
  190. , BGFX_CLEAR_COLOR|BGFX_CLEAR_DEPTH
  191. , 0x303030ff
  192. , 1.0f
  193. , 0
  194. );
  195. // Imgui.
  196. imguiCreate();
  197. // Uniforms.
  198. s_uniforms.init();
  199. // Vertex declarations.
  200. PosColorTexCoord0Vertex::init();
  201. LightProbe lightProbes[LightProbe::Count];
  202. lightProbes[LightProbe::Wells ].load("wells");
  203. lightProbes[LightProbe::Uffizi].load("uffizi");
  204. lightProbes[LightProbe::Pisa ].load("pisa");
  205. lightProbes[LightProbe::Ennis ].load("ennis");
  206. lightProbes[LightProbe::Grace ].load("grace");
  207. LightProbe::Enum currentLightProbe = LightProbe::Wells;
  208. bgfx::UniformHandle u_mtx = bgfx::createUniform("u_mtx", bgfx::UniformType::Mat4);
  209. bgfx::UniformHandle u_params = bgfx::createUniform("u_params", bgfx::UniformType::Vec4);
  210. bgfx::UniformHandle u_flags = bgfx::createUniform("u_flags", bgfx::UniformType::Vec4);
  211. bgfx::UniformHandle u_camPos = bgfx::createUniform("u_camPos", bgfx::UniformType::Vec4);
  212. bgfx::UniformHandle s_texCube = bgfx::createUniform("s_texCube", bgfx::UniformType::Int1);
  213. bgfx::UniformHandle s_texCubeIrr = bgfx::createUniform("s_texCubeIrr", bgfx::UniformType::Int1);
  214. bgfx::ProgramHandle programMesh = loadProgram("vs_ibl_mesh", "fs_ibl_mesh");
  215. bgfx::ProgramHandle programSky = loadProgram("vs_ibl_skybox", "fs_ibl_skybox");
  216. Mesh* meshBunny;
  217. meshBunny = meshLoad("meshes/bunny.bin");
  218. struct Settings
  219. {
  220. float m_speed;
  221. float m_glossiness;
  222. float m_exposure;
  223. float m_diffspec;
  224. float m_rgbDiff[3];
  225. float m_rgbSpec[3];
  226. bool m_diffuse;
  227. bool m_specular;
  228. bool m_diffuseIbl;
  229. bool m_specularIbl;
  230. bool m_showDiffColorWheel;
  231. bool m_showSpecColorWheel;
  232. ImguiCubemap::Enum m_crossCubemapPreview;
  233. };
  234. Settings settings;
  235. settings.m_speed = 0.37f;
  236. settings.m_glossiness = 1.0f;
  237. settings.m_exposure = 0.0f;
  238. settings.m_diffspec = 0.65f;
  239. settings.m_rgbDiff[0] = 0.2f;
  240. settings.m_rgbDiff[1] = 0.2f;
  241. settings.m_rgbDiff[2] = 0.2f;
  242. settings.m_rgbSpec[0] = 1.0f;
  243. settings.m_rgbSpec[1] = 1.0f;
  244. settings.m_rgbSpec[2] = 1.0f;
  245. settings.m_diffuse = true;
  246. settings.m_specular = true;
  247. settings.m_diffuseIbl = true;
  248. settings.m_specularIbl = true;
  249. settings.m_showDiffColorWheel = true;
  250. settings.m_showSpecColorWheel = false;
  251. settings.m_crossCubemapPreview = ImguiCubemap::Cross;
  252. float time = 0.0f;
  253. int32_t leftScrollArea = 0;
  254. entry::MouseState mouseState;
  255. while (!entry::processEvents(width, height, debug, reset, &mouseState) )
  256. {
  257. imguiBeginFrame(mouseState.m_mx
  258. , mouseState.m_my
  259. , (mouseState.m_buttons[entry::MouseButton::Left ] ? IMGUI_MBUT_LEFT : 0)
  260. | (mouseState.m_buttons[entry::MouseButton::Right ] ? IMGUI_MBUT_RIGHT : 0)
  261. | (mouseState.m_buttons[entry::MouseButton::Middle] ? IMGUI_MBUT_MIDDLE : 0)
  262. , mouseState.m_mz
  263. , width
  264. , height
  265. );
  266. static int32_t rightScrollArea = 0;
  267. imguiBeginScrollArea("Settings", width - 256 - 10, 10, 256, 540, &rightScrollArea);
  268. imguiLabel("Shade:");
  269. imguiSeparator();
  270. imguiBool("Diffuse", settings.m_diffuse);
  271. imguiBool("Specular", settings.m_specular);
  272. imguiBool("IBL Diffuse", settings.m_diffuseIbl);
  273. imguiBool("IBL Specular", settings.m_specularIbl);
  274. imguiSeparatorLine();
  275. imguiSlider("Speed", settings.m_speed, 0.0f, 1.0f, 0.01f);
  276. imguiSeparatorLine();
  277. imguiSeparator();
  278. imguiSlider("Exposure", settings.m_exposure, -8.0f, 8.0f, 0.01f);
  279. imguiSeparator();
  280. imguiLabel("Environment:");
  281. currentLightProbe = LightProbe::Enum(imguiChoose(currentLightProbe
  282. , "Wells"
  283. , "Uffizi"
  284. , "Pisa"
  285. , "Ennis"
  286. , "Grace"
  287. ) );
  288. static float lod = 0.0f;
  289. if (imguiCube(lightProbes[currentLightProbe].m_tex, lod, settings.m_crossCubemapPreview, true) )
  290. {
  291. settings.m_crossCubemapPreview = ImguiCubemap::Enum( (settings.m_crossCubemapPreview+1) % ImguiCubemap::Count);
  292. }
  293. imguiSlider("Texture LOD", lod, 0.0f, 10.1f, 0.1f);
  294. imguiEndScrollArea();
  295. imguiBeginScrollArea("Settings", 10, 70, 256, 576, &leftScrollArea);
  296. imguiLabel("Material properties:");
  297. imguiSeparator();
  298. imguiSlider("Diffuse - Specular", settings.m_diffspec, 0.0f, 1.0f, 0.01f);
  299. imguiSlider("Glossiness" , settings.m_glossiness, 0.0f, 1.0f, 0.01f);
  300. imguiSeparator();
  301. imguiColorWheel("Diffuse color:", &settings.m_rgbDiff[0], settings.m_showDiffColorWheel);
  302. imguiSeparator();
  303. imguiColorWheel("Specular color:", &settings.m_rgbSpec[0], settings.m_showSpecColorWheel);
  304. imguiSeparator();
  305. imguiLabel("Predefined materials:");
  306. imguiSeparator();
  307. if (imguiButton("Gold") )
  308. {
  309. settings.m_glossiness = 0.8f;
  310. settings.m_diffspec = 1.0f;
  311. settings.m_rgbDiff[0] = 0.0f;
  312. settings.m_rgbDiff[1] = 0.0f;
  313. settings.m_rgbDiff[2] = 0.0f;
  314. settings.m_rgbSpec[0] = 1.0f;
  315. settings.m_rgbSpec[1] = 0.86f;
  316. settings.m_rgbSpec[2] = 0.58f;
  317. }
  318. if (imguiButton("Copper") )
  319. {
  320. settings.m_glossiness = 0.67f;
  321. settings.m_diffspec = 1.0f;
  322. settings.m_rgbDiff[0] = 0.0f;
  323. settings.m_rgbDiff[1] = 0.0f;
  324. settings.m_rgbDiff[2] = 0.0f;
  325. settings.m_rgbSpec[0] = 0.98f;
  326. settings.m_rgbSpec[1] = 0.82f;
  327. settings.m_rgbSpec[2] = 0.76f;
  328. }
  329. if (imguiButton("Titanium") )
  330. {
  331. settings.m_glossiness = 0.57f;
  332. settings.m_diffspec = 1.0f;
  333. settings.m_rgbDiff[0] = 0.0f;
  334. settings.m_rgbDiff[1] = 0.0f;
  335. settings.m_rgbDiff[2] = 0.0f;
  336. settings.m_rgbSpec[0] = 0.76f;
  337. settings.m_rgbSpec[1] = 0.73f;
  338. settings.m_rgbSpec[2] = 0.71f;
  339. }
  340. if (imguiButton("Steel") )
  341. {
  342. settings.m_glossiness = 0.82f;
  343. settings.m_diffspec = 1.0f;
  344. settings.m_rgbDiff[0] = 0.0f;
  345. settings.m_rgbDiff[1] = 0.0f;
  346. settings.m_rgbDiff[2] = 0.0f;
  347. settings.m_rgbSpec[0] = 0.77f;
  348. settings.m_rgbSpec[1] = 0.78f;
  349. settings.m_rgbSpec[2] = 0.77f;
  350. }
  351. imguiEndScrollArea();
  352. imguiEndFrame();
  353. s_uniforms.m_glossiness = settings.m_glossiness;
  354. s_uniforms.m_exposure = settings.m_exposure;
  355. s_uniforms.m_diffspec = settings.m_diffspec;
  356. s_uniforms.m_flags[0] = float(settings.m_diffuse);
  357. s_uniforms.m_flags[1] = float(settings.m_specular);
  358. s_uniforms.m_flags[2] = float(settings.m_diffuseIbl);
  359. s_uniforms.m_flags[3] = float(settings.m_specularIbl);
  360. memcpy(s_uniforms.m_rgbDiff, settings.m_rgbDiff, 3*sizeof(float) );
  361. memcpy(s_uniforms.m_rgbSpec, settings.m_rgbSpec, 3*sizeof(float) );
  362. s_uniforms.submitPerFrameUniforms();
  363. int64_t now = bx::getHPCounter();
  364. static int64_t last = now;
  365. const int64_t frameTime = now - last;
  366. last = now;
  367. const double freq = double(bx::getHPFrequency() );
  368. const double toMs = 1000.0/freq;
  369. time += (float)(frameTime*settings.m_speed/freq);
  370. s_uniforms.m_camPosTime[3] = time;
  371. // Use debug font to print information about this example.
  372. bgfx::dbgTextClear();
  373. bgfx::dbgTextPrintf(0, 1, 0x4f, "bgfx/examples/18-ibl");
  374. bgfx::dbgTextPrintf(0, 2, 0x6f, "Description: Image based lightning.");
  375. bgfx::dbgTextPrintf(0, 3, 0x0f, "Frame: % 7.3f[ms]", double(frameTime)*toMs);
  376. float at[3] = { 0.0f, 0.0f, 0.0f };
  377. float eye[3] = { 0.0f, 0.0f, -3.0f };
  378. bx::mtxRotateXY(s_uniforms.m_mtx
  379. , 0.0f
  380. , time
  381. );
  382. float view[16];
  383. float proj[16];
  384. bx::mtxIdentity(view);
  385. bx::mtxOrtho(proj, 0.0f, 1.0f, 1.0f, 0.0f, 0.0f, 100.0f);
  386. bgfx::setViewTransform(0, view, proj);
  387. bx::mtxLookAt(view, eye, at);
  388. memcpy(s_uniforms.m_camPosTime, eye, 3*sizeof(float) );
  389. bx::mtxProj(proj, 60.0f, float(width)/float(height), 0.1f, 100.0f);
  390. bgfx::setViewTransform(1, view, proj);
  391. bgfx::setViewRect(0, 0, 0, width, height);
  392. bgfx::setViewRect(1, 0, 0, width, height);
  393. // View 0.
  394. bgfx::setTexture(0, s_texCube, lightProbes[currentLightProbe].m_tex);
  395. bgfx::setState(BGFX_STATE_RGB_WRITE|BGFX_STATE_ALPHA_WRITE);
  396. screenSpaceQuad( (float)width, (float)height, true);
  397. s_uniforms.submitPerDrawUniforms();
  398. bgfx::submit(0, programSky);
  399. // View 1.
  400. float mtx[16];
  401. bx::mtxSRT(mtx
  402. , 1.0f
  403. , 1.0f
  404. , 1.0f
  405. , 0.0f
  406. , bx::pi+time
  407. , 0.0f
  408. , 0.0f
  409. , -1.0f
  410. , 0.0f
  411. );
  412. bgfx::setTexture(0, s_texCube, lightProbes[currentLightProbe].m_tex);
  413. bgfx::setTexture(1, s_texCubeIrr, lightProbes[currentLightProbe].m_texIrr);
  414. meshSubmit(meshBunny, 1, programMesh, mtx);
  415. // Advance to next frame. Rendering thread will be kicked to
  416. // process submitted rendering primitives.
  417. bgfx::frame();
  418. }
  419. meshUnload(meshBunny);
  420. // Cleanup.
  421. bgfx::destroyProgram(programMesh);
  422. bgfx::destroyProgram(programSky);
  423. bgfx::destroyUniform(u_camPos);
  424. bgfx::destroyUniform(u_flags);
  425. bgfx::destroyUniform(u_params);
  426. bgfx::destroyUniform(u_mtx);
  427. bgfx::destroyUniform(s_texCube);
  428. bgfx::destroyUniform(s_texCubeIrr);
  429. for (uint8_t ii = 0; ii < LightProbe::Count; ++ii)
  430. {
  431. lightProbes[ii].destroy();
  432. }
  433. s_uniforms.destroy();
  434. imguiDestroy();
  435. // Shutdown bgfx.
  436. bgfx::shutdown();
  437. return 0;
  438. }