mirror of
https://github.com/Ale32bit/Capy64.git
synced 2025-12-14 10:05:44 +00:00
193 lines
4.2 KiB
C#
193 lines
4.2 KiB
C#
// This file is part of Capy64 - https://github.com/Ale32bit/Capy64
|
|
// Copyright 2023 Alessandro "AlexDevs" Proto
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License").
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
using Capy64.API;
|
|
using KeraLua;
|
|
using System;
|
|
|
|
namespace Capy64.Runtime.Objects;
|
|
|
|
public class GPUBuffer : IComponent
|
|
{
|
|
public const string ObjectType = "GPUBuffer";
|
|
|
|
private static LuaRegister[] MetaMethods = new LuaRegister[]
|
|
{
|
|
new()
|
|
{
|
|
name = "__index",
|
|
function = LM_Index,
|
|
},
|
|
new()
|
|
{
|
|
name = "__newindex",
|
|
function = LM_NewIndex,
|
|
},
|
|
new()
|
|
{
|
|
name = "__len",
|
|
function = LM_Length,
|
|
},
|
|
new()
|
|
{
|
|
name = "__gc",
|
|
function = LM_GC,
|
|
},
|
|
new()
|
|
{
|
|
name = "__close",
|
|
function = LM_GC,
|
|
},
|
|
new()
|
|
{
|
|
name = "__tostring",
|
|
function = LM_ToString,
|
|
},
|
|
|
|
new(),
|
|
};
|
|
|
|
public void LuaInit(Lua L)
|
|
{
|
|
CreateMeta(L);
|
|
}
|
|
|
|
public static void CreateMeta(Lua L)
|
|
{
|
|
L.NewMetaTable(ObjectType);
|
|
L.SetFuncs(MetaMethods, 0);
|
|
}
|
|
|
|
public static uint[] ToBuffer(Lua L, bool gc = false)
|
|
{
|
|
return ObjectManager.ToObject<uint[]>(L, 1, gc);
|
|
}
|
|
|
|
public static uint[] CheckBuffer(Lua L, bool gc = false)
|
|
{
|
|
var obj = ObjectManager.CheckObject<uint[]>(L, 1, ObjectType, gc);
|
|
if (obj is null)
|
|
{
|
|
L.Error("attempt to use a closed buffer");
|
|
return null;
|
|
}
|
|
return obj;
|
|
}
|
|
|
|
private static int LM_Index(IntPtr state)
|
|
{
|
|
var L = Lua.FromIntPtr(state);
|
|
|
|
var buffer = CheckBuffer(L, false);
|
|
|
|
if (!L.IsInteger(2))
|
|
{
|
|
L.PushNil();
|
|
return 1;
|
|
}
|
|
|
|
var key = L.ToInteger(2);
|
|
|
|
if (key < 0 || key >= buffer.Length)
|
|
{
|
|
L.PushNil();
|
|
return 1;
|
|
}
|
|
|
|
var value = buffer[key];
|
|
|
|
// ABGR to RGB
|
|
value =
|
|
((value & 0x00_00_00_FFU) << 16) | // move R
|
|
(value & 0x00_00_FF_00U) | // move G
|
|
((value & 0x00_FF_00_00U) >> 16); // move B
|
|
|
|
L.PushInteger(value);
|
|
|
|
return 1;
|
|
}
|
|
|
|
private static int LM_NewIndex(IntPtr state)
|
|
{
|
|
var L = Lua.FromIntPtr(state);
|
|
|
|
var buffer = CheckBuffer(L, false);
|
|
if (!L.IsInteger(2))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
var key = L.ToInteger(2);
|
|
|
|
if (key < 0 || key >= buffer.Length)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (!L.IsInteger(3))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
var value = (uint)L.ToInteger(3);
|
|
|
|
// RGB to ABGR
|
|
value =
|
|
((value & 0x00_FF_00_00U) >> 16) | // move R
|
|
(value & 0x00_00_FF_00U) | // move G
|
|
((value & 0x00_00_00_FFU) << 16) | // move B
|
|
0xFF_00_00_00U;
|
|
|
|
|
|
buffer[key] = value;
|
|
|
|
return 0;
|
|
}
|
|
|
|
private static int LM_GC(IntPtr state)
|
|
{
|
|
var L = Lua.FromIntPtr(state);
|
|
|
|
CheckBuffer(L, true);
|
|
|
|
return 0;
|
|
}
|
|
|
|
private static int LM_Length(IntPtr state)
|
|
{
|
|
var L = Lua.FromIntPtr(state);
|
|
|
|
var buffer = CheckBuffer(L, false);
|
|
|
|
L.PushInteger(buffer.LongLength);
|
|
|
|
return 1;
|
|
}
|
|
|
|
private static unsafe int LM_ToString(IntPtr state)
|
|
{
|
|
var L = Lua.FromIntPtr(state);
|
|
var buffer = ToBuffer(L);
|
|
if (buffer is not null)
|
|
{
|
|
L.PushString("GPUBuffer ({0:X})", (ulong)&buffer);
|
|
}
|
|
else
|
|
{
|
|
L.PushString("GPUBuffer (closed)");
|
|
}
|
|
return 1;
|
|
}
|
|
}
|