184 lines
7.1 KiB
C#
184 lines
7.1 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using Switch_Toolbox.Library;
|
|
using Syroot.NintenTools.Bfres.GX2;
|
|
|
|
namespace FirstPlugin
|
|
{
|
|
public class GTXSwizzle
|
|
{
|
|
public static GX2.GX2Surface CreateGx2Texture(byte[] imageData, GTXImporterSettings setting, uint tileMode, uint AAMode)
|
|
{
|
|
var Format = (GX2.GX2SurfaceFormat)setting.Format;
|
|
|
|
Console.WriteLine("Format " + Format + " " + setting.TexName);
|
|
|
|
var surfOut = GX2.getSurfaceInfo(Format, setting.TexWidth, setting.TexHeight, 1, 1, tileMode, 0, 0);
|
|
uint imageSize = (uint)surfOut.surfSize;
|
|
uint alignment = surfOut.baseAlign;
|
|
uint pitch = surfOut.pitch;
|
|
uint mipSize = 0;
|
|
uint dataSize = (uint)imageData.Length;
|
|
uint bpp = GX2.surfaceGetBitsPerPixel((uint)setting.Format) >> 3;
|
|
int DepthLevel = 1;
|
|
|
|
if (dataSize <= 0)
|
|
throw new Exception($"Image is empty!!");
|
|
|
|
if (surfOut.depth != 1)
|
|
throw new Exception($"Unsupported Depth {surfOut.depth}!");
|
|
|
|
uint s = (uint)(setting.swizzle << 8);
|
|
|
|
uint blkWidth, blkHeight;
|
|
if (GX2.IsFormatBCN(Format))
|
|
{
|
|
blkWidth = 4;
|
|
blkHeight = 4;
|
|
}
|
|
else
|
|
{
|
|
blkWidth = 1;
|
|
blkHeight = 1;
|
|
}
|
|
|
|
if (tileMode == 0)
|
|
tileMode = GX2.getDefaultGX2TileMode(1, setting.TexWidth, setting.TexHeight, 1,(uint)setting.Format, 0, 1);
|
|
|
|
int tiling1dLevel = 0;
|
|
bool tiling1dLevelSet = false;
|
|
|
|
List<uint> mipOffsets = new List<uint>();
|
|
List<byte[]> Swizzled = new List<byte[]>();
|
|
|
|
for (int mipLevel = 0; mipLevel < setting.MipCount; mipLevel++)
|
|
{
|
|
var result = TextureHelper.GetCurrentMipSize(setting.TexWidth, setting.TexHeight, blkWidth, blkHeight, bpp, mipLevel);
|
|
|
|
uint offset = result.Item1;
|
|
uint size = result.Item2;
|
|
|
|
Console.WriteLine("Swizzle Size " + size);
|
|
Console.WriteLine("Swizzle offset " + offset);
|
|
Console.WriteLine("bpp " + bpp);
|
|
Console.WriteLine("TexWidth " + setting.TexWidth);
|
|
Console.WriteLine("TexHeight " + setting.TexHeight);
|
|
Console.WriteLine("blkWidth " + blkWidth);
|
|
Console.WriteLine("blkHeight " + blkHeight);
|
|
Console.WriteLine("mipLevel " + mipLevel);
|
|
|
|
byte[] data_ = new byte[size];
|
|
Array.Copy(imageData, offset, data_, 0, size);
|
|
|
|
uint width_ = Math.Max(1, setting.TexWidth >> mipLevel);
|
|
uint height_ = Math.Max(1, setting.TexHeight >> mipLevel);
|
|
|
|
if (mipLevel != 0)
|
|
{
|
|
surfOut = GX2.getSurfaceInfo(Format, setting.TexWidth, setting.TexHeight, 1, 1, tileMode, 0, mipLevel);
|
|
|
|
if (mipLevel == 1)
|
|
mipOffsets.Add(imageSize);
|
|
else
|
|
mipOffsets.Add(mipSize);
|
|
}
|
|
|
|
data_ = Utils.CombineByteArray(data_, new byte[surfOut.surfSize - size]);
|
|
byte[] dataAlignBytes = new byte[RoundUp(mipSize, surfOut.baseAlign) - mipSize];
|
|
|
|
if (mipLevel != 0)
|
|
mipSize += (uint)(surfOut.surfSize + dataAlignBytes.Length);
|
|
|
|
byte[] SwizzledData = GX2.swizzle(width_, height_, surfOut.depth, surfOut.height, (uint)Format, surfOut.tileMode, s,
|
|
surfOut.pitch, surfOut.bpp, data_, DepthLevel);
|
|
|
|
Swizzled.Add(dataAlignBytes.Concat(SwizzledData).ToArray());
|
|
|
|
if (surfOut.tileMode == 1 || surfOut.tileMode == 2 ||
|
|
surfOut.tileMode == 3 || surfOut.tileMode == 16)
|
|
{
|
|
tiling1dLevelSet = true;
|
|
}
|
|
|
|
if (tiling1dLevelSet == false)
|
|
tiling1dLevel += 1;
|
|
}
|
|
|
|
if (tiling1dLevelSet)
|
|
s |= (uint)(tiling1dLevel << 16);
|
|
else
|
|
s |= (uint)(13 << 16);
|
|
|
|
GX2.GX2Surface surf = new GX2.GX2Surface();
|
|
surf.depth = setting.Depth;
|
|
surf.width = setting.TexWidth;
|
|
surf.height = setting.TexHeight;
|
|
surf.depth = 1;
|
|
surf.use = 1;
|
|
surf.dim = (uint)setting.SurfaceDim;
|
|
surf.tileMode = tileMode;
|
|
surf.swizzle = s;
|
|
surf.resourceFlags = 0;
|
|
surf.pitch = pitch;
|
|
surf.bpp = bpp;
|
|
surf.format = (uint)setting.Format;
|
|
surf.numMips = setting.MipCount;
|
|
surf.aa = AAMode;
|
|
surf.mipOffset = mipOffsets.ToArray();
|
|
surf.numMips = (uint)Swizzled.Count;
|
|
surf.alignment = alignment;
|
|
surf.mipSize = mipSize;
|
|
surf.imageSize = imageSize;
|
|
surf.data = Swizzled[0];
|
|
|
|
List<byte[]> mips = new List<byte[]>();
|
|
for (int mipLevel = 1; mipLevel < Swizzled.Count; mipLevel++)
|
|
{
|
|
mips.Add(Swizzled[mipLevel]);
|
|
Console.WriteLine(Swizzled[mipLevel].Length);
|
|
}
|
|
surf.mipData = Utils.CombineByteArray(mips.ToArray());
|
|
mips.Clear();
|
|
|
|
|
|
Console.WriteLine("");
|
|
Console.WriteLine("// ----- GX2Surface Swizzled Info ----- ");
|
|
Console.WriteLine(" dim = 1");
|
|
Console.WriteLine(" width = " + surf.width);
|
|
Console.WriteLine(" height = " + surf.height);
|
|
Console.WriteLine(" depth = 1");
|
|
Console.WriteLine(" numMips = " + surf.numMips);
|
|
Console.WriteLine(" format = " + surf.format);
|
|
Console.WriteLine(" aa = 0");
|
|
Console.WriteLine(" use = 1");
|
|
Console.WriteLine(" imageSize = " + surf.imageSize);
|
|
Console.WriteLine(" mipSize = " + surf.mipSize);
|
|
Console.WriteLine(" tileMode = " + surf.tileMode);
|
|
Console.WriteLine(" swizzle = " + surf.swizzle);
|
|
Console.WriteLine(" alignment = " + surf.alignment);
|
|
Console.WriteLine(" pitch = " + surf.pitch);
|
|
Console.WriteLine(" data = " + surf.data.Length);
|
|
Console.WriteLine(" mipData = " + surf.mipData.Length);
|
|
Console.WriteLine("");
|
|
Console.WriteLine(" GX2 Component Selector:");
|
|
Console.WriteLine("");
|
|
Console.WriteLine(" bits per pixel = " + (surf.bpp << 3));
|
|
Console.WriteLine(" bytes per pixel = " + surf.bpp);
|
|
Console.WriteLine(" realSize = " + imageData.Length);
|
|
|
|
return surf;
|
|
}
|
|
private static int RoundUp(int X, int Y)
|
|
{
|
|
return ((X - 1) | (Y - 1)) + 1;
|
|
}
|
|
private static uint RoundUp(uint X, uint Y)
|
|
{
|
|
return ((X - 1) | (Y - 1)) + 1;
|
|
}
|
|
}
|
|
}
|