blob: 290bc7cf250437e14dc28fc9da62a24138cd19e5 [file] [log] [blame]
Sylvain Munaute7c15cd2010-12-07 10:01:55 +01001# -*- coding: utf-8 -*-
2
3""" pySim: PCSC reader transport link base
4"""
5
Harald Welte6e0458d2021-04-03 11:52:37 +02006from typing import Optional
7
Harald Weltee79cc802021-01-21 14:10:43 +01008from pySim.exceptions import *
Harald Weltee0f9ef12021-04-10 17:22:35 +02009from pySim.construct import filter_dict
10from pySim.utils import sw_match, b2h, h2b, i2h
Harald Weltee79cc802021-01-21 14:10:43 +010011
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010012#
13# Copyright (C) 2009-2010 Sylvain Munaut <tnt@246tNt.com>
Harald Weltee0f9ef12021-04-10 17:22:35 +020014# Copyright (C) 2021 Harald Welte <laforge@osmocom.org>
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010015#
16# This program is free software: you can redistribute it and/or modify
17# it under the terms of the GNU General Public License as published by
18# the Free Software Foundation, either version 2 of the License, or
19# (at your option) any later version.
20#
21# This program is distributed in the hope that it will be useful,
22# but WITHOUT ANY WARRANTY; without even the implied warranty of
23# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
24# GNU General Public License for more details.
25#
26# You should have received a copy of the GNU General Public License
27# along with this program. If not, see <http://www.gnu.org/licenses/>.
28#
29
Harald Welte7829d8a2021-04-10 11:28:53 +020030class ApduTracer:
31 def trace_command(self, cmd):
32 pass
33
34 def trace_response(self, cmd, sw, resp):
35 pass
36
37
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010038class LinkBase(object):
Harald Welteee3501f2021-04-02 13:00:18 +020039 """Base class for link/transport to card."""
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010040
Harald Welte7829d8a2021-04-10 11:28:53 +020041 def __init__(self, sw_interpreter=None, apdu_tracer=None):
Harald Welteeb05b2f2021-04-10 11:01:56 +020042 self.sw_interpreter = sw_interpreter
Harald Welte7829d8a2021-04-10 11:28:53 +020043 self.apdu_tracer = apdu_tracer
Harald Welte4f2c5462021-04-03 11:48:22 +020044
45 def set_sw_interpreter(self, interp):
46 """Set an (optional) status word interpreter."""
47 self.sw_interpreter = interp
48
Harald Welteee3501f2021-04-02 13:00:18 +020049 def wait_for_card(self, timeout:int=None, newcardonly:bool=False):
50 """Wait for a card and connect to it
Sylvain Munautbdca2522010-12-09 13:31:58 +010051
Harald Welteee3501f2021-04-02 13:00:18 +020052 Args:
53 timeout : Maximum wait time in seconds (None=no timeout)
54 newcardonly : Should we wait for a new card, or an already inserted one ?
Sylvain Munautbdca2522010-12-09 13:31:58 +010055 """
56 pass
57
58 def connect(self):
Harald Welteee3501f2021-04-02 13:00:18 +020059 """Connect to a card immediately
Sylvain Munautbdca2522010-12-09 13:31:58 +010060 """
61 pass
62
63 def disconnect(self):
Harald Welteee3501f2021-04-02 13:00:18 +020064 """Disconnect from card
Sylvain Munautbdca2522010-12-09 13:31:58 +010065 """
66 pass
67
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010068 def reset_card(self):
Harald Welteee3501f2021-04-02 13:00:18 +020069 """Resets the card (power down/up)
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010070 """
71 pass
72
Harald Welteee3501f2021-04-02 13:00:18 +020073 def send_apdu_raw(self, pdu:str):
74 """Sends an APDU with minimal processing
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010075
Harald Welteee3501f2021-04-02 13:00:18 +020076 Args:
77 pdu : string of hexadecimal characters (ex. "A0A40000023F00")
78 Returns:
79 tuple(data, sw), where
80 data : string (in hex) of returned data (ex. "074F4EFFFF")
81 sw : string (in hex) of status word (ex. "9000")
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010082 """
Harald Welte7829d8a2021-04-10 11:28:53 +020083 if self.apdu_tracer:
84 self.apdu_tracer.trace_command(pdu)
85 (data, sw) = self._send_apdu_raw(pdu)
86 if self.apdu_tracer:
87 self.apdu_tracer.trace_response(pdu, sw, data)
88 return (data, sw)
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010089
90 def send_apdu(self, pdu):
Harald Welteee3501f2021-04-02 13:00:18 +020091 """Sends an APDU and auto fetch response data
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010092
Harald Welteee3501f2021-04-02 13:00:18 +020093 Args:
94 pdu : string of hexadecimal characters (ex. "A0A40000023F00")
95 Returns:
96 tuple(data, sw), where
97 data : string (in hex) of returned data (ex. "074F4EFFFF")
98 sw : string (in hex) of status word (ex. "9000")
Sylvain Munaute7c15cd2010-12-07 10:01:55 +010099 """
100 data, sw = self.send_apdu_raw(pdu)
101
Philipp Maier859e0fd2018-06-12 18:40:24 +0200102 # When whe have sent the first APDU, the SW may indicate that there are response bytes
103 # available. There are two SWs commonly used for this 9fxx (sim) and 61xx (usim), where
104 # xx is the number of response bytes available.
105 # See also:
106 # SW1=9F: 3GPP TS 51.011 9.4.1, Responses to commands which are correctly executed
107 # SW1=61: ISO/IEC 7816-4, Table 5 — General meaning of the interindustry values of SW1-SW2
108 if (sw is not None) and ((sw[0:2] == '9f') or (sw[0:2] == '61')):
Sylvain Munaute7c15cd2010-12-07 10:01:55 +0100109 pdu_gr = pdu[0:2] + 'c00000' + sw[2:4]
110 data, sw = self.send_apdu_raw(pdu_gr)
111
112 return data, sw
113
114 def send_apdu_checksw(self, pdu, sw="9000"):
Harald Welteee3501f2021-04-02 13:00:18 +0200115 """Sends an APDU and check returned SW
Sylvain Munaute7c15cd2010-12-07 10:01:55 +0100116
Harald Welteee3501f2021-04-02 13:00:18 +0200117 Args:
118 pdu : string of hexadecimal characters (ex. "A0A40000023F00")
119 sw : string of 4 hexadecimal characters (ex. "9000"). The user may mask out certain
120 digits using a '?' to add some ambiguity if needed.
121 Returns:
122 tuple(data, sw), where
123 data : string (in hex) of returned data (ex. "074F4EFFFF")
124 sw : string (in hex) of status word (ex. "9000")
Sylvain Munaute7c15cd2010-12-07 10:01:55 +0100125 """
126 rv = self.send_apdu(pdu)
Philipp Maierd4ebb6f2018-06-12 17:56:07 +0200127
Harald Welte67d551a2021-01-21 14:50:01 +0100128 if not sw_match(rv[1], sw):
Harald Welte4f2c5462021-04-03 11:48:22 +0200129 raise SwMatchError(rv[1], sw.lower(), self.sw_interpreter)
Sylvain Munaute7c15cd2010-12-07 10:01:55 +0100130 return rv
Harald Welte6e0458d2021-04-03 11:52:37 +0200131
Harald Weltee0f9ef12021-04-10 17:22:35 +0200132 def send_apdu_constr(self, cla, ins, p1, p2, cmd_constr, cmd_data, resp_constr):
133 """Build and sends an APDU using a 'construct' definition; parses response.
134
135 Args:
136 cla : string (in hex) ISO 7816 class byte
137 ins : string (in hex) ISO 7816 instruction byte
138 p1 : string (in hex) ISO 7116 Parameter 1 byte
139 p2 : string (in hex) ISO 7116 Parameter 2 byte
140 cmd_cosntr : defining how to generate binary APDU command data
141 cmd_data : command data passed to cmd_constr
142 resp_cosntr : defining how to decode binary APDU response data
143 Returns:
144 Tuple of (decoded_data, sw)
145 """
146 cmd = cmd_constr.build(cmd_data) if cmd_data else ''
147 p3 = i2h([len(cmd)])
148 pdu = ''.join([cla, ins, p1, p2, p3, b2h(cmd)])
149 (data, sw) = self.send_apdu(pdu)
150 if data:
151 # filter the resulting dict to avoid '_io' members inside
152 rsp = filter_dict(resp_constr.parse(h2b(data)))
153 else:
154 rsp = None
155 return (rsp, sw)
156
157 def send_apdu_constr_checksw(self, cla, ins, p1, p2, cmd_constr, cmd_data, resp_constr,
158 sw_exp="9000"):
159 """Build and sends an APDU using a 'construct' definition; parses response.
160
161 Args:
162 cla : string (in hex) ISO 7816 class byte
163 ins : string (in hex) ISO 7816 instruction byte
164 p1 : string (in hex) ISO 7116 Parameter 1 byte
165 p2 : string (in hex) ISO 7116 Parameter 2 byte
166 cmd_cosntr : defining how to generate binary APDU command data
167 cmd_data : command data passed to cmd_constr
168 resp_cosntr : defining how to decode binary APDU response data
169 exp_sw : string (in hex) of status word (ex. "9000")
170 Returns:
171 Tuple of (decoded_data, sw)
172 """
173 (rsp, sw) = self.send_apdu_constr(cla, ins, p1, p2, cmd_constr, cmd_data, resp_constr)
174 if not sw_match(sw, sw_exp):
175 raise SwMatchError(sw, sw_exp.lower(), self.sw_interpreter)
176 return (rsp, sw)
177
Harald Welteeb05b2f2021-04-10 11:01:56 +0200178def init_reader(opts, **kwargs) -> Optional[LinkBase]:
Harald Welte6e0458d2021-04-03 11:52:37 +0200179 """
180 Init card reader driver
181 """
182 sl = None # type : :Optional[LinkBase]
183 try:
184 if opts.pcsc_dev is not None:
185 print("Using PC/SC reader interface")
186 from pySim.transport.pcsc import PcscSimLink
Harald Welteeb05b2f2021-04-10 11:01:56 +0200187 sl = PcscSimLink(opts.pcsc_dev, **kwargs)
Harald Welte6e0458d2021-04-03 11:52:37 +0200188 elif opts.osmocon_sock is not None:
189 print("Using Calypso-based (OsmocomBB) reader interface")
190 from pySim.transport.calypso import CalypsoSimLink
Harald Welteeb05b2f2021-04-10 11:01:56 +0200191 sl = CalypsoSimLink(sock_path=opts.osmocon_sock, **kwargs)
Harald Welte6e0458d2021-04-03 11:52:37 +0200192 elif opts.modem_dev is not None:
193 print("Using modem for Generic SIM Access (3GPP TS 27.007)")
194 from pySim.transport.modem_atcmd import ModemATCommandLink
Harald Welteeb05b2f2021-04-10 11:01:56 +0200195 sl = ModemATCommandLink(device=opts.modem_dev, baudrate=opts.modem_baud, **kwargs)
Harald Welte6e0458d2021-04-03 11:52:37 +0200196 else: # Serial reader is default
197 print("Using serial reader interface")
198 from pySim.transport.serial import SerialSimLink
Harald Welteeb05b2f2021-04-10 11:01:56 +0200199 sl = SerialSimLink(device=opts.device, baudrate=opts.baudrate, **kwargs)
Harald Welte6e0458d2021-04-03 11:52:37 +0200200 return sl
201 except Exception as e:
202 print("Card reader initialization failed with exception:\n" + str(e))
203 return None