proxmark3/client/scripts/ufodump.lua

181 lines
4.9 KiB
Lua
Raw Normal View History

local cmds = require('commands')
local getopt = require('getopt')
local lib14a = require('read14a')
local utils = require('utils')
copyright = ''
author = 'Iceman'
version = 'v1.0.1'
desc =
[[
This is a script that reads AZTEK iso14443a tags.
It starts from block 0, and ends at default block 20. Use 'b' to say different endblock.
xor: the first three block (0,1,2) is not XORED. The rest seems to be xored.
]]
example = [[
-- default
script run ufodump
2019-04-29 04:54:00 +08:00
-- stop at block 10
script run ufodump -b 10
]]
usage = [[
script run ufudump -h -b
Arguments:
h this helptext
b endblock in decimal (1-255, default 20)
]]
-- Some globals
local DEBUG = false -- the debug flag
2019-03-09 17:34:43 +08:00
---
-- A debug printout-function
local function dbg(args)
if not DEBUG then return end
if type(args) == 'table' then
local i = 1
while args[i] do
dbg(args[i])
i = i+1
end
else
print('###', args)
2019-03-09 17:34:43 +08:00
end
end
---
-- This is only meant to be used when errors occur
local function oops(err)
print('ERROR:', err)
2019-03-09 17:34:43 +08:00
core.clearCommandBuffer()
2019-04-29 04:54:00 +08:00
return nil, err
end
2019-03-09 17:34:43 +08:00
---
-- Usage help
local function help()
print(copyright)
print(author)
print(version)
2019-03-09 17:34:43 +08:00
print(desc)
print('Example usage')
2019-03-09 17:34:43 +08:00
print(example)
2019-04-29 04:54:00 +08:00
print(usage)
end
--
--- Picks out and displays the data read from a tag
-- Specifically, takes a usb packet, converts to a Command
2019-03-09 17:34:43 +08:00
-- (as in commands.lua), takes the data-array and
-- reads the number of bytes specified in arg1 (arg0 in c-struct)
-- and displays the data
-- @blockno just to print which block the data belong to
-- @param usbpacket the data received from the device
function showdata(blockno, data)
2019-03-09 17:34:43 +08:00
local xorkey = '55AA55AA55AA55AA6262'
local s = data.." | "
local dex = ''
local rs
for i = 1, 20-4, 4 do
local item = string.sub(data, i, i+3)
local xor = string.sub(xorkey, i, i+3)
if blockno > 2 then
rs = bit32.bxor(tonumber(item,16) , tonumber(xor,16))
else
rs = tonumber(item, 16)
end
dex = (dex..'%04X'):format(rs)
end
s = s..dex.." | "
print( (" %02d | %s"):format(blockno,s))
end
--
-- Send a "raw" iso14443a package, ie "hf 14a raw" command
function sendRaw(rawdata, options)
local flags = lib14a.ISO14A_COMMAND.ISO14A_NO_DISCONNECT
+ lib14a.ISO14A_COMMAND.ISO14A_RAW
+ lib14a.ISO14A_COMMAND.ISO14A_APPEND_CRC
+ lib14a.ISO14A_COMMAND.ISO14A_NO_RATS
local command = Command:newMIX{cmd = cmds.CMD_READER_ISO_14443a,
2019-03-09 17:34:43 +08:00
arg1 = flags, -- Send raw
-- arg2 contains the length, which is half the length
-- of the ASCII-string rawdata
arg2 = string.len(rawdata)/2,
data = rawdata}
2019-04-29 04:54:00 +08:00
return command:sendMIX(options.ignore_response)
end
--
-- Sends an instruction to do nothing, only disconnect
function disconnect()
local command = Command:newMIX{cmd = cmds.CMD_READER_ISO_14443a, arg1 = 0,}
2019-03-09 17:34:43 +08:00
-- We can ignore the response here, no ACK is returned for this command
-- Check /armsrc/iso14443a.c, ReaderIso14443a() for details
return command:sendMIX(true)
2019-03-09 17:34:43 +08:00
end
---
-- The main entry point
function main(args)
2019-03-09 17:34:43 +08:00
local ignore_response = false
local endblock = 20
-- Read the parameters
for o, a in getopt.getopt(args, 'hb:') do
if o == 'h' then return help() end
if o == 'b' then endblock = a end
2019-03-09 17:34:43 +08:00
end
endblock = endblock or 20
-- First of all, connect
info, err = lib14a.read(true, true)
if err then disconnect() return oops(err) end
core.clearCommandBuffer()
local blockData = {}
-- Show tag info
print(('\nFound Card UID [%s]\n'):format(info.uid))
2019-03-09 17:34:43 +08:00
print('blk | data | xored')
print('----+------------------+-------------------')
2019-03-09 17:34:43 +08:00
for block = 00, endblock do
local cmd = string.format('10%02x00', block)
2019-03-09 17:34:43 +08:00
res, err = sendRaw(cmd , {ignore_response = ignore_response})
if err then disconnect() return oops(err) end
local cmd_response = Command.parse(res)
local len = tonumber(cmd_response.arg1) * 2
local data = string.sub(tostring(cmd_response.data), 0, len-4)
showdata(block, data)
table.insert(blockData, data)
end
print("----+------------------+-------------------")
disconnect()
local filename, err = utils.WriteDumpFile(info.uid, blockData)
2019-03-09 17:34:43 +08:00
if err then return oops(err) end
print(string.format('\nDumped data into %s', filename))
end
2019-03-09 17:34:43 +08:00
-------------------------
2019-03-09 17:34:43 +08:00
-- Testing
-------------------------
function selftest()
2019-03-09 17:34:43 +08:00
DEBUG = true
dbg('Performing test')
2019-03-09 17:34:43 +08:00
main()
dbg('Tests done')
end
2019-03-09 17:34:43 +08:00
-- Flip the switch here to perform a sanity check.
-- It read a nonce in two different ways, as specified in the usage-section
if '--test' == args then
2019-03-09 17:34:43 +08:00
selftest()
else
-- Call the main
main(args)
2019-03-12 07:12:26 +08:00
end