mirror of
				https://github.com/Ryujinx/Ryujinx.git
				synced 2025-10-25 19:43:56 -07:00 
			
		
		
		
	* Initial implementation of KProcess * Some improvements to the memory manager, implement back guest stack trace printing * Better GetInfo implementation, improve checking in some places with information from process capabilities * Allow the cpu to read/write from the correct memory locations for accesses crossing a page boundary * Change long -> ulong for address/size on memory related methods to avoid unnecessary casts * Attempt at implementing ldr:ro with new KProcess * Allow BSS with size 0 on ldr:ro * Add checking for memory block slab heap usage, return errors if full, exit gracefully * Use KMemoryBlockSize const from KMemoryManager * Allow all methods to read from non-contiguous locations * Fix for TransactParcelAuto * Address PR feedback, additionally fix some small issues related to the KIP loader and implement SVCs GetProcessId, GetProcessList, GetSystemInfo, CreatePort and ManageNamedPort * Fix wrong check for source pages count from page list on MapPhysicalMemory * Fix some issues with UnloadNro on ldr:ro
		
			
				
	
	
		
			207 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			207 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using OpenTK.Graphics.OpenGL;
 | |
| using System;
 | |
| 
 | |
| namespace Ryujinx.Graphics.Gal.OpenGL
 | |
| {
 | |
|     class OGLRasterizer : IGalRasterizer
 | |
|     {
 | |
|         private const long MaxVertexBufferCacheSize = 128 * 1024 * 1024;
 | |
|         private const long MaxIndexBufferCacheSize  = 64  * 1024 * 1024;
 | |
| 
 | |
|         private int[] VertexBuffers;
 | |
| 
 | |
|         private OGLCachedResource<int> VboCache;
 | |
|         private OGLCachedResource<int> IboCache;
 | |
| 
 | |
|         private struct IbInfo
 | |
|         {
 | |
|             public int Count;
 | |
|             public int ElemSizeLog2;
 | |
| 
 | |
|             public DrawElementsType Type;
 | |
|         }
 | |
| 
 | |
|         private IbInfo IndexBuffer;
 | |
| 
 | |
|         public OGLRasterizer()
 | |
|         {
 | |
|             VertexBuffers = new int[32];
 | |
| 
 | |
|             VboCache = new OGLCachedResource<int>(GL.DeleteBuffer, MaxVertexBufferCacheSize);
 | |
|             IboCache = new OGLCachedResource<int>(GL.DeleteBuffer, MaxIndexBufferCacheSize);
 | |
| 
 | |
|             IndexBuffer = new IbInfo();
 | |
|         }
 | |
| 
 | |
|         public void LockCaches()
 | |
|         {
 | |
|             VboCache.Lock();
 | |
|             IboCache.Lock();
 | |
|         }
 | |
| 
 | |
|         public void UnlockCaches()
 | |
|         {
 | |
|             VboCache.Unlock();
 | |
|             IboCache.Unlock();
 | |
|         }
 | |
| 
 | |
|         public void ClearBuffers(
 | |
|             GalClearBufferFlags Flags,
 | |
|             int Attachment,
 | |
|             float Red,
 | |
|             float Green,
 | |
|             float Blue,
 | |
|             float Alpha,
 | |
|             float Depth,
 | |
|             int Stencil)
 | |
|         {
 | |
|             GL.ColorMask(
 | |
|                 Attachment,
 | |
|                 Flags.HasFlag(GalClearBufferFlags.ColorRed),
 | |
|                 Flags.HasFlag(GalClearBufferFlags.ColorGreen),
 | |
|                 Flags.HasFlag(GalClearBufferFlags.ColorBlue),
 | |
|                 Flags.HasFlag(GalClearBufferFlags.ColorAlpha));
 | |
| 
 | |
|             GL.ClearBuffer(ClearBuffer.Color, Attachment, new float[] { Red, Green, Blue, Alpha });
 | |
| 
 | |
|             GL.ColorMask(Attachment, true, true, true, true);
 | |
|             GL.DepthMask(true);
 | |
| 
 | |
|             if (Flags.HasFlag(GalClearBufferFlags.Depth))
 | |
|             {
 | |
|                 GL.ClearBuffer(ClearBuffer.Depth, 0, ref Depth);
 | |
|             }
 | |
| 
 | |
|             if (Flags.HasFlag(GalClearBufferFlags.Stencil))
 | |
|             {
 | |
|                 GL.ClearBuffer(ClearBuffer.Stencil, 0, ref Stencil);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool IsVboCached(long Key, long DataSize)
 | |
|         {
 | |
|             return VboCache.TryGetSize(Key, out long Size) && Size == DataSize;
 | |
|         }
 | |
| 
 | |
|         public bool IsIboCached(long Key, long DataSize)
 | |
|         {
 | |
|             return IboCache.TryGetSize(Key, out long Size) && Size == DataSize;
 | |
|         }
 | |
| 
 | |
|         public void CreateVbo(long Key, int DataSize, IntPtr HostAddress)
 | |
|         {
 | |
|             int Handle = GL.GenBuffer();
 | |
| 
 | |
|             VboCache.AddOrUpdate(Key, Handle, DataSize);
 | |
| 
 | |
|             IntPtr Length = new IntPtr(DataSize);
 | |
| 
 | |
|             GL.BindBuffer(BufferTarget.ArrayBuffer, Handle);
 | |
|             GL.BufferData(BufferTarget.ArrayBuffer, Length, HostAddress, BufferUsageHint.StreamDraw);
 | |
|         }
 | |
| 
 | |
|         public void CreateVbo(long Key, byte[] Data)
 | |
|         {
 | |
|             int Handle = GL.GenBuffer();
 | |
| 
 | |
|             VboCache.AddOrUpdate(Key, Handle, Data.Length);
 | |
| 
 | |
|             IntPtr Length = new IntPtr(Data.Length);
 | |
| 
 | |
|             GL.BindBuffer(BufferTarget.ArrayBuffer, Handle);
 | |
|             GL.BufferData(BufferTarget.ArrayBuffer, Length, Data, BufferUsageHint.StreamDraw);
 | |
|         }
 | |
| 
 | |
|         public void CreateIbo(long Key, int DataSize, IntPtr HostAddress)
 | |
|         {
 | |
|             int Handle = GL.GenBuffer();
 | |
| 
 | |
|             IboCache.AddOrUpdate(Key, Handle, (uint)DataSize);
 | |
| 
 | |
|             IntPtr Length = new IntPtr(DataSize);
 | |
| 
 | |
|             GL.BindBuffer(BufferTarget.ElementArrayBuffer, Handle);
 | |
|             GL.BufferData(BufferTarget.ElementArrayBuffer, Length, HostAddress, BufferUsageHint.StreamDraw);
 | |
|         }
 | |
| 
 | |
|         public void CreateIbo(long Key, int DataSize, byte[] Buffer)
 | |
|         {
 | |
|             int Handle = GL.GenBuffer();
 | |
| 
 | |
|             IboCache.AddOrUpdate(Key, Handle, DataSize);
 | |
| 
 | |
|             IntPtr Length = new IntPtr(Buffer.Length);
 | |
| 
 | |
|             GL.BindBuffer(BufferTarget.ElementArrayBuffer, Handle);
 | |
|             GL.BufferData(BufferTarget.ElementArrayBuffer, Length, Buffer, BufferUsageHint.StreamDraw);
 | |
|         }
 | |
| 
 | |
|         public void SetIndexArray(int Size, GalIndexFormat Format)
 | |
|         {
 | |
|             IndexBuffer.Type = OGLEnumConverter.GetDrawElementsType(Format);
 | |
| 
 | |
|             IndexBuffer.Count = Size >> (int)Format;
 | |
| 
 | |
|             IndexBuffer.ElemSizeLog2 = (int)Format;
 | |
|         }
 | |
| 
 | |
|         public void DrawArrays(int First, int Count, GalPrimitiveType PrimType)
 | |
|         {
 | |
|             if (Count == 0)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             if (PrimType == GalPrimitiveType.Quads)
 | |
|             {
 | |
|                 for (int Offset = 0; Offset < Count; Offset += 4)
 | |
|                 {
 | |
|                     GL.DrawArrays(PrimitiveType.TriangleFan, First + Offset, 4);
 | |
|                 }
 | |
|             }
 | |
|             else if (PrimType == GalPrimitiveType.QuadStrip)
 | |
|             {
 | |
|                 GL.DrawArrays(PrimitiveType.TriangleFan, First, 4);
 | |
| 
 | |
|                 for (int Offset = 2; Offset < Count; Offset += 2)
 | |
|                 {
 | |
|                     GL.DrawArrays(PrimitiveType.TriangleFan, First + Offset, 4);
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 GL.DrawArrays(OGLEnumConverter.GetPrimitiveType(PrimType), First, Count);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void DrawElements(long IboKey, int First, int VertexBase, GalPrimitiveType PrimType)
 | |
|         {
 | |
|             if (!IboCache.TryGetValue(IboKey, out int IboHandle))
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             PrimitiveType Mode = OGLEnumConverter.GetPrimitiveType(PrimType);
 | |
| 
 | |
|             GL.BindBuffer(BufferTarget.ElementArrayBuffer, IboHandle);
 | |
| 
 | |
|             First <<= IndexBuffer.ElemSizeLog2;
 | |
| 
 | |
|             if (VertexBase != 0)
 | |
|             {
 | |
|                 IntPtr Indices = new IntPtr(First);
 | |
| 
 | |
|                 GL.DrawElementsBaseVertex(Mode, IndexBuffer.Count, IndexBuffer.Type, Indices, VertexBase);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 GL.DrawElements(Mode, IndexBuffer.Count, IndexBuffer.Type, First);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool TryGetVbo(long VboKey, out int VboHandle)
 | |
|         {
 | |
|             return VboCache.TryGetValue(VboKey, out VboHandle);
 | |
|         }
 | |
|     }
 | |
| } |