2018-04-08 15:17:35 -04:00
|
|
|
using OpenTK.Graphics.OpenGL;
|
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
|
|
namespace Ryujinx.Graphics.Gal.OpenGL
|
|
|
|
{
|
2018-06-23 20:39:25 -04:00
|
|
|
public class OGLRasterizer : IGalRasterizer
|
2018-04-08 15:17:35 -04:00
|
|
|
{
|
|
|
|
private static Dictionary<GalVertexAttribSize, int> AttribElements =
|
|
|
|
new Dictionary<GalVertexAttribSize, int>()
|
|
|
|
{
|
|
|
|
{ GalVertexAttribSize._32_32_32_32, 4 },
|
|
|
|
{ GalVertexAttribSize._32_32_32, 3 },
|
|
|
|
{ GalVertexAttribSize._16_16_16_16, 4 },
|
|
|
|
{ GalVertexAttribSize._32_32, 2 },
|
|
|
|
{ GalVertexAttribSize._16_16_16, 3 },
|
|
|
|
{ GalVertexAttribSize._8_8_8_8, 4 },
|
|
|
|
{ GalVertexAttribSize._16_16, 2 },
|
|
|
|
{ GalVertexAttribSize._32, 1 },
|
|
|
|
{ GalVertexAttribSize._8_8_8, 3 },
|
|
|
|
{ GalVertexAttribSize._8_8, 2 },
|
|
|
|
{ GalVertexAttribSize._16, 1 },
|
|
|
|
{ GalVertexAttribSize._8, 1 },
|
|
|
|
{ GalVertexAttribSize._10_10_10_2, 4 },
|
|
|
|
{ GalVertexAttribSize._11_11_10, 3 }
|
|
|
|
};
|
|
|
|
|
|
|
|
private static Dictionary<GalVertexAttribSize, VertexAttribPointerType> AttribTypes =
|
|
|
|
new Dictionary<GalVertexAttribSize, VertexAttribPointerType>()
|
|
|
|
{
|
|
|
|
{ GalVertexAttribSize._32_32_32_32, VertexAttribPointerType.Int },
|
|
|
|
{ GalVertexAttribSize._32_32_32, VertexAttribPointerType.Int },
|
|
|
|
{ GalVertexAttribSize._16_16_16_16, VertexAttribPointerType.Short },
|
|
|
|
{ GalVertexAttribSize._32_32, VertexAttribPointerType.Int },
|
|
|
|
{ GalVertexAttribSize._16_16_16, VertexAttribPointerType.Short },
|
|
|
|
{ GalVertexAttribSize._8_8_8_8, VertexAttribPointerType.Byte },
|
|
|
|
{ GalVertexAttribSize._16_16, VertexAttribPointerType.Short },
|
|
|
|
{ GalVertexAttribSize._32, VertexAttribPointerType.Int },
|
|
|
|
{ GalVertexAttribSize._8_8_8, VertexAttribPointerType.Byte },
|
|
|
|
{ GalVertexAttribSize._8_8, VertexAttribPointerType.Byte },
|
|
|
|
{ GalVertexAttribSize._16, VertexAttribPointerType.Short },
|
|
|
|
{ GalVertexAttribSize._8, VertexAttribPointerType.Byte },
|
|
|
|
{ GalVertexAttribSize._10_10_10_2, VertexAttribPointerType.Int }, //?
|
|
|
|
{ GalVertexAttribSize._11_11_10, VertexAttribPointerType.Int } //?
|
|
|
|
};
|
|
|
|
|
2018-06-08 20:15:56 -04:00
|
|
|
private int VaoHandle;
|
|
|
|
|
|
|
|
private int[] VertexBuffers;
|
|
|
|
|
|
|
|
private OGLCachedResource<int> VboCache;
|
|
|
|
private OGLCachedResource<int> IboCache;
|
|
|
|
|
2018-04-08 15:17:35 -04:00
|
|
|
private struct IbInfo
|
|
|
|
{
|
|
|
|
public int Count;
|
|
|
|
|
|
|
|
public DrawElementsType Type;
|
|
|
|
}
|
|
|
|
|
|
|
|
private IbInfo IndexBuffer;
|
|
|
|
|
|
|
|
public OGLRasterizer()
|
|
|
|
{
|
2018-04-29 16:58:38 -04:00
|
|
|
VertexBuffers = new int[32];
|
2018-04-08 15:17:35 -04:00
|
|
|
|
2018-06-08 20:15:56 -04:00
|
|
|
VboCache = new OGLCachedResource<int>(GL.DeleteBuffer);
|
|
|
|
IboCache = new OGLCachedResource<int>(GL.DeleteBuffer);
|
|
|
|
|
2018-04-08 15:17:35 -04:00
|
|
|
IndexBuffer = new IbInfo();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void ClearBuffers(int RtIndex, GalClearBufferFlags Flags)
|
|
|
|
{
|
|
|
|
ClearBufferMask Mask = 0;
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
//TODO: Use glColorMask to clear just the specified channels.
|
2018-04-08 15:17:35 -04:00
|
|
|
if (Flags.HasFlag(GalClearBufferFlags.ColorRed) &&
|
|
|
|
Flags.HasFlag(GalClearBufferFlags.ColorGreen) &&
|
|
|
|
Flags.HasFlag(GalClearBufferFlags.ColorBlue) &&
|
|
|
|
Flags.HasFlag(GalClearBufferFlags.ColorAlpha))
|
|
|
|
{
|
|
|
|
Mask = ClearBufferMask.ColorBufferBit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Flags.HasFlag(GalClearBufferFlags.Depth))
|
|
|
|
{
|
|
|
|
Mask |= ClearBufferMask.DepthBufferBit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Flags.HasFlag(GalClearBufferFlags.Stencil))
|
|
|
|
{
|
|
|
|
Mask |= ClearBufferMask.StencilBufferBit;
|
|
|
|
}
|
|
|
|
|
|
|
|
GL.Clear(Mask);
|
|
|
|
}
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
public bool IsVboCached(long Key, long DataSize)
|
2018-06-08 20:15:56 -04:00
|
|
|
{
|
2018-06-23 20:39:25 -04:00
|
|
|
return VboCache.TryGetSize(Key, out long Size) && Size == DataSize;
|
2018-06-08 20:15:56 -04:00
|
|
|
}
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
public bool IsIboCached(long Key, long DataSize)
|
2018-06-08 20:15:56 -04:00
|
|
|
{
|
2018-06-23 20:39:25 -04:00
|
|
|
return IboCache.TryGetSize(Key, out long Size) && Size == DataSize;
|
2018-06-08 20:15:56 -04:00
|
|
|
}
|
|
|
|
|
2018-06-27 00:32:28 -04:00
|
|
|
public void EnableCullFace()
|
|
|
|
{
|
|
|
|
GL.Enable(EnableCap.CullFace);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void DisableCullFace()
|
|
|
|
{
|
|
|
|
GL.Disable(EnableCap.CullFace);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void EnableDepthTest()
|
|
|
|
{
|
|
|
|
GL.Enable(EnableCap.DepthTest);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void DisableDepthTest()
|
|
|
|
{
|
|
|
|
GL.Disable(EnableCap.DepthTest);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void SetDepthFunction(GalComparisonOp Func)
|
|
|
|
{
|
|
|
|
GL.DepthFunc(OGLEnumConverter.GetDepthFunc(Func));
|
|
|
|
}
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
public void CreateVbo(long Key, byte[] Buffer)
|
2018-04-08 15:17:35 -04:00
|
|
|
{
|
2018-06-08 20:15:56 -04:00
|
|
|
int Handle = GL.GenBuffer();
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
VboCache.AddOrUpdate(Key, Handle, (uint)Buffer.Length);
|
2018-04-08 15:17:35 -04:00
|
|
|
|
|
|
|
IntPtr Length = new IntPtr(Buffer.Length);
|
|
|
|
|
2018-06-08 20:15:56 -04:00
|
|
|
GL.BindBuffer(BufferTarget.ArrayBuffer, Handle);
|
2018-04-08 15:17:35 -04:00
|
|
|
GL.BufferData(BufferTarget.ArrayBuffer, Length, Buffer, BufferUsageHint.StreamDraw);
|
2018-06-08 20:15:56 -04:00
|
|
|
}
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
public void CreateIbo(long Key, byte[] Buffer)
|
2018-06-08 20:15:56 -04:00
|
|
|
{
|
|
|
|
int Handle = GL.GenBuffer();
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
IboCache.AddOrUpdate(Key, Handle, (uint)Buffer.Length);
|
2018-06-08 20:15:56 -04:00
|
|
|
|
|
|
|
IntPtr Length = new IntPtr(Buffer.Length);
|
|
|
|
|
|
|
|
GL.BindBuffer(BufferTarget.ElementArrayBuffer, Handle);
|
|
|
|
GL.BufferData(BufferTarget.ElementArrayBuffer, Length, Buffer, BufferUsageHint.StreamDraw);
|
|
|
|
}
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
public void SetVertexArray(int VbIndex, int Stride, long VboKey, GalVertexAttrib[] Attribs)
|
2018-06-08 20:15:56 -04:00
|
|
|
{
|
2018-06-23 20:39:25 -04:00
|
|
|
if (!VboCache.TryGetValue(VboKey, out int VboHandle))
|
2018-06-08 20:15:56 -04:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VaoHandle == 0)
|
|
|
|
{
|
|
|
|
VaoHandle = GL.GenVertexArray();
|
|
|
|
}
|
2018-04-08 15:17:35 -04:00
|
|
|
|
2018-04-29 16:58:38 -04:00
|
|
|
GL.BindVertexArray(VaoHandle);
|
2018-04-08 15:17:35 -04:00
|
|
|
|
2018-04-29 16:58:38 -04:00
|
|
|
foreach (GalVertexAttrib Attrib in Attribs)
|
2018-04-08 15:17:35 -04:00
|
|
|
{
|
2018-04-29 16:58:38 -04:00
|
|
|
GL.EnableVertexAttribArray(Attrib.Index);
|
2018-04-08 15:17:35 -04:00
|
|
|
|
2018-06-08 20:15:56 -04:00
|
|
|
GL.BindBuffer(BufferTarget.ArrayBuffer, VboHandle);
|
2018-04-08 15:17:35 -04:00
|
|
|
|
|
|
|
bool Unsigned =
|
|
|
|
Attrib.Type == GalVertexAttribType.Unorm ||
|
|
|
|
Attrib.Type == GalVertexAttribType.Uint ||
|
|
|
|
Attrib.Type == GalVertexAttribType.Uscaled;
|
|
|
|
|
|
|
|
bool Normalize =
|
|
|
|
Attrib.Type == GalVertexAttribType.Snorm ||
|
|
|
|
Attrib.Type == GalVertexAttribType.Unorm;
|
|
|
|
|
|
|
|
VertexAttribPointerType Type = 0;
|
|
|
|
|
|
|
|
if (Attrib.Type == GalVertexAttribType.Float)
|
|
|
|
{
|
|
|
|
Type = VertexAttribPointerType.Float;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Type = AttribTypes[Attrib.Size] + (Unsigned ? 1 : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int Size = AttribElements[Attrib.Size];
|
|
|
|
int Offset = Attrib.Offset;
|
|
|
|
|
2018-04-29 16:58:38 -04:00
|
|
|
GL.VertexAttribPointer(Attrib.Index, Size, Type, Normalize, Stride, Offset);
|
2018-04-08 15:17:35 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
public void SetIndexArray(long Key, int Size, GalIndexFormat Format)
|
2018-04-08 15:17:35 -04:00
|
|
|
{
|
|
|
|
IndexBuffer.Type = OGLEnumConverter.GetDrawElementsType(Format);
|
|
|
|
|
2018-06-08 20:15:56 -04:00
|
|
|
IndexBuffer.Count = Size >> (int)Format;
|
2018-04-08 15:17:35 -04:00
|
|
|
}
|
|
|
|
|
2018-06-08 20:15:56 -04:00
|
|
|
public void DrawArrays(int First, int PrimCount, GalPrimitiveType PrimType)
|
2018-04-08 15:17:35 -04:00
|
|
|
{
|
2018-04-25 22:11:26 -04:00
|
|
|
if (PrimCount == 0)
|
2018-04-08 15:17:35 -04:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-04-29 16:58:38 -04:00
|
|
|
GL.BindVertexArray(VaoHandle);
|
2018-04-08 15:17:35 -04:00
|
|
|
|
2018-04-25 22:11:26 -04:00
|
|
|
GL.DrawArrays(OGLEnumConverter.GetPrimitiveType(PrimType), First, PrimCount);
|
2018-04-08 15:17:35 -04:00
|
|
|
}
|
|
|
|
|
2018-06-23 20:39:25 -04:00
|
|
|
public void DrawElements(long IboKey, int First, GalPrimitiveType PrimType)
|
2018-04-08 15:17:35 -04:00
|
|
|
{
|
2018-06-23 20:39:25 -04:00
|
|
|
if (!IboCache.TryGetValue(IboKey, out int IboHandle))
|
2018-06-08 20:15:56 -04:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-04-08 15:17:35 -04:00
|
|
|
PrimitiveType Mode = OGLEnumConverter.GetPrimitiveType(PrimType);
|
|
|
|
|
2018-04-29 16:58:38 -04:00
|
|
|
GL.BindVertexArray(VaoHandle);
|
2018-04-08 15:17:35 -04:00
|
|
|
|
2018-06-08 20:15:56 -04:00
|
|
|
GL.BindBuffer(BufferTarget.ElementArrayBuffer, IboHandle);
|
2018-04-08 15:17:35 -04:00
|
|
|
|
|
|
|
GL.DrawElements(Mode, IndexBuffer.Count, IndexBuffer.Type, First);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|