memory.usememorydomain("IWRAM")
local R4u= memory.read_u32_le
local R4s= memory.read_s32_le
local R2u= memory.read_u16_le
local R2s= memory.read_s16_le
local R1u= memory.read_u8
local R1s= memory.read_s8
local R4x= function(a) return string.format("%X",R4u(a)) end
local R2x= function(a) return string.format("%X",R2u(a)) end
local RAMList2= {
{a=0x15DC,m=R4s,c=0xFFFFFFFF},
{a=0x1514,m=R4x,c=0xFFFFFFFF},
{a=0x1EA4,m=R2s,c=0xFFFFFF00},
{a=0x1504,m=R4s,c=0xFFFFFFFF},
{a=0x1E5C,m=R2s,c=0xFFFFFFFF},
{a=0x1500,m=R4s,c=0xFFFFFFFF},
{a=0x1E58,m=R2s,c=0xFFFFFFFF},
{a=0x17A0,m=R4x,c=0xFFFFFFFF},
{a=0x1530,m=R4s,c=0xFF00FFFF},
{a=0x15E0,m=R4s,c=0xFFFFFFFF},
{a=0x1608,m=R4s,c=0xFFFFFFFF},
{a=0x1E6C,m=R2s,c=0xFF00FF00},
{a=0x16E0,m=R4s,c=0xFFC0C0C0},
{a=0x16F8,m=R4s,c=0xFFC0C0C0},
{a=0x1708,m=R4s,c=0xFFFF8000},
{a=0x170C,m=R4s,c=0xFFFF8000},
}
local RAMList= {
0x15E0,
0x1608,
0x1530,
0x153C,
0x1540
}
local function PixelTextRight(x,y,t,c1,c2,f)
x= x - 4*(#tostring(t))
gui.pixelText(x,y,t,c1,c2,f)
end
local function FetchAddrDomainGBA(a)
if (a >= 0x02000000) and (a < (0x02000000+memory.getmemorydomainsize("EWRAM"))) then
return a-0x02000000, "EWRAM"
elseif (a >= 0x03000000) and (a < (0x03000000+memory.getmemorydomainsize("IWRAM"))) then
return a-0x03000000, "IWRAM"
elseif (a >= 0x08000000) and (a < (0x08000000+memory.getmemorydomainsize("ROM"))) then
return a-0x08000000, "ROM"
else
error("Unknown address " .. a,2)
end
end
local ObjColors= {
[0]=0xFFC0C0C0,
0xFF20FF20,
0xFFC0C0C0,
0xFFC0C0C0,
0xFFC0C0C0,
0xFFFFFFFF,
0xFFC0C0C0,
0xFFC0C0C0,
0xFFC0C0C0,
0xFFC0C0C0,
0xFF00FF00,
0xFFFFFFFF,
0xFFFFFFFF,
0xFFFFFFFF,
0xFFFFFFFF,
0xFFFFFFFF,
0xFFFFFFFF,
0xFFFFFFFF,
0xFFFFFFFF,
0xFFC0C0C0,
0xFF00FFFF,
0xFF00FFFF,
0xFF00FFFF,
0xFF00FFFF,
0xFF00FFFF,
0xFF00FFFF,
0xFF00FFFF,
0xFF00FFFF,
0xFF00FFFF,
0xFF00FFFF,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFFF00,
0xFFFFC000,
0xFFFFC000,
0xFFFFC000,
0xFFFFC000,
0xFFFFC000,
0xFFFFC000,
0xFFFFC000,
0xFFFFC000,
0xFFFFC000,
0xFFFFC000,
}
local function DrawBlocks(x,y,a)
local PANIC= 0
while R2s(a,"ROM") ~= -1 do
local bX,bY= R1s(a,"ROM"),R1s(a+1,"ROM")
bX= bX*4+x; bY= bY*4+y
gui.drawRectangle(bX,bY,4,4,0x60FF00FF,0x60FFFF00)
a= a+2
PANIC=PANIC+1; if PANIC > 400 then error("PANIC") end
end
end
local function ObjList()
local ObjCount= 0
for i= 0, 59 do
local addr= 0x1DE4 + 0x74*i
local X= R4s(addr+0x00)
local Y= R4s(addr+0x04)
local clr= ObjColors[i] or 0xFFFF00FF
gui.pixelText(X,Y ,string.format("%d",R2s(addr+0x2C)),clr)
gui.pixelText(X,Y+ 7,R2x(addr+0x1C),clr)
if R2s(addr+0x0A) ~= 0 then ObjCount= ObjCount+1 end
end
gui.pixelText(10,153,ObjCount)
end
local AssignmentArrayColors= {}
local function ResetColors() for i= 0, 0xBF do AssignmentArrayColors[i]= 0xFFFF0000 end end
local function NullFn() end
local function DoPlyr(addr,x,y)
gui.pixelText(x,y ,string.format("%d",R2s(addr+0x08)),0xFF00FF20)
gui.pixelText(x,y+ 7,string.format("%d",R2s(addr+0x10)),0xFF00FF20)
gui.pixelText(x,y+14,(R4u(0x17A0)-1)%4,0xFFFFFFFF)
end
local Bcount= 0
local function DoBllt(addr,x,y)
if R2s(addr+0x08) == 0 then return end
Bcount= Bcount+1
local ArrayStart= math.floor((R2u(addr+0x0E)-256)/4)
local ArrayCount= R2u(addr+0x40)
for i= ArrayStart, ArrayStart+ArrayCount-1 do
AssignmentArrayColors[i]= 0xFF00FFFF
end
gui.pixelText(x,y ,string.format("%d",R2s(addr+0x1C)),0xFF00FFFF)
gui.pixelText(x,y+ 7,string.format("%d",R2s(addr+0x48)),0xFF00FFFF)
end
local Ecount= 0
local function DoOthr(addr,x,y)
if R2s(addr+0x0A) == 0 then return end
Ecount= Ecount+1
local clr= 0xFFFFFFFF
if R2s(addr+0x3C) == 2 then
gui.pixelText(x,y ,string.format("%d",R2s(addr+0x1C)),0xFF000000,0xFF00FF00)
gui.pixelText(x,y+7,string.format("%d",R2s(addr+0x0A)),0xFF000000,0xFF00FF00)
clr= 0xFF00FF00
elseif R2s(addr+0x2C) ~= -1 then
gui.pixelText(x,y ,string.format("%d",R2s(addr+0x2C)),0xFFFF8000)
gui.pixelText(x,y+7,string.format("%d",R2s(addr+0x58)),0xFFFF8000)
clr= 0xFFFFFF00
else
gui.pixelText(x,y ,string.format("%d",R2s(addr+0x1C)),0xFF000000,0xFFFF8000)
gui.pixelText(x,y+7,string.format("%d",R2s(addr+0x08)),0xFF000000,0xFFFF8000)
clr= 0xFFFF00FF
end
local ArrayStart= math.floor((R2u(addr+0x0E)-256)/4)
local ArrayCount= R2u(addr+0x40)
for i= ArrayStart, ArrayStart+ArrayCount-1 do
AssignmentArrayColors[i]= clr
end
end
local ObjTbl= {
[0]=NullFn, DoPlyr, NullFn, NullFn, NullFn,
NullFn, NullFn, NullFn, NullFn, NullFn,
DoBllt, DoBllt, DoBllt, DoBllt, DoBllt,
DoBllt, DoBllt, DoBllt, DoBllt, DoBllt,
DoBllt, DoBllt, DoBllt, DoBllt, DoBllt,
DoBllt, DoBllt, DoBllt, DoBllt, DoBllt,
DoOthr, DoOthr, DoOthr, DoOthr, DoOthr,
DoOthr, DoOthr, DoOthr, DoOthr, DoOthr,
DoOthr, DoOthr, DoOthr, DoOthr, DoOthr,
DoOthr, DoOthr, DoOthr, DoOthr, DoOthr,
DoOthr, DoOthr, DoOthr, DoOthr, DoOthr,
DoOthr, DoOthr, DoOthr, DoOthr, DoOthr
}
local function ObjList2()
Bcount= 0
Ecount= 0
local ObjCount= 0
for i= 0, 59 do
local addr= 0x1DE4 + 0x74*i
local X= R4s(addr+0x00)
local Y= R4s(addr+0x04)
ObjTbl[i](addr,X,Y)
end
PixelTextRight( 9,153,Bcount)
PixelTextRight(19,153,Ecount)
end
local HeartStr= " h H + H +h Hh H + "
local AmmoStr = " a A Aa a A * A* a"
local LifeStr = " L "
local DirStr = " < > <>> < ><<> <<>> <> < >> < >"
local function ItemDrop()
gui.pixelText(40,153,HeartStr,0xFFFF8000)
gui.pixelText(40,153,AmmoStr ,0xFF00C0FF,0)
gui.pixelText(39,153,LifeStr ,0xFF00FF00,0)
gui.pixelText(40,146,DirStr ,0xFFC0C0C0)
local RNG= R4u(0x17A0)%0x20
local X= RNG*4 + 40
gui.drawRectangle(X,153,5,6)
end
local DashDir= {[0]=">","<"}
local function TimerWatch()
if R4s(0x15C0,"IWRAM") ~= 0 then
gui.pixelText( 24,6,DashDir[R4s(0x15C4)] or "?",0xFFFF80FF)
PixelTextRight(23,6, R4s(0x15C0) ,0xFFFF80FF)
end
if R4s(0x15D4,"IWRAM") ~= 0 then
PixelTextRight(32,6, R4s(0x15D0) ,0xFF80FF80)
end
if R4s(0x1624,"IWRAM") ~= 0 then
PixelTextRight(40,6, R4s(0x1628) ,0xFFFF8000)
end
if R4s(0x16B4,"IWRAM") ~= 0 then
PixelTextRight(60,6, R4s(0x16B4) ,0xFF0080FF)
PixelTextRight(70,6, R4s(0x16B8) ,0xFF0080FF)
end
end
local DashFrames= 0
local function DashCount()
if R4s(0x15D4,"IWRAM") ~= 0 then DashFrames= DashFrames+1 end
PixelTextRight(32,50,DashFrames,0xFF80FF80)
end
local function RAMWatch()
local Y= 153 - (#RAMList)*7
for i= 1, #RAMList do
PixelTextRight(239,Y+i*7,R2s(RAMList[i]))
end
end
local function RAMWatch2()
local Y= 153 - (#RAMList2)*7
for i= 1, #RAMList2 do
local stat= RAMList2[i]
PixelTextRight(239,Y+i*7,stat.m(stat.a),stat.c)
end
end
local function ListVals(ptr,x)
local i= 0
while R2s(ptr,"ROM") ~= -1 do
gui.pixelText(x ,i*7,string.format("%04X",R2u(ptr ,"ROM")))
ptr= ptr+2
i=i+1; if i > 30 then return end
end
end
local function FinalHitboxAttempt()
local addr= 0x1DE4 + 0x74*1
local index= R2s(addr+0x08,"IWRAM")
local ptr= R4s(addr+0x70,"IWRAM")
if index == 0 then gui.pixelText(50,0,"Null sprite")
elseif ptr == 0 then gui.pixelText(50,0,"Null pointer")
else
ptr= ptr + (index-1)*4
ptr= R4s(FetchAddrDomainGBA(ptr))
if (ptr >= 0x08000000) and (ptr < 0x08000000+memory.getmemorydomainsize("ROM")) then
ptr= ptr - 0x08000000
ListVals(ptr,50)
else
gui.pixelText(50,0,"Bad pointer")
end
end
end
local function ProgressColors(v)
if v > 0 then return 0xFF00C0FF end
if v < 0 then return 0xFFFFC000 end
return 0xFFFF00FF
end
local function ImportantPlayerPixels(PlX,PlY)
local PlAir= R4u(0x1554,"IWRAM") == 1
local PlLeft=R4u(0x1E64,"IWRAM") == 1
local PlJump=R4u(0x1E6C,"IWRAM")
if (not PlAir) or ((PlJump >= 20) and not PlLeft) then
gui.drawPixel(PlX+16,PlY+48,0xFFFFFFFF)
gui.drawPixel(PlX+15,PlY+47,0xFFFFFFFF)
gui.drawPixel(PlX+16,PlY+47,0xFFFFFFFF)
gui.drawPixel(PlX+17,PlY+47,0xFFFFFFFF)
end
if (not PlAir) or ((PlJump >= 20) and PlLeft) then
gui.drawPixel(PlX+24,PlY+48,0xFFFFFFFF)
gui.drawPixel(PlX+23,PlY+47,0xFFFFFFFF)
gui.drawPixel(PlX+24,PlY+47,0xFFFFFFFF)
gui.drawPixel(PlX+25,PlY+47,0xFFFFFFFF)
end
if PlAir and (PlJump < 20) and (not PlLeft) then
gui.drawPixel(PlX+16,PlY+ 8,0xFFFF80FF)
gui.drawPixel(PlX+15,PlY+ 9,0xFFFF80FF)
gui.drawPixel(PlX+16,PlY+ 9,0xFFFF80FF)
gui.drawPixel(PlX+17,PlY+ 9,0xFFFF80FF)
end
if PlAir and (PlJump < 20) and PlLeft then
gui.drawPixel(PlX+24,PlY+ 8,0xFFFF80FF)
gui.drawPixel(PlX+23,PlY+ 9,0xFFFF80FF)
gui.drawPixel(PlX+24,PlY+ 9,0xFFFF80FF)
gui.drawPixel(PlX+25,PlY+ 9,0xFFFF80FF)
end
if PlLeft then
gui.drawPixel(PlX+ 8,PlY+47,0xFFFFFF00)
gui.drawPixel(PlX+ 9,PlY+46,0xFFFFFF00)
gui.drawPixel(PlX+ 9,PlY+47,0xFFFFFF00)
gui.drawPixel(PlX+ 9,PlY+48,0xFFFFFF00)
end
if not PlLeft then
gui.drawPixel(PlX+32,PlY+47,0xFFFFFF00)
gui.drawPixel(PlX+31,PlY+46,0xFFFFFF00)
gui.drawPixel(PlX+31,PlY+47,0xFFFFFF00)
gui.drawPixel(PlX+31,PlY+48,0xFFFFFF00)
end
if PlAir and (PlJump >= 20) and PlLeft then
gui.drawPixel(PlX+ 8,PlY+32,0xFF00FFFF)
gui.drawPixel(PlX+ 9,PlY+32,0xFF00FFFF)
gui.drawPixel(PlX+10,PlY+32,0xFF00FFFF)
gui.drawPixel(PlX+ 9,PlY+31,0xFF00FFFF)
gui.drawPixel(PlX+10,PlY+30,0xFF00FFFF)
end
if PlAir and (PlJump >= 20) and (not PlLeft) then
gui.drawPixel(PlX+32,PlY+32,0xFF00FFFF)
gui.drawPixel(PlX+31,PlY+32,0xFF00FFFF)
gui.drawPixel(PlX+30,PlY+32,0xFF00FFFF)
gui.drawPixel(PlX+31,PlY+31,0xFF00FFFF)
gui.drawPixel(PlX+30,PlY+30,0xFF00FFFF)
end
end
local function ReadAllocationArray()
for i= 0, 0xBF do
local addr= 0x7C44 + i
local clr= 0xFF000000
if R1u(addr,"IWRAM") ~= 0 then clr= AssignmentArrayColors[i] end
gui.drawLine(i,0,i,3,clr)
end
gui.drawLine(0xC0,0,0xC0,3,0xFFFFFFFF)
end
local Left,Zero,DamageR,DamageL= 0,0,0,0
local cyan= 0xFF00FFFF
local LastX,LastY= 0,0
local function BasicHUD()
ReadAllocationArray()
local CamX,CamY= R4s(0x1500,"IWRAM"),R4s(0x1504,"IWRAM")
local PlScrX,PlScrY= R4s(0x1E58,"IWRAM"),R4s(0x1E5C,"IWRAM")
local PlX,PlY= PlScrX+CamX, PlScrY+CamY
PixelTextRight(239, 0,R4s(0x1560,"IWRAM"),cyan)
PixelTextRight(239, 7,PlX)
PixelTextRight(239, 14,PlX-LastX,ProgressColors(PlX-LastX))
PixelTextRight(239, 25,R4s(0x1564,"IWRAM"),cyan)
PixelTextRight(239, 32,PlY)
PixelTextRight(239, 39,PlY-LastY,ProgressColors(PlY-LastY))
local BossHP= (R4s(0x1708,"IWRAM")-1)*5 + R4s(0x170C,"IWRAM")
PixelTextRight(224,153,BossHP)
ImportantPlayerPixels(PlScrX,PlScrY)
local move= PlX-LastX
if move == 1 then DamageR= DamageR+1
elseif move == 0 then Zero= Zero+1
elseif move == -1 then DamageL= DamageL+1
elseif move == -2 then Left= Left+1
end
PixelTextRight(219, 0,DamageR,0xFFFFFF80)
PixelTextRight(219, 7,Zero ,0xFFFFFF00)
PixelTextRight(219, 14,DamageL,0xFFFFC000)
PixelTextRight(219, 21,Left ,0xFFFF4000)
LastX,LastY= PlX,PlY
end
local Idle= 0
local BossScan
local function BossRepeat()
local inv= R4s(0x16F8,"IWRAM")
if inv == 0 then
Idle= Idle+1
elseif Idle ~= 0 then
print(string.format("f%d - %d delayed",emu.framecount(),Idle-1))
Idle= 0
end
end
local function BossFirst()
local inv= R4s(0x16F8,"IWRAM")
if inv ~= 0 then
print(string.format("f%d - First hit",emu.framecount()))
BossScan= BossRepeat
end
end
BossScan= BossFirst
while true do
ResetColors()
ObjList2()
ItemDrop()
RAMWatch2()
TimerWatch()
BasicHUD()
emu.frameadvance()
end