2018-02-05 00:08:20 +01:00
|
|
|
// This code was written for the OpenTK library and has been released
|
|
|
|
// to the Public Domain.
|
|
|
|
// It is provided "as is" without express or implied warranty of any kind.
|
|
|
|
|
|
|
|
using OpenTK;
|
|
|
|
using OpenTK.Graphics;
|
|
|
|
using OpenTK.Graphics.OpenGL;
|
2018-02-20 21:09:23 +01:00
|
|
|
using Ryujinx.Core;
|
|
|
|
using Ryujinx.Graphics.Gal;
|
2018-02-05 00:08:20 +01:00
|
|
|
using System;
|
|
|
|
|
|
|
|
namespace Ryujinx
|
|
|
|
{
|
|
|
|
public class GLScreen : GameWindow
|
|
|
|
{
|
|
|
|
class ScreenTexture : IDisposable
|
|
|
|
{
|
|
|
|
private Switch Ns;
|
|
|
|
private IGalRenderer Renderer;
|
|
|
|
|
|
|
|
private int Width;
|
|
|
|
private int Height;
|
|
|
|
private int TexHandle;
|
|
|
|
|
|
|
|
private int[] Pixels;
|
|
|
|
|
|
|
|
public ScreenTexture(Switch Ns, IGalRenderer Renderer, int Width, int Height)
|
|
|
|
{
|
|
|
|
this.Ns = Ns;
|
|
|
|
this.Renderer = Renderer;
|
|
|
|
this.Width = Width;
|
|
|
|
this.Height = Height;
|
|
|
|
|
|
|
|
Pixels = new int[Width * Height];
|
|
|
|
|
|
|
|
TexHandle = GL.GenTexture();
|
|
|
|
|
|
|
|
GL.BindTexture(TextureTarget.Texture2D, TexHandle);
|
|
|
|
GL.TexParameter(TextureTarget.Texture2D, TextureParameterName.TextureMinFilter, (int)TextureMinFilter.Linear);
|
|
|
|
GL.TexParameter(TextureTarget.Texture2D, TextureParameterName.TextureMagFilter, (int)TextureMagFilter.Linear);
|
|
|
|
GL.TexImage2D(TextureTarget.Texture2D,
|
|
|
|
0,
|
|
|
|
PixelInternalFormat.Rgba,
|
|
|
|
Width,
|
|
|
|
Height,
|
|
|
|
0,
|
|
|
|
PixelFormat.Rgba,
|
|
|
|
PixelType.UnsignedByte,
|
|
|
|
IntPtr.Zero);
|
|
|
|
}
|
|
|
|
|
|
|
|
public int Texture
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
UploadBitmap();
|
|
|
|
|
|
|
|
return TexHandle;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe void UploadBitmap()
|
|
|
|
{
|
2018-02-07 17:44:48 +01:00
|
|
|
int FbSize = Width * Height * 4;
|
|
|
|
|
|
|
|
if (Renderer.FrameBufferPtr == 0 || Renderer.FrameBufferPtr + FbSize > uint.MaxValue)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-02-07 17:44:48 +01:00
|
|
|
byte* SrcPtr = (byte*)Ns.Ram + (uint)Renderer.FrameBufferPtr;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
{
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
int SrcOffs = GetSwizzleOffset(X, Y, 4);
|
|
|
|
|
|
|
|
Pixels[X + Y * Width] = *((int*)(SrcPtr + SrcOffs));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GL.BindTexture(TextureTarget.Texture2D, TexHandle);
|
|
|
|
GL.TexSubImage2D(TextureTarget.Texture2D,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
Width,
|
|
|
|
Height,
|
|
|
|
PixelFormat.Rgba,
|
|
|
|
PixelType.UnsignedByte,
|
|
|
|
Pixels);
|
|
|
|
}
|
|
|
|
|
|
|
|
private int GetSwizzleOffset(int X, int Y, int Bpp)
|
|
|
|
{
|
|
|
|
int Pos;
|
|
|
|
|
|
|
|
Pos = (Y & 0x7f) >> 4;
|
|
|
|
Pos += (X >> 4) << 3;
|
|
|
|
Pos += (Y >> 7) * ((Width >> 4) << 3);
|
|
|
|
Pos *= 1024;
|
|
|
|
Pos += ((Y & 0xf) >> 3) << 9;
|
|
|
|
Pos += ((X & 0xf) >> 3) << 8;
|
|
|
|
Pos += ((Y & 0x7) >> 1) << 6;
|
|
|
|
Pos += ((X & 0x7) >> 2) << 5;
|
|
|
|
Pos += ((Y & 0x1) >> 0) << 4;
|
|
|
|
Pos += ((X & 0x3) >> 0) << 2;
|
|
|
|
|
|
|
|
return Pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
private bool disposed;
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
{
|
|
|
|
Dispose(true);
|
|
|
|
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Dispose(bool disposing)
|
|
|
|
{
|
|
|
|
if (!disposed)
|
|
|
|
{
|
|
|
|
if (disposing)
|
|
|
|
{
|
|
|
|
GL.DeleteTexture(TexHandle);
|
|
|
|
}
|
|
|
|
|
|
|
|
disposed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private string VtxShaderSource = @"
|
|
|
|
#version 330 core
|
|
|
|
|
|
|
|
precision highp float;
|
|
|
|
|
2018-02-20 11:52:35 +01:00
|
|
|
uniform vec2 window_size;
|
|
|
|
|
2018-02-05 00:08:20 +01:00
|
|
|
layout(location = 0) in vec3 in_position;
|
|
|
|
layout(location = 1) in vec4 in_color;
|
|
|
|
layout(location = 2) in vec2 in_tex_coord;
|
|
|
|
|
|
|
|
out vec4 color;
|
|
|
|
out vec2 tex_coord;
|
|
|
|
|
2018-02-20 11:52:35 +01:00
|
|
|
// Have a fixed aspect ratio, fit the image within the available space.
|
|
|
|
vec3 get_scale_ratio() {
|
|
|
|
vec2 native_size = vec2(1280, 720);
|
|
|
|
vec2 ratio = vec2(
|
|
|
|
(window_size.y * native_size.x) / (native_size.y * window_size.x),
|
|
|
|
(window_size.x * native_size.y) / (native_size.x * window_size.y)
|
|
|
|
);
|
2018-02-21 22:56:52 +01:00
|
|
|
return vec3(min(ratio, vec2(1, 1)) * vec2(1, -1), 1);
|
2018-02-20 11:52:35 +01:00
|
|
|
}
|
|
|
|
|
2018-02-05 00:08:20 +01:00
|
|
|
void main(void) {
|
|
|
|
color = in_color;
|
|
|
|
tex_coord = in_tex_coord;
|
2018-02-20 11:52:35 +01:00
|
|
|
gl_Position = vec4(in_position * get_scale_ratio(), 1);
|
2018-02-05 00:08:20 +01:00
|
|
|
}";
|
|
|
|
|
|
|
|
private string FragShaderSource = @"
|
|
|
|
#version 330 core
|
|
|
|
|
|
|
|
precision highp float;
|
|
|
|
|
|
|
|
uniform sampler2D tex;
|
|
|
|
|
|
|
|
in vec4 color;
|
|
|
|
in vec2 tex_coord;
|
|
|
|
out vec4 out_frag_color;
|
|
|
|
|
|
|
|
void main(void) {
|
|
|
|
out_frag_color = vec4(texture(tex, tex_coord).rgb, color.a);
|
|
|
|
}";
|
|
|
|
|
|
|
|
private int VtxShaderHandle,
|
|
|
|
FragShaderHandle,
|
|
|
|
PrgShaderHandle;
|
2018-02-20 11:52:35 +01:00
|
|
|
|
|
|
|
private int WindowSizeUniformLocation;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
|
|
|
private int VaoHandle;
|
|
|
|
private int VboHandle;
|
|
|
|
|
|
|
|
private Switch Ns;
|
|
|
|
|
|
|
|
private IGalRenderer Renderer;
|
|
|
|
|
|
|
|
private ScreenTexture ScreenTex;
|
|
|
|
|
|
|
|
public GLScreen(Switch Ns, IGalRenderer Renderer)
|
|
|
|
: base(1280, 720,
|
|
|
|
new GraphicsMode(), "Ryujinx", 0,
|
|
|
|
DisplayDevice.Default, 3, 3,
|
|
|
|
GraphicsContextFlags.ForwardCompatible)
|
|
|
|
{
|
|
|
|
this.Ns = Ns;
|
|
|
|
this.Renderer = Renderer;
|
|
|
|
|
|
|
|
ScreenTex = new ScreenTexture(Ns, Renderer, 1280, 720);
|
|
|
|
}
|
|
|
|
|
2018-02-14 03:43:08 +01:00
|
|
|
protected override void OnLoad(EventArgs e)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
|
|
|
VSync = VSyncMode.On;
|
|
|
|
|
|
|
|
CreateShaders();
|
|
|
|
CreateVbo();
|
|
|
|
|
|
|
|
GL.Enable(EnableCap.Blend);
|
|
|
|
GL.BlendFunc(BlendingFactorSrc.SrcAlpha, BlendingFactorDest.OneMinusSrcAlpha);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnUnload(EventArgs e)
|
|
|
|
{
|
|
|
|
ScreenTex.Dispose();
|
|
|
|
|
|
|
|
GL.DeleteVertexArray(VaoHandle);
|
|
|
|
GL.DeleteBuffer(VboHandle);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void CreateVbo()
|
|
|
|
{
|
|
|
|
VaoHandle = GL.GenVertexArray();
|
|
|
|
VboHandle = GL.GenBuffer();
|
|
|
|
|
|
|
|
uint[] Buffer = new uint[]
|
|
|
|
{
|
2018-02-20 11:52:35 +01:00
|
|
|
0xbf800000, 0x3f800000, 0x00000000, 0xffffffff, 0x00000000, 0x00000000, 0x00000000,
|
|
|
|
0x3f800000, 0x3f800000, 0x00000000, 0xffffffff, 0x00000000, 0x3f800000, 0x00000000,
|
|
|
|
0xbf800000, 0xbf800000, 0x00000000, 0xffffffff, 0x00000000, 0x00000000, 0x3f800000,
|
|
|
|
0x3f800000, 0xbf800000, 0x00000000, 0xffffffff, 0x00000000, 0x3f800000, 0x3f800000
|
2018-02-05 00:08:20 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
IntPtr Length = new IntPtr(Buffer.Length * 4);
|
|
|
|
|
|
|
|
GL.BindBuffer(BufferTarget.ArrayBuffer, VboHandle);
|
|
|
|
GL.BufferData(BufferTarget.ArrayBuffer, Length, Buffer, BufferUsageHint.StreamDraw);
|
|
|
|
GL.BindBuffer(BufferTarget.ArrayBuffer, 0);
|
|
|
|
|
|
|
|
GL.BindVertexArray(VaoHandle);
|
|
|
|
|
|
|
|
GL.EnableVertexAttribArray(0);
|
|
|
|
|
|
|
|
GL.BindBuffer(BufferTarget.ArrayBuffer, VboHandle);
|
|
|
|
|
|
|
|
GL.VertexAttribPointer(0, 3, VertexAttribPointerType.Float, false, 28, 0);
|
|
|
|
|
|
|
|
GL.EnableVertexAttribArray(1);
|
|
|
|
|
|
|
|
GL.BindBuffer(BufferTarget.ArrayBuffer, VboHandle);
|
|
|
|
|
|
|
|
GL.VertexAttribPointer(1, 4, VertexAttribPointerType.UnsignedByte, false, 28, 12);
|
|
|
|
|
|
|
|
GL.EnableVertexAttribArray(2);
|
|
|
|
|
|
|
|
GL.BindBuffer(BufferTarget.ArrayBuffer, VboHandle);
|
|
|
|
|
|
|
|
GL.VertexAttribPointer(2, 2, VertexAttribPointerType.Float, false, 28, 20);
|
|
|
|
|
|
|
|
GL.BindVertexArray(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void CreateShaders()
|
|
|
|
{
|
|
|
|
VtxShaderHandle = GL.CreateShader(ShaderType.VertexShader);
|
|
|
|
FragShaderHandle = GL.CreateShader(ShaderType.FragmentShader);
|
|
|
|
|
|
|
|
GL.ShaderSource(VtxShaderHandle, VtxShaderSource);
|
|
|
|
GL.ShaderSource(FragShaderHandle, FragShaderSource);
|
|
|
|
GL.CompileShader(VtxShaderHandle);
|
|
|
|
GL.CompileShader(FragShaderHandle);
|
|
|
|
|
|
|
|
PrgShaderHandle = GL.CreateProgram();
|
|
|
|
|
|
|
|
GL.AttachShader(PrgShaderHandle, VtxShaderHandle);
|
|
|
|
GL.AttachShader(PrgShaderHandle, FragShaderHandle);
|
|
|
|
GL.LinkProgram(PrgShaderHandle);
|
|
|
|
GL.UseProgram(PrgShaderHandle);
|
|
|
|
|
|
|
|
int TexLocation = GL.GetUniformLocation(PrgShaderHandle, "tex");
|
|
|
|
GL.Uniform1(TexLocation, 0);
|
2018-02-20 11:52:35 +01:00
|
|
|
|
|
|
|
WindowSizeUniformLocation = GL.GetUniformLocation(PrgShaderHandle, "window_size");
|
|
|
|
GL.Uniform2(WindowSizeUniformLocation, new Vector2(1280.0f, 720.0f));
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnUpdateFrame(FrameEventArgs e)
|
|
|
|
{
|
2018-02-18 00:54:19 +01:00
|
|
|
HidControllerKeys CurrentButton = 0;
|
|
|
|
JoystickPosition LeftJoystick;
|
|
|
|
JoystickPosition RightJoystick;
|
|
|
|
|
|
|
|
if (Keyboard[OpenTK.Input.Key.Escape]) this.Exit();
|
|
|
|
|
|
|
|
//RightJoystick
|
|
|
|
int LeftJoystickDX = 0;
|
|
|
|
int LeftJoystickDY = 0;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.StickUp]) LeftJoystickDY = short.MaxValue;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.StickDown]) LeftJoystickDY = -short.MaxValue;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.StickLeft]) LeftJoystickDX = -short.MaxValue;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.StickRight]) LeftJoystickDX = short.MaxValue;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.StickButton]) CurrentButton |= HidControllerKeys.KEY_LSTICK;
|
|
|
|
|
|
|
|
//LeftButtons
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.DPadUp]) CurrentButton |= HidControllerKeys.KEY_DUP;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.DPadDown]) CurrentButton |= HidControllerKeys.KEY_DDOWN;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.DPadLeft]) CurrentButton |= HidControllerKeys.KEY_DLEFT;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.DPadRight]) CurrentButton |= HidControllerKeys.KEY_DRIGHT;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.ButtonMinus]) CurrentButton |= HidControllerKeys.KEY_MINUS;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.ButtonL]) CurrentButton |= HidControllerKeys.KEY_L;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Left.ButtonZL]) CurrentButton |= HidControllerKeys.KEY_ZL;
|
|
|
|
|
|
|
|
//RightJoystick
|
|
|
|
int RightJoystickDX = 0;
|
|
|
|
int RightJoystickDY = 0;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.StickUp]) RightJoystickDY = short.MaxValue;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.StickDown]) RightJoystickDY = -short.MaxValue;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.StickLeft]) RightJoystickDX = -short.MaxValue;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.StickRight]) RightJoystickDX = short.MaxValue;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.StickButton]) CurrentButton |= HidControllerKeys.KEY_RSTICK;
|
|
|
|
|
|
|
|
//RightButtons
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.ButtonA]) CurrentButton |= HidControllerKeys.KEY_A;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.ButtonB]) CurrentButton |= HidControllerKeys.KEY_B;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.ButtonX]) CurrentButton |= HidControllerKeys.KEY_X;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.ButtonY]) CurrentButton |= HidControllerKeys.KEY_Y;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.ButtonPlus]) CurrentButton |= HidControllerKeys.KEY_PLUS;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.ButtonR]) CurrentButton |= HidControllerKeys.KEY_R;
|
|
|
|
if (Keyboard[(OpenTK.Input.Key)Config.FakeJoyCon.Right.ButtonZR]) CurrentButton |= HidControllerKeys.KEY_ZR;
|
|
|
|
|
|
|
|
LeftJoystick = new JoystickPosition
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-02-18 00:54:19 +01:00
|
|
|
DX = LeftJoystickDX,
|
|
|
|
DY = LeftJoystickDY
|
|
|
|
};
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-02-18 00:54:19 +01:00
|
|
|
RightJoystick = new JoystickPosition
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-02-18 00:54:19 +01:00
|
|
|
DX = RightJoystickDX,
|
|
|
|
DY = RightJoystickDY
|
|
|
|
};
|
|
|
|
|
|
|
|
//We just need one pair of JoyCon because it's emulate by the keyboard.
|
2018-02-20 21:09:23 +01:00
|
|
|
Ns.SendControllerButtons(HidControllerID.CONTROLLER_HANDHELD, HidControllerLayouts.Main, CurrentButton, LeftJoystick, RightJoystick);
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnRenderFrame(FrameEventArgs e)
|
|
|
|
{
|
2018-02-20 11:52:35 +01:00
|
|
|
GL.Viewport(0, 0, Width, Height);
|
2018-02-09 01:43:22 +01:00
|
|
|
|
|
|
|
Title = $"Ryujinx Screen - (Vsync: {VSync} - FPS: {1f / e.Time:0})";
|
|
|
|
|
2018-02-05 00:08:20 +01:00
|
|
|
GL.Clear(ClearBufferMask.ColorBufferBit | ClearBufferMask.DepthBufferBit);
|
|
|
|
|
|
|
|
RenderFb();
|
|
|
|
|
|
|
|
GL.UseProgram(PrgShaderHandle);
|
|
|
|
|
|
|
|
Renderer.RunActions();
|
|
|
|
Renderer.BindTexture(0);
|
|
|
|
Renderer.Render();
|
|
|
|
|
|
|
|
SwapBuffers();
|
|
|
|
}
|
|
|
|
|
2018-02-20 11:52:35 +01:00
|
|
|
protected override void OnResize(EventArgs e)
|
|
|
|
{
|
|
|
|
GL.UseProgram(PrgShaderHandle);
|
|
|
|
GL.Uniform2(WindowSizeUniformLocation, new Vector2(Width, Height));
|
|
|
|
}
|
|
|
|
|
2018-02-05 00:08:20 +01:00
|
|
|
void RenderFb()
|
|
|
|
{
|
|
|
|
GL.ActiveTexture(TextureUnit.Texture0);
|
|
|
|
GL.BindTexture(TextureTarget.Texture2D, ScreenTex.Texture);
|
|
|
|
GL.BindVertexArray(VaoHandle);
|
|
|
|
GL.DrawArrays(PrimitiveType.TriangleStrip, 0, 4);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|