forked from public/pysim
Add a new pySim-shell program
pySim-prog was nice when there were only 5 parameters on a SIM that we could program, and where the use case was pretty limited. Today, we have SIM/USIM/ISIM cards with hundreds of files and even more parameters to program. We cannot add a command line argument for each file to pySim-prog. Instead, this introduces an interactive command-line shell / REPL, in which one can navigate the file system of the card, read and update files both in raw format and in decoded/parsed format. The idea is primarily inspired by Henryk Ploatz' venerable cyberflex-shell, but implemented on a more modern basis using the cmd2 python module. See https://lists.osmocom.org/pipermail/simtrace/2021-January/000860.html and https://lists.osmocom.org/pipermail/simtrace/2021-February/000864.html for some related background. Most code by Harald Welte. Some bug fixes by Philipp Maier have been squashed. Change-Id: Iad117596e922223bdc1e5b956f84844b7c577e02 Related: OS#4963
This commit is contained in:
@@ -1,10 +1,15 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
""" Various constants from ETSI TS 151.011
|
||||
""" Various constants from ETSI TS 151.011 +
|
||||
Representation of the GSM SIM/USIM/ISIM filesystem hierarchy.
|
||||
|
||||
The File (and its derived classes) uses the classes of pySim.filesystem in
|
||||
order to describe the files specified in the relevant ETSI + 3GPP specifications.
|
||||
"""
|
||||
|
||||
#
|
||||
# Copyright (C) 2017 Alexander.Chemeris <Alexander.Chemeris@gmail.com>
|
||||
# Copyright (C) 2021 Harald Welte <laforge@osmocom.org>
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
@@ -323,3 +328,301 @@ EF_AD_mode_map = {
|
||||
'02' : 'maintenance (off line)',
|
||||
'04' : 'cell test operation',
|
||||
}
|
||||
|
||||
|
||||
from pySim.utils import *
|
||||
from struct import pack, unpack
|
||||
|
||||
from pySim.filesystem import *
|
||||
import pySim.ts_102_221
|
||||
|
||||
######################################################################
|
||||
# DF.TELECOM
|
||||
######################################################################
|
||||
|
||||
# TS 51.011 Section 10.5.1
|
||||
class EF_ADN(LinFixedEF):
|
||||
def __init__(self, fid='6f3a', sfid=None, name='EF.ADN', desc='Abbreviated Dialing Numbers'):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, rec_len={14, 30})
|
||||
def _decode_record_bin(self, raw_bin_data):
|
||||
alpha_id_len = len(raw_bin_data) - 14
|
||||
alpha_id = raw_bin_data[:alpha_id_len]
|
||||
u = unpack('!BB10sBB', raw_bin_data[-14:])
|
||||
return {'alpha_id': alpha_id, 'len_of_bcd': u[0], 'ton_npi': u[1],
|
||||
'dialing_nr': u[2], 'cap_conf_id': u[3], 'ext1_record_id': u[4]}
|
||||
|
||||
# TS 51.011 Section 10.5.5
|
||||
class EF_MSISDN(LinFixedEF):
|
||||
def __init__(self, fid='6f4f', sfid=None, name='EF.MSISDN', desc='MSISDN'):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, rec_len={15, None})
|
||||
def _decode_record_hex(self, raw_hex_data):
|
||||
return {'msisdn': dec_msisdn(raw_hex_data)}
|
||||
def _encode_record_hex(self, abstract):
|
||||
return enc_msisdn(abstract['msisdn'])
|
||||
|
||||
# TS 51.011 Section 10.5.6
|
||||
class EF_SMSP(LinFixedEF):
|
||||
def __init__(self, fid='6f42', sfid=None, name='EF.SMSP', desc='Short message service parameters'):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, rec_len={28, None})
|
||||
|
||||
class DF_TELECOM(CardDF):
|
||||
def __init__(self, fid='7f10', name='DF.TELECOM', desc=None):
|
||||
super().__init__(fid=fid, name=name, desc=desc)
|
||||
files = [
|
||||
EF_ADN(),
|
||||
# FDN, SMS, CCP, ECCP
|
||||
EF_MSISDN(),
|
||||
EF_SMSP(),
|
||||
# SMSS, LND, SDN, EXT1, EXT2, EXT3, BDN, EXT4, SMSR, CMI
|
||||
]
|
||||
self.add_files(files)
|
||||
|
||||
def decode_select_response(self, data_hex):
|
||||
return decode_select_response(data_hex)
|
||||
|
||||
######################################################################
|
||||
# DF.GSM
|
||||
######################################################################
|
||||
|
||||
# TS 51.011 Section 10.3.1
|
||||
class EF_LP(TransRecEF):
|
||||
def __init__(self, fid='6f05', sfid=None, name='EF.LP', size={1,None}, rec_len=1,
|
||||
desc='Language Preference'):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size, rec_len=rec_len)
|
||||
def _decode_record_bin(self, in_bin):
|
||||
return b2h(in_bin)
|
||||
def _encode_record_bin(self, in_json):
|
||||
return h2b(in_json)
|
||||
|
||||
# TS 51.011 Section 10.3.2
|
||||
class EF_IMSI(TransparentEF):
|
||||
def __init__(self, fid='6f07', sfid=None, name='EF.IMSI', desc='IMSI', size={9,9}):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size)
|
||||
def _decode_hex(self, raw_hex):
|
||||
return {'imsi': dec_imsi(raw_hex)}
|
||||
def _encode_hex(self, abstract):
|
||||
return enc_imsi(abstract['imsi'])
|
||||
|
||||
# TS 51.011 Section 10.3.4
|
||||
class EF_PLMNsel(TransRecEF):
|
||||
def __init__(self, fid='6f30', sfid=None, name='EF.PLMNsel', desc='PLMN selector',
|
||||
size={24,None}, rec_len=3):
|
||||
super().__init__(fid, name=name, sfid=sfid, desc=desc, size=size, rec_len=rec_len)
|
||||
def _decode_record_hex(self, in_hex):
|
||||
if in_hex[:6] == "ffffff":
|
||||
return None
|
||||
else:
|
||||
return dec_plmn(in_hex)
|
||||
def _encode_record_hex(self, in_json):
|
||||
if in_json == None:
|
||||
return "ffffff"
|
||||
else:
|
||||
return enc_plmn(in_json['mcc'], in_json['mnc'])
|
||||
|
||||
# TS 51.011 Section 10.3.7
|
||||
class EF_ServiceTable(TransparentEF):
|
||||
def __init__(self, fid, sfid, name, desc, size, table):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size)
|
||||
self.table = table
|
||||
def _decode_bin(self, raw_bin):
|
||||
ret = {}
|
||||
for i in range(0, len(raw_bin)*4):
|
||||
service_nr = i+1
|
||||
byte = int(raw_bin[i//4])
|
||||
bit_offset = (i % 4) * 2
|
||||
bits = (byte >> bit_offset) & 3
|
||||
ret[service_nr] = {
|
||||
'description': self.table[service_nr] or None,
|
||||
'allocated': True if bits & 1 else False,
|
||||
'activated': True if bits & 2 else False,
|
||||
}
|
||||
return ret
|
||||
# TODO: encoder
|
||||
|
||||
# TS 51.011 Section 10.3.11
|
||||
class EF_SPN(TransparentEF):
|
||||
def __init__(self, fid='6f46', sfid=None, name='EF.SPN', desc='Service Provider Name', size={17,17}):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size)
|
||||
def _decode_hex(self, raw_hex):
|
||||
return {'spn': dec_spn(raw_hex)}
|
||||
def _encode_hex(self, abstract):
|
||||
return enc_spn(abstract['spn'])
|
||||
|
||||
# TS 51.011 Section 10.3.13
|
||||
class EF_CBMI(TransRecEF):
|
||||
def __init__(self, fid='6f45', sfid=None, name='EF.CBMI', size={2,None}, rec_len=2,
|
||||
desc='Cell Broadcast message identifier selection'):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size, rec_len=rec_len)
|
||||
|
||||
# TS 51.011 Section 10.3.15
|
||||
class EF_ACC(TransparentEF):
|
||||
def __init__(self, fid='6f78', sfid=None, name='EF.ACC', desc='Access Control Class', size={2,2}):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size)
|
||||
def _decode_bin(self, raw_bin):
|
||||
return {'acc': unpack('!H', raw_bin)[0]}
|
||||
def _encode_bin(self, abstract):
|
||||
return pack('!H', abstract['acc'])
|
||||
|
||||
# TS 51.011 Section 10.3.18
|
||||
class EF_AD(TransparentEF):
|
||||
OP_MODE = {
|
||||
0x00: 'normal operation',
|
||||
0x80: 'type approval operations',
|
||||
0x01: 'normal operation + specific facilities',
|
||||
0x81: 'type approval + specific facilities',
|
||||
0x02: 'maintenance (off line)',
|
||||
0x04: 'cell test operation',
|
||||
}
|
||||
def __init__(self, fid='6fad', sfid=None, name='EF.AD', desc='Administrative Data', size={3,4}):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size)
|
||||
def _decode_bin(self, raw_bin):
|
||||
u = unpack('!BH', raw_bin[:3])
|
||||
|
||||
# TS 51.011 Section 10.3.13
|
||||
class EF_CBMID(EF_CBMI):
|
||||
def __init__(self, fid='6f48', sfid=None, name='EF.CBMID', size={2,None}, rec_len=2,
|
||||
desc='Cell Broadcast Message Identifier for Data Download'):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size, rec_len=rec_len)
|
||||
|
||||
# TS 51.011 Section 10.3.26
|
||||
class EF_ECC(LinFixedEF):
|
||||
def __init__(self, fid='6fb7', sfid=None, name='EF.ECC', desc='Emergency Call Codes'):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, rec_len={4, 20})
|
||||
|
||||
# TS 51.011 Section 10.3.28
|
||||
class EF_CBMIR(TransRecEF):
|
||||
def __init__(self, fid='6f50', sfid=None, name='EF.CBMIR', size={4,None}, rec_len=4,
|
||||
desc='Cell Broadcast message identifier range selection'):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size, rec_len=rec_len)
|
||||
|
||||
|
||||
# TS 51.011 Section 10.3.35..37
|
||||
class EF_xPLMNwAcT(TransRecEF):
|
||||
def __init__(self, fid, sfid=None, name=None, desc=None, size={40,None}, rec_len=5):
|
||||
super().__init__(fid, sfid=sfid, name=name, desc=desc, size=size, rec_len=rec_len)
|
||||
def _decode_record_hex(self, in_hex):
|
||||
if in_hex[:6] == "ffffff":
|
||||
return None
|
||||
else:
|
||||
return dec_xplmn_w_act(in_hex)
|
||||
def _encode_record_hex(self, in_json):
|
||||
if in_json == None:
|
||||
return "ffffff0000"
|
||||
else:
|
||||
hplmn = enc_plmn(in_json['mcc'], in_json['mnc'])
|
||||
act = self.enc_act(in_json['act'])
|
||||
return hplmn + act
|
||||
@staticmethod
|
||||
def enc_act(in_list):
|
||||
u16 = 0
|
||||
# first the simple ones
|
||||
if 'UTRAN' in in_list:
|
||||
u16 |= 0x8000
|
||||
if 'NG-RAN' in in_list:
|
||||
u16 |= 0x0800
|
||||
if 'GSM COMPACT' in in_list:
|
||||
u16 |= 0x0040
|
||||
if 'cdma2000 HRPD' in in_list:
|
||||
u16 |= 0x0020
|
||||
if 'cdma2000 1xRTT' in in_list:
|
||||
u16 |= 0x0010
|
||||
# E-UTRAN
|
||||
if 'E-UTRAN WB-S1' and 'E-UTRAN NB-S1' in in_list:
|
||||
u16 |= 0x7000 # WB-S1 and NB-S1
|
||||
elif 'E-UTRAN NB-S1' in in_list:
|
||||
u16 |= 0x6000 # only WB-S1
|
||||
elif 'E-UTRAN NB-S1' in in_list:
|
||||
u16 |= 0x5000 # only NB-S1
|
||||
# GSM mess
|
||||
if 'GSM' in in_list and 'EC-GSM-IoT' in in_list:
|
||||
u16 |= 0x008C
|
||||
elif 'GSM' in in_list:
|
||||
u16 |= 0x0084
|
||||
elif 'EC-GSM-IuT' in in_list:
|
||||
u16 |= 0x0088
|
||||
return '%04X'%(u16)
|
||||
|
||||
|
||||
class DF_GSM(CardDF):
|
||||
def __init__(self, fid='7f20', name='DF.GSM', desc='GSM Network related files'):
|
||||
super().__init__(fid=fid, name=name, desc=desc)
|
||||
files = [
|
||||
EF_LP(),
|
||||
EF_IMSI(),
|
||||
TransparentEF('5f20', None, 'EF.Kc', 'Ciphering key Kc'),
|
||||
EF_PLMNsel(),
|
||||
TransparentEF('6f31', None, 'EF.HPPLMN', 'Higher Priority PLMN search period'),
|
||||
# ACMmax
|
||||
EF_ServiceTable('6f37', None, 'EF.SST', 'SIM service table', table=EF_SST_map, size={2,16}),
|
||||
CyclicEF('6f39', None, 'EF.ACM', 'Accumulated call meter', rec_len={4,3}),
|
||||
TransparentEF('6f3e', None, 'EF.GID1', 'Group Identifier Level 1'),
|
||||
TransparentEF('6f3f', None, 'EF.GID2', 'Group Identifier Level 2'),
|
||||
EF_SPN(),
|
||||
TransparentEF('6f41', None, 'EF.PUCT', 'Price per unit and currency table', size={5,5}),
|
||||
EF_CBMI(),
|
||||
TransparentEF('6f7f', None, 'EF.BCCH', 'Broadcast control channels', size={16,16}),
|
||||
EF_ACC(),
|
||||
EF_PLMNsel('6f7b', None, 'EF.FPLMN', 'Forbidden PLMNs', size={12,12}),
|
||||
TransparentEF('6f7e', None, 'EF.LOCI', 'Locationn information', size={11,11}),
|
||||
EF_AD(),
|
||||
TransparentEF('6fa3', None, 'EF.Phase', 'Phase identification', size={1,1}),
|
||||
# TODO EF.VGCS VGCSS, VBS, VBSS, eMLPP, AAeM
|
||||
EF_CBMID(),
|
||||
EF_ECC(),
|
||||
EF_CBMIR(),
|
||||
# DCK, CNL, NIA, KcGRS, LOCIGPRS, SUME
|
||||
EF_xPLMNwAcT('6f60', None, 'EF.PLMNwAcT',
|
||||
'User controlled PLMN Selector with Access Technology'),
|
||||
EF_xPLMNwAcT('6f61', None, 'EF.OPLMNwAcT',
|
||||
'Operator controlled PLMN Selector with Access Technology'),
|
||||
EF_xPLMNwAcT('6f62', None, 'EF.HPLMNwAcT', 'HPLMN Selector with Access Technology'),
|
||||
# CPBCCH, InvScan, PNN, OPL, MBDN, MBI, MWIS, CFIS, EXT5, EXT6, EXT7, SPDI, MMSN, EXT8
|
||||
# MMSICP, MMSUP, MMSUCP
|
||||
]
|
||||
self.add_files(files)
|
||||
|
||||
def decode_select_response(self, data_hex):
|
||||
return decode_select_response(data_hex)
|
||||
|
||||
def decode_select_response(resp_hex):
|
||||
resp_bin = h2b(resp_hex)
|
||||
if resp_bin[0] == 0x62:
|
||||
return pySim.ts_102_221.decode_select_response(resp_hex)
|
||||
struct_of_file_map = {
|
||||
0: 'transparent',
|
||||
1: 'linear_fixed',
|
||||
3: 'cyclic'
|
||||
}
|
||||
type_of_file_map = {
|
||||
1: 'mf',
|
||||
2: 'df',
|
||||
4: 'working_ef'
|
||||
}
|
||||
ret = {
|
||||
'file_descriptor': {},
|
||||
'proprietary_info': {},
|
||||
}
|
||||
ret['file_id'] = b2h(resp_bin[4:6])
|
||||
ret['proprietary_info']['available_memory'] = int.from_bytes(resp_bin[2:4], 'big')
|
||||
file_type = type_of_file_map[resp_bin[6]] if resp_bin[6] in type_of_file_map else resp_bin[6]
|
||||
ret['file_descriptor']['file_type'] = file_type
|
||||
if file_type in ['mf', 'df']:
|
||||
ret['file_characteristics'] = b2h(resp_bin[13])
|
||||
ret['num_direct_child_df'] = int(resp_bin[14], 16)
|
||||
ret['num_direct_child_ef'] = int(resp_bin[15], 16)
|
||||
ret['num_chv_unbkock_adm_codes'] = int(resp_bin[16])
|
||||
# CHV / UNBLOCK CHV stats
|
||||
elif file_type in ['working_ef']:
|
||||
file_struct = struct_of_file_map[resp_bin[13]] if resp_bin[13] in struct_of_file_map else resp_bin[13]
|
||||
ret['file_descriptor']['structure'] = file_struct
|
||||
ret['access_conditions'] = b2h(resp_bin[8:10])
|
||||
if resp_bin[11] & 0x01 == 0:
|
||||
ret['life_cycle_status_int'] = 'operational_activated'
|
||||
elif resp_bin[11] & 0x04:
|
||||
ret['life_cycle_status_int'] = 'operational_deactivated'
|
||||
else:
|
||||
ret['life_cycle_status_int'] = 'terminated'
|
||||
|
||||
return ret
|
||||
|
||||
CardProfileSIM = CardProfile('SIM', desc='GSM SIM Card', files_in_mf=[DF_TELECOM(), DF_GSM()])
|
||||
|
||||
Reference in New Issue
Block a user