plugins/cheatfind: add width and endian awareness

---
Need to handled alignment
This commit is contained in:
cracyc 2016-04-09 22:40:02 -05:00
parent a107408a08
commit 1132ce2870

View File

@ -61,8 +61,45 @@ function cheatfind.startplugin()
end end
-- compare two data blocks -- compare two data blocks
function cheat.comp(olddata, newdata, oper, val) function cheat.comp(olddata, newdata, oper, val, signed, width, endian)
local ret = {} local ret = {}
local format = ""
local size = olddata.size
if endian == 1 then
format = format .. ">"
else
format = format .. "<"
end
if width == 16 then
size = size & 0xfffe
if signed == "signed" then
format = format .. "h"
else
format = format .. "H"
end
elseif width == 32 then
size = size & 0xfffc
if signed == "signed" then
format = format .. "l"
else
format = format .. "L"
end
elseif width == 64 then
size = size & 0xfff8
if signed == "signed" then
format = format .. "j"
else
format = format .. "J"
end
else
if signed == "signed" then
format = format .. "b"
else
format = format .. "B"
end
end
if olddata.start ~= newdata.start or olddata.size ~= newdata.size then if olddata.start ~= newdata.start or olddata.size ~= newdata.size then
return {} return {}
end end
@ -70,9 +107,9 @@ function cheatfind.startplugin()
val = 0 val = 0
end end
if oper == "+" or oper == "inc" or oper == "<" or oper == "lt" then if oper == "+" or oper == "inc" or oper == "<" or oper == "lt" then
for i = 1, olddata.size do for i = 1, size do
local old = string.unpack("B", olddata.block, i) local old = string.unpack(format, olddata.block, i)
local new = string.unpack("B", newdata.block, i) local new = string.unpack(format, newdata.block, i)
if old < new then if old < new then
if (val > 0 and (old + val) == new) or val == 0 then if (val > 0 and (old + val) == new) or val == 0 then
ret[#ret + 1] = { addr = olddata.start + i - 1, ret[#ret + 1] = { addr = olddata.start + i - 1,
@ -82,9 +119,9 @@ function cheatfind.startplugin()
end end
end end
elseif oper == "-" or oper == "dec" or oper == ">" or oper == "gt" then elseif oper == "-" or oper == "dec" or oper == ">" or oper == "gt" then
for i = 1, olddata.size do for i = 1, size do
local old = string.unpack("B", olddata.block, i) local old = string.unpack(format, olddata.block, i)
local new = string.unpack("B", newdata.block, i) local new = string.unpack(format, newdata.block, i)
if old > new then if old > new then
if (val > 0 and (old - val) == new) or val == 0 then if (val > 0 and (old - val) == new) or val == 0 then
ret[#ret + 1] = { addr = olddata.start + i - 1, ret[#ret + 1] = { addr = olddata.start + i - 1,
@ -94,9 +131,9 @@ function cheatfind.startplugin()
end end
end end
elseif oper == "=" or oper == "eq" then elseif oper == "=" or oper == "eq" then
for i = 1, olddata.size do for i = 1, size do
local old = string.unpack("B", olddata.block, i) local old = string.unpack(format, olddata.block, i)
local new = string.unpack("B", newdata.block, i) local new = string.unpack(format, newdata.block, i)
if old == new then if old == new then
ret[#ret + 1] = { addr = olddata.start + i - 1, ret[#ret + 1] = { addr = olddata.start + i - 1,
oldval = old, oldval = old,
@ -108,9 +145,9 @@ function cheatfind.startplugin()
end end
-- compare a data block to the current state -- compare a data block to the current state
function cheat.compcur(olddata, oper, val) function cheat.compcur(olddata, oper, val, signed, width, endian)
local newdata = cheat.save(olddata.space, olddata.start, olddata.size, olddata.space) local newdata = cheat.save(olddata.space, olddata.start, olddata.size, olddata.space)
return cheat.comp(olddata, newdata, oper, val) return cheat.comp(olddata, newdata, oper, val, signed, width, endian)
end end
_G.cf = cheat _G.cf = cheat
@ -119,6 +156,7 @@ function cheatfind.startplugin()
local devsel = 1 local devsel = 1
local bitwidth = 3 local bitwidth = 3
local signed = 0 local signed = 0
local endian = 0
local optable = { "<", ">", "=" } local optable = { "<", ">", "=" }
local opsel = 1 local opsel = 1
local value = 0 local value = 0
@ -127,12 +165,16 @@ function cheatfind.startplugin()
local matches = {} local matches = {}
local matchsel = 1 local matchsel = 1
local menu_blocks = {} local menu_blocks = {}
local midx = { region = 1, init = 2, width = 3, signed = 4, save = 3, op = 5, val = 6, local midx = { region = 1, init = 2, save = 3, width = 5, signed = 6, endian = 7, op = 8, val = 9,
lop = 7, rop = 8, comp = 9, match = 11 } lop = 10, rop = 11, comp = 12, match = 14 }
local function start() local function start()
devtable = {} devtable = {}
devsel = 1 devsel = 1
devsel = 1
bitwidth = 3
signed = 0
endian = 0
opsel = 1 opsel = 1
value = 0 value = 0
leftop = 1 leftop = 1
@ -183,17 +225,21 @@ function cheatfind.startplugin()
menu_lim(devsel, 1, #devtable, menu[midx.region]) menu_lim(devsel, 1, #devtable, menu[midx.region])
end end
menu[midx.init] = { "Init", "", 0 } menu[midx.init] = { "Init", "", 0 }
if not next(menu_blocks) then if next(menu_blocks) then
menu[midx.width] = { "Bit Width", 1 << bitwidth , 0 }
menu_lim(bitwidth, 3, 6, menu[midx.width])
menu[midx.signed] = { "Signed", "true", 0 }
menu_lim(signed, 0, 1, menu[midx.signed])
if signed == 0 then
menu[midx.signed][2] = "false"
end
else
menu[midx.save] = { "Save current", "", 0 } menu[midx.save] = { "Save current", "", 0 }
menu[midx.save + 1] = { "---", "", "off" } menu[midx.save + 1] = { "---", "", "off" }
menu[midx.width] = { "Bit Width", 1 << bitwidth , 0 }
menu_lim(bitwidth, 3, 6, menu[midx.width])
menu[midx.signed] = { "Signed", "false", 0 }
menu_lim(signed, 0, 1, menu[midx.signed])
if signed == 1 then
menu[midx.signed][2] = "true"
end
menu[midx.endian] = { "Endian", "little", 0 }
menu_lim(endian, 0, 1, menu[midx.endian])
if endian == 1 then
menu[midx.endian][2] = "big"
end
menu[midx.op] = { "Operator", optable[opsel], "" } menu[midx.op] = { "Operator", optable[opsel], "" }
menu_lim(opsel, 1, #optable, menu[midx.op]) menu_lim(opsel, 1, #optable, menu[midx.op])
menu[midx.val] = { "Value", value, "" } menu[midx.val] = { "Value", value, "" }
@ -225,7 +271,18 @@ function cheatfind.startplugin()
if #menu > 50 then if #menu > 50 then
break break
end end
menu[#menu + 1] = { string.format("%08x %02x %02x", match.addr, match.oldval, match.newval), "", 0 } local numform = ""
if bitwidth == 4 then
numform = " %04x"
elseif bitwidth == 5 then
numform = " %08x"
elseif bitwidth == 6 then
numform = " %016x"
else
numform = " %02x"
end
menu[#menu + 1] = { string.format("%08x" .. numform .. numform, match.addr, match.oldval,
match.newval), "", 0 }
end end
end end
end end
@ -258,11 +315,15 @@ function cheatfind.startplugin()
manager:machine():popmessage("Data cleared and current state saved") manager:machine():popmessage("Data cleared and current state saved")
ret = true ret = true
end end
elseif not next(menu_blocks) then elseif index == midx.init then
if index == midx.bitwidth then if event == "select" then
bitwidth = incdec(bitwidth, 3, 6) menu_blocks = {{}}
elseif index == midx.signed then matches = {}
signed = incdec(signed, 0, 1) for num, region in ipairs(devtable[devsel].ram) do
menu_blocks[1][num] = cheat.save(devtable[devsel].space, region.offset, region.size)
end
manager:machine():popmessage("Data cleared and current state saved")
ret = true
end end
elseif index == midx.save then elseif index == midx.save then
if event == "select" then if event == "select" then
@ -273,6 +334,12 @@ function cheatfind.startplugin()
manager:machine():popmessage("Current state saved") manager:machine():popmessage("Current state saved")
ret = true ret = true
end end
elseif index == midx.width then
bitwidth = incdec(bitwidth, 3, 6)
elseif index == midx.signed then
signed = incdec(signed, 0, 1)
elseif index == midx.endian then
endian = incdec(endian, 0, 1)
elseif index == midx.op then elseif index == midx.op then
if event == "up" or event == "down" or event == "comment" then if event == "up" or event == "down" or event == "comment" then
if optable[opsel] == "<" then if optable[opsel] == "<" then
@ -296,10 +363,11 @@ function cheatfind.startplugin()
matches = {} matches = {}
for num = 1, #menu_blocks[1] do for num = 1, #menu_blocks[1] do
if rightop == 0 then if rightop == 0 then
matches[#matches + 1] = cheat.compcur(menu_blocks[leftop][num], optable[opsel], value) matches[#matches + 1] = cheat.compcur(menu_blocks[leftop][num], optable[opsel], value,
signed, 1 << bitwidth, endian)
else else
matches[#matches + 1] = cheat.comp(menu_blocks[leftop][num], menu_blocks[rightop][num], matches[#matches + 1] = cheat.comp(menu_blocks[leftop][num], menu_blocks[rightop][num],
optable[opsel], value) optable[opsel], value, signed, 1 << bitwidth, endian)
end end
end end
ret = true ret = true