mirror of
https://github.com/Ryujinx/Ryujinx.git
synced 2025-01-22 16:41:39 +00:00
f5b4f6ccc4
* Handle negative viewport coordinates * Disable scissor before framebuffer blit * Comment to explain scissor disable will be reenabled if needed * Comma and spelling mistake
527 lines
16 KiB
C#
527 lines
16 KiB
C#
using OpenTK.Graphics.OpenGL;
|
|
using Ryujinx.Graphics.Texture;
|
|
using System;
|
|
|
|
namespace Ryujinx.Graphics.Gal.OpenGL
|
|
{
|
|
class OGLRenderTarget : IGalRenderTarget
|
|
{
|
|
private const int NativeWidth = 1280;
|
|
private const int NativeHeight = 720;
|
|
|
|
private const int RenderTargetsCount = GalPipelineState.RenderTargetsCount;
|
|
|
|
private struct Rect
|
|
{
|
|
public int X { get; private set; }
|
|
public int Y { get; private set; }
|
|
public int Width { get; private set; }
|
|
public int Height { get; private set; }
|
|
|
|
public Rect(int X, int Y, int Width, int Height)
|
|
{
|
|
this.X = X;
|
|
this.Y = Y;
|
|
this.Width = Width;
|
|
this.Height = Height;
|
|
}
|
|
}
|
|
|
|
private class FrameBufferAttachments
|
|
{
|
|
public int MapCount { get; set; }
|
|
|
|
public DrawBuffersEnum[] Map { get; private set; }
|
|
|
|
public long[] Colors { get; private set; }
|
|
|
|
public long Zeta { get; set; }
|
|
|
|
public FrameBufferAttachments()
|
|
{
|
|
Colors = new long[RenderTargetsCount];
|
|
|
|
Map = new DrawBuffersEnum[RenderTargetsCount];
|
|
}
|
|
|
|
public void Update(FrameBufferAttachments Source)
|
|
{
|
|
for (int Index = 0; Index < RenderTargetsCount; Index++)
|
|
{
|
|
Map[Index] = Source.Map[Index];
|
|
|
|
Colors[Index] = Source.Colors[Index];
|
|
}
|
|
|
|
MapCount = Source.MapCount;
|
|
Zeta = Source.Zeta;
|
|
}
|
|
}
|
|
|
|
private int[] ColorHandles;
|
|
private int ZetaHandle;
|
|
|
|
private OGLTexture Texture;
|
|
|
|
private ImageHandler ReadTex;
|
|
|
|
private Rect Window;
|
|
|
|
private float[] Viewports;
|
|
|
|
private bool FlipX;
|
|
private bool FlipY;
|
|
|
|
private int CropTop;
|
|
private int CropLeft;
|
|
private int CropRight;
|
|
private int CropBottom;
|
|
|
|
//This framebuffer is used to attach guest rendertargets,
|
|
//think of it as a dummy OpenGL VAO
|
|
private int DummyFrameBuffer;
|
|
|
|
//These framebuffers are used to blit images
|
|
private int SrcFb;
|
|
private int DstFb;
|
|
|
|
private FrameBufferAttachments Attachments;
|
|
private FrameBufferAttachments OldAttachments;
|
|
|
|
private int CopyPBO;
|
|
|
|
public bool FramebufferSrgb { get; set; }
|
|
|
|
public OGLRenderTarget(OGLTexture Texture)
|
|
{
|
|
Attachments = new FrameBufferAttachments();
|
|
|
|
OldAttachments = new FrameBufferAttachments();
|
|
|
|
ColorHandles = new int[RenderTargetsCount];
|
|
|
|
Viewports = new float[RenderTargetsCount * 4];
|
|
|
|
this.Texture = Texture;
|
|
|
|
Texture.TextureDeleted += TextureDeletionHandler;
|
|
}
|
|
|
|
private void TextureDeletionHandler(object Sender, int Handle)
|
|
{
|
|
//Texture was deleted, the handle is no longer valid, so
|
|
//reset all uses of this handle on a render target.
|
|
for (int Attachment = 0; Attachment < RenderTargetsCount; Attachment++)
|
|
{
|
|
if (ColorHandles[Attachment] == Handle)
|
|
{
|
|
ColorHandles[Attachment] = 0;
|
|
}
|
|
}
|
|
|
|
if (ZetaHandle == Handle)
|
|
{
|
|
ZetaHandle = 0;
|
|
}
|
|
}
|
|
|
|
public void Bind()
|
|
{
|
|
if (DummyFrameBuffer == 0)
|
|
{
|
|
DummyFrameBuffer = GL.GenFramebuffer();
|
|
}
|
|
|
|
GL.BindFramebuffer(FramebufferTarget.DrawFramebuffer, DummyFrameBuffer);
|
|
|
|
ImageHandler CachedImage;
|
|
|
|
for (int Attachment = 0; Attachment < RenderTargetsCount; Attachment++)
|
|
{
|
|
long Key = Attachments.Colors[Attachment];
|
|
|
|
int Handle = 0;
|
|
|
|
if (Key != 0 && Texture.TryGetImageHandler(Key, out CachedImage))
|
|
{
|
|
Handle = CachedImage.Handle;
|
|
}
|
|
|
|
if (Handle == ColorHandles[Attachment])
|
|
{
|
|
continue;
|
|
}
|
|
|
|
GL.FramebufferTexture(
|
|
FramebufferTarget.DrawFramebuffer,
|
|
FramebufferAttachment.ColorAttachment0 + Attachment,
|
|
Handle,
|
|
0);
|
|
|
|
ColorHandles[Attachment] = Handle;
|
|
}
|
|
|
|
if (Attachments.Zeta != 0 && Texture.TryGetImageHandler(Attachments.Zeta, out CachedImage))
|
|
{
|
|
if (CachedImage.Handle != ZetaHandle)
|
|
{
|
|
if (CachedImage.HasDepth && CachedImage.HasStencil)
|
|
{
|
|
GL.FramebufferTexture(
|
|
FramebufferTarget.DrawFramebuffer,
|
|
FramebufferAttachment.DepthStencilAttachment,
|
|
CachedImage.Handle,
|
|
0);
|
|
}
|
|
else if (CachedImage.HasDepth)
|
|
{
|
|
GL.FramebufferTexture(
|
|
FramebufferTarget.DrawFramebuffer,
|
|
FramebufferAttachment.DepthAttachment,
|
|
CachedImage.Handle,
|
|
0);
|
|
|
|
GL.FramebufferTexture(
|
|
FramebufferTarget.DrawFramebuffer,
|
|
FramebufferAttachment.StencilAttachment,
|
|
0,
|
|
0);
|
|
}
|
|
else
|
|
{
|
|
throw new InvalidOperationException("Invalid image format \"" + CachedImage.Format + "\" used as Zeta!");
|
|
}
|
|
|
|
ZetaHandle = CachedImage.Handle;
|
|
}
|
|
}
|
|
else if (ZetaHandle != 0)
|
|
{
|
|
GL.FramebufferTexture(
|
|
FramebufferTarget.DrawFramebuffer,
|
|
FramebufferAttachment.DepthStencilAttachment,
|
|
0,
|
|
0);
|
|
|
|
ZetaHandle = 0;
|
|
}
|
|
|
|
if (OGLExtension.ViewportArray)
|
|
{
|
|
GL.ViewportArray(0, RenderTargetsCount, Viewports);
|
|
}
|
|
else
|
|
{
|
|
GL.Viewport(
|
|
(int)Viewports[0],
|
|
(int)Viewports[1],
|
|
(int)Viewports[2],
|
|
(int)Viewports[3]);
|
|
}
|
|
|
|
if (Attachments.MapCount > 1)
|
|
{
|
|
GL.DrawBuffers(Attachments.MapCount, Attachments.Map);
|
|
}
|
|
else if (Attachments.MapCount == 1)
|
|
{
|
|
GL.DrawBuffer((DrawBufferMode)Attachments.Map[0]);
|
|
}
|
|
else
|
|
{
|
|
GL.DrawBuffer(DrawBufferMode.None);
|
|
}
|
|
|
|
OldAttachments.Update(Attachments);
|
|
}
|
|
|
|
public void BindColor(long Key, int Attachment)
|
|
{
|
|
Attachments.Colors[Attachment] = Key;
|
|
}
|
|
|
|
public void UnbindColor(int Attachment)
|
|
{
|
|
Attachments.Colors[Attachment] = 0;
|
|
}
|
|
|
|
public void BindZeta(long Key)
|
|
{
|
|
Attachments.Zeta = Key;
|
|
}
|
|
|
|
public void UnbindZeta()
|
|
{
|
|
Attachments.Zeta = 0;
|
|
}
|
|
|
|
public void Present(long Key)
|
|
{
|
|
Texture.TryGetImageHandler(Key, out ReadTex);
|
|
}
|
|
|
|
public void SetMap(int[] Map)
|
|
{
|
|
if (Map != null)
|
|
{
|
|
Attachments.MapCount = Map.Length;
|
|
|
|
for (int Attachment = 0; Attachment < Attachments.MapCount; Attachment++)
|
|
{
|
|
Attachments.Map[Attachment] = DrawBuffersEnum.ColorAttachment0 + Map[Attachment];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Attachments.MapCount = 0;
|
|
}
|
|
}
|
|
|
|
public void SetTransform(bool FlipX, bool FlipY, int Top, int Left, int Right, int Bottom)
|
|
{
|
|
this.FlipX = FlipX;
|
|
this.FlipY = FlipY;
|
|
|
|
CropTop = Top;
|
|
CropLeft = Left;
|
|
CropRight = Right;
|
|
CropBottom = Bottom;
|
|
}
|
|
|
|
public void SetWindowSize(int Width, int Height)
|
|
{
|
|
Window = new Rect(0, 0, Width, Height);
|
|
}
|
|
|
|
public void SetViewport(int Attachment, int X, int Y, int Width, int Height)
|
|
{
|
|
int Offset = Attachment * 4;
|
|
|
|
Viewports[Offset + 0] = X;
|
|
Viewports[Offset + 1] = Y;
|
|
Viewports[Offset + 2] = Width;
|
|
Viewports[Offset + 3] = Height;
|
|
}
|
|
|
|
public void Render()
|
|
{
|
|
if (ReadTex == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
int SrcX0, SrcX1, SrcY0, SrcY1;
|
|
|
|
if (CropLeft == 0 && CropRight == 0)
|
|
{
|
|
SrcX0 = 0;
|
|
SrcX1 = ReadTex.Width;
|
|
}
|
|
else
|
|
{
|
|
SrcX0 = CropLeft;
|
|
SrcX1 = CropRight;
|
|
}
|
|
|
|
if (CropTop == 0 && CropBottom == 0)
|
|
{
|
|
SrcY0 = 0;
|
|
SrcY1 = ReadTex.Height;
|
|
}
|
|
else
|
|
{
|
|
SrcY0 = CropTop;
|
|
SrcY1 = CropBottom;
|
|
}
|
|
|
|
float RatioX = MathF.Min(1f, (Window.Height * (float)NativeWidth) / ((float)NativeHeight * Window.Width));
|
|
float RatioY = MathF.Min(1f, (Window.Width * (float)NativeHeight) / ((float)NativeWidth * Window.Height));
|
|
|
|
int DstWidth = (int)(Window.Width * RatioX);
|
|
int DstHeight = (int)(Window.Height * RatioY);
|
|
|
|
int DstPaddingX = (Window.Width - DstWidth) / 2;
|
|
int DstPaddingY = (Window.Height - DstHeight) / 2;
|
|
|
|
int DstX0 = FlipX ? Window.Width - DstPaddingX : DstPaddingX;
|
|
int DstX1 = FlipX ? DstPaddingX : Window.Width - DstPaddingX;
|
|
|
|
int DstY0 = FlipY ? DstPaddingY : Window.Height - DstPaddingY;
|
|
int DstY1 = FlipY ? Window.Height - DstPaddingY : DstPaddingY;
|
|
|
|
GL.Viewport(0, 0, Window.Width, Window.Height);
|
|
|
|
if (SrcFb == 0)
|
|
{
|
|
SrcFb = GL.GenFramebuffer();
|
|
}
|
|
|
|
GL.BindFramebuffer(FramebufferTarget.ReadFramebuffer, SrcFb);
|
|
GL.BindFramebuffer(FramebufferTarget.DrawFramebuffer, 0);
|
|
|
|
GL.FramebufferTexture(FramebufferTarget.ReadFramebuffer, FramebufferAttachment.ColorAttachment0, ReadTex.Handle, 0);
|
|
|
|
GL.ReadBuffer(ReadBufferMode.ColorAttachment0);
|
|
|
|
GL.Clear(ClearBufferMask.ColorBufferBit);
|
|
|
|
GL.Disable(EnableCap.FramebufferSrgb);
|
|
|
|
// Will be re-enabled if needed while binding, called before any game GL calls
|
|
GL.Disable(EnableCap.ScissorTest);
|
|
|
|
GL.BlitFramebuffer(
|
|
SrcX0,
|
|
SrcY0,
|
|
SrcX1,
|
|
SrcY1,
|
|
DstX0,
|
|
DstY0,
|
|
DstX1,
|
|
DstY1,
|
|
ClearBufferMask.ColorBufferBit,
|
|
BlitFramebufferFilter.Linear);
|
|
|
|
if (FramebufferSrgb)
|
|
{
|
|
GL.Enable(EnableCap.FramebufferSrgb);
|
|
}
|
|
}
|
|
|
|
public void Copy(
|
|
long SrcKey,
|
|
long DstKey,
|
|
int SrcX0,
|
|
int SrcY0,
|
|
int SrcX1,
|
|
int SrcY1,
|
|
int DstX0,
|
|
int DstY0,
|
|
int DstX1,
|
|
int DstY1)
|
|
{
|
|
if (Texture.TryGetImageHandler(SrcKey, out ImageHandler SrcTex) &&
|
|
Texture.TryGetImageHandler(DstKey, out ImageHandler DstTex))
|
|
{
|
|
if (SrcTex.HasColor != DstTex.HasColor ||
|
|
SrcTex.HasDepth != DstTex.HasDepth ||
|
|
SrcTex.HasStencil != DstTex.HasStencil)
|
|
{
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
if (SrcFb == 0)
|
|
{
|
|
SrcFb = GL.GenFramebuffer();
|
|
}
|
|
|
|
if (DstFb == 0)
|
|
{
|
|
DstFb = GL.GenFramebuffer();
|
|
}
|
|
|
|
GL.BindFramebuffer(FramebufferTarget.ReadFramebuffer, SrcFb);
|
|
GL.BindFramebuffer(FramebufferTarget.DrawFramebuffer, DstFb);
|
|
|
|
FramebufferAttachment Attachment = GetAttachment(SrcTex);
|
|
|
|
GL.FramebufferTexture(FramebufferTarget.ReadFramebuffer, Attachment, SrcTex.Handle, 0);
|
|
GL.FramebufferTexture(FramebufferTarget.DrawFramebuffer, Attachment, DstTex.Handle, 0);
|
|
|
|
BlitFramebufferFilter Filter = BlitFramebufferFilter.Nearest;
|
|
|
|
if (SrcTex.HasColor)
|
|
{
|
|
GL.DrawBuffer(DrawBufferMode.ColorAttachment0);
|
|
|
|
Filter = BlitFramebufferFilter.Linear;
|
|
}
|
|
|
|
ClearBufferMask Mask = GetClearMask(SrcTex);
|
|
|
|
GL.BlitFramebuffer(SrcX0, SrcY0, SrcX1, SrcY1, DstX0, DstY0, DstX1, DstY1, Mask, Filter);
|
|
}
|
|
}
|
|
|
|
public void Reinterpret(long Key, GalImage NewImage)
|
|
{
|
|
if (!Texture.TryGetImage(Key, out GalImage OldImage))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (NewImage.Format == OldImage.Format &&
|
|
NewImage.Width == OldImage.Width &&
|
|
NewImage.Height == OldImage.Height)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (CopyPBO == 0)
|
|
{
|
|
CopyPBO = GL.GenBuffer();
|
|
}
|
|
|
|
GL.BindBuffer(BufferTarget.PixelPackBuffer, CopyPBO);
|
|
|
|
//The buffer should be large enough to hold the largest texture.
|
|
int BufferSize = Math.Max(ImageUtils.GetSize(OldImage),
|
|
ImageUtils.GetSize(NewImage));
|
|
|
|
GL.BufferData(BufferTarget.PixelPackBuffer, BufferSize, IntPtr.Zero, BufferUsageHint.StreamCopy);
|
|
|
|
if (!Texture.TryGetImageHandler(Key, out ImageHandler CachedImage))
|
|
{
|
|
throw new InvalidOperationException();
|
|
}
|
|
|
|
(_, PixelFormat Format, PixelType Type) = OGLEnumConverter.GetImageFormat(CachedImage.Format);
|
|
|
|
GL.BindTexture(TextureTarget.Texture2D, CachedImage.Handle);
|
|
|
|
GL.GetTexImage(TextureTarget.Texture2D, 0, Format, Type, IntPtr.Zero);
|
|
|
|
GL.BindBuffer(BufferTarget.PixelPackBuffer, 0);
|
|
GL.BindBuffer(BufferTarget.PixelUnpackBuffer, CopyPBO);
|
|
|
|
GL.PixelStore(PixelStoreParameter.UnpackRowLength, OldImage.Width);
|
|
|
|
Texture.Create(Key, ImageUtils.GetSize(NewImage), NewImage);
|
|
|
|
GL.PixelStore(PixelStoreParameter.UnpackRowLength, 0);
|
|
|
|
GL.BindBuffer(BufferTarget.PixelUnpackBuffer, 0);
|
|
}
|
|
|
|
private static FramebufferAttachment GetAttachment(ImageHandler CachedImage)
|
|
{
|
|
if (CachedImage.HasColor)
|
|
{
|
|
return FramebufferAttachment.ColorAttachment0;
|
|
}
|
|
else if (CachedImage.HasDepth && CachedImage.HasStencil)
|
|
{
|
|
return FramebufferAttachment.DepthStencilAttachment;
|
|
}
|
|
else if (CachedImage.HasDepth)
|
|
{
|
|
return FramebufferAttachment.DepthAttachment;
|
|
}
|
|
else if (CachedImage.HasStencil)
|
|
{
|
|
return FramebufferAttachment.StencilAttachment;
|
|
}
|
|
else
|
|
{
|
|
throw new InvalidOperationException();
|
|
}
|
|
}
|
|
|
|
private static ClearBufferMask GetClearMask(ImageHandler CachedImage)
|
|
{
|
|
return (CachedImage.HasColor ? ClearBufferMask.ColorBufferBit : 0) |
|
|
(CachedImage.HasDepth ? ClearBufferMask.DepthBufferBit : 0) |
|
|
(CachedImage.HasStencil ? ClearBufferMask.StencilBufferBit : 0);
|
|
}
|
|
}
|
|
} |