2013-06-07 21:23:32 +08:00
|
|
|
--[[
|
2019-03-09 17:40:45 +08:00
|
|
|
This is a library to read 14443a tags. It can be used something like this
|
2013-06-07 21:23:32 +08:00
|
|
|
|
2019-03-09 17:40:45 +08:00
|
|
|
local reader = require('read14a')
|
|
|
|
result, err = reader.read14443a()
|
|
|
|
if not result then
|
|
|
|
print(err)
|
|
|
|
return
|
|
|
|
end
|
|
|
|
print(result.name)
|
2013-06-07 21:23:32 +08:00
|
|
|
|
|
|
|
--]]
|
|
|
|
-- Loads the commands-library
|
2019-04-29 05:43:55 +08:00
|
|
|
local taglib = require('taglib')
|
2013-06-07 21:23:32 +08:00
|
|
|
local cmds = require('commands')
|
2019-04-29 08:07:40 +08:00
|
|
|
|
|
|
|
-- Shouldn't take longer than 2 seconds
|
|
|
|
local TIMEOUT = 2000
|
|
|
|
|
2013-06-07 21:23:32 +08:00
|
|
|
local ISO14A_COMMAND = {
|
2019-03-09 17:40:45 +08:00
|
|
|
ISO14A_CONNECT = 1,
|
|
|
|
ISO14A_NO_DISCONNECT = 2,
|
|
|
|
ISO14A_APDU = 4,
|
|
|
|
ISO14A_RAW = 8,
|
|
|
|
ISO14A_REQUEST_TRIGGER = 0x10,
|
|
|
|
ISO14A_APPEND_CRC = 0x20,
|
|
|
|
ISO14A_SET_TIMEOUT = 0x40,
|
|
|
|
ISO14A_NO_SELECT = 0x80,
|
|
|
|
ISO14A_TOPAZMODE = 0x100,
|
|
|
|
ISO14A_NO_RATS = 0x200
|
2013-06-07 21:23:32 +08:00
|
|
|
}
|
|
|
|
|
2019-03-09 17:40:45 +08:00
|
|
|
local ISO14443a_TYPES = {}
|
2017-07-05 02:05:50 +08:00
|
|
|
ISO14443a_TYPES[0x00] = "NXP MIFARE Ultralight | Ultralight C | NTAG"
|
2014-11-03 20:49:19 +08:00
|
|
|
ISO14443a_TYPES[0x01] = "NXP MIFARE TNP3xxx Activision Game Appliance"
|
2013-06-07 21:23:32 +08:00
|
|
|
ISO14443a_TYPES[0x04] = "NXP MIFARE (various !DESFire !DESFire EV1)"
|
|
|
|
ISO14443a_TYPES[0x08] = "NXP MIFARE CLASSIC 1k | Plus 2k"
|
|
|
|
ISO14443a_TYPES[0x09] = "NXP MIFARE Mini 0.3k"
|
2018-05-13 13:24:49 +08:00
|
|
|
ISO14443a_TYPES[0x0A] = "FM11RF005SH (Shanghai Metro)"
|
2013-06-07 21:23:32 +08:00
|
|
|
ISO14443a_TYPES[0x10] = "NXP MIFARE Plus 2k"
|
|
|
|
ISO14443a_TYPES[0x11] = "NXP MIFARE Plus 4k"
|
|
|
|
ISO14443a_TYPES[0x18] = "NXP MIFARE Classic 4k | Plus 4k"
|
|
|
|
ISO14443a_TYPES[0x20] = "NXP MIFARE DESFire 4k | DESFire EV1 2k/4k/8k | Plus 2k/4k | JCOP 31/41"
|
|
|
|
ISO14443a_TYPES[0x24] = "NXP MIFARE DESFire | DESFire EV1"
|
|
|
|
ISO14443a_TYPES[0x28] = "JCOP31 or JCOP41 v2.3.1"
|
|
|
|
ISO14443a_TYPES[0x38] = "Nokia 6212 or 6131 MIFARE CLASSIC 4K"
|
|
|
|
ISO14443a_TYPES[0x88] = "Infineon MIFARE CLASSIC 1K"
|
|
|
|
ISO14443a_TYPES[0x98] = "Gemplus MPCOS"
|
|
|
|
|
2018-01-25 16:51:49 +08:00
|
|
|
local function tostring_14443a(sak)
|
2019-03-09 17:40:45 +08:00
|
|
|
return ISO14443a_TYPES[sak] or ("Unknown (SAK=%x)"):format(sak)
|
2013-06-07 21:23:32 +08:00
|
|
|
end
|
|
|
|
|
2018-01-25 16:51:49 +08:00
|
|
|
local function parse14443a(data)
|
2019-03-09 17:40:45 +08:00
|
|
|
--[[
|
|
|
|
typedef struct {
|
2019-03-21 22:19:18 +08:00
|
|
|
uint8_t uid[10];
|
|
|
|
uint8_t uidlen;
|
|
|
|
uint8_t atqa[2];
|
|
|
|
uint8_t sak;
|
|
|
|
uint8_t ats_len;
|
|
|
|
uint8_t ats[256];
|
2019-03-09 17:40:45 +08:00
|
|
|
} __attribute__((__packed__)) iso14a_card_select_t;
|
|
|
|
--]]
|
|
|
|
|
2019-06-01 01:10:01 +08:00
|
|
|
local count, uid, uidlen, atqa, sak, ats_len, ats = bin.unpack('H10CH2CCH', data)
|
2019-04-29 05:43:55 +08:00
|
|
|
uid = uid:sub(1, 2 * uidlen)
|
|
|
|
local man_byte = tonumber(uid:sub(1,2), 16)
|
|
|
|
|
2019-04-30 04:41:28 +08:00
|
|
|
return {
|
2019-04-29 05:43:55 +08:00
|
|
|
uid = uid,
|
|
|
|
atqa = atqa,
|
|
|
|
sak = sak,
|
|
|
|
name = tostring_14443a(sak),
|
|
|
|
data = data,
|
2019-06-01 01:10:01 +08:00
|
|
|
manufacturer = taglib.lookupManufacturer(man_byte),
|
|
|
|
ats = ats
|
2019-04-29 05:43:55 +08:00
|
|
|
}
|
2013-06-07 21:23:32 +08:00
|
|
|
end
|
|
|
|
|
2013-10-18 03:21:04 +08:00
|
|
|
-- This function does a connect and retrieves som einfo
|
|
|
|
-- @param dont_disconnect - if true, does not disable the field
|
|
|
|
-- @return if successfull: an table containing card info
|
|
|
|
-- @return if unsuccessfull : nil, error
|
2017-10-05 22:17:09 +08:00
|
|
|
local function read14443a(dont_disconnect, no_rats)
|
2019-03-09 17:40:45 +08:00
|
|
|
local command, result, info, err, data
|
|
|
|
|
2019-04-29 00:46:06 +08:00
|
|
|
command = Command:newMIX{
|
|
|
|
cmd = cmds.CMD_READER_ISO_14443a,
|
|
|
|
arg1 = ISO14A_COMMAND.ISO14A_CONNECT
|
|
|
|
}
|
2019-03-09 17:40:45 +08:00
|
|
|
|
|
|
|
if dont_disconnect then
|
|
|
|
command.arg1 = command.arg1 + ISO14A_COMMAND.ISO14A_NO_DISCONNECT
|
|
|
|
end
|
|
|
|
if no_rats then
|
|
|
|
command.arg1 = command.arg1 + ISO14A_COMMAND.ISO14A_NO_RATS
|
|
|
|
end
|
2019-04-29 04:54:00 +08:00
|
|
|
|
2019-05-08 04:46:00 +08:00
|
|
|
local result, err = command:sendMIX()
|
2019-03-09 17:40:45 +08:00
|
|
|
if result then
|
2019-05-25 21:06:44 +08:00
|
|
|
local count, cmd, arg1, arg2, arg3 = bin.unpack('LLLL',result)
|
|
|
|
if arg1 == 0 then
|
2019-04-29 00:46:06 +08:00
|
|
|
return nil, 'iso14443a card select failed'
|
2019-03-09 17:40:45 +08:00
|
|
|
end
|
2019-04-29 08:07:40 +08:00
|
|
|
data = string.sub(result, count)
|
2019-06-01 01:10:01 +08:00
|
|
|
info = parse14443a(data)
|
2019-03-09 17:40:45 +08:00
|
|
|
else
|
2019-04-29 00:46:06 +08:00
|
|
|
err = 'No response from card'
|
2019-03-09 17:40:45 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
if err then
|
|
|
|
print(err)
|
|
|
|
return nil, err
|
|
|
|
end
|
2019-06-01 01:10:01 +08:00
|
|
|
return info, nil
|
2013-10-18 03:21:04 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
---
|
2018-01-25 16:51:49 +08:00
|
|
|
-- Waits for a mifare card to be placed within the vicinity of the reader.
|
2013-10-18 03:21:04 +08:00
|
|
|
-- @return if successfull: an table containing card info
|
|
|
|
-- @return if unsuccessfull : nil, error
|
|
|
|
local function waitFor14443a()
|
2019-05-08 04:33:26 +08:00
|
|
|
print('Waiting for card... press any key to quit')
|
2019-03-09 17:40:45 +08:00
|
|
|
while not core.ukbhit() do
|
|
|
|
res, err = read14443a()
|
|
|
|
if res then return res end
|
|
|
|
-- err means that there was no response from card
|
|
|
|
end
|
2019-05-08 04:33:26 +08:00
|
|
|
return nil, 'Aborted by user'
|
2013-10-18 03:21:04 +08:00
|
|
|
end
|
2019-05-02 00:26:12 +08:00
|
|
|
|
|
|
|
-- Sends an instruction to do nothing, only disconnect
|
|
|
|
local function disconnect14443a()
|
|
|
|
local c = Command:newMIX{cmd = cmds.CMD_READER_ISO_14443a}
|
|
|
|
-- We can ignore the response here, no ACK is returned for this command
|
|
|
|
-- Check /armsrc/iso14443a.c, ReaderIso14443a() for details
|
2019-06-09 16:59:16 +08:00
|
|
|
return c:sendMIX(true)
|
2019-05-02 00:26:12 +08:00
|
|
|
end
|
|
|
|
|
2013-10-18 03:21:04 +08:00
|
|
|
local library = {
|
2019-03-09 17:40:45 +08:00
|
|
|
read = read14443a,
|
|
|
|
waitFor14443a = waitFor14443a,
|
|
|
|
parse14443a = parse14443a,
|
2019-05-02 00:26:12 +08:00
|
|
|
disconnect = disconnect14443a,
|
2019-03-09 17:40:45 +08:00
|
|
|
ISO14A_COMMAND = ISO14A_COMMAND,
|
2013-06-07 21:23:32 +08:00
|
|
|
}
|
|
|
|
|
2018-01-25 16:51:49 +08:00
|
|
|
return library
|