proxmark3/client/luascripts/calc_ev1_it.lua

181 lines
4.1 KiB
Lua
Raw Normal View History

2015-10-30 16:07:04 +08:00
local bin = require('bin')
local getopt = require('getopt')
local lib14a = require('read14a')
2015-10-30 16:07:04 +08:00
local utils = require('utils')
copyright = ''
author = "Iceman"
version = 'v1.0.0'
desc = [[
This script calculates mifare Ultralight-EV1 pwd based on uid diversification for an Italian ticketsystem
Algo not found by me.
]]
2015-10-30 16:07:04 +08:00
example =[[
-- if called without, it reads tag uid
2019-03-09 17:34:43 +08:00
script run calc_ev1_it
--
script run calc_ev1_it -u 11223344556677
2015-10-30 16:07:04 +08:00
]]
usage = [[
script run calc_ev1_it -h -u <uid> "
2015-10-30 16:07:04 +08:00
Arguments:
2019-03-09 17:34:43 +08:00
-h : this help
-u <UID> : UID
2015-10-30 16:07:04 +08:00
]]
local DEBUG = true
local bxor = bit32.bxor
2019-03-09 17:34:43 +08:00
---
2015-10-30 16:07:04 +08:00
-- A debug printout-function
local function dbg(args)
2019-03-09 17:34:43 +08:00
if not DEBUG then return end
2019-05-08 04:12:18 +08:00
if type(args) == 'table' then
2019-03-09 17:34:43 +08:00
local i = 1
while args[i] do
dbg(args[i])
i = i+1
end
else
2019-05-08 04:12:18 +08:00
print('###', args)
2019-03-09 17:34:43 +08:00
end
end
---
2015-10-30 16:07:04 +08:00
-- This is only meant to be used when errors occur
local function oops(err)
2019-05-08 04:12:18 +08:00
print('ERROR: ', err)
core.clearCommandBuffer()
return nil, err
2015-10-30 16:07:04 +08:00
end
2019-03-09 17:34:43 +08:00
---
2015-10-30 16:07:04 +08:00
-- Usage help
local function help()
2019-03-09 17:34:43 +08:00
print(copyright)
print(author)
print(version)
print(desc)
print("Example usage")
print(example)
2019-05-08 07:35:51 +08:00
print(usage)
2015-10-30 16:07:04 +08:00
end
--
-- Exit message
2019-05-08 04:12:18 +08:00
local function exitMsg(msg)
2019-03-09 17:34:43 +08:00
print( string.rep('--',20) )
print( string.rep('--',20) )
print(msg)
print()
2015-10-30 16:07:04 +08:00
end
local _xortable = {
--[[ position, 4byte xor
--]]
2019-03-09 17:34:43 +08:00
{"00","4f2711c1"},
{"01","07D7BB83"},
{"02","9636EF07"},
{"03","B5F4460E"},
{"04","F271141C"},
{"05","7D7BB038"},
{"06","636EF871"},
{"07","5F4468E3"},
{"08","271149C7"},
{"09","D7BB0B8F"},
{"0A","36EF8F1E"},
{"0B","F446863D"},
{"0C","7114947A"},
{"0D","7BB0B0F5"},
{"0E","6EF8F9EB"},
{"0F","44686BD7"},
{"10","11494fAF"},
{"11","BB0B075F"},
{"12","EF8F96BE"},
{"13","4686B57C"},
{"14","1494F2F9"},
{"15","B0B07DF3"},
{"16","F8F963E6"},
{"17","686B5FCC"},
{"18","494F2799"},
{"19","0B07D733"},
{"1A","8F963667"},
{"1B","86B5F4CE"},
{"1C","94F2719C"},
{"1D","B07D7B38"},
{"1E","F9636E70"},
{"1F","6B5F44E0"},
2015-10-30 16:07:04 +08:00
}
local function findEntryByUid( uid )
2019-03-09 17:34:43 +08:00
-- xor UID4,UID5,UID6,UID7
-- mod 0x20 (dec 32)
local pos = (bxor(uid[4], uid[5], uid[6], uid[7])) % 32
-- convert to hexstring
pos = string.format('%02X', pos)
2015-10-30 16:07:04 +08:00
for k, v in pairs(_xortable) do
2019-03-09 17:34:43 +08:00
if ( v[1] == pos ) then
return utils.ConvertHexToBytes(v[2])
end
end
return nil
2015-10-30 16:07:04 +08:00
end
---
-- create pwd
local function pwdgen(uid)
2019-03-09 17:34:43 +08:00
-- PWD CALC
-- PWD0 = T0 xor B xor C xor D
-- PWD1 = T1 xor A xor C xor E
-- PWD2 = T2 xor A xor B xor F
-- PWD3 = T3 xor G
local uidbytes = utils.ConvertHexToBytes(uid)
local entry = findEntryByUid(uidbytes)
2019-03-09 17:34:43 +08:00
if entry == nil then return nil, "Can't find a xor entry" end
2015-10-30 16:07:04 +08:00
local pwd0 = bxor( entry[1], uidbytes[2], uidbytes[3], uidbytes[4])
2019-03-09 17:34:43 +08:00
local pwd1 = bxor( entry[2], uidbytes[1], uidbytes[3], uidbytes[5])
local pwd2 = bxor( entry[3], uidbytes[1], uidbytes[2], uidbytes[6])
local pwd3 = bxor( entry[4], uidbytes[7])
return string.format('%02X%02X%02X%02X', pwd0, pwd1, pwd2, pwd3)
end
2019-05-08 04:12:18 +08:00
--
-- main
2015-10-30 16:07:04 +08:00
local function main(args)
2019-03-09 17:34:43 +08:00
print( string.rep('--',20) )
print( string.rep('--',20) )
print()
local uid = '04111211121110'
local useUID = false
-- Arguments for the script
for o, a in getopt.getopt(args, 'hu:') do
2019-05-08 04:12:18 +08:00
if o == 'h' then return help() end
if o == 'u' then uid = a; useUID = true end
2019-03-09 17:34:43 +08:00
end
if useUID then
-- uid string checks
if uid == nil then return oops('empty uid string') end
if #uid == 0 then return oops('empty uid string') end
if #uid ~= 14 then return oops('uid wrong length. Should be 7 hex bytes') end
else
-- GET TAG UID
local tag, err = lib14a.read(false, true)
if not tag then return oops(err) end
core.clearCommandBuffer()
uid = tag.uid
end
print('UID | '..uid)
local pwd, err = pwdgen(uid)
if not pwd then return ooops(err) end
print(string.format('PWD | %s', pwd))
2015-10-30 16:07:04 +08:00
end
2019-03-12 07:12:26 +08:00
main(args)