598 lines
16 KiB

  1. //
  2. // Lol Engine
  3. //
  4. // Copyright: (c) 2010-2013 Sam Hocevar <sam@hocevar.net>
  5. // This program is free software; you can redistribute it and/or
  6. // modify it under the terms of the Do What The Fuck You Want To
  7. // Public License, Version 2, as published by Sam Hocevar. See
  8. // http://www.wtfpl.net/ for more details.
  9. //
  10. #if defined HAVE_CONFIG_H
  11. # include "config.h"
  12. #endif
  13. #include <cstdlib>
  14. #if defined _XBOX
  15. # include <xtl.h>
  16. # undef near /* Fuck Microsoft */
  17. # undef far /* Fuck Microsoft again */
  18. #elif defined _WIN32
  19. # if defined USE_D3D9
  20. # include <d3d9.h>
  21. # endif
  22. # define WIN32_LEAN_AND_MEAN
  23. # include <windows.h>
  24. # undef near /* Fuck Microsoft */
  25. # undef far /* Fuck Microsoft again */
  26. #endif
  27. #include "core.h"
  28. #include "lolgl.h"
  29. #if defined USE_D3D9
  30. extern IDirect3DDevice9 *g_d3ddevice;
  31. #elif defined _XBOX
  32. extern D3DDevice *g_d3ddevice;
  33. #endif
  34. namespace lol
  35. {
  36. /*
  37. * The global g_renderer object, initialised by Video::Init
  38. */
  39. Renderer *g_renderer;
  40. /*
  41. * Private RendererData class
  42. */
  43. class RendererData
  44. {
  45. friend class Renderer;
  46. private:
  47. ibox2 m_viewport;
  48. vec4 m_clear_color;
  49. float m_clear_depth;
  50. AlphaFunc m_alpha_func;
  51. float m_alpha_value;
  52. BlendFunc m_blend_src, m_blend_dst;
  53. DepthFunc m_depth_func;
  54. CullMode m_face_culling;
  55. #if defined USE_D3D9
  56. IDirect3DDevice9 *m_d3d_dev;
  57. #elif defined _XBOX
  58. D3DDevice *m_d3d_dev;
  59. #endif
  60. };
  61. /*
  62. * Public Renderer class
  63. */
  64. Renderer::Renderer(ivec2 size)
  65. : m_data(new RendererData())
  66. {
  67. #if defined USE_D3D9 || defined _XBOX
  68. /* FIXME: we should be in charge of creating this */
  69. m_data->m_d3d_dev = g_d3ddevice;
  70. #else
  71. # if defined USE_GLEW && !defined __APPLE__
  72. /* Initialise GLEW if necessary */
  73. GLenum glerr = glewInit();
  74. if (glerr != GLEW_OK)
  75. {
  76. Log::Error("cannot initialise GLEW: %s\n", glewGetErrorString(glerr));
  77. exit(EXIT_FAILURE);
  78. }
  79. # endif
  80. #endif
  81. /* Initialise rendering states */
  82. m_data->m_viewport = ibox2(0, 0, 0, 0);
  83. SetViewport(ibox2(vec2(0), size));
  84. m_data->m_clear_color = vec4(-1.f);
  85. SetClearColor(vec4(0.1f, 0.2f, 0.3f, 1.0f));
  86. m_data->m_clear_depth = -1.f;
  87. SetClearDepth(1.f);
  88. m_data->m_alpha_func = AlphaFunc::Never;
  89. m_data->m_alpha_value = -1.0f;
  90. SetAlphaFunc(AlphaFunc::Disabled, 0.0f);
  91. m_data->m_blend_src = BlendFunc::Disabled;
  92. m_data->m_blend_dst = BlendFunc::Disabled;
  93. SetBlendFunc(BlendFunc::SrcAlpha, BlendFunc::OneMinusSrcAlpha);
  94. m_data->m_depth_func = DepthFunc::Disabled;
  95. SetDepthFunc(DepthFunc::LessOrEqual);
  96. m_data->m_face_culling = CullMode::Disabled;
  97. SetFaceCulling(CullMode::CounterClockwise);
  98. /* Add some rendering states that we don't export to the user */
  99. #if defined USE_D3D9 || defined _XBOX
  100. /* TODO */
  101. #else
  102. # if defined HAVE_GL_2X && !defined __APPLE__
  103. glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST);
  104. # endif
  105. #endif
  106. }
  107. Renderer::~Renderer()
  108. {
  109. delete m_data;
  110. }
  111. /*
  112. * Buffer clearing
  113. */
  114. void Renderer::Clear(ClearMask mask)
  115. {
  116. #if defined USE_D3D9 || defined _XBOX
  117. int m = 0;
  118. if (mask & ClearMask::Color)
  119. m |= D3DCLEAR_TARGET;
  120. if (mask & ClearMask::Depth)
  121. m |= D3DCLEAR_ZBUFFER;
  122. if (mask & ClearMask::Stencil)
  123. m |= D3DCLEAR_STENCIL;
  124. vec3 tmp = 255.999f * GetClearColor().rgb;
  125. D3DCOLOR clear_color = D3DCOLOR_XRGB((int)tmp.r, (int)tmp.g, (int)tmp.b);
  126. if (FAILED(VideoData::d3d_dev->Clear(0, nullptr, m, clear_color,
  127. g_renderer->GetClearDepth(), 0)))
  128. Abort();
  129. #else
  130. GLbitfield m = 0;
  131. if (mask & ClearMask::Color)
  132. m |= GL_COLOR_BUFFER_BIT;
  133. if (mask & ClearMask::Depth)
  134. m |= GL_DEPTH_BUFFER_BIT;
  135. if (mask & ClearMask::Stencil)
  136. m |= GL_STENCIL_BUFFER_BIT;
  137. glClear(m);
  138. #endif
  139. }
  140. /*
  141. * Viewport dimensions
  142. */
  143. void Renderer::SetViewport(ibox2 viewport)
  144. {
  145. if (m_data->m_viewport == viewport)
  146. return;
  147. #if defined USE_D3D9 || defined _XBOX
  148. D3DVIEWPORT9 vp = { viewport.A.x, viewport.A.y,
  149. viewport.B.x, viewport.B.y,
  150. 0.0f, 1.0f };
  151. m_data->m_d3d_dev->SetViewport(&vp);
  152. #else
  153. glViewport(viewport.A.x, viewport.A.y, viewport.B.x, viewport.B.y);
  154. #endif
  155. m_data->m_viewport = viewport;
  156. }
  157. ibox2 Renderer::GetViewport() const
  158. {
  159. return m_data->m_viewport;
  160. }
  161. /*
  162. * Clear color
  163. */
  164. void Renderer::SetClearColor(vec4 color)
  165. {
  166. if (m_data->m_clear_color == color)
  167. return;
  168. #if defined USE_D3D9 || defined _XBOX
  169. /* Nothing to do */
  170. #else
  171. glClearColor(color.r, color.g, color.b, color.a);
  172. #endif
  173. m_data->m_clear_color = color;
  174. }
  175. vec4 Renderer::GetClearColor() const
  176. {
  177. return m_data->m_clear_color;
  178. }
  179. /*
  180. * Clear depth
  181. */
  182. void Renderer::SetClearDepth(float depth)
  183. {
  184. if (m_data->m_clear_depth == depth)
  185. return;
  186. #if defined USE_D3D9 || defined _XBOX
  187. /* Nothing to do */
  188. #elif defined HAVE_GLES_2X
  189. glClearDepthf(depth);
  190. #else
  191. glClearDepth(depth);
  192. #endif
  193. m_data->m_clear_depth = depth;
  194. }
  195. float Renderer::GetClearDepth() const
  196. {
  197. return m_data->m_clear_depth;
  198. }
  199. /*
  200. * Alpha testing
  201. */
  202. void Renderer::SetAlphaFunc(AlphaFunc func, float alpha)
  203. {
  204. if (m_data->m_alpha_func == func && m_data->m_alpha_value == alpha)
  205. return;
  206. #if defined USE_D3D9 || defined _XBOX
  207. switch (func)
  208. {
  209. case AlphaFunc::Disabled:
  210. break; /* Nothing to do */
  211. case AlphaFunc::Never:
  212. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHAFUNC, D3DCMP_NEVER);
  213. break;
  214. case AlphaFunc::Less:
  215. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHAFUNC, D3DCMP_LESS);
  216. break;
  217. case AlphaFunc::Equal:
  218. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHAFUNC, D3DCMP_EQUAL);
  219. break;
  220. case AlphaFunc::LessOrEqual:
  221. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHAFUNC, D3DCMP_LESSEQUAL);
  222. break;
  223. case AlphaFunc::Greater:
  224. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHAFUNC, D3DCMP_GREATER);
  225. break;
  226. case AlphaFunc::NotEqual:
  227. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHAFUNC, D3DCMP_NOTEQUAL);
  228. break;
  229. case AlphaFunc::GreaterOrEqual:
  230. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHAFUNC, D3DCMP_GREATEREQUAL);
  231. break;
  232. case AlphaFunc::Always:
  233. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHAFUNC, D3DCMP_ALWAYS);
  234. break;
  235. }
  236. if (func == AlphaFunc::Disabled)
  237. {
  238. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHATESTENABLE, FALSE);
  239. }
  240. else
  241. {
  242. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHATESTENABLE, TRUE);
  243. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHAREF,
  244. (DWORD)(alpha * 255.999f));
  245. }
  246. #elif defined HAVE_GL_2X
  247. switch (func)
  248. {
  249. case AlphaFunc::Disabled:
  250. break; /* Nothing to do */
  251. case AlphaFunc::Never:
  252. glAlphaFunc(GL_NEVER, alpha); break;
  253. case AlphaFunc::Less:
  254. glAlphaFunc(GL_LESS, alpha); break;
  255. case AlphaFunc::Equal:
  256. glAlphaFunc(GL_EQUAL, alpha); break;
  257. case AlphaFunc::LessOrEqual:
  258. glAlphaFunc(GL_LEQUAL, alpha); break;
  259. case AlphaFunc::Greater:
  260. glAlphaFunc(GL_GREATER, alpha); break;
  261. case AlphaFunc::NotEqual:
  262. glAlphaFunc(GL_NOTEQUAL, alpha); break;
  263. case AlphaFunc::GreaterOrEqual:
  264. glAlphaFunc(GL_GEQUAL, alpha); break;
  265. case AlphaFunc::Always:
  266. glAlphaFunc(GL_ALWAYS, alpha); break;
  267. }
  268. if (func == AlphaFunc::Disabled)
  269. glDisable(GL_ALPHA_TEST);
  270. else
  271. glEnable(GL_ALPHA_TEST);
  272. #else
  273. /* XXX: alpha test not available in GL ES */
  274. #endif
  275. m_data->m_alpha_func = func;
  276. m_data->m_alpha_value = alpha;
  277. }
  278. AlphaFunc Renderer::GetAlphaFunc() const
  279. {
  280. return m_data->m_alpha_func;
  281. }
  282. float Renderer::GetAlphaValue() const
  283. {
  284. return m_data->m_alpha_value;
  285. }
  286. /*
  287. * Blend function
  288. */
  289. void Renderer::SetBlendFunc(BlendFunc src, BlendFunc dst)
  290. {
  291. if (m_data->m_blend_src == src && m_data->m_blend_dst == dst)
  292. return;
  293. #if defined USE_D3D9 || defined _XBOX
  294. enum D3DBLEND s1[2] = { D3DBLEND_ONE, D3DBLEND_ZERO };
  295. BlendFunc s2[2] = { src, dst };
  296. for (int i = 0; i < 2; ++i)
  297. {
  298. switch (s2[i])
  299. {
  300. case BlendFunc::Disabled:
  301. break; /* Nothing to do */
  302. case BlendFunc::Zero:
  303. s1[i] = D3DBLEND_ZERO; break;
  304. case BlendFunc::One:
  305. s1[i] = D3DBLEND_ONE; break;
  306. case BlendFunc::SrcColor:
  307. s1[i] = D3DBLEND_SRCCOLOR; break;
  308. case BlendFunc::OneMinusSrcColor:
  309. s1[i] = D3DBLEND_INVSRCCOLOR; break;
  310. case BlendFunc::DstColor:
  311. s1[i] = D3DBLEND_DESTCOLOR; break;
  312. case BlendFunc::OneMinusDstColor:
  313. s1[i] = D3DBLEND_INVDESTCOLOR; break;
  314. case BlendFunc::SrcAlpha:
  315. s1[i] = D3DBLEND_SRCALPHA; break;
  316. case BlendFunc::OneMinusSrcAlpha:
  317. s1[i] = D3DBLEND_INVSRCALPHA; break;
  318. case BlendFunc::DstAlpha:
  319. s1[i] = D3DBLEND_DESTALPHA; break;
  320. case BlendFunc::OneMinusDstAlpha:
  321. s1[i] = D3DBLEND_INVDESTALPHA; break;
  322. /* FiXME: these can be supported using D3DPBLENDCAPS_BLENDFACTOR */
  323. case BlendFunc::ConstantColor:
  324. assert(0, "BlendFunc::ConstantColor not supported");
  325. break;
  326. case BlendFunc::OneMinusConstantColor:
  327. assert(0, "BlendFunc::OneMinusConstantColor not supported");
  328. break;
  329. case BlendFunc::ConstantAlpha:
  330. assert(0, "BlendFunc::ConstantAlpha not supported");
  331. break;
  332. case BlendFunc::OneMinusConstantAlpha:
  333. assert(0, "BlendFunc::OneMinusConstantAlpha not supported");
  334. break;
  335. }
  336. }
  337. if (src == BlendFunc::Disabled)
  338. {
  339. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHABLENDENABLE, 0);
  340. }
  341. else
  342. {
  343. m_data->m_d3d_dev->SetRenderState(D3DRS_ALPHABLENDENABLE, 1);
  344. m_data->m_d3d_dev->SetRenderState(D3DRS_SRCBLEND, s1[0]);
  345. m_data->m_d3d_dev->SetRenderState(D3DRS_DESTBLEND, s1[1]);
  346. }
  347. #else
  348. GLenum s1[2] = { GL_ONE, GL_ZERO };
  349. BlendFunc s2[2] = { src, dst };
  350. for (int i = 0; i < 2; ++i)
  351. {
  352. switch (s2[i])
  353. {
  354. case BlendFunc::Disabled:
  355. break; /* Nothing to do */
  356. case BlendFunc::Zero:
  357. s1[i] = GL_ZERO; break;
  358. case BlendFunc::One:
  359. s1[i] = GL_ONE; break;
  360. case BlendFunc::SrcColor:
  361. s1[i] = GL_SRC_COLOR; break;
  362. case BlendFunc::OneMinusSrcColor:
  363. s1[i] = GL_ONE_MINUS_SRC_COLOR; break;
  364. case BlendFunc::DstColor:
  365. s1[i] = GL_DST_COLOR; break;
  366. case BlendFunc::OneMinusDstColor:
  367. s1[i] = GL_ONE_MINUS_DST_COLOR; break;
  368. case BlendFunc::SrcAlpha:
  369. s1[i] = GL_SRC_ALPHA; break;
  370. case BlendFunc::OneMinusSrcAlpha:
  371. s1[i] = GL_ONE_MINUS_SRC_ALPHA; break;
  372. case BlendFunc::DstAlpha:
  373. s1[i] = GL_DST_ALPHA; break;
  374. case BlendFunc::OneMinusDstAlpha:
  375. s1[i] = GL_ONE_MINUS_DST_ALPHA; break;
  376. case BlendFunc::ConstantColor:
  377. s1[i] = GL_CONSTANT_COLOR; break;
  378. case BlendFunc::OneMinusConstantColor:
  379. s1[i] = GL_ONE_MINUS_CONSTANT_COLOR; break;
  380. case BlendFunc::ConstantAlpha:
  381. s1[i] = GL_CONSTANT_ALPHA; break;
  382. case BlendFunc::OneMinusConstantAlpha:
  383. s1[i] = GL_ONE_MINUS_CONSTANT_ALPHA; break;
  384. }
  385. }
  386. if (src == BlendFunc::Disabled)
  387. {
  388. glDisable(GL_BLEND);
  389. }
  390. else
  391. {
  392. glEnable(GL_BLEND);
  393. glBlendFunc(s1[0], s1[1]);
  394. }
  395. #endif
  396. m_data->m_blend_src = src;
  397. m_data->m_blend_dst = dst;
  398. }
  399. BlendFunc Renderer::GetBlendFuncSrc() const
  400. {
  401. return m_data->m_blend_src;
  402. }
  403. BlendFunc Renderer::GetBlendFuncDst() const
  404. {
  405. return m_data->m_blend_dst;
  406. }
  407. /*
  408. * Depth test
  409. */
  410. void Renderer::SetDepthFunc(DepthFunc func)
  411. {
  412. if (m_data->m_depth_func == func)
  413. return;
  414. #if defined USE_D3D9 || defined _XBOX
  415. switch (func)
  416. {
  417. case DepthFunc::Disabled:
  418. break; /* Nothing to do */
  419. case DepthFunc::Never:
  420. m_data->m_d3d_dev->SetRenderState(D3DRS_ZFUNC, D3DCMP_NEVER);
  421. break;
  422. case DepthFunc::Less:
  423. m_data->m_d3d_dev->SetRenderState(D3DRS_ZFUNC, D3DCMP_LESS);
  424. break;
  425. case DepthFunc::Equal:
  426. m_data->m_d3d_dev->SetRenderState(D3DRS_ZFUNC, D3DCMP_EQUAL);
  427. break;
  428. case DepthFunc::LessOrEqual:
  429. m_data->m_d3d_dev->SetRenderState(D3DRS_ZFUNC, D3DCMP_LESSEQUAL);
  430. break;
  431. case DepthFunc::Greater:
  432. m_data->m_d3d_dev->SetRenderState(D3DRS_ZFUNC, D3DCMP_GREATER);
  433. break;
  434. case DepthFunc::NotEqual:
  435. m_data->m_d3d_dev->SetRenderState(D3DRS_ZFUNC, D3DCMP_NOTEQUAL);
  436. break;
  437. case DepthFunc::GreaterOrEqual:
  438. m_data->m_d3d_dev->SetRenderState(D3DRS_ZFUNC, D3DCMP_GREATEREQUAL);
  439. break;
  440. case DepthFunc::Always:
  441. m_data->m_d3d_dev->SetRenderState(D3DRS_ZFUNC, D3DCMP_ALWAYS);
  442. break;
  443. }
  444. if (func == DepthFunc::Disabled)
  445. m_data->m_d3d_dev->SetRenderState(D3DRS_ZENABLE, D3DZB_FALSE);
  446. else
  447. m_data->m_d3d_dev->SetRenderState(D3DRS_ZENABLE, D3DZB_TRUE);
  448. #else
  449. switch (func)
  450. {
  451. case DepthFunc::Disabled:
  452. break; /* Nothing to do */
  453. case DepthFunc::Never:
  454. glDepthFunc(GL_NEVER); break;
  455. case DepthFunc::Less:
  456. glDepthFunc(GL_LESS); break;
  457. case DepthFunc::Equal:
  458. glDepthFunc(GL_EQUAL); break;
  459. case DepthFunc::LessOrEqual:
  460. glDepthFunc(GL_LEQUAL); break;
  461. case DepthFunc::Greater:
  462. glDepthFunc(GL_GREATER); break;
  463. case DepthFunc::NotEqual:
  464. glDepthFunc(GL_NOTEQUAL); break;
  465. case DepthFunc::GreaterOrEqual:
  466. glDepthFunc(GL_GEQUAL); break;
  467. case DepthFunc::Always:
  468. glDepthFunc(GL_ALWAYS); break;
  469. }
  470. if (func == DepthFunc::Disabled)
  471. glDisable(GL_DEPTH_TEST);
  472. else
  473. glEnable(GL_DEPTH_TEST);
  474. #endif
  475. m_data->m_depth_func = func;
  476. }
  477. DepthFunc Renderer::GetDepthFunc() const
  478. {
  479. return m_data->m_depth_func;
  480. }
  481. /*
  482. * Face culling
  483. */
  484. void Renderer::SetFaceCulling(CullMode mode)
  485. {
  486. if (m_data->m_face_culling == mode)
  487. return;
  488. #if defined USE_D3D9 || defined _XBOX
  489. switch (mode)
  490. {
  491. case CullMode::Disabled:
  492. m_data->m_d3d_dev->SetRenderState(D3DRS_CULLMODE, D3DCULL_NONE);
  493. break;
  494. case CullMode::Clockwise:
  495. m_data->m_d3d_dev->SetRenderState(D3DRS_CULLMODE, D3DCULL_CW);
  496. break;
  497. case CullMode::CounterClockwise:
  498. m_data->m_d3d_dev->SetRenderState(D3DRS_CULLMODE, D3DCULL_CCW);
  499. break;
  500. }
  501. #else
  502. switch (mode)
  503. {
  504. case CullMode::Disabled:
  505. glDisable(GL_CULL_FACE);
  506. break;
  507. case CullMode::Clockwise:
  508. glEnable(GL_CULL_FACE);
  509. glCullFace(GL_BACK);
  510. glFrontFace(GL_CW);
  511. break;
  512. case CullMode::CounterClockwise:
  513. glEnable(GL_CULL_FACE);
  514. glCullFace(GL_BACK);
  515. glFrontFace(GL_CCW);
  516. break;
  517. }
  518. #endif
  519. m_data->m_face_culling = mode;
  520. }
  521. CullMode Renderer::GetFaceCulling() const
  522. {
  523. return m_data->m_face_culling;
  524. }
  525. } /* namespace lol */