OGLRasterizer.cs 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. using OpenTK.Graphics.OpenGL;
  2. using System;
  3. using System.Collections.Generic;
  4. namespace Ryujinx.Graphics.Gal.OpenGL
  5. {
  6. public class OGLRasterizer : IGalRasterizer
  7. {
  8. private static Dictionary<GalVertexAttribSize, int> AttribElements =
  9. new Dictionary<GalVertexAttribSize, int>()
  10. {
  11. { GalVertexAttribSize._32_32_32_32, 4 },
  12. { GalVertexAttribSize._32_32_32, 3 },
  13. { GalVertexAttribSize._16_16_16_16, 4 },
  14. { GalVertexAttribSize._32_32, 2 },
  15. { GalVertexAttribSize._16_16_16, 3 },
  16. { GalVertexAttribSize._8_8_8_8, 4 },
  17. { GalVertexAttribSize._16_16, 2 },
  18. { GalVertexAttribSize._32, 1 },
  19. { GalVertexAttribSize._8_8_8, 3 },
  20. { GalVertexAttribSize._8_8, 2 },
  21. { GalVertexAttribSize._16, 1 },
  22. { GalVertexAttribSize._8, 1 },
  23. { GalVertexAttribSize._10_10_10_2, 4 },
  24. { GalVertexAttribSize._11_11_10, 3 }
  25. };
  26. private static Dictionary<GalVertexAttribSize, VertexAttribPointerType> AttribTypes =
  27. new Dictionary<GalVertexAttribSize, VertexAttribPointerType>()
  28. {
  29. { GalVertexAttribSize._32_32_32_32, VertexAttribPointerType.Int },
  30. { GalVertexAttribSize._32_32_32, VertexAttribPointerType.Int },
  31. { GalVertexAttribSize._16_16_16_16, VertexAttribPointerType.Short },
  32. { GalVertexAttribSize._32_32, VertexAttribPointerType.Int },
  33. { GalVertexAttribSize._16_16_16, VertexAttribPointerType.Short },
  34. { GalVertexAttribSize._8_8_8_8, VertexAttribPointerType.Byte },
  35. { GalVertexAttribSize._16_16, VertexAttribPointerType.Short },
  36. { GalVertexAttribSize._32, VertexAttribPointerType.Int },
  37. { GalVertexAttribSize._8_8_8, VertexAttribPointerType.Byte },
  38. { GalVertexAttribSize._8_8, VertexAttribPointerType.Byte },
  39. { GalVertexAttribSize._16, VertexAttribPointerType.Short },
  40. { GalVertexAttribSize._8, VertexAttribPointerType.Byte },
  41. { GalVertexAttribSize._10_10_10_2, VertexAttribPointerType.Int }, //?
  42. { GalVertexAttribSize._11_11_10, VertexAttribPointerType.Int } //?
  43. };
  44. private int VaoHandle;
  45. private int[] VertexBuffers;
  46. private OGLCachedResource<int> VboCache;
  47. private OGLCachedResource<int> IboCache;
  48. private struct IbInfo
  49. {
  50. public int Count;
  51. public DrawElementsType Type;
  52. }
  53. private IbInfo IndexBuffer;
  54. public OGLRasterizer()
  55. {
  56. VertexBuffers = new int[32];
  57. VboCache = new OGLCachedResource<int>(GL.DeleteBuffer);
  58. IboCache = new OGLCachedResource<int>(GL.DeleteBuffer);
  59. IndexBuffer = new IbInfo();
  60. }
  61. public void ClearBuffers(int RtIndex, GalClearBufferFlags Flags)
  62. {
  63. ClearBufferMask Mask = 0;
  64. //TODO: Use glColorMask to clear just the specified channels.
  65. if (Flags.HasFlag(GalClearBufferFlags.ColorRed) &&
  66. Flags.HasFlag(GalClearBufferFlags.ColorGreen) &&
  67. Flags.HasFlag(GalClearBufferFlags.ColorBlue) &&
  68. Flags.HasFlag(GalClearBufferFlags.ColorAlpha))
  69. {
  70. Mask = ClearBufferMask.ColorBufferBit;
  71. }
  72. if (Flags.HasFlag(GalClearBufferFlags.Depth))
  73. {
  74. Mask |= ClearBufferMask.DepthBufferBit;
  75. }
  76. if (Flags.HasFlag(GalClearBufferFlags.Stencil))
  77. {
  78. Mask |= ClearBufferMask.StencilBufferBit;
  79. }
  80. GL.Clear(Mask);
  81. }
  82. public bool IsVboCached(long Key, long DataSize)
  83. {
  84. return VboCache.TryGetSize(Key, out long Size) && Size == DataSize;
  85. }
  86. public bool IsIboCached(long Key, long DataSize)
  87. {
  88. return IboCache.TryGetSize(Key, out long Size) && Size == DataSize;
  89. }
  90. public void EnableCullFace()
  91. {
  92. GL.Enable(EnableCap.CullFace);
  93. }
  94. public void DisableCullFace()
  95. {
  96. GL.Disable(EnableCap.CullFace);
  97. }
  98. public void EnableDepthTest()
  99. {
  100. GL.Enable(EnableCap.DepthTest);
  101. }
  102. public void DisableDepthTest()
  103. {
  104. GL.Disable(EnableCap.DepthTest);
  105. }
  106. public void SetDepthFunction(GalComparisonOp Func)
  107. {
  108. GL.DepthFunc(OGLEnumConverter.GetDepthFunc(Func));
  109. }
  110. public void CreateVbo(long Key, byte[] Buffer)
  111. {
  112. int Handle = GL.GenBuffer();
  113. VboCache.AddOrUpdate(Key, Handle, (uint)Buffer.Length);
  114. IntPtr Length = new IntPtr(Buffer.Length);
  115. GL.BindBuffer(BufferTarget.ArrayBuffer, Handle);
  116. GL.BufferData(BufferTarget.ArrayBuffer, Length, Buffer, BufferUsageHint.StreamDraw);
  117. }
  118. public void CreateIbo(long Key, byte[] Buffer)
  119. {
  120. int Handle = GL.GenBuffer();
  121. IboCache.AddOrUpdate(Key, Handle, (uint)Buffer.Length);
  122. IntPtr Length = new IntPtr(Buffer.Length);
  123. GL.BindBuffer(BufferTarget.ElementArrayBuffer, Handle);
  124. GL.BufferData(BufferTarget.ElementArrayBuffer, Length, Buffer, BufferUsageHint.StreamDraw);
  125. }
  126. public void SetVertexArray(int VbIndex, int Stride, long VboKey, GalVertexAttrib[] Attribs)
  127. {
  128. if (!VboCache.TryGetValue(VboKey, out int VboHandle))
  129. {
  130. return;
  131. }
  132. if (VaoHandle == 0)
  133. {
  134. VaoHandle = GL.GenVertexArray();
  135. }
  136. GL.BindVertexArray(VaoHandle);
  137. foreach (GalVertexAttrib Attrib in Attribs)
  138. {
  139. GL.EnableVertexAttribArray(Attrib.Index);
  140. GL.BindBuffer(BufferTarget.ArrayBuffer, VboHandle);
  141. bool Unsigned =
  142. Attrib.Type == GalVertexAttribType.Unorm ||
  143. Attrib.Type == GalVertexAttribType.Uint ||
  144. Attrib.Type == GalVertexAttribType.Uscaled;
  145. bool Normalize =
  146. Attrib.Type == GalVertexAttribType.Snorm ||
  147. Attrib.Type == GalVertexAttribType.Unorm;
  148. VertexAttribPointerType Type = 0;
  149. if (Attrib.Type == GalVertexAttribType.Float)
  150. {
  151. Type = VertexAttribPointerType.Float;
  152. }
  153. else
  154. {
  155. Type = AttribTypes[Attrib.Size] + (Unsigned ? 1 : 0);
  156. }
  157. int Size = AttribElements[Attrib.Size];
  158. int Offset = Attrib.Offset;
  159. GL.VertexAttribPointer(Attrib.Index, Size, Type, Normalize, Stride, Offset);
  160. }
  161. }
  162. public void SetIndexArray(long Key, int Size, GalIndexFormat Format)
  163. {
  164. IndexBuffer.Type = OGLEnumConverter.GetDrawElementsType(Format);
  165. IndexBuffer.Count = Size >> (int)Format;
  166. }
  167. public void DrawArrays(int First, int PrimCount, GalPrimitiveType PrimType)
  168. {
  169. if (PrimCount == 0)
  170. {
  171. return;
  172. }
  173. GL.BindVertexArray(VaoHandle);
  174. GL.DrawArrays(OGLEnumConverter.GetPrimitiveType(PrimType), First, PrimCount);
  175. }
  176. public void DrawElements(long IboKey, int First, GalPrimitiveType PrimType)
  177. {
  178. if (!IboCache.TryGetValue(IboKey, out int IboHandle))
  179. {
  180. return;
  181. }
  182. PrimitiveType Mode = OGLEnumConverter.GetPrimitiveType(PrimType);
  183. GL.BindVertexArray(VaoHandle);
  184. GL.BindBuffer(BufferTarget.ElementArrayBuffer, IboHandle);
  185. GL.DrawElements(Mode, IndexBuffer.Count, IndexBuffer.Type, First);
  186. }
  187. }
  188. }