Massive juggling of core AFP/AP2 implementation into its own files.
This commit is contained in:
parent
f1294df839
commit
30a51f48e6
File diff suppressed because it is too large
Load Diff
1681
bemani/format/afp/container.py
Normal file
1681
bemani/format/afp/container.py
Normal file
File diff suppressed because it is too large
Load Diff
230
bemani/format/afp/geo.py
Normal file
230
bemani/format/afp/geo.py
Normal file
@ -0,0 +1,230 @@
|
||||
import os
|
||||
import struct
|
||||
from typing import Any, Dict, List, Optional
|
||||
|
||||
from .types import Color, Point
|
||||
from .util import descramble_text
|
||||
|
||||
|
||||
class Shape:
|
||||
def __init__(
|
||||
self,
|
||||
name: str,
|
||||
data: bytes,
|
||||
) -> None:
|
||||
self.name = name
|
||||
self.data = data
|
||||
|
||||
# Vertex points outlining this shape.
|
||||
self.vertex_points: List[Point] = []
|
||||
|
||||
# Texture points, as used alongside vertex chunks when the shape contains a texture.
|
||||
self.tex_points: List[Point] = []
|
||||
|
||||
# Colors for texture points, if they exist in the file.
|
||||
self.tex_colors: List[Color] = []
|
||||
|
||||
# Actual shape drawing parameters.
|
||||
self.draw_params: List[DrawParams] = []
|
||||
|
||||
def as_dict(self) -> Dict[str, Any]:
|
||||
return {
|
||||
'name': self.name,
|
||||
'vertex_points': [p.as_dict() for p in self.vertex_points],
|
||||
'tex_points': [p.as_dict() for p in self.tex_points],
|
||||
'tex_colors': [c.as_dict() for c in self.tex_colors],
|
||||
'draw_params': [d.as_dict() for d in self.draw_params],
|
||||
}
|
||||
|
||||
def __repr__(self) -> str:
|
||||
return os.linesep.join([
|
||||
*[f"vertex point: {vertex}" for vertex in self.vertex_points],
|
||||
*[f"tex point: {tex}" for tex in self.tex_points],
|
||||
*[f"tex color: {color}" for color in self.tex_colors],
|
||||
*[f"draw params: {params}" for params in self.draw_params],
|
||||
])
|
||||
|
||||
def get_until_null(self, offset: int) -> bytes:
|
||||
out = b""
|
||||
while self.data[offset] != 0:
|
||||
out += self.data[offset:(offset + 1)]
|
||||
offset += 1
|
||||
return out
|
||||
|
||||
def parse(self, text_obfuscated: bool = True) -> None:
|
||||
# First, grab the header bytes.
|
||||
magic = self.data[0:4]
|
||||
|
||||
if magic == b"D2EG":
|
||||
endian = "<"
|
||||
elif magic == b"GE2D":
|
||||
endian = ">"
|
||||
else:
|
||||
raise Exception("Invalid magic value in GE2D structure!")
|
||||
|
||||
# There are two integers at 0x4 and 0x8 which are basically file versions.
|
||||
|
||||
filesize = struct.unpack(f"{endian}I", self.data[12:16])[0]
|
||||
if filesize != len(self.data):
|
||||
raise Exception("Unexpected file size for GE2D structure!")
|
||||
|
||||
# There is an integer at 0x16 which always appears to be zero. It should be
|
||||
# file flags, but I don't know what it does since no code I've found cares.
|
||||
if self.data[16:20] != b"\0\0\0\0":
|
||||
raise Exception("Unhandled flag data bytes in GE2D structure!")
|
||||
|
||||
vertex_count, tex_count, color_count, label_count, render_params_count, _ = struct.unpack(
|
||||
f"{endian}HHHHHH",
|
||||
self.data[20:32],
|
||||
)
|
||||
|
||||
vertex_offset, tex_offset, color_offset, label_offset, render_params_offset = struct.unpack(
|
||||
f"{endian}IIIII",
|
||||
self.data[32:52],
|
||||
)
|
||||
|
||||
vertex_points: List[Point] = []
|
||||
if vertex_offset != 0:
|
||||
for vertexno in range(vertex_count):
|
||||
vertexno_offset = vertex_offset + (8 * vertexno)
|
||||
x, y = struct.unpack(f"{endian}ff", self.data[vertexno_offset:vertexno_offset + 8])
|
||||
vertex_points.append(Point(x, y))
|
||||
self.vertex_points = vertex_points
|
||||
|
||||
tex_points: List[Point] = []
|
||||
if tex_offset != 0:
|
||||
for texno in range(tex_count):
|
||||
texno_offset = tex_offset + (8 * texno)
|
||||
x, y = struct.unpack(f"{endian}ff", self.data[texno_offset:texno_offset + 8])
|
||||
tex_points.append(Point(x, y))
|
||||
self.tex_points = tex_points
|
||||
|
||||
colors: List[Color] = []
|
||||
if color_offset != 0:
|
||||
for colorno in range(color_count):
|
||||
colorno_offset = color_offset + (4 * colorno)
|
||||
rgba = struct.unpack(f"{endian}I", self.data[colorno_offset:colorno_offset + 4])[0]
|
||||
color = Color(
|
||||
a=(rgba & 0xFF) / 255.0,
|
||||
b=((rgba >> 8) & 0xFF) / 255.0,
|
||||
g=((rgba >> 16) & 0xFF) / 255.0,
|
||||
r=((rgba >> 24) & 0xFF) / 255.0,
|
||||
)
|
||||
colors.append(color)
|
||||
self.tex_colors = colors
|
||||
|
||||
labels: List[str] = []
|
||||
if label_offset != 0:
|
||||
for labelno in range(label_count):
|
||||
labelno_offset = label_offset + (4 * labelno)
|
||||
labelptr = struct.unpack(f"{endian}I", self.data[labelno_offset:labelno_offset + 4])[0]
|
||||
|
||||
bytedata = self.get_until_null(labelptr)
|
||||
labels.append(descramble_text(bytedata, text_obfuscated))
|
||||
|
||||
draw_params: List[DrawParams] = []
|
||||
if render_params_offset != 0:
|
||||
# The actual render parameters for the shape. This dictates how the texture values
|
||||
# are used when drawing shapes, whether to use a blend value or draw a primitive, etc.
|
||||
for render_paramsno in range(render_params_count):
|
||||
render_paramsno_offset = render_params_offset + (16 * render_paramsno)
|
||||
mode, flags, tex1, tex2, trianglecount, _, rgba, triangleoffset = struct.unpack(
|
||||
f"{endian}BBBBHHII",
|
||||
self.data[(render_paramsno_offset):(render_paramsno_offset + 16)]
|
||||
)
|
||||
|
||||
if mode != 4:
|
||||
raise Exception("Unexpected mode in GE2D structure!")
|
||||
if (flags & 0x2) and len(labels) == 0:
|
||||
raise Exception("GE2D structure has a texture, but no region labels present!")
|
||||
if (flags & 0x2) and (tex1 == 0xFF):
|
||||
raise Exception("GE2D structure requests a texture, but no texture pointer present!")
|
||||
if tex2 != 0xFF:
|
||||
raise Exception("GE2D structure requests a second texture, but we don't support this!")
|
||||
|
||||
color = Color(
|
||||
r=(rgba & 0xFF) / 255.0,
|
||||
g=((rgba >> 8) & 0xFF) / 255.0,
|
||||
b=((rgba >> 16) & 0xFF) / 255.0,
|
||||
a=((rgba >> 24) & 0xFF) / 255.0,
|
||||
)
|
||||
|
||||
verticies: List[int] = []
|
||||
for render_paramstriangleno in range(trianglecount):
|
||||
render_paramstriangleno_offset = triangleoffset + (2 * render_paramstriangleno)
|
||||
tex_offset = struct.unpack(f"{endian}H", self.data[render_paramstriangleno_offset:(render_paramstriangleno_offset + 2)])[0]
|
||||
verticies.append(tex_offset)
|
||||
|
||||
# Seen bits are 0x1, 0x2, 0x4, 0x8 so far.
|
||||
# 0x1 Is a "this shape is instantiable/drawable" bit.
|
||||
# 0x2 Is the shape having a texture.
|
||||
# 0x4 Is the shape having a texture color per texture point.
|
||||
# 0x8 Is "draw background color/blend" flag.
|
||||
# 0x40 Is a "normalize texture coordinates" flag. It performs the below algorithm.
|
||||
|
||||
if (flags & (0x2 | 0x40)) == (0x2 | 0x40):
|
||||
# The tex offsets point at the tex vals parsed above, and are used in conjunction with
|
||||
# texture/region metrics to calcuate some offsets. First, the region left/right/top/bottom
|
||||
# is divided by 2 (looks like a scaling of 2 for regions to textures is hardcoded) and then
|
||||
# divided by the texture width/height (as relevant). The returned metrics are in texture space
|
||||
# where 0.0 is the origin and 1.0 is the furthest right/down. The metrics are then multiplied
|
||||
# by the texture point pairs that appear above, meaning they should be treated as percentages.
|
||||
pass
|
||||
|
||||
draw_params.append(
|
||||
DrawParams(
|
||||
flags=flags,
|
||||
region=labels[tex1] if (flags & 0x2) else None,
|
||||
vertexes=verticies if (flags & 0x6) else [],
|
||||
blend=color if (flags & 0x8) else None,
|
||||
)
|
||||
)
|
||||
self.draw_params = draw_params
|
||||
|
||||
|
||||
class DrawParams:
|
||||
def __init__(
|
||||
self,
|
||||
flags: int,
|
||||
region: Optional[str] = None,
|
||||
vertexes: List[int] = [],
|
||||
blend: Optional[Color] = None,
|
||||
) -> None:
|
||||
self.flags = flags
|
||||
self.region = region
|
||||
self.vertexes = vertexes
|
||||
self.blend = blend
|
||||
|
||||
def as_dict(self) -> Dict[str, Any]:
|
||||
return {
|
||||
'flags': self.flags,
|
||||
'region': self.region,
|
||||
'vertexes': self.vertexes,
|
||||
'blend': self.blend.as_dict() if self.blend else None,
|
||||
}
|
||||
|
||||
def __repr__(self) -> str:
|
||||
flagbits: List[str] = []
|
||||
if self.flags & 0x1:
|
||||
flagbits.append("(Instantiable)")
|
||||
if self.flags & 0x2:
|
||||
flagbits.append("(Includes Texture)")
|
||||
if self.flags & 0x4:
|
||||
flagbits.append("(Includes Texture Color)")
|
||||
if self.flags & 0x8:
|
||||
flagbits.append("(Includes Blend Color)")
|
||||
if self.flags & 0x40:
|
||||
flagbits.append("(Needs Tex Point Normalization)")
|
||||
|
||||
flagspart = f"flags: {hex(self.flags)} {' '.join(flagbits)}"
|
||||
if self.flags & 0x2:
|
||||
texpart = f", region: {self.region}, vertexes: {', '.join(str(x) for x in self.vertexes)}"
|
||||
else:
|
||||
texpart = ""
|
||||
|
||||
if self.flags & 0x8:
|
||||
blendpart = f", blend: {self.blend}"
|
||||
else:
|
||||
blendpart = ""
|
||||
|
||||
return f"{flagspart}{texpart}{blendpart}"
|
1068
bemani/format/afp/swf.py
Normal file
1068
bemani/format/afp/swf.py
Normal file
File diff suppressed because it is too large
Load Diff
36
bemani/format/afp/util.py
Normal file
36
bemani/format/afp/util.py
Normal file
@ -0,0 +1,36 @@
|
||||
def _hex(data: int) -> str:
|
||||
hexval = hex(data)[2:]
|
||||
if len(hexval) == 1:
|
||||
return "0" + hexval
|
||||
return hexval
|
||||
|
||||
|
||||
def align(val: int) -> int:
|
||||
return (val + 3) & 0xFFFFFFFFC
|
||||
|
||||
|
||||
def pad(data: bytes, length: int) -> bytes:
|
||||
if len(data) == length:
|
||||
return data
|
||||
elif len(data) > length:
|
||||
raise Exception("Logic error, padding request in data already written!")
|
||||
return data + (b"\0" * (length - len(data)))
|
||||
|
||||
|
||||
def descramble_text(text: bytes, obfuscated: bool) -> str:
|
||||
if len(text):
|
||||
if obfuscated and (text[0] - 0x20) > 0x7F:
|
||||
# Gotta do a weird demangling where we swap the
|
||||
# top bit.
|
||||
return bytes(((x + 0x80) & 0xFF) for x in text).decode('ascii')
|
||||
else:
|
||||
return text.decode('ascii')
|
||||
else:
|
||||
return ""
|
||||
|
||||
|
||||
def scramble_text(text: str, obfuscated: bool) -> bytes:
|
||||
if obfuscated:
|
||||
return bytes(((x + 0x80) & 0xFF) for x in text.encode('ascii')) + b'\0'
|
||||
else:
|
||||
return text.encode('ascii') + b'\0'
|
@ -8,7 +8,7 @@ import textwrap
|
||||
from PIL import Image, ImageDraw # type: ignore
|
||||
from typing import Any, Dict
|
||||
|
||||
from bemani.format.afp import AFPFile, Shape, SWF
|
||||
from bemani.format.afp import TXP2File, Shape, SWF
|
||||
|
||||
|
||||
def main() -> int:
|
||||
@ -141,7 +141,7 @@ def main() -> int:
|
||||
raise Exception("Cannot generate mapping overlays when splitting sprites!")
|
||||
|
||||
with open(args.file, "rb") as bfp:
|
||||
afpfile = AFPFile(bfp.read(), verbose=args.verbose)
|
||||
afpfile = TXP2File(bfp.read(), verbose=args.verbose)
|
||||
|
||||
# Actually place the files down.
|
||||
os.makedirs(args.dir, exist_ok=True)
|
||||
@ -297,7 +297,7 @@ def main() -> int:
|
||||
if args.action == "update":
|
||||
# First, parse the file out
|
||||
with open(args.file, "rb") as bfp:
|
||||
afpfile = AFPFile(bfp.read(), verbose=args.verbose)
|
||||
afpfile = TXP2File(bfp.read(), verbose=args.verbose)
|
||||
|
||||
# Now, find any PNG files that match texture names.
|
||||
for texture in afpfile.textures:
|
||||
@ -339,7 +339,7 @@ def main() -> int:
|
||||
if args.action == "print":
|
||||
# First, parse the file out
|
||||
with open(args.file, "rb") as bfp:
|
||||
afpfile = AFPFile(bfp.read(), verbose=args.verbose)
|
||||
afpfile = TXP2File(bfp.read(), verbose=args.verbose)
|
||||
|
||||
# Now, print it
|
||||
print(json.dumps(afpfile.as_dict(), sort_keys=True, indent=4))
|
||||
|
Loading…
x
Reference in New Issue
Block a user