192 lines
		
	
	
	
		
			5.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			192 lines
		
	
	
	
		
			5.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using OpenTK.Graphics.OpenGL;
 | |
| using Ryujinx.Graphics.GAL;
 | |
| using System;
 | |
| using System.Runtime.CompilerServices;
 | |
| 
 | |
| namespace Ryujinx.Graphics.OpenGL
 | |
| {
 | |
|     class VertexArray : IDisposable
 | |
|     {
 | |
|         public int Handle { get; private set; }
 | |
| 
 | |
|         private bool _needsAttribsUpdate;
 | |
| 
 | |
|         private readonly VertexAttribDescriptor[] _vertexAttribs;
 | |
|         private readonly VertexBufferDescriptor[] _vertexBuffers;
 | |
| 
 | |
|         private int _vertexAttribsCount;
 | |
|         private int _vertexBuffersCount;
 | |
| 
 | |
|         private uint _vertexAttribsInUse;
 | |
|         private uint _vertexBuffersInUse;
 | |
| 
 | |
|         public VertexArray()
 | |
|         {
 | |
|             Handle = GL.GenVertexArray();
 | |
| 
 | |
|             _vertexAttribs = new VertexAttribDescriptor[Constants.MaxVertexAttribs];
 | |
|             _vertexBuffers = new VertexBufferDescriptor[Constants.MaxVertexBuffers];
 | |
|         }
 | |
| 
 | |
|         public void Bind()
 | |
|         {
 | |
|             GL.BindVertexArray(Handle);
 | |
|         }
 | |
| 
 | |
|         public void SetVertexBuffers(ReadOnlySpan<VertexBufferDescriptor> vertexBuffers)
 | |
|         {
 | |
|             int bindingIndex;
 | |
|             for (bindingIndex = 0; bindingIndex < vertexBuffers.Length; bindingIndex++)
 | |
|             {
 | |
|                 VertexBufferDescriptor vb = vertexBuffers[bindingIndex];
 | |
| 
 | |
|                 if (vb.Buffer.Handle != BufferHandle.Null)
 | |
|                 {
 | |
|                     GL.BindVertexBuffer(bindingIndex, vb.Buffer.Handle.ToInt32(), (IntPtr)vb.Buffer.Offset, vb.Stride);
 | |
|                     GL.VertexBindingDivisor(bindingIndex, vb.Divisor);
 | |
|                     _vertexBuffersInUse |= 1u << bindingIndex;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     if ((_vertexBuffersInUse & (1u << bindingIndex)) != 0)
 | |
|                     {
 | |
|                         GL.BindVertexBuffer(bindingIndex, 0, IntPtr.Zero, 0);
 | |
|                         _vertexBuffersInUse &= ~(1u << bindingIndex);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 _vertexBuffers[bindingIndex] = vb;
 | |
|             }
 | |
| 
 | |
|             _vertexBuffersCount = bindingIndex;
 | |
|             _needsAttribsUpdate = true;
 | |
|         }
 | |
| 
 | |
|         public void SetVertexAttributes(ReadOnlySpan<VertexAttribDescriptor> vertexAttribs)
 | |
|         {
 | |
|             int index = 0;
 | |
| 
 | |
|             for (; index < vertexAttribs.Length; index++)
 | |
|             {
 | |
|                 VertexAttribDescriptor attrib = vertexAttribs[index];
 | |
| 
 | |
|                 if (attrib.Equals(_vertexAttribs[index]))
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 FormatInfo fmtInfo = FormatTable.GetFormatInfo(attrib.Format);
 | |
| 
 | |
|                 if (attrib.IsZero)
 | |
|                 {
 | |
|                     // Disabling the attribute causes the shader to read a constant value.
 | |
|                     // The value is configurable, but by default is a vector of (0, 0, 0, 1).
 | |
|                     DisableVertexAttrib(index);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     EnableVertexAttrib(index);
 | |
|                 }
 | |
| 
 | |
|                 int offset = attrib.Offset;
 | |
|                 int size   = fmtInfo.Components;
 | |
| 
 | |
|                 bool isFloat = fmtInfo.PixelType == PixelType.Float ||
 | |
|                                fmtInfo.PixelType == PixelType.HalfFloat;
 | |
| 
 | |
|                 if (isFloat || fmtInfo.Normalized || fmtInfo.Scaled)
 | |
|                 {
 | |
|                     VertexAttribType type = (VertexAttribType)fmtInfo.PixelType;
 | |
| 
 | |
|                     GL.VertexAttribFormat(index, size, type, fmtInfo.Normalized, offset);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     VertexAttribIntegerType type = (VertexAttribIntegerType)fmtInfo.PixelType;
 | |
| 
 | |
|                     GL.VertexAttribIFormat(index, size, type, offset);
 | |
|                 }
 | |
| 
 | |
|                 GL.VertexAttribBinding(index, attrib.BufferIndex);
 | |
| 
 | |
|                 _vertexAttribs[index] = attrib;
 | |
|             }
 | |
| 
 | |
|             _vertexAttribsCount = index;
 | |
| 
 | |
|             for (; index < Constants.MaxVertexAttribs; index++)
 | |
|             {
 | |
|                 DisableVertexAttrib(index);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void SetIndexBuffer(BufferHandle buffer)
 | |
|         {
 | |
|             GL.BindBuffer(BufferTarget.ElementArrayBuffer, buffer.ToInt32());
 | |
|         }
 | |
| 
 | |
|         public void Validate()
 | |
|         {
 | |
|             for (int attribIndex = 0; attribIndex < _vertexAttribsCount; attribIndex++)
 | |
|             {
 | |
|                 VertexAttribDescriptor attrib = _vertexAttribs[attribIndex];
 | |
| 
 | |
|                 if (!attrib.IsZero)
 | |
|                 {
 | |
|                     if ((uint)attrib.BufferIndex >= _vertexBuffersCount)
 | |
|                     {
 | |
|                         DisableVertexAttrib(attribIndex);
 | |
|                         continue;
 | |
|                     }
 | |
| 
 | |
|                     if (_vertexBuffers[attrib.BufferIndex].Buffer.Handle == BufferHandle.Null)
 | |
|                     {
 | |
|                         DisableVertexAttrib(attribIndex);
 | |
|                         continue;
 | |
|                     }
 | |
| 
 | |
|                     if (_needsAttribsUpdate)
 | |
|                     {
 | |
|                         EnableVertexAttrib(attribIndex);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             _needsAttribsUpdate = false;
 | |
|         }
 | |
| 
 | |
|         [MethodImpl(MethodImplOptions.AggressiveInlining)]
 | |
|         private void EnableVertexAttrib(int index)
 | |
|         {
 | |
|             uint mask = 1u << index;
 | |
| 
 | |
|             if ((_vertexAttribsInUse & mask) == 0)
 | |
|             {
 | |
|                 _vertexAttribsInUse |= mask;
 | |
|                 GL.EnableVertexAttribArray(index);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [MethodImpl(MethodImplOptions.AggressiveInlining)]
 | |
|         private void DisableVertexAttrib(int index)
 | |
|         {
 | |
|             uint mask = 1u << index;
 | |
| 
 | |
|             if ((_vertexAttribsInUse & mask) != 0)
 | |
|             {
 | |
|                 _vertexAttribsInUse &= ~mask;
 | |
|                 GL.DisableVertexAttribArray(index);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void Dispose()
 | |
|         {
 | |
|             if (Handle != 0)
 | |
|             {
 | |
|                 GL.DeleteVertexArray(Handle);
 | |
| 
 | |
|                 Handle = 0;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | 
