simtrace2/usb_application/apdu_split.py

106 lines
3.2 KiB
Python
Raw Normal View History

2015-04-14 13:07:33 +00:00
#!/usr/bin/env python
2015-04-14 12:55:13 +00:00
# Code ported from simtrace host program apdu_split.c
#
# (C) 2010 by Harald Welte <hwelte@hmw-consulting.de>
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License version 2
# as published by the Free Software Foundation
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
from enum import Enum
2015-04-14 13:53:13 +00:00
class apdu_states(Enum):
2015-04-14 13:07:33 +00:00
APDU_S_CLA = 1
APDU_S_INS = 2
APDU_S_P1 = 3
APDU_S_P2 = 4
APDU_S_P3 = 5
APDU_S_DATA = 6
APDU_S_DATA_SINGLE = 7
APDU_S_SW1 = 8
APDU_S_SW2 = 9
APDU_S_FIN = 10
2015-04-14 13:53:13 +00:00
class Apdu_splitter:
2015-04-14 12:55:13 +00:00
def __init__(self):
2015-04-14 13:53:13 +00:00
self.state = apdu_states.APDU_S_CLA
2015-04-14 13:07:33 +00:00
self.buf = []
2015-04-14 12:55:13 +00:00
def func_APDU_S_INS(self, c):
self.ins = c
def func_APDU_S_CLA_P1_P2(self, c):
self.buf.append(c)
2015-04-14 13:53:13 +00:00
self.state = apdu_states(self.state.value + 1)
2015-04-14 12:55:13 +00:00
2015-04-14 13:07:33 +00:00
def func_APDU_S_P3(self, c):
2015-04-14 12:55:13 +00:00
self.buf.append(c)
2015-04-14 13:07:33 +00:00
self.data_remaining = 256 if c == 0 else c
2015-04-14 13:53:13 +00:00
self.state = apdu_states.APDU_S_SW1
2015-04-14 12:55:13 +00:00
def func_APDU_S_DATA(self, c):
self.buf.append(c)
self.data_remaining -= 1
if data_remaining == 0:
2015-04-14 13:53:13 +00:00
self.state = apdu_states.APDU_S_SW1;
2015-04-14 12:55:13 +00:00
def func_APDU_S_DATA_SINGLE(self, c):
self.buf.append(c)
self.data_remaining -= 1
2015-04-14 13:53:13 +00:00
self.state = apdu_states.APDU_S_SW1
2015-04-14 12:55:13 +00:00
def func_APDU_S_SW1(self, c):
2015-04-14 13:07:33 +00:00
if (c == 0x60):
2015-04-14 12:55:13 +00:00
print("APDU_S_SW1: NULL")
else:
# check for 'all remaining' type ACK
if c == self.ins or c == self.ins + 1 or c == ~(self.ins+1):
print("ACK")
2015-04-14 13:53:13 +00:00
self.state = apdu_states.APDU_S_DATA
2015-04-14 12:55:13 +00:00
else:
# check for 'only next byte' type ACK */
2015-04-14 13:07:33 +00:00
if c == ~(self.ins):
2015-04-14 13:53:13 +00:00
self.state = apdu_states.APDU_S_DATA_SINGLE
2015-04-14 12:55:13 +00:00
else:
# must be SW1
self.buf.append(c)
2015-04-14 13:53:13 +00:00
self.state = apdu_states.APDU_S_SW2
2015-04-14 12:55:13 +00:00
2015-04-14 13:07:33 +00:00
def func_APDU_S_SW2(self, c):
2015-04-14 12:55:13 +00:00
self.buf.append(c)
2015-04-14 13:07:33 +00:00
print("APDU:", self.buf)
2015-04-14 13:53:13 +00:00
self.state = apdu_states.APDU_S_FIN
2015-04-14 13:07:33 +00:00
Apdu_S = {
2015-04-14 13:53:13 +00:00
apdu_states.APDU_S_CLA : func_APDU_S_CLA_P1_P2,
apdu_states.APDU_S_INS : func_APDU_S_INS,
apdu_states.APDU_S_P1 : func_APDU_S_CLA_P1_P2,
apdu_states.APDU_S_P2 : func_APDU_S_CLA_P1_P2,
apdu_states.APDU_S_P3 : func_APDU_S_P3,
apdu_states.APDU_S_DATA : func_APDU_S_DATA,
apdu_states.APDU_S_DATA_SINGLE : func_APDU_S_DATA_SINGLE,
apdu_states.APDU_S_SW1 : func_APDU_S_SW1,
apdu_states.APDU_S_SW2 : func_APDU_S_SW2 }
def split(self, c):
self.Apdu_S[self.state](self, c)
2015-04-14 13:07:33 +00:00
if __name__ == '__main__':
msg = [0xA0, 0xA4, 0x00, 0x00, 0x02]
apdus = Apdu_splitter()
for c in msg:
2015-04-14 13:53:13 +00:00
print(hex(c))
2015-04-14 13:07:33 +00:00
apdus.split(c)