2015-10-30 16:07:04 +08:00
|
|
|
local bin = require('bin')
|
|
|
|
local getopt = require('getopt')
|
2017-08-16 03:23:55 +08:00
|
|
|
local lib14a = require('read14a')
|
2015-10-30 16:07:04 +08:00
|
|
|
local utils = require('utils')
|
|
|
|
|
2017-08-16 03:23:55 +08:00
|
|
|
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 =[[
|
2017-08-16 03:23:55 +08:00
|
|
|
-- if called without, it reads tag uid
|
2015-10-30 16:07:04 +08:00
|
|
|
script run calc_ev1_it
|
2017-08-16 03:23:55 +08:00
|
|
|
|
|
|
|
--
|
2015-10-30 16:07:04 +08:00
|
|
|
script run calc_ev1_it -u 11223344556677
|
|
|
|
]]
|
2017-08-16 03:23:55 +08:00
|
|
|
usage = [[
|
|
|
|
script run calc_ev1_it -h -u <uid> "
|
|
|
|
|
2015-10-30 16:07:04 +08:00
|
|
|
Arguments:
|
|
|
|
-h : this help
|
|
|
|
-u <UID> : UID
|
|
|
|
]]
|
2017-08-16 03:23:55 +08:00
|
|
|
|
|
|
|
local DEBUG = true
|
|
|
|
local bxor = bit32.bxor
|
2015-10-30 16:07:04 +08:00
|
|
|
---
|
|
|
|
-- A debug printout-function
|
2017-08-16 03:23:55 +08:00
|
|
|
local function dbg(args)
|
2017-12-03 01:59:47 +08:00
|
|
|
if not DEBUG then return end
|
|
|
|
|
2015-10-30 16:07:04 +08:00
|
|
|
if type(args) == "table" then
|
|
|
|
local i = 1
|
|
|
|
while args[i] do
|
|
|
|
dbg(args[i])
|
|
|
|
i = i+1
|
|
|
|
end
|
|
|
|
else
|
|
|
|
print("###", args)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
---
|
|
|
|
-- This is only meant to be used when errors occur
|
2017-08-16 03:23:55 +08:00
|
|
|
local function oops(err)
|
2015-10-30 16:07:04 +08:00
|
|
|
print("ERROR: ",err)
|
|
|
|
return nil,err
|
|
|
|
end
|
|
|
|
---
|
|
|
|
-- Usage help
|
2017-08-16 03:23:55 +08:00
|
|
|
local function help()
|
|
|
|
print(copyright)
|
|
|
|
print(version)
|
2015-10-30 16:07:04 +08:00
|
|
|
print(desc)
|
|
|
|
print("Example usage")
|
|
|
|
print(example)
|
|
|
|
end
|
|
|
|
--
|
|
|
|
-- Exit message
|
|
|
|
function exitMsg(msg)
|
|
|
|
print( string.rep('--',20) )
|
|
|
|
print( string.rep('--',20) )
|
|
|
|
print(msg)
|
|
|
|
print()
|
|
|
|
end
|
|
|
|
|
|
|
|
local _xortable = {
|
|
|
|
--[[ position, 4byte xor
|
|
|
|
--]]
|
|
|
|
{"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"},
|
|
|
|
}
|
|
|
|
|
|
|
|
local function findEntryByUid( uid )
|
|
|
|
|
|
|
|
-- xor UID4,UID5,UID6,UID7
|
|
|
|
-- mod 0x20 (dec 32)
|
|
|
|
local pos = (bxor(bxor(bxor(uid[4],uid[5]), uid[6]),uid[7])) % 32
|
|
|
|
|
|
|
|
-- convert to hexstring
|
|
|
|
pos = string.format('%02X', pos)
|
|
|
|
|
|
|
|
for k, v in pairs(_xortable) do
|
|
|
|
if ( v[1] == pos ) then
|
|
|
|
return utils.ConvertHexToBytes(v[2])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return nil
|
|
|
|
end
|
2017-08-16 03:23:55 +08:00
|
|
|
---
|
|
|
|
-- create pwd
|
|
|
|
local function pwdgen(uid)
|
|
|
|
-- 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)
|
|
|
|
if entry == nil then return nil, "Can't find a xor entry" end
|
2015-10-30 16:07:04 +08:00
|
|
|
|
2017-08-16 03:23:55 +08:00
|
|
|
local pwd0 = bxor( bxor( bxor( entry[1], uidbytes[2]), uidbytes[3]), uidbytes[4])
|
|
|
|
local pwd1 = bxor( bxor( bxor( entry[2], uidbytes[1]), uidbytes[3]), uidbytes[5])
|
|
|
|
local pwd2 = bxor( bxor( 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
|
|
|
|
--
|
|
|
|
-- main
|
2015-10-30 16:07:04 +08:00
|
|
|
local function main(args)
|
|
|
|
|
|
|
|
print( string.rep('--',20) )
|
|
|
|
print( string.rep('--',20) )
|
|
|
|
print()
|
|
|
|
|
|
|
|
local uid = '04111211121110'
|
2017-08-16 03:23:55 +08:00
|
|
|
local useUID = false
|
2015-10-30 16:07:04 +08:00
|
|
|
|
|
|
|
-- Arguments for the script
|
|
|
|
for o, a in getopt.getopt(args, 'hu:') do
|
|
|
|
if o == "h" then return help() end
|
2017-08-16 03:23:55 +08:00
|
|
|
if o == "u" then uid = a; useUID = true end
|
2015-10-30 16:07:04 +08:00
|
|
|
end
|
|
|
|
|
2017-08-16 03:23:55 +08:00
|
|
|
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
|
2018-01-25 16:51:49 +08:00
|
|
|
local tag, err = lib14a.read(false, true)
|
2017-08-16 03:23:55 +08:00
|
|
|
if not tag then return oops(err) end
|
|
|
|
core.clearCommandBuffer()
|
|
|
|
uid = tag.uid
|
|
|
|
end
|
2015-10-30 16:07:04 +08:00
|
|
|
|
|
|
|
print('UID | '..uid)
|
2017-08-16 03:23:55 +08:00
|
|
|
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
|
|
|
|
|
|
|
|
main(args)
|