aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorHarald Welte <laforge@gnumonks.org>2014-10-26 20:14:04 +0100
committerHarald Welte <laforge@gnumonks.org>2014-10-26 20:14:04 +0100
commita97ebc9599d4b3210d7127a2cc75d552e69eeb9a (patch)
tree60eee24f5b78ad5115b21cb9996406e5cbf4fd3d /src
parent42645d7d78a22db7b76039407b7b43bc0cc79c54 (diff)
parent534163badec879a644af985c9f5d04b7c957a355 (diff)
Merge branch 'laforge/sim'
This contains some work from 2012 about creating a general library about the data structure of data contained on SIM cards. This 'libosmosim' is supposed to be used by (among others) * tools to review/modify/personalize SIM card filesystem data * virtual SIM card implementation
Diffstat (limited to 'src')
-rw-r--r--src/sim/Makefile.am21
-rw-r--r--src/sim/card_fs_isim.c105
-rw-r--r--src/sim/card_fs_sim.c477
-rw-r--r--src/sim/card_fs_tetra.c267
-rw-r--r--src/sim/card_fs_uicc.c208
-rw-r--r--src/sim/card_fs_usim.c380
-rw-r--r--src/sim/core.c312
-rw-r--r--src/sim/gsm_int.h17
-rw-r--r--src/sim/reader.c271
-rw-r--r--src/sim/reader_pcsc.c161
-rw-r--r--src/sim/sim_int.h41
11 files changed, 2260 insertions, 0 deletions
diff --git a/src/sim/Makefile.am b/src/sim/Makefile.am
new file mode 100644
index 00000000..2b30ae99
--- /dev/null
+++ b/src/sim/Makefile.am
@@ -0,0 +1,21 @@
+# This is _NOT_ the library release version, it's an API version.
+# Please read Chapter 6 "Library interface versions" of the libtool documentation before making any modification
+LIBVERSION=0:0:0
+
+INCLUDES = $(all_includes) -I$(top_srcdir)/include
+AM_CFLAGS = -fPIC -Wall $(LIBOSMOCORE_CFLAGS) $(LIBOSMOGSM_CFLAGS) $(PCSC_CFLAGS)
+AM_LDFLAGS = $(COVERAGE_LDFLAGS)
+
+if ENABLE_PCSC
+# FIXME: only build the PC/SC dependent part conditional, but always build other parts
+
+noinst_HEADERS = sim_int.h gsm_int.h
+
+lib_LTLIBRARIES = libosmosim.la
+
+libosmosim_la_SOURCES = core.c reader.c reader_pcsc.c \
+ card_fs_sim.c card_fs_usim.c card_fs_uicc.c card_fs_isim.c card_fs_tetra.c
+libosmosim_la_LDFLAGS = -version-info $(LIBVERSION)
+libosmosim_la_LIBADD = $(LIBOSMOCORE_LIBS) $(LIBOSMOGSM_LIBS) $(PCSC_LIBS)
+
+endif
diff --git a/src/sim/card_fs_isim.c b/src/sim/card_fs_isim.c
new file mode 100644
index 00000000..339e8627
--- /dev/null
+++ b/src/sim/card_fs_isim.c
@@ -0,0 +1,105 @@
+/* 3GPP ISIM specific structures / routines */
+/*
+ * (C) 2014 by Harald Welte <laforge@gnumonks.org>
+ *
+ * All Rights Reserved
+ *
+ * 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
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+
+#include <errno.h>
+#include <string.h>
+
+#include <osmocom/sim/sim.h>
+#include <osmocom/core/talloc.h>
+#include <osmocom/gsm/gsm48.h>
+
+#include "sim_int.h"
+#include "gsm_int.h"
+
+/* TS 31.103 Version 11.2.0 Release 11 / Chapoter 7.1.3 */
+const struct osim_card_sw ts31_103_sw[] = {
+ {
+ 0x9862, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - Authentication error, incorrect MAC",
+ },
+ OSIM_CARD_SW_LAST
+};
+
+static const struct osim_card_sw *isim_card_sws[] = {
+ ts31_103_sw,
+ ts102221_uicc_sw,
+ NULL
+};
+
+/* TS 31.103 Version 11.2.0 Release 11 / Chapoter 4.2 */
+static const struct osim_file_desc isim_ef_in_adf_isim[] = {
+ EF_TRANSP_N(0x6F02, 0x02, "EF.IMPI", 0, 1, 256,
+ "IMS private user identity"),
+ EF_TRANSP_N(0x6F03, 0x05, "EF.DOMAIN", 0, 1, 256,
+ "Home Network Domain Name"),
+ EF_LIN_FIX_N(0x6F04, 0x04, "EF.IMPU", 0, 1, 256,
+ "IMS public user identity"),
+ EF_TRANSP_N(0x6FAD, 0x03, "EF.AD", 0, 4, 16,
+ "Administrative Data"),
+ EF_LIN_FIX_N(0x6F06, 0x06, "EF.ARR", 0, 1, 256,
+ "Access Rule TLV data objects"),
+ EF_TRANSP_N(0x6F07, 0x07, "EF.IST", F_OPTIONAL, 1, 16,
+ "ISIM Service Table"),
+ EF_LIN_FIX_N(0x6F09, SFI_NONE, "EF.P-CSCF", F_OPTIONAL, 1, 256,
+ "P-CSCF Address"),
+ EF_TRANSP_N(0x6FD5, SFI_NONE, "EF.GBABP", F_OPTIONAL, 1, 35,
+ "GBA Bootstrapping parameters"),
+ EF_LIN_FIX_N(0x6FD7, SFI_NONE, "EF.GBANL", F_OPTIONAL, 1, 256,
+ "NAF Key Identifier TLV Objects"),
+ EF_LIN_FIX_N(0x6FDD, SFI_NONE, "EF.NAFKCA", F_OPTIONAL, 1, 256,
+ "NAF Key Centre Address"),
+ EF_LIN_FIX_N(0x6F3C, SFI_NONE, "EF.SMS", F_OPTIONAL, 176, 176,
+ "Short messages"),
+ EF_TRANSP_N(0x6F43, SFI_NONE, "EF.SMSS", F_OPTIONAL, 2, 4,
+ "SMS status"),
+ EF_LIN_FIX_N(0x6F47, SFI_NONE, "EF.SMSR", F_OPTIONAL, 30, 30,
+ "Short message status reports"),
+ EF_LIN_FIX_N(0x6F42, SFI_NONE, "EF.SMSP", F_OPTIONAL, 29, 64,
+ "Short message service parameters"),
+ EF_LIN_FIX_N(0x6FE7, SFI_NONE, "EF.UICCIARI", F_OPTIONAL, 1, 256,
+ "UICC IARI"),
+};
+
+/* Annex E - TS 101 220 */
+static const uint8_t adf_isim_aid[] = { 0xA0, 0x00, 0x00, 0x00, 0x87, 0x10, 0x04 };
+
+struct osim_card_profile *osim_cprof_isim(void *ctx)
+{
+ struct osim_card_profile *cprof;
+ struct osim_file_desc *mf;
+
+ cprof = talloc_zero(ctx, struct osim_card_profile);
+ cprof->name = "3GPP ISIM";
+ cprof->sws = isim_card_sws;
+
+ mf = alloc_df(cprof, 0x3f00, "MF");
+
+ cprof->mf = mf;
+
+ /* ADF.USIM with its EF siblings */
+ add_adf_with_ef(mf, adf_isim_aid, sizeof(adf_isim_aid),
+ "ADF.ISIM", isim_ef_in_adf_isim,
+ ARRAY_SIZE(isim_ef_in_adf_isim));
+
+ return cprof;
+}
diff --git a/src/sim/card_fs_sim.c b/src/sim/card_fs_sim.c
new file mode 100644
index 00000000..432c945b
--- /dev/null
+++ b/src/sim/card_fs_sim.c
@@ -0,0 +1,477 @@
+/* classic SIM card specific structures/routines */
+/*
+ * (C) 2012-2014 by Harald Welte <laforge@gnumonks.org>
+ *
+ * All Rights Reserved
+ *
+ * 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
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+#include <errno.h>
+#include <string.h>
+
+#include <osmocom/sim/sim.h>
+#include <osmocom/core/talloc.h>
+#include <osmocom/gsm/gsm48.h>
+
+#include "sim_int.h"
+
+/* 3GPP TS 51.011 / Chapter 9.4 */
+static const struct osim_card_sw ts11_11_sw[] = {
+ {
+ 0x9000, 0xffff, SW_TYPE_STR, SW_CLS_OK,
+ .u.str = "Normal ending of the command",
+ }, {
+ 0x9100, 0xff00, SW_TYPE_STR, SW_CLS_OK,
+ .u.str = "Normal ending of the command - proactive command from SIM pending",
+ }, {
+ 0x9e00, 0xff00, SW_TYPE_STR, SW_CLS_OK,
+ .u.str = "Normal ending of the command - response data for SIM data download",
+ }, {
+ 0x9f00, 0xff00, SW_TYPE_STR, SW_CLS_OK,
+ .u.str = "Normal ending of the command - response data available",
+ }, {
+ 0x9300, 0xffff, SW_TYPE_STR, SW_CLS_POSTP,
+ .u.str = "SIM Application Toolkit is busy, command cannot be executed at present",
+ }, {
+ 0x9200, 0xfff0, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "Memory management - Command successful but after using an internal updat retry X times",
+ }, {
+ 0x9240, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Memory management - Memory problem",
+ }, {
+ 0x9400, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Referencing management - no EF selected",
+ }, {
+ 0x9402, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Referencing management - out of range (invalid address)",
+ }, {
+ 0x9404, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Referencing management - file ID not found / pattern not found",
+ }, {
+ 0x9408, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Referencing management - file is inconsistent with the command",
+ }, {
+ 0x9802, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - no CHV initialized",
+ }, {
+ 0x9804, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - access condition not fulfilled",
+ }, {
+ 0x9808, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - in contradiction with CHV status",
+ }, {
+ 0x9810, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - in contradiction with invalidation status",
+ }, {
+ 0x9840, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - unsuccessful CHV verification, no attempt left",
+ }, {
+ 0x9850, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - increase cannot be performed, max value reached",
+ }, {
+ 0x6700, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - incorrect parameter P3",
+ }, {
+ 0x6b00, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - incorrect parameter P1 or P2",
+ }, {
+ 0x6d00, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - unknown instruction code",
+ }, {
+ 0x6e00, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - wrong instruction class",
+ }, {
+ 0x6f00, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - technical problem with no diagnostic given",
+ },
+ OSIM_CARD_SW_LAST
+};
+
+static const struct osim_card_sw *sim_card_sws[] = {
+ ts11_11_sw,
+ NULL
+};
+
+static int iccid_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data)
+{
+ struct osim_decoded_element *elem;
+
+ elem = element_alloc(dd, "ICCID", ELEM_T_BCD, ELEM_REPR_DEC);
+ elem->length = len;
+ elem->u.buf = talloc_memdup(elem, data, len);
+
+ return 0;
+}
+
+static int elp_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data)
+{
+ int i, num_lp = len / 2;
+
+ for (i = 0; i < num_lp; i++) {
+ uint8_t *cur = data + i*2;
+ struct osim_decoded_element *elem;
+ elem = element_alloc(dd, "Language Code", ELEM_T_STRING, ELEM_REPR_NONE);
+ elem->u.buf = (uint8_t *) talloc_strndup(elem, (const char *) cur, 2);
+ }
+
+ return 0;
+}
+
+/* 10.3.1 */
+int gsm_lp_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data)
+{
+ int i;
+
+ for (i = 0; i < len; i++) {
+ struct osim_decoded_element *elem;
+ elem = element_alloc(dd, "Language Code", ELEM_T_UINT8, ELEM_REPR_DEC);
+ elem->u.u8 = data[i];
+ }
+
+ return 0;
+}
+
+/* 10.3.2 */
+int gsm_imsi_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data)
+{
+ struct osim_decoded_element *elem;
+
+ if (len < 2)
+ return -EINVAL;
+
+ elem = element_alloc(dd, "IMSI", ELEM_T_BCD, ELEM_REPR_DEC);
+ elem->length = data[0];
+ elem->u.buf = talloc_memdup(elem, data+1, len-1);
+
+ return 0;
+}
+
+/* 10.3.3 */
+static int gsm_kc_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data)
+{
+ struct osim_decoded_element *kc, *cksn;
+
+ if (len < 9)
+ return -EINVAL;
+
+ kc = element_alloc(dd, "Kc", ELEM_T_BYTES, ELEM_REPR_HEX);
+ kc->u.buf = talloc_memdup(kc, data, 8);
+ cksn = element_alloc(dd, "CKSN", ELEM_T_UINT8, ELEM_REPR_DEC);
+ cksn->u.u8 = data[8];
+
+ return 0;
+}
+
+/* 10.3.4 */
+static int gsm_plmnsel_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data)
+{
+ int i, n_plmn = len / 3;
+
+ if (n_plmn < 1)
+ return -EINVAL;
+
+ for (i = 0; i < n_plmn; i++) {
+ uint8_t *cur = data + 3*i;
+ struct osim_decoded_element *elem, *mcc, *mnc;
+ uint8_t ra_buf[6];
+ struct gprs_ra_id ra_id;
+
+ memset(ra_buf, 0, sizeof(ra_buf));
+ memcpy(ra_buf, cur, 3);
+ gsm48_parse_ra(&ra_id, ra_buf);
+
+ elem = element_alloc(dd, "PLMN", ELEM_T_GROUP, ELEM_REPR_NONE);
+
+ mcc = element_alloc_sub(elem, "MCC", ELEM_T_UINT16, ELEM_REPR_DEC);
+ mcc->u.u16 = ra_id.mcc;
+
+ mnc = element_alloc_sub(elem, "MNC", ELEM_T_UINT16, ELEM_REPR_DEC);
+ mnc->u.u16 = ra_id.mnc;
+ }
+
+ return 0;
+}
+
+/* 10.3.5 */
+int gsm_hpplmn_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data)
+{
+ struct osim_decoded_element *elem;
+
+ elem = element_alloc(dd, "Time interval", ELEM_T_UINT8, ELEM_REPR_DEC);
+ elem->u.u8 = *data;
+
+ return 0;
+}
+
+/* Chapter 10.1. Contents of the EFs at the MF level */
+static const struct osim_file_desc sim_ef_in_mf[] = {
+ EF_TRANSP(0x2FE2, SFI_NONE, "EF.ICCID", 0, 10, 10,
+ "ICC Identification", &iccid_decode, NULL),
+ EF_TRANSP(0x2F05, SFI_NONE, "EF.PL", F_OPTIONAL, 2, 20,
+ "Preferred language", &elp_decode, NULL),
+};
+
+/* Chapter 10.3.x Contents of files at the GSM application level */
+static const struct osim_file_desc sim_ef_in_gsm[] = {
+ EF_TRANSP(0x6F05, SFI_NONE, "EF.LP", 0, 1, 16,
+ "Language preference", &gsm_lp_decode, NULL),
+ EF_TRANSP(0x6F07, SFI_NONE, "EF.IMSI", 0, 9, 9,
+ "IMSI", &gsm_imsi_decode, NULL),
+ EF_TRANSP(0x6F20, SFI_NONE, "EF.Kc", 0, 9, 9,
+ "Ciphering key Kc", &gsm_kc_decode, NULL),
+ EF_TRANSP(0x6F30, SFI_NONE, "EF.PLMNsel", F_OPTIONAL, 24, 72,
+ "PLMN selector", &gsm_plmnsel_decode, NULL),
+ EF_TRANSP(0x6F31, SFI_NONE, "EF.HPPLMN", 0, 1, 1,
+ "Higher Priority PLMN search period", &gsm_hpplmn_decode, NULL),
+ EF_TRANSP_N(0x6F37, SFI_NONE, "EF.ACMmax", F_OPTIONAL, 3, 3,
+ "ACM maximum value"),
+ EF_TRANSP_N(0x6F38, SFI_NONE, "EF.SST", 0, 2, 16,
+ "SIM service table"),
+ EF_CYCLIC_N(0x6F39, SFI_NONE, "EF.ACM", F_OPTIONAL, 3, 3,
+ "Accumulated call meter"),
+ EF_TRANSP_N(0x6F3E, SFI_NONE, "EF.GID1", F_OPTIONAL, 1, 8,
+ "Group Identifier Level 1"),
+ EF_TRANSP_N(0x6F3F, SFI_NONE, "EF.GID2", F_OPTIONAL, 1, 8,
+ "Group Identifier Level 2"),
+ EF_TRANSP_N(0x6F46, SFI_NONE, "EF.SPN", F_OPTIONAL, 17, 17,
+ "Service Provider Name"),
+ EF_TRANSP_N(0x6F41, SFI_NONE, "EF.PUCT", F_OPTIONAL, 5, 5,
+ "Price per unit and currency table"),
+ EF_TRANSP_N(0x6F45, SFI_NONE, "EF.CBMI", F_OPTIONAL, 2, 32,
+ "Cell broadcast massage identifier selection"),
+ EF_TRANSP_N(0x6F74, SFI_NONE, "EF.BCCH", 0, 16, 16,
+ "Broadcast control channels"),
+ EF_TRANSP_N(0x6F78, SFI_NONE, "EF.ACC", 0, 2, 2,
+ "Access control class"),
+ EF_TRANSP_N(0x6F7B, SFI_NONE, "EF.FPLMN", 0, 12, 12,
+ "Forbidden PLMNs"),
+ EF_TRANSP_N(0x6F7E, SFI_NONE, "EF.LOCI", 0, 11, 11,
+ "Location information"),
+ EF_TRANSP_N(0x6FAD, SFI_NONE, "EF.AD", 0, 3, 8,
+ "Administrative data"),
+ EF_TRANSP_N(0x6FAE, SFI_NONE, "EF.Phase", 0, 1, 1,
+ "Phase identification"),
+ EF_TRANSP_N(0x6FB1, SFI_NONE, "EF.VGCS", F_OPTIONAL, 4, 80,
+ "Voice Group Call Service"),
+ EF_TRANSP_N(0x6FB2, SFI_NONE, "EF.VGCSS", F_OPTIONAL, 7, 7,
+ "Voice Group Call Service Status"),
+ EF_TRANSP_N(0x6FB3, SFI_NONE, "EF.VBS", F_OPTIONAL, 4, 80,
+ "Voice Broadcast Service"),
+ EF_TRANSP_N(0x6FB4, SFI_NONE, "EF.VBSS", F_OPTIONAL, 7, 7,
+ "Voice Broadcast Service Status"),
+ EF_TRANSP_N(0x6FB5, SFI_NONE, "EF.eMLPP", F_OPTIONAL, 2, 2,
+ "enhanced Mult Level Pre-emption and Priority"),
+ EF_TRANSP_N(0x6FB6, SFI_NONE, "EF.AAeM", F_OPTIONAL, 1, 1,
+ "Automatic Answer for eMLPP Service"),
+ EF_TRANSP_N(0x6F48, SFI_NONE, "EF.CBMID", F_OPTIONAL, 2, 32,
+ "Cell Broadcast Message Identifier for Data Download"),
+ EF_TRANSP_N(0x6FB7, SFI_NONE, "EF.ECC", F_OPTIONAL, 3, 15,
+ "Emergency Call Code"),
+ EF_TRANSP_N(0x6F50, SFI_NONE, "EF.CBMIR", F_OPTIONAL, 4, 64,
+ "Cell broadcast message identifier range selection"),
+ EF_TRANSP_N(0x6F2C, SFI_NONE, "EF.DCK", F_OPTIONAL, 16, 16,
+ "De-personalization Control Keys"),
+ EF_TRANSP_N(0x6F32, SFI_NONE, "EF.CNL", F_OPTIONAL, 6, 60,
+ "Co-operative Network List"),
+ EF_LIN_FIX_N(0x6F51, SFI_NONE, "EF.NIA", F_OPTIONAL, 1, 17,
+ "Network's Indication of Alerting"),
+ EF_TRANSP_N(0x6F52, SFI_NONE, "EF.KcGPRS", F_OPTIONAL, 9, 9,
+ "GPRS Ciphering key KcGPRS"),
+ EF_TRANSP_N(0x6F53, SFI_NONE, "EF.LOCIGPRS", F_OPTIONAL, 14, 14,
+ "GPRS location information"),
+ EF_TRANSP_N(0x6F54, SFI_NONE, "EF.SUME", F_OPTIONAL, 1, 64,
+ "SetUpMenu Elements"),
+ EF_TRANSP_N(0x6F60, SFI_NONE, "EF.PLMNwAcT", F_OPTIONAL, 40, 80,
+ "User controlled PLMN Selector with Access Technology"),
+ EF_TRANSP_N(0x6F61, SFI_NONE, "EF.OPLMNwAcT", F_OPTIONAL, 40, 80,
+ "Operator controlled PLMN Selector with Access Technology"),
+ EF_TRANSP_N(0x6F62, SFI_NONE, "EF.HPLMNwAcT", F_OPTIONAL, 5, 20,
+ "HPLMN Selector with Access Technology"),
+ EF_TRANSP_N(0x6F63, SFI_NONE, "EF.CPBCCH", F_OPTIONAL, 2, 20,
+ "CPBCCH Information"),
+ EF_TRANSP_N(0x6F64, SFI_NONE, "EF.InvScan", F_OPTIONAL, 1, 1,
+ "Investigation Scan"),
+ EF_LIN_FIX_N(0x6FC5, SFI_NONE, "EF.PNN", F_OPTIONAL, 3, 20,
+ "PLMN Network Name"),
+ EF_LIN_FIX_N(0x6FC6, SFI_NONE, "EF.OPL", F_OPTIONAL, 8, 8,
+ "PLMN Operator PLMN List"),
+ EF_LIN_FIX_N(0x6FC7, SFI_NONE, "EF.MBDN", F_OPTIONAL, 14, 30,
+ "Mailbox Dialling Number"),
+ EF_LIN_FIX_N(0x6FC9, SFI_NONE, "EF.MBI", F_OPTIONAL, 4, 4,
+ "Maibox Identifier"),
+ EF_LIN_FIX_N(0x6FCA, SFI_NONE, "EF.MWIS", F_OPTIONAL, 5, 5,
+ "Message Waiting Indication Status"),
+ EF_LIN_FIX_N(0x6FCB, SFI_NONE, "EF.CFIS", F_OPTIONAL, 16, 16,
+ "Call Forwarding Indication Status"),
+ EF_LIN_FIX_N(0x6FC8, SFI_NONE, "EF.EXT6", F_OPTIONAL, 13, 13,
+ "Extension6 (MBDN)"),
+ EF_LIN_FIX_N(0x6FCC, SFI_NONE, "EF.EXT7", F_OPTIONAL, 13, 13,
+ "Extension7 (CFIS)"),
+ EF_TRANSP_N(0x6FCD, SFI_NONE, "EF.SPDI", F_OPTIONAL, 1, 32,
+ "Extension7 (CFIS)"),
+ EF_LIN_FIX_N(0x6FCE, SFI_NONE, "EF.MMSN", F_OPTIONAL, 4, 32,
+ "MMS Notification"),
+ EF_LIN_FIX_N(0x6FCF, SFI_NONE, "EF.EXT8", F_OPTIONAL, 2, 18,
+ "Extension8 (MMSN)"),
+ EF_TRANSP_N(0x6FD0, SFI_NONE, "EF.MMSICP", F_OPTIONAL, 1, 64,
+ "MMS Issuer Connectivity Parameters"),
+ EF_LIN_FIX_N(0x6FD1, SFI_NONE, "EF.MMSUP", F_OPTIONAL, 1, 64,
+ "MMS User Preferences"),
+ EF_TRANSP_N(0x6FD2, SFI_NONE, "EF.MMSUCP", F_OPTIONAL, 1, 64,
+ "MMS User Connectivity Parameters"),
+};
+
+/* 10.4.1 Contents of the files at the SoLSA level */
+static const struct osim_file_desc sim_ef_in_solsa[] = {
+ EF_TRANSP_N(0x4F30, SFI_NONE, "EF.SAI", F_OPTIONAL, 1, 32,
+ "SoLSA Access Indicator"),
+ EF_LIN_FIX_N(0x4F31, SFI_NONE, "EF.SLL", F_OPTIONAL, 1, 32,
+ "SoLSA LSA List"),
+ /* LSA Descriptor files */
+};
+
+/* 10.4.2 Contents of files at the MExE level */
+static const struct osim_file_desc sim_ef_in_mexe[] = {
+ EF_TRANSP_N(0x4F40, SFI_NONE, "EF.MExE-ST", F_OPTIONAL, 1, 8,
+ "MExE Service table"),
+ EF_LIN_FIX_N(0x4F41, SFI_NONE, "EF.ORPK", F_OPTIONAL, 11, 32,
+ "Operator Root Public Key"),
+ EF_LIN_FIX_N(0x4F42, SFI_NONE, "EF.ARPK", F_OPTIONAL, 11, 32,
+ "Administrator Root Public Key"),
+ EF_LIN_FIX_N(0x4F43, SFI_NONE, "EF.TRPK", F_OPTIONAL, 11, 32,
+ "Third Party Root Public Key"),
+};
+
+/* 10.5 Contents of files at the telecom level */
+static const struct osim_file_desc sim_ef_in_telecom[] = {
+ EF_LIN_FIX_N(0x6F3A, SFI_NONE, "EF.ADN", F_OPTIONAL, 14, 30,
+ "Abbreviated dialling numbers"),
+ EF_LIN_FIX_N(0x6F3B, SFI_NONE, "EF.FDN", F_OPTIONAL, 14, 30,
+ "Fixed dialling numbers"),
+ EF_LIN_FIX_N(0x6F3C, SFI_NONE, "EF.SMS", F_OPTIONAL, 176, 176,
+ "Short messages"),
+ EF_LIN_FIX_N(0x6F3D, SFI_NONE, "EF.CCP", F_OPTIONAL, 14, 14,
+ "Capability configuration parameters"),
+ EF_LIN_FIX_N(0x6F4F, SFI_NONE, "EF.ECCP", F_OPTIONAL, 15, 15,
+ "Extended Capability configuration parameters"),
+ EF_LIN_FIX_N(0x6F40, SFI_NONE, "EF.MSISDN", F_OPTIONAL, 14, 30,
+ "MSISDN"),
+ EF_LIN_FIX_N(0x6F42, SFI_NONE, "EF.SMSP", F_OPTIONAL, 28, 44,
+ "Short message service parameters"),
+ EF_TRANSP_N(0x6F43, SFI_NONE, "EF.SMSS", F_OPTIONAL, 2, 3,
+ "SMS Status"),
+ EF_CYCLIC_N(0x6F44, SFI_NONE, "EF.LND", F_OPTIONAL, 14, 30,
+ "Last number dialled"),
+ EF_LIN_FIX_N(0x6F49, SFI_NONE, "EF.SDN", F_OPTIONAL, 14, 30,
+ "Service Dialling Numbers"),
+ EF_LIN_FIX_N(0x6F4A, SFI_NONE, "EF.EXT1", F_OPTIONAL, 13, 13,
+ "Extension 1 (ADN/SSC, MSISDN, LND)"),
+ EF_LIN_FIX_N(0x6F4B, SFI_NONE, "EF.EXT2", F_OPTIONAL, 13, 13,
+ "Extension 2 (FDN/SSC)"),
+ EF_LIN_FIX_N(0x6F4C, SFI_NONE, "EF.EXT3", F_OPTIONAL, 13, 13,
+ "Extension 3 (SDN)"),
+ EF_LIN_FIX_N(0x6F4D, SFI_NONE, "EF.BDN", F_OPTIONAL, 15, 31,
+ "Barred dialling numbers"),
+ EF_LIN_FIX_N(0x6F4E, SFI_NONE, "EF.EXT4", F_OPTIONAL, 13, 13,
+ "Extension 4 (BDN/SSC)"),
+ EF_LIN_FIX_N(0x6F47, SFI_NONE, "EF.SMSR", F_OPTIONAL, 30, 30,
+ "Short message status reports"),
+ EF_LIN_FIX_N(0x6F58, SFI_NONE, "EF.CMI", F_OPTIONAL, 1, 17,
+ "Comparison Method Information"),
+};
+
+/* 10.6.1 Contents of files at the telecom graphics level */
+const struct osim_file_desc sim_ef_in_graphics[] = {
+ EF_LIN_FIX_N(0x4F20, SFI_NONE, "EF.IMG", F_OPTIONAL, 11, 38,
+ "Image"),
+};
+
+int osim_int_cprof_add_gsm(struct osim_file_desc *mf)
+{
+ struct osim_file_desc *gsm;
+
+ gsm = add_df_with_ef(mf, 0x7F20, "DF.GSM", sim_ef_in_gsm,
+ ARRAY_SIZE(sim_ef_in_gsm));
+ /* Chapter 10.2: DFs at the GSM Application Level */
+ add_df_with_ef(gsm, 0x5F30, "DF.IRIDIUM", NULL, 0);
+ add_df_with_ef(gsm, 0x5F31, "DF.GLOBALSTAR", NULL, 0);
+ add_df_with_ef(gsm, 0x5F32, "DF.ICO", NULL, 0);
+ add_df_with_ef(gsm, 0x5F33, "DF.ACeS", NULL, 0);
+ add_df_with_ef(gsm, 0x5F3C, "DF.MExE", sim_ef_in_mexe,
+ ARRAY_SIZE(sim_ef_in_mexe));
+ add_df_with_ef(gsm, 0x5F40, "DF.EIA/TIA-533", NULL, 0);
+ add_df_with_ef(gsm, 0x5F60, "DF.CTS", NULL, 0);
+ add_df_with_ef(gsm, 0x5F70, "DF.SoLSA", sim_ef_in_solsa,
+ ARRAY_SIZE(sim_ef_in_solsa));
+
+ return 0;
+}
+
+int osim_int_cprof_add_telecom(struct osim_file_desc *mf)
+{
+ struct osim_file_desc *tc;
+
+ tc = add_df_with_ef(mf, 0x7F10, "DF.TELECOM", sim_ef_in_telecom,
+ ARRAY_SIZE(sim_ef_in_telecom));
+ add_df_with_ef(tc, 0x5F50, "DF.GRAPHICS", sim_ef_in_graphics,
+ ARRAY_SIZE(sim_ef_in_graphics));
+ add_df_with_ef(mf, 0x7F22, "DF.IS-41", NULL, 0);
+ add_df_with_ef(mf, 0x7F23, "DF.FP-CTS", NULL, 0); /* TS 11.19 */
+
+ return 0;
+}
+
+struct osim_card_profile *osim_cprof_sim(void *ctx)
+{
+ struct osim_card_profile *cprof;
+ struct osim_file_desc *mf;
+ int rc;
+
+ cprof = talloc_zero(ctx, struct osim_card_profile);
+ cprof->name = "GSM SIM";
+ cprof->sws = sim_card_sws;
+
+ mf = alloc_df(cprof, 0x3f00, "MF");
+
+ cprof->mf = mf;
+
+ /* According to Figure 8 */
+ add_filedesc(mf, sim_ef_in_mf, ARRAY_SIZE(sim_ef_in_mf));
+
+ rc = osim_int_cprof_add_gsm(mf);
+ rc |= osim_int_cprof_add_telecom(mf);
+ if (rc != 0) {
+ talloc_free(cprof);
+ return NULL;
+ }
+
+ return cprof;
+}
diff --git a/src/sim/card_fs_tetra.c b/src/sim/card_fs_tetra.c
new file mode 100644
index 00000000..625adbd2
--- /dev/null
+++ b/src/sim/card_fs_tetra.c
@@ -0,0 +1,267 @@
+/* TETRA SIM card specific structures/routines */
+/*
+ * (C) 2014 by Harald Welte <laforge@gnumonks.org>
+ *
+ * All Rights Reserved
+ *
+ * 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
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+#include <errno.h>
+#include <string.h>
+
+#include <osmocom/sim/sim.h>
+#include <osmocom/core/talloc.h>
+#include <osmocom/gsm/gsm48.h>
+
+#include "sim_int.h"
+#include "gsm_int.h"
+
+/* EN 300 812 V2.1.1 (2001-12) 9.4 */
+static const struct osim_card_sw tsim_sw[] = {
+ {
+ 0x9000, 0xffff, SW_TYPE_STR, SW_CLS_OK,
+ .u.str = "Normal ending of the command",
+ }, {
+ 0x9f00, 0xff00, SW_TYPE_STR, SW_CLS_OK,
+ .u.str = "Normal ending of the command - response data available",
+ }, {
+ 0x9300, 0xffff, SW_TYPE_STR, SW_CLS_POSTP,
+ .u.str = "SIM Application Toolkit is busy, command cannot be executed at present",
+ }, {
+ 0x9200, 0xfff0, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "Memory management - Command successful but after using an internal updat retry X times",
+ }, {
+ 0x9240, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Memory management - Memory problem",
+ }, {
+ 0x9400, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Referencing management - no EF selected",
+ }, {
+ 0x9402, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Referencing management - out of range (invalid address)",
+ }, {
+ 0x9404, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Referencing management - file ID not found / pattern not found",
+ }, {
+ 0x9408, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Referencing management - file is inconsistent with the command",
+ }, {
+ 0x9802, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - no CHV initialized",
+ }, {
+ 0x9804, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - access condition not fulfilled",
+ }, {
+ 0x9808, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - in contradiction with CHV status",
+ }, {
+ 0x9810, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - in contradiction with invalidation status",
+ }, {
+ 0x9840, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - unsuccessful CHV verification, no attempt left",
+ }, {
+ 0x9860, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - manipulation flag set",
+ }, {
+ 0x9870, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - SwMI authentication unsuccessful",
+ }, {
+ 0x6700, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - incorrect parameter P3",
+ }, {
+ 0x6b00, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - incorrect parameter P1 or P2",
+ }, {
+ 0x6d00, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - unknown instruction code",
+ }, {
+ 0x6e00, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - wrong instruction class",
+ }, {
+ 0x6f00, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application independent - technical problem with no diagnostic given",
+ },
+ OSIM_CARD_SW_LAST
+};
+
+static const struct osim_card_sw *tsim_card_sws[] = {
+ tsim_sw,
+ NULL
+};
+
+/* Chapter 10.2.x */
+static const struct osim_file_desc sim_ef_in_mf[] = {
+ EF_TRANSP_N(0x2FE2, SFI_NONE, "EF.ICCID", 0, 10, 10,
+ "ICC Identification"),
+ EF_TRANSP_N(0x2F00, SFI_NONE, "EF.DIR", F_OPTIONAL, 8, 54,
+ "Application directory"),
+ EF_TRANSP_N(0x2F05, SFI_NONE, "EF.LP", 0, 2, 32,
+ "Language preference"),
+};
+
+////////////////////////////
+
+/* Chapter 10.3.x */
+static const struct osim_file_desc sim_ef_in_tetra[] = {
+ EF_TRANSP_N(0x6F01, SFI_NONE, "EF.SST", 0, 4, 8,
+ "SIM service table"),
+ EF_TRANSP_N(0x6F02, SFI_NONE, "EF.ITSI", 0, 6, 6,
+ "ITSI"),
+ EF_TRANSP_N(0x6F03, SFI_NONE, "EF.ITSIDIS", 0, 1, 1,
+ "ITSI Disable"),
+ EF_TRANSP_N(0x6F05, SFI_NONE, "EF.SCT", 0, 4, 4,
+ "Subscriber Class Table"),
+ EF_TRANSP_N(0x6F06, SFI_NONE, "EF.PHASE", 0, 1, 1,
+ "Phase identification"),
+ EF_KEY_N(0x6F07, SFI_NONE, "EF.CCK", F_OPTIONAL, 12, 12,
+ "Common Cipher Key"),
+ EF_TRANSP_N(0x6F08, SFI_NONE, "EF.CCKLOC", F_OPTIONAL, 31, 31,
+ "CCK location areas"),
+ EF_KEY_N(0x6F09, SFI_NONE, "EF.SCK", F_OPTIONAL, 12, 12,
+ "Static Cipher Keys"),
+ /* X+4 for each record, suggested 1 to 10 */
+ EF_LIN_FIX_N(0x6F0A, SFI_NONE, "EF.GSSIS", 0, 5, 21,
+ "Static Cipher Keys"),
+ /* 2 for each record, one for each recod in GSSIS */
+ EF_LIN_FIX_N(0x6F0B, SFI_NONE, "EF.GRDS", 0, 2, 2,
+ "Group related data for static GSSIS"),
+ EF_LIN_FIX_N(0x6F0C, SFI_NONE, "EF.GSSID", 0, 5, 21,
+ "Dynamic GSSIs"),
+ /* 2 for each record, one for each recod in GSSID */
+ EF_LIN_FIX_N(0x6F0D, SFI_NONE, "EF.GRDD", 0, 2, 2,
+ "Dynamic GSSIs"),
+ EF_LIN_FIX_N(0x6F0E, SFI_NONE, "EF.GCK", F_OPTIONAL, 12, 12,
+ "Group Cipher Keys"),
+ EF_KEY_N(0x6F0F, SFI_NONE, "EF.MGCK", F_OPTIONAL, 12, 12,
+ "Modified Group Cipher Keys"),
+ EF_TRANSP_N(0x6F10, SFI_NONE, "EF.GINFO", 0, 9, 9,
+ "User's group information"),
+ EF_TRANSP_N(0x6F11, SFI_NONE, "EF.SEC", 0, 1, 1,
+ "Security settings"),
+ EF_CYCLIC_N(0x6F12, SFI_NONE, "EF.FORBID", 0, 3, 3,
+ "Security settings"),
+ EF_CYCLIC_N(0x6F13, SFI_NONE, "EF.PREF", F_OPTIONAL, 3, 3,
+ "Preferred networks"),
+ EF_TRANSP_N(0x6F14, SFI_NONE, "EF.SPN", F_OPTIONAL, 17, 17,
+ "Service Provider Name"),
+ EF_TRANSP_N(0x6F15, SFI_NONE, "EF.LOCI", F_OPTIONAL, 7, 7,
+ "Location Information"),
+ EF_TRANSP_N(0x6F16, SFI_NONE, "EF.DNWRK", 0, 3, 3,
+ "Broadcast network information"),
+ EF_LIN_FIX_N(0x6F17, SFI_NONE, "EF.NWT", 0, 5, 5,
+ "Network table"),
+ EF_LIN_FIX_N(0x6F18, SFI_NONE, "EF.GWT", F_OPTIONAL, 13, 13,
+ "Gateway table"),
+ EF_LIN_FIX_N(0x6F19, SFI_NONE, "EF.CMT", F_OPTIONAL, 5, 20,
+ "Call Modifier Table"),
+ EF_LIN_FIX_N(0x6F1A, SFI_NONE, "EF.ADNGWT", F_OPTIONAL, 13, 28,
+ "Abbreviated Dialling Number with Gateways"),
+ EF_LIN_FIX_N(0x6F1C, SFI_NONE, "EF.ADNTETRA", F_OPTIONAL, 9, 23,
+ "Abbreviated dialling numbers for TETRA network"),
+ EF_LIN_FIX_N(0x6F1D, SFI_NONE, "EF.EXTA", F_OPTIONAL, 20, 20,
+ "Extension A"),
+ EF_LIN_FIX_N(0x6F1E, SFI_NONE, "EF.FDNGWT", F_OPTIONAL, 13, 28,
+ "Fixed dialling numbers with Gateways"),
+ EF_LIN_FIX_N(0x6F1F, SFI_NONE, "EF.GWTEXT2", F_OPTIONAL, 13, 13,
+ "Gateway Extension2"),
+ EF_LIN_FIX_N(0x6F20, SFI_NONE, "EF.FDNTETRA", F_OPTIONAL, 9, 25,
+ "Fixed dialling numbers for TETRA network"),
+ EF_LIN_FIX_N(0x6F21, SFI_NONE, "EF.EXTB", F_OPTIONAL, 20, 20,
+ "Extension B"),
+ EF_LIN_FIX_N(0x6F22, SFI_NONE, "EF.LNDGWT", F_OPTIONAL, 13, 28,
+ "Last number dialled with Gateways"),
+ EF_CYCLIC_N(0x6F23, SFI_NONE, "EF.LNDTETRA", F_OPTIONAL, 9, 23,
+ "Last numbers dialled for TETRA network"),
+ EF_LIN_FIX_N(0x6F24, SFI_NONE, "EF.SDNGWT", F_OPTIONAL, 13, 28,
+ "Service Dialling Numbers with gateway"),
+ EF_LIN_FIX_N(0x6F25, SFI_NONE, "EF.GWTEXT3", F_OPTIONAL, 13, 13,
+ "Gateway Extension3"),
+ EF_LIN_FIX_N(0x6F26, SFI_NONE, "EF.SDNTETRA", F_OPTIONAL, 8, 22,
+ "Service Dialling Nubers for TETRA network"),
+ EF_LIN_FIX_N(0x6F27, SFI_NONE, "EF.STXT", F_OPTIONAL, 5, 18,
+ "Status message texts"),
+ EF_LIN_FIX_N(0x6F28, SFI_NONE, "EF.MSGTXT", F_OPTIONAL, 5, 18,
+ "SDS-1 message texts"),
+ EF_LIN_FIX_N(0x6F29, SFI_NONE, "EF.SDS123", F_OPTIONAL, 46, 46,
+ "Status and SDS type 1, 2 and 3 message storage"),
+ EF_LIN_FIX_N(0x6F2A, SFI_NONE, "EF.SDS4", F_OPTIONAL, 255, 255,
+ "Status and SDS type 4 message storage"),
+ EF_LIN_FIX_N(0x6F2B, SFI_NONE, "EF.MSGEXT", F_OPTIONAL, 16, 16,
+ "Message Extension"),
+ EF_LIN_FIX_N(0x6F2C, SFI_NONE, "EF.EADDR", 0, 17, 17,
+ "Emergency adresses"),
+ EF_TRANSP_N(0x6F2D, SFI_NONE, "EF.EINFO", 0, 2, 2,
+ "Emergency call information"),
+ EF_LIN_FIX_N(0x6F2E, SFI_NONE, "EF.DMOCh", F_OPTIONAL, 4, 4,
+ "DMO channel information"),
+ EF_TRANSP_N(0x6F2F, SFI_NONE, "EF.MSCh", F_OPTIONAL, 1, 16,
+ "MS allocation of DMO channels"),
+ EF_TRANSP_N(0x6F30, SFI_NONE, "EF.KH", F_OPTIONAL, 6, 6,
+ "List of Key Holders"),
+ EF_LIN_FIX_N(0x6F31, SFI_NONE, "EF.REPGATE", F_OPTIONAL, 2, 2,
+ "DMO repeater and gateway list"),
+ EF_TRANSP_N(0x6F32, SFI_NONE, "EF.AD", F_OPTIONAL, 2, 2,
+ "Administrative Data"),
+ EF_TRANSP_N(0x6F33, SFI_NONE, "EF.PREF_LA", F_OPTIONAL, 2, 2,
+ "Preferred location areas"),
+ EF_CYCLIC_N(0x6F34, SFI_NONE, "EF.LNDComp", F_OPTIONAL, 3, 3,
+ "Composite LND file"),
+ EF_TRANSP_N(0x6F35, SFI_NONE, "EF.DFLTSTSTGGT", F_OPTIONAL, 16, 16,
+ "Status Default Target"),
+ EF_TRANSP_N(0x6F36, SFI_NONE, "EF.SDSMEM_STATUS", F_OPTIONAL, 7, 7,
+ "SDS Memory Status"),
+ EF_TRANSP_N(0x6F37, SFI_NONE, "EF.WELCOME", F_OPTIONAL, 32, 32,
+ "Welcome Message"),
+ EF_LIN_FIX_N(0x6F38, SFI_NONE, "EF.SDSR", F_OPTIONAL, 2, 2,
+ "SDS delivery report"),
+ EF_LIN_FIX_N(0x6F39, SFI_NONE, "EF.SDSP", F_OPTIONAL, 20, 35,
+ "SDS parameters"),
+ EF_TRANSP_N(0x6F46, SFI_NONE, "EF.DIALSC", 0, 5, 5,
+ "Dialling schemes for TETRA network"),
+ EF_TRANSP_N(0x6F3E, SFI_NONE, "EF.APN", F_OPTIONAL, 65, 65,
+ "APN table"),
+ EF_LIN_FIX_N(0x6FC0, SFI_NONE, "EF.PNI", F_OPTIONAL, 14, 14,
+ "Private Number Information"),
+};
+
+struct osim_card_profile *osim_cprof_tsim(void *ctx)
+{
+ struct osim_card_profile *cprof;
+ struct osim_file_desc *mf;
+ int rc;
+
+ cprof = talloc_zero(ctx, struct osim_card_profile);
+ cprof->name = "TETRA SIM";
+ cprof->sws = tsim_card_sws;
+
+ mf = alloc_df(cprof, 0x3f00, "MF");
+
+ cprof->mf = mf;
+
+ add_filedesc(mf, sim_ef_in_mf, ARRAY_SIZE(sim_ef_in_mf));
+ add_df_with_ef(mf, 0x7F20, "DF.TETRA", sim_ef_in_tetra,
+ ARRAY_SIZE(sim_ef_in_tetra));
+
+ rc = osim_int_cprof_add_telecom(mf);
+ if (rc != 0) {
+ talloc_free(cprof);
+ return NULL;
+ }
+
+ return cprof;
+}
diff --git a/src/sim/card_fs_uicc.c b/src/sim/card_fs_uicc.c
new file mode 100644
index 00000000..70737b6f
--- /dev/null
+++ b/src/sim/card_fs_uicc.c
@@ -0,0 +1,208 @@
+/* ETSI UICC specific structures / routines */
+/*
+ * (C) 2012 by Harald Welte <laforge@gnumonks.org>
+ *
+ * All Rights Reserved
+ *
+ * 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
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+
+#include <osmocom/sim/sim.h>
+#include <osmocom/gsm/tlv.h>
+
+/* TS 102 221 V10.0.0 / 10.2.1 */
+const struct osim_card_sw ts102221_uicc_sw[] = {
+ {
+ 0x9000, 0xffff, SW_TYPE_STR, SW_CLS_OK,
+ .u.str = "Normal ending of the command",
+ }, {
+ 0x9100, 0xff00, SW_TYPE_STR, SW_CLS_OK,
+ .u.str = "Normal ending of the command, extra info proactive",
+ }, {
+ 0x9200, 0xff00, SW_TYPE_STR, SW_CLS_OK,
+ .u.str = "Normal ending of the command, extra info regarding transfer session",
+ }, {
+ 0x9300, 0xff00, SW_TYPE_STR, SW_CLS_POSTP,
+ .u.str = "SIM Application Toolkit is busy, command cannot be executed at present",
+ }, {
+ 0x6200, 0xffff, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "No information given, state of non volatile memory unchanged",
+ }, {
+ 0x6281, 0xffff, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "Part of returned data may be corrupted",
+ }, {
+ 0x6282, 0xffff, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "End of file/record reached before reading Le bytes",
+ }, {
+ 0x6283, 0xffff, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "Selected file invalidated",
+ }, {
+ 0x6285, 0xffff, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "Selected file in termination state",
+ }, {
+ 0x62f1, 0xffff, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "More data available",
+ }, {
+ 0x62f2, 0xffff, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "More data available and proactive command pending",
+ }, {
+ 0x62f3, 0xffff, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "Response data available",
+ }, {
+ 0x63f1, 0xffff, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "More data expected",
+ }, {
+ 0x63c0, 0xfff0, SW_TYPE_STR, SW_CLS_WARN,
+ .u.str = "Verification falied, X retries remaining",
+ }, {
+ 0x6400, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Execution - No information given, state of non-volatile memory unchanged",
+ }, {
+ 0x6500, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Execution - No information given, state of non-volatile memory changed",
+ }, {
+ 0x6581, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Execution - Memory problem",
+ }, {
+ 0x6700, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Checking - Wrong length",
+ }, {
+ 0x6700, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Checking - Command dependent error",
+ }, {
+ 0x6b00, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Checking - Wrong parameter(s) P1-P2",
+ }, {
+ 0x6d00, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Checking - Instruction code not supported or valid",
+ }, {
+ 0x6e00, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Checking - Class not supported",
+ }, {
+ 0x6f00, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Checking - Technical problem, no precise diagnostics",
+ }, {
+ 0x6f00, 0xff00, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Checking - Command dependent error",
+ }, {
+ 0x6800, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Function in CLA not supported - No information given",
+ }, {
+ 0x6881, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Function in CLA not supported - Logical channel not supported",
+ }, {
+ 0x6882, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Function in CLA not supportied - Secure messaging not supported",
+ }, {
+ 0x6900, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Command not allowed - No information given",
+ }, {
+ 0x6981, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Command not allowed - Command incompatible with file structure",
+ }, {
+ 0x6982, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Command not allowed - Security status not satisfied",
+ }, {
+ 0x6983, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Command not allowed - Authentication/PIN method blocked",
+ }, {
+ 0x6984, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Command not allowed - Referenced data invalidated",
+ }, {
+ 0x6985, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Command not allowed - Conditions of use not satisfied",
+ }, {
+ 0x6986, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Command not allowed - Noe EF selected",
+ }, {
+ 0x6989, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Command not allowed - secure channel - security not satisfied",
+ }, {
+ 0x6a80, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Wrong parameters - Incorrect parameters in the data field",
+ }, {
+ 0x6a81, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Wrong parameters - Function not supported",
+ }, {
+ 0x6a82, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Wrong parameters - File not found",
+ }, {
+ 0x6a83, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Wrong parameters - Record not found",
+ }, {
+ 0x6a84, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Wrong parameters - Not enough memory space",
+ }, {
+ 0x6a86, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Wrong parameters - Incorrect parameters P1 to P2",
+ }, {
+ 0x6a87, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Wrong parameters - Lc inconsistent with P1 ot P2",
+ }, {
+ 0x6a88, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Wrong parameters - Referenced data not found",
+ }, {
+ 0x9850, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application error - INCREASE cannot be performed, max value reached",
+ }, {
+ 0x9862, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application error - Authentication error, application specific",
+ }, {
+ 0x9863, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Application error - Security session or association expired",
+ },
+ OSIM_CARD_SW_LAST
+};
+
+const struct value_string ts102221_fcp_vals[14] = {
+ { UICC_FCP_T_FCP, "File control parameters" },
+ { UICC_FCP_T_FILE_SIZE, "File size" },
+ { UICC_FCP_T_TOT_F_SIZE, "Total size of files" },
+ { UICC_FCP_T_FILE_DESC, "File descriptor" },
+ { UICC_FCP_T_FILE_ID, "File identifier" },
+ { UICC_FCP_T_DF_NAME, "DF name" },
+ { UICC_FCP_T_SFID, "Short file identifier" },
+ { UICC_FCP_T_LIFEC_STS, "Lifecycle status integer" },
+ { UICC_FCP_T_SEC_ATTR_REFEXP, "Security attributes (Referenced/Expanded)" },
+ { UICC_FCP_T_SEC_ATTR_COMP, "Security attributes (Compact)" },
+ { UICC_FCP_T_PROPRIETARY, "Proprietary" },
+ { UICC_FCP_T_SEC_ATTR_EXP, "Security attributes (Expanded)" },
+ { UICC_FCP_T_PIN_STS_DO, "PIN Status DO" },
+ { 0, NULL }
+};
+
+/* FIXME: Ber-TLV ?? */
+const struct tlv_definition ts102221_fcp_tlv_def = {
+ .def = {
+ [UICC_FCP_T_FCP] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_FILE_SIZE] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_TOT_F_SIZE] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_FILE_DESC] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_FILE_ID] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_DF_NAME] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_SFID] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_LIFEC_STS] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_SEC_ATTR_REFEXP] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_SEC_ATTR_COMP] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_PROPRIETARY] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_SEC_ATTR_EXP] = { TLV_TYPE_TLV },
+ [UICC_FCP_T_PIN_STS_DO] = { TLV_TYPE_TLV },
+ },
+};
+
+/* Annex E - TS 101 220 */
+static const uint8_t adf_uicc_aid[] = { 0xA0, 0x00, 0x00, 0x00, 0x87, 0x10, 0x01 };
diff --git a/src/sim/card_fs_usim.c b/src/sim/card_fs_usim.c
new file mode 100644
index 00000000..22c193f8
--- /dev/null
+++ b/src/sim/card_fs_usim.c
@@ -0,0 +1,380 @@
+/* 3GPP USIM specific structures / routines */
+/*
+ * (C) 2012-2014 by Harald Welte <laforge@gnumonks.org>
+ *
+ * All Rights Reserved
+ *
+ * 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
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+
+#include <errno.h>
+#include <string.h>
+
+#include <osmocom/sim/sim.h>
+#include <osmocom/core/talloc.h>
+#include <osmocom/gsm/gsm48.h>
+
+#include "sim_int.h"
+#include "gsm_int.h"
+
+/* TS 31.102 Version 7.7.0 / Chapter 7.3 */
+const struct osim_card_sw ts31_102_sw[] = {
+ {
+ 0x9862, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - Authentication error, incorrect MAC",
+ }, {
+ 0x9864, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - Authentication error, security context not supported",
+ }, {
+ 0x9865, 0xffff, SW_TYPE_STR, SW_CLS_ERROR,
+ .u.str = "Security management - Key freshness error",
+ },
+ OSIM_CARD_SW_LAST
+};
+
+static const struct osim_card_sw *usim_card_sws[] = {
+ ts31_102_sw,
+ ts102221_uicc_sw,
+ NULL
+};
+
+/* TS 102 221 Chapter 13.1 */
+static const struct osim_file_desc uicc_ef_in_mf[] = {
+ EF_LIN_FIX_N(0x2f00, SFI_NONE, "EF.DIR", 0, 1, 32,
+ "Application directory"),
+ EF_TRANSP_N(0x2FE2, SFI_NONE, "EF.ICCID", 0, 10, 10,
+ "ICC Identification"),
+ EF_TRANSP_N(0x2F05, SFI_NONE, "EF.PL", 0, 2, 20,
+ "Preferred Languages"),
+ EF_LIN_FIX_N(0x2F06, SFI_NONE, "EF.ARR", F_OPTIONAL, 1, 256,
+ "Access Rule Reference"),
+};
+
+/* 31.102 Chapter 4.4.3 */
+static const struct osim_file_desc usim_ef_in_df_gsm_access[] = {
+ EF_TRANSP_N(0x4f20, 0x01, "EF.Kc", 0, 9, 9,
+ "Ciphering Key Kc"),
+ EF_TRANSP_N(0x4f52, 0x02, "EF.KcGPRS", F_OPTIONAL, 9, 9,
+ "GPRS Ciphering key KcGPRS"),
+ EF_TRANSP_N(0x4f63, SFI_NONE, "EF.CPBCCH", F_OPTIONAL, 2, 20,
+ "CPBCCH Information"),
+ EF_TRANSP_N(0x4f64, SFI_NONE, "EF.invSCAN", F_OPTIONAL, 1, 1,
+ "Investigation Scan"),
+};
+
+/* 31.102 Chapter 4.2 */
+static const struct osim_file_desc usim_ef_in_adf_usim[] = {
+ EF_TRANSP(0x6F05, 0x02, "EF.LI", 0, 2, 16,
+ "Language Indication", &gsm_lp_decode, NULL),
+ EF_TRANSP(0x6F07, 0x07, "EF.IMSI", 0, 9, 9,
+ "IMSI", &gsm_imsi_decode, NULL),
+ EF_TRANSP_N(0x6F08, 0x08, "EF.Keys", 0, 33, 33,
+ "Ciphering and Integrity Keys"),
+ EF_TRANSP_N(0x6F09, 0x09, "EF.KeysPS", 0, 33, 33,
+ "Ciphering and Integrity Keys for Packet Switched domain"),
+ EF_TRANSP_N(0x6F60, 0x0A, "EF.PLMNwAcT", F_OPTIONAL, 40, 80,
+ "User controlled PLMN Selector with Access Technology"),
+ EF_TRANSP(0x6F31, 0x12, "EF.HPPLMN", 0, 1, 1,
+ "Higher Priority PLMN search period", &gsm_hpplmn_decode, NULL),
+ EF_TRANSP_N(0x6F37, SFI_NONE, "EF.ACMmax", F_OPTIONAL, 3, 3,
+ "ACM maximum value"),
+ EF_TRANSP_N(0x6F38, 0x04, "EF.UST", 0, 1, 16,
+ "USIM Service Table"),
+ EF_CYCLIC_N(0x6F39, SFI_NONE, "EF.ACM", F_OPTIONAL, 3, 3,
+ "Accumulated call meter"),
+ EF_TRANSP_N(0x6F3E, SFI_NONE, "EF.GID1", F_OPTIONAL, 1, 4,
+ "Group Identifier Level 1"),
+ EF_TRANSP_N(0x6F3F, SFI_NONE, "EF.GID2", F_OPTIONAL, 1, 4,
+ "Group Identifier Level 2"),
+ EF_TRANSP_N(0x6F46, SFI_NONE, "EF.SPN", F_OPTIONAL, 17, 17,
+ "Service Provider Name"),
+ EF_TRANSP_N(0x6F41, SFI_NONE, "EF.PUCT", F_OPTIONAL, 5, 5,
+ "Price per unit and currency table"),
+ EF_TRANSP_N(0x6F45, SFI_NONE, "EF.CBMI", F_OPTIONAL, 2, 32,
+ "Cell broadcast massage identifier selection"),
+ EF_TRANSP_N(0x6F78, 0x06, "EF.ACC", 0, 2, 2,
+ "Access control class"),
+ EF_TRANSP_N(0x6F7B, 0x0D, "EF.FPLMN", 0, 12, 36,
+ "Forbidden PLMNs"),
+ EF_TRANSP_N(0x6F7E, 0x0B, "EF.LOCI", 0, 11, 11,
+ "Location information"),
+ EF_TRANSP_N(0x6FAD, 0x03, "EF.AD", 0, 5, 16,
+ "Administrative data"),
+ EF_TRANSP_N(0x6F48, 0x0E, "EF.CBMID", F_OPTIONAL, 2, 32,
+ "Cell Broadcast Message Identifier for Data Download"),
+ EF_TRANSP_N(0x6FB7, 0x01, "EF.ECC", F_OPTIONAL, 5, 32,
+ "Emergency Call Code"),
+ EF_TRANSP_N(0x6F50, SFI_NONE, "EF.CBMIR", F_OPTIONAL, 4, 32,
+ "Cell broadcast message identifier range selection"),
+ EF_TRANSP_N(0x6F73, 0x0C, "EF.PSLOCI", 0, 14, 14,
+ "Pacet Switched location information"),
+ EF_LIN_FIX_N(0x6F3B, SFI_NONE, "EF.FDN", F_OPTIONAL, 14, 32,
+ "Fixed dialling numbers"),
+ EF_LIN_FIX_N(0x6F3C, SFI_NONE, "EF.SMS", F_OPTIONAL, 176, 176,
+ "Short messages"),
+ EF_LIN_FIX_N(0x6F40, SFI_NONE, "EF.MSISDN", F_OPTIONAL, 14, 32,
+ "MSISDN"),
+ EF_LIN_FIX_N(0x6F42, SFI_NONE, "EF.SMSP", F_OPTIONAL, 28, 64,
+ "Short message service parameters"),
+ EF_TRANSP_N(0x6F43, SFI_NONE, "EF.SMSS", F_OPTIONAL, 2, 8,
+ "SMS Status"),
+ EF_LIN_FIX_N(0x6F49, SFI_NONE, "EF.SDN", F_OPTIONAL, 14, 32,
+ "Service Dialling Numbers"),
+ EF_LIN_FIX_N(0x6F4B, SFI_NONE, "EF.EXT2", F_OPTIONAL, 13, 13,
+ "Extension 2"),
+ EF_LIN_FIX_N(0x6F4C, SFI_NONE, "EF.EXT3", F_OPTIONAL, 13, 13,
+ "Extension 3"),
+ EF_LIN_FIX_N(0x6F47, SFI_NONE, "EF.SMSR", F_OPTIONAL, 30, 30,
+ "Short message status reports"),
+ EF_CYCLIC_N(0x6F80, 0x14, "EF.ICI", F_OPTIONAL, 28, 64,
+ "Incoming Calling Information"),
+ EF_CYCLIC_N(0x6F81, 0x15, "EF.OCI", F_OPTIONAL, 27, 64,
+ "Outgoing Calling Information"),
+ EF_CYCLIC_N(0x6F82, SFI_NONE, "EF.ICT", F_OPTIONAL, 3, 3,
+ "Incoming Call Timer"),
+ EF_CYCLIC_N(0x6F83, SFI_NONE, "EF.OCT", F_OPTIONAL, 3, 3,
+ "Outgoing Call Timer"),
+ EF_LIN_FIX_N(0x6F4E, SFI_NONE, "EF.EXT5", F_OPTIONAL, 13, 13,
+ "Extension 5"),
+ EF_LIN_FIX_N(0x6F4F, 0x16, "EF.CCP2", F_OPTIONAL, 15, 32,
+ "Capability Configuration Parameters 2"),
+ EF_TRANSP_N(0x6FB5, SFI_NONE, "EF.eMLPP", F_OPTIONAL, 2, 2,
+ "enhanced Multi Level Precedence and Pre-emption"),
+ EF_TRANSP_N(0x6FB6, SFI_NONE, "EF.AAeM", F_OPTIONAL, 1, 1,
+ "Automatic Answer for eMLPP Service"),
+ EF_TRANSP_N(0x6FC3, SFI_NONE, "EF.Hiddenkey", F_OPTIONAL, 4, 4,
+ "Key for hidden phone book entries"),
+ EF_LIN_FIX_N(0x6F4D, SFI_NONE, "EF.BDN", F_OPTIONAL, 15, 32,
+ "Barred Dialling Numbers"),
+ EF_LIN_FIX_N(0x6F4E, SFI_NONE, "EF.EXT4", F_OPTIONAL, 13, 13,
+ "Extension 4"),
+ EF_LIN_FIX_N(0x6F58, SFI_NONE, "EF.CMI", F_OPTIONAL, 2, 16,
+ "Comparison Method Information"),
+ EF_TRANSP_N(0x6F56, 0x05, "EF.EST", F_OPTIONAL, 1, 8,
+ "Enhanced Services Table"),
+ EF_TRANSP_N(0x6F57, SFI_NONE, "EF.ACL", F_OPTIONAL, 2, 256,
+ "Access Point Name Control List"),
+ EF_TRANSP_N(0x6F2C, SFI_NONE, "EF.DCK", F_OPTIONAL, 16, 16,
+ "Depersonalisation Control Keys"),
+ EF_TRANSP_N(0x6F32, SFI_NONE, "EF.CNL", F_OPTIONAL, 6, 46,
+ "Co-operative Network List"),
+ EF_TRANSP_N(0x6F5B, 0x0F, "EF.START-HFN", 0, 6, 6,
+ "Initialisation values for Hyperframe number"),
+ EF_TRANSP_N(0x6F5C, 0x10, "EF.THRESHOLD", 0, 3, 3,
+ "Maximum value of START"),
+ EF_TRANSP_N(0x6F61, 0x11, "EF.OPLMNwAcT", F_OPTIONAL, 40, 80,
+ "Operator controlled PLMN Selector with Access Technology"),
+ EF_TRANSP_N(0x6F62, 0x13, "EF.HPLMNwAcT", F_OPTIONAL, 5, 30,
+ "HPLMN Selector with Access Technology"),
+ EF_LIN_FIX_N(0x6F06, 0x17, "EF.ARR", 0, 1, 256,
+ "Access Rule Reference"),
+ EF_TRANSP_N(0x6FC4, SFI_NONE, "EF.NETPAR", 0, 46, 256,
+ "Network Parameters"),
+ EF_LIN_FIX_N(0x6FC5, 0x19, "EF.PNN", F_OPTIONAL, 3, 128,
+ "PLMN Network Name"),
+ EF_LIN_FIX_N(0x6FC6, 0x1A, "EF.OPL", F_OPTIONAL, 8, 32,
+ "Operator PLMN List"),
+ EF_LIN_FIX_N(0x6FC7, SFI_NONE, "EF.MBDN", F_OPTIONAL, 14, 32,
+ "Mailbox Dialling Numbers"),
+ EF_LIN_FIX_N(0x6FC8, SFI_NONE, "EF.EXT6", F_OPTIONAL, 13, 13,
+ "Extension 6"),
+ EF_LIN_FIX_N(0x6FC9, SFI_NONE, "EF.MBI", F_OPTIONAL, 4, 5,
+ "Mailbox Identifier"),
+ EF_LIN_FIX_N(0x6FCA, SFI_NONE, "EF.MWIS", F_OPTIONAL, 5, 6,
+ "Message Waiting Indication Status"),
+ EF_LIN_FIX_N(0x6FCB, SFI_NONE, "EF.CFIS", F_OPTIONAL, 16, 16,
+ "Call Forwarding Indication Status"),
+ EF_LIN_FIX_N(0x6FCC, SFI_NONE, "EF.EXT7", F_OPTIONAL, 13, 13,
+ "Extension 7"),
+ EF_TRANSP_N(0x6FCD, 0x1B, "EF.SPDI", F_OPTIONAL, 1, 64,
+ "Service Provider Display Information"),
+ EF_LIN_FIX_N(0x6FCE, SFI_NONE, "EF.MMSN", F_OPTIONAL, 4, 32,
+ "MMS Notification"),
+ EF_LIN_FIX_N(0x6FCF, SFI_NONE, "EF.EXT8", F_OPTIONAL, 3, 16,
+ "Extension 8"),
+ EF_TRANSP_N(0x6FD0, SFI_NONE, "EF.MMSICP", F_OPTIONAL, 3, 256,
+ "MMS Issuer Connectivity Parameters"),
+ EF_LIN_FIX_N(0x6FD1, SFI_NONE, "EF.MMSUP", F_OPTIONAL, 1, 64,
+ "MMS User Preferences"),
+ EF_TRANSP_N(0x6FD2, SFI_NONE, "EF.MMSUCP", F_OPTIONAL, 1, 64,
+ "MMS User Connectivity Parameters"),
+ EF_LIN_FIX_N(0x6FD3, SFI_NONE, "EF.NIA", F_OPTIONAL, 2, 64,
+ "Network's Indication of Alerting"),
+ EF_TRANSP_N(0x6FB1, SFI_NONE, "EF.VGCS", F_OPTIONAL, 4, 80,
+ "Voice Group Call Service"),
+ EF_TRANSP_N(0x6FB2, SFI_NONE, "EF.VGCSS", F_OPTIONAL, 7, 7,
+ "Voice Group Call Service Status"),
+ EF_TRANSP_N(0x6FB3, SFI_NONE, "EF.VBS", F_OPTIONAL, 4, 80,
+ "Voice Broadcast Service"),
+ EF_TRANSP_N(0x6FB4, SFI_NONE, "EF.VBSS", F_OPTIONAL, 7, 7,
+ "Voice Broadcast Service Status"),
+ EF_TRANSP_N(0x6FD4, SFI_NONE, "EF.VGCSCA", F_OPTIONAL, 2, 40,
+ "Voice Group Call Service Ciphering Algorithm"),
+ EF_TRANSP_N(0x6FD5, SFI_NONE, "EF.VBSCA", F_OPTIONAL, 2, 40,
+ "Voice Broadcast Service Ciphering Algorithm"),
+ EF_TRANSP_N(0x6FD6, SFI_NONE, "EF.GBABP", F_OPTIONAL, 4, 128,
+ "GBA Bootstrapping parameters"),
+ EF_LIN_FIX_N(0x6FD7, SFI_NONE, "EF.MSK", F_OPTIONAL, 20, 84,
+ "MBMS Serviec Key List"),
+ EF_LIN_FIX_N(0x6FD8, SFI_NONE, "EF.MUK", F_OPTIONAL, 1, 128,
+ "MBMS User Key"),
+ EF_LIN_FIX_N(0x6FDA, SFI_NONE, "EF.GBANL", F_OPTIONAL, 1, 128,
+ "GBA NAF List"),
+ EF_TRANSP_N(0x6FD9, 0x1D, "EF.EHPLMN", F_OPTIONAL, 3, 30,
+ "Equivalent HPLMN"),
+ EF_TRANSP_N(0x6FDB, SFI_NONE, "EF.EHPLMNPI", F_OPTIONAL, 1, 1,
+ "Equivalent HPLMN Presentation Indication"),
+ EF_TRANSP_N(0x6FDC, SFI_NONE, "EF.LRPLMNSI", F_OPTIONAL, 1, 1,
+ "Last RPLMN Selection Indication"),
+ EF_LIN_FIX_N(0x6FDD, SFI_NONE, "EF.NAFKCA", F_OPTIONAL, 1, 128,
+ "NAF Key Centre Address"),
+ EF_TRANSP_N(0x6FDE, SFI_NONE, "EF.SPNI", F_OPTIONAL, 1, 128,
+ "Service Provider Name Icon"),
+ EF_LIN_FIX_N(0x6FDF, SFI_NONE, "EF.PNNI", F_OPTIONAL, 1, 128,
+ "PLMN Network Name Icon"),
+ EF_LIN_FIX_N(0x6FE2, SFI_NONE, "EF.NCP-IP", F_OPTIONAL, 2, 256,
+ "Network Connectivity Parameters for USIM IP Connections"),
+ EF_TRANSP_N(0x6FE3, 0x1E, "EF.EPSLOCI", F_OPTIONAL, 18, 18,
+ "EPS location information"),
+ EF_LIN_FIX_N(0x6FE4, 0x18, "EF.EPSNSC", F_OPTIONAL, 54, 128,
+ "EPS NAS Security Context"),
+ EF_TRANSP_N(0x6FE6, SFI_NONE, "EF.UFC", F_OPTIONAL, 1, 8,
+ "USAT Facility Control"),
+ EF_TRANSP_N(0x6FE8, SFI_NONE, "EF.NASCONFIG", F_OPTIONAL, 1, 128,
+ "Non Access Stratum Configuration"),
+ EF_LIN_FIX_N(0x6FE7, SFI_NONE, "EF.UICCIARI", F_OPTIONAL, 1, 32,
+ "UICC IARI"),
+ EF_TRANSP_N(0x6FEC, SFI_NONE, "EF.PWS", F_OPTIONAL, 1, 32,
+ "Public Warning System"),
+};
+
+
+
+/* 31.102 Chapter 4.4.1 */
+static const struct osim_file_desc usim_ef_in_solsa[] = {
+ EF_TRANSP_N(0x4F30, SFI_NONE, "EF.SAI", F_OPTIONAL, 2, 32,
+ "SoLSA Access Indicator"),
+ EF_LIN_FIX_N(0x4F31, SFI_NONE, "EF.SLL", F_OPTIONAL, 11, 32,
+ "SoLSA LSA List"),
+ /* LSA descriptor files 4Fxx, hard to represent here */
+};
+
+/* 31.102 Chapter 4.4.4 */
+static const struct osim_file_desc usim_ef_in_df_mexe[] = {
+ EF_TRANSP_N(0x4F40, SFI_NONE, "EF.MexE-ST", F_OPTIONAL, 1, 4,
+ "MexE Service table"),
+ EF_LIN_FIX_N(0x4F41, SFI_NONE, "EF.ORPK", F_OPTIONAL, 10, 16,
+ "Operator Root Public Key"),
+ EF_LIN_FIX_N(0x4F42, SFI_NONE, "EF.ARPK", F_OPTIONAL, 10, 16,
+ "Administrator Root Public Key"),
+ EF_LIN_FIX_N(0x4F43, SFI_NONE, "EF.TPRPK", F_OPTIONAL, 10, 16,
+ "Third Party Root Public Key"),
+ /* TKCDF files 4Fxx, hard to represent here */
+};
+
+/* 31.102 Chapter 4.4.5 */
+static const struct osim_file_desc usim_ef_in_df_wlan[] = {
+ EF_TRANSP_N(0x4F41, 0x01, "EF.Pseudo", F_OPTIONAL, 2, 16,
+ "Pseudonym"),
+ EF_TRANSP_N(0x4F42, 0x02, "EF.UPLMNWLAN", F_OPTIONAL, 30, 60,
+ "User controlled PLMN selector for I-WLAN Access"),
+ EF_TRANSP_N(0x4F43, 0x03, "EF.OPLMNWLAN", F_OPTIONAL, 30, 60,
+ "Operator controlled PLMN selector for I-WLAN Access"),
+ EF_LIN_FIX_N(0x4F44, 0x04, "EF.UWSIDL", F_OPTIONAL, 1, 16,
+ "User controlled WLAN Specific Identifier List"),
+ EF_LIN_FIX_N(0x4F45, 0x05, "EF.OWSIDL", F_OPTIONAL, 1, 16,
+ "Operator controlled WLAN Specific Identifier List"),
+ EF_TRANSP_N(0x4F46, 0x06, "EF.WRI", F_OPTIONAL, 1, 64,
+ "WLAN Reauthentication Identity"),
+ EF_LIN_FIX_N(0x4F47, 0x07, "EF.HWSIDL", F_OPTIONAL, 1, 16,
+ "Home I-WLAN Specific Identifier List"),
+ EF_TRANSP_N(0x4F48, 0x08, "EF.WEPLMNPI", F_OPTIONAL, 1, 1,
+ "I-WLAN Equivalent HPLMN Presentation Indication"),
+ EF_TRANSP_N(0x4F49, 0x09, "EF.WHPI", F_OPTIONAL, 1, 1,
+ "I-WLAN HPLMN Priority Indiation"),
+ EF_TRANSP_N(0x4F4A, 0x0a, "EF.WLRPLMN", F_OPTIONAL, 3, 3,
+ "I-WLAN Last Registered PLMN"),
+};
+
+/* 31.102 Chapter 4.4.6 */
+static const struct osim_file_desc usim_ef_in_df_hnb[] = {
+ EF_LIN_FIX_N(0x4F81, 0x01, "EF.ACSGL", F_OPTIONAL, 1, 128,
+ "Allowed CSG Lists"),
+ EF_LIN_FIX_N(0x4F82, 0x02, "EF.CSGT", F_OPTIONAL, 1, 64,
+ "CSG Type"),
+ EF_LIN_FIX_N(0x4F83, 0x03, "EF.HNBN", F_OPTIONAL, 1, 64,
+ "Home NodeB Name"),
+ EF_LIN_FIX_N(0x4F84, 0x04, "EF.OCSGL", F_OPTIONAL, 1, 64,
+ "Operator CSG List"),
+ EF_LIN_FIX_N(0x4F85, 0x05, "EF.OCSGT", F_OPTIONAL, 1, 64,
+ "Operator CSG Type"),
+ EF_LIN_FIX_N(0x4F86, 0x06, "EF.OHNBN", F_OPTIONAL, 1, 64,
+ "Oprator Home NodeB Name"),
+};
+
+/* Annex E - TS 101 220 */
+static const uint8_t adf_usim_aid[] = { 0xA0, 0x00, 0x00, 0x00, 0x87, 0x10, 0x02 };
+
+struct osim_card_profile *osim_cprof_usim(void *ctx)
+{
+ struct osim_card_profile *cprof;
+ struct osim_file_desc *mf, *uadf;
+ int rc;
+
+ cprof = talloc_zero(ctx, struct osim_card_profile);
+ cprof->name = "3GPP USIM";
+ cprof->sws = usim_card_sws;
+
+ mf = alloc_df(cprof, 0x3f00, "MF");
+
+ cprof->mf = mf;
+
+ /* Core UICC Files */
+ add_filedesc(mf, uicc_ef_in_mf, ARRAY_SIZE(uicc_ef_in_mf));
+
+ /* ADF.USIM with its EF siblings */
+ uadf = add_adf_with_ef(mf, adf_usim_aid, sizeof(adf_usim_aid),
+ "ADF.USIM", usim_ef_in_adf_usim,
+ ARRAY_SIZE(usim_ef_in_adf_usim));
+
+ /* DFs under ADF.USIM */
+ add_df_with_ef(uadf, 0x5F3A, "DF.PHONEBOOK", NULL, 0);
+ add_df_with_ef(uadf, 0x5F3B, "DF.GSM-ACCESS", usim_ef_in_df_gsm_access,
+ ARRAY_SIZE(usim_ef_in_df_gsm_access));
+ add_df_with_ef(uadf, 0x5F3C, "DF.MexE", usim_ef_in_df_mexe,
+ ARRAY_SIZE(usim_ef_in_df_mexe));
+ add_df_with_ef(uadf, 0x5F40, "DF.WLAN", usim_ef_in_df_wlan,
+ ARRAY_SIZE(usim_ef_in_df_wlan));
+ /* Home-NodeB (femtocell) */
+ add_df_with_ef(uadf, 0x5F50, "DF.HNB", usim_ef_in_df_hnb,
+ ARRAY_SIZE(usim_ef_in_df_hnb));
+ /* Support of Localised Service Areas */
+ add_df_with_ef(uadf, 0x5F70, "DF.SoLSA", usim_ef_in_solsa,
+ ARRAY_SIZE(usim_ef_in_solsa));
+ /* OMA BCAST Smart Card Profile */
+ add_df_with_ef(uadf, 0x5F80, "DF.BCAST", NULL, 0);
+
+ /* DF.GSM and DF.TELECOM hierarchy as sub-directory of MF */
+ rc = osim_int_cprof_add_gsm(mf);
+ rc |= osim_int_cprof_add_telecom(mf);
+ if (rc != 0) {
+ talloc_free(cprof);
+ return NULL;
+ }
+
+ return cprof;
+}
diff --git a/src/sim/core.c b/src/sim/core.c
new file mode 100644
index 00000000..890df667
--- /dev/null
+++ b/src/sim/core.c
@@ -0,0 +1,312 @@
+/* Core routines for SIM/UICC/USIM access */
+/*
+ * (C) 2012 by Harald Welte <laforge@gnumonks.org>
+ *
+ * All Rights Reserved
+ *
+ * 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
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+
+#include <stdlib.h>
+#include <stdint.h>
+#include <string.h>
+
+#include <osmocom/core/talloc.h>
+#include <osmocom/sim/sim.h>
+
+struct osim_decoded_data *osim_file_decode(struct osim_file *file,
+ int len, uint8_t *data)
+{
+ struct osim_decoded_data *dd;
+
+ if (!file->desc->ops.parse)
+ return NULL;
+
+ dd = talloc_zero(file, struct osim_decoded_data);
+ if (!dd)
+ return NULL;
+ dd->file = file;
+
+ if (file->desc->ops.parse(dd, file->desc, len, data) < 0) {
+ talloc_free(dd);
+ return NULL;
+ } else
+ return dd;
+}
+
+struct msgb *osim_file_encode(const struct osim_file_desc *desc,
+ const struct osim_decoded_data *data)
+{
+ if (!desc->ops.encode)
+ return NULL;
+
+ return desc->ops.encode(desc, data);
+}
+
+static struct osim_decoded_element *
+__element_alloc(void *ctx, const char *name, enum osim_element_type type,
+ enum osim_element_repr repr)
+{
+ struct osim_decoded_element *elem;
+
+ elem = talloc_zero(ctx, struct osim_decoded_element);
+ if (!elem)
+ return NULL;
+ elem->name = name;
+ elem->type = type;
+ elem->representation = repr;
+
+ if (elem->type == ELEM_T_GROUP)
+ INIT_LLIST_HEAD(&elem->u.siblings);
+
+ return elem;
+}
+
+
+struct osim_decoded_element *
+element_alloc(struct osim_decoded_data *dd, const char *name,
+ enum osim_element_type type, enum osim_element_repr repr)
+{
+ struct osim_decoded_element *elem;
+
+ elem = __element_alloc(dd, name, type, repr);
+ if (!elem)
+ return NULL;
+
+ llist_add_tail(&elem->list, &dd->decoded_elements);
+
+ return elem;
+}
+
+struct osim_decoded_element *
+element_alloc_sub(struct osim_decoded_element *ee, const char *name,
+ enum osim_element_type type, enum osim_element_repr repr)
+{
+ struct osim_decoded_element *elem;
+
+ elem = __element_alloc(ee, name, type, repr);
+ if (!elem)
+ return NULL;
+
+ llist_add(&elem->list, &ee->u.siblings);
+
+ return elem;
+}
+
+
+void add_filedesc(struct osim_file_desc *root, const struct osim_file_desc *in, int num)
+{
+ int i;
+
+ for (i = 0; i < num; i++) {
+ struct osim_file_desc *ofd = talloc_memdup(root, &in[i], sizeof(*in));
+ llist_add_tail(&ofd->list, &root->child_list);
+ }
+}
+
+struct osim_file_desc *alloc_df(void *ctx, uint16_t fid, const char *name)
+{
+ struct osim_file_desc *mf;
+
+ mf = talloc_zero(ctx, struct osim_file_desc);
+ if (!mf)
+ return NULL;
+ mf->type = TYPE_DF;
+ mf->fid = fid;
+ mf->short_name = name;
+ INIT_LLIST_HEAD(&mf->child_list);
+
+ return mf;
+}
+
+struct osim_file_desc *
+add_df_with_ef(struct osim_file_desc *parent,
+ uint16_t fid, const char *name,
+ const struct osim_file_desc *in, int num)
+{
+ struct osim_file_desc *df;
+
+ df = alloc_df(parent, fid, name);
+ if (!df)
+ return NULL;
+ df->parent = parent;
+ llist_add_tail(&df->list, &parent->child_list);
+ add_filedesc(df, in, num);
+
+ return df;
+}
+
+struct osim_file_desc *
+add_adf_with_ef(struct osim_file_desc *parent,
+ const uint8_t *adf_name, uint8_t adf_name_len,
+ const char *name, const struct osim_file_desc *in,
+ int num)
+{
+ struct osim_file_desc *df;
+
+ df = alloc_df(parent, 0xffff, name);
+ if (!df)
+ return NULL;
+ df->type = TYPE_ADF;
+ df->df_name = adf_name;
+ df->df_name_len = adf_name_len;
+ df->parent = parent;
+ llist_add_tail(&df->list, &parent->child_list);
+ add_filedesc(df, in, num);
+
+ return df;
+}
+
+struct osim_file_desc *
+osim_file_find_name(struct osim_file_desc *parent, const char *name)
+{
+ struct osim_file_desc *ofd;
+ llist_for_each_entry(ofd, &parent->child_list, list) {
+ if (!strcmp(ofd->short_name, name)) {
+ return ofd;
+ }
+ }
+ return NULL;
+}
+
+/*! \brief Generate an APDU message and initialize APDU command header
+ * \param[in] cla CLASS byte
+ * \param[in] ins INSTRUCTION byte
+ * \param[in] p1 Parameter 1 byte
+ * \param[in] p2 Parameter 2 byte
+ * \param[in] lc number of bytes in the command data field Nc, which will encoded in 0, 1 or 3 bytes into Lc
+ * \param[in] le maximum number of bytes expected in the response data field, which will encoded in 0, 1, 2 or 3 bytes into Le
+ * \returns an APDU message generated using provided APDU parameters
+ *
+ * This function generates an APDU message, as defined in ISO/IEC 7816-4:2005(E) ยง5.1.
+ * The APDU command header, command and response fields lengths are initialized using the parameters.
+ * The APDU case is determined by the command and response fields lengths.
+ */
+struct msgb *osim_new_apdumsg(uint8_t cla, uint8_t ins, uint8_t p1,
+ uint8_t p2, uint16_t lc, uint16_t le)
+{
+ struct osim_apdu_cmd_hdr *ch;
+ struct msgb *msg = msgb_alloc(lc+le+sizeof(*ch)+2, "APDU");
+ if (!msg)
+ return NULL;
+
+ ch = (struct osim_apdu_cmd_hdr *) msgb_put(msg, sizeof(*ch));
+ msg->l2h = (uint8_t *) ch;
+
+ ch->cla = cla;
+ ch->ins = ins;
+ ch->p1 = p1;
+ ch->p2 = p2;
+
+ msgb_apdu_lc(msg) = lc;
+ msgb_apdu_le(msg) = le;
+
+ if (lc == 0 && le == 0)
+ msgb_apdu_case(msg) = APDU_CASE_1;
+ else if (lc == 0 && le >= 1) {
+ if (le <= 256)
+ msgb_apdu_case(msg) = APDU_CASE_2S;
+ else
+ msgb_apdu_case(msg) = APDU_CASE_2E;
+ } else if (le == 0 && lc >= 1) {
+ if (lc <= 255)
+ msgb_apdu_case(msg) = APDU_CASE_3S;
+ else
+ msgb_apdu_case(msg) = APDU_CASE_3E;
+ } else if (lc >= 1 && le >= 1) {
+ if (lc <= 255 && le <= 256)
+ msgb_apdu_case(msg) = APDU_CASE_4S;
+ else
+ msgb_apdu_case(msg) = APDU_CASE_4E;
+ }
+
+ return msg;
+}
+
+/* FIXME: do we want to mark this as __thread? */
+static char sw_print_buf[256];
+
+char *osim_print_sw(const struct osim_card_hdl *ch, uint16_t sw_in)
+{
+ const struct osim_card_sw *csw;
+
+ if (!ch || !ch->prof)
+ goto ret_def;
+
+ csw = osim_find_sw(ch->prof, sw_in);
+ if (!csw)
+ goto ret_def;
+
+ switch (csw->type) {
+ case SW_TYPE_STR:
+ snprintf(sw_print_buf, sizeof(sw_print_buf),
+ "%04x (%s)", sw_in, csw->u.str);
+ break;
+ default:
+ goto ret_def;
+ }
+
+ sw_print_buf[sizeof(sw_print_buf)-1] = '\0';
+
+ return sw_print_buf;
+
+ret_def:
+ snprintf(sw_print_buf, sizeof(sw_print_buf),
+ "%04x (Unknown)", sw_in);
+ sw_print_buf[sizeof(sw_print_buf)-1] = '\0';
+
+ return sw_print_buf;
+}
+
+
+const struct osim_card_sw *osim_find_sw(const struct osim_card_profile *cp,
+ uint16_t sw_in)
+{
+ const struct osim_card_sw **sw_lists = cp->sws;
+ const struct osim_card_sw *sw_list, *sw;
+
+ for (sw_list = *sw_lists++; sw_list != NULL; sw = sw_list = *sw_lists++) {
+ for (sw = sw_list; sw->code != 0 && sw->mask != 0; sw++) {
+ if ((sw_in & sw->mask) == sw->code)
+ return sw;
+ }
+ }
+ return NULL;
+}
+
+enum osim_card_sw_class osim_sw_class(const struct osim_card_profile *cp,
+ uint16_t sw_in)
+{
+ const struct osim_card_sw *csw = osim_find_sw(cp, sw_in);
+
+ if (!csw)
+ return SW_CLS_NONE;
+
+ return csw->class;
+}
+
+int default_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data)
+{
+ struct osim_decoded_element *elem;
+
+ elem = element_alloc(dd, "Unknown Payload", ELEM_T_BYTES, ELEM_REPR_HEX);
+ elem->u.buf = talloc_memdup(elem, data, len);
+
+ return 0;
+}
diff --git a/src/sim/gsm_int.h b/src/sim/gsm_int.h
new file mode 100644
index 00000000..54a2fbf2
--- /dev/null
+++ b/src/sim/gsm_int.h
@@ -0,0 +1,17 @@
+#include <sys/types.h>
+#include <osmocom/sim/sim.h>
+
+int osim_int_cprof_add_gsm(struct osim_file_desc *mf);
+int osim_int_cprof_add_telecom(struct osim_file_desc *mf);
+
+int gsm_hpplmn_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data);
+
+int gsm_lp_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data);
+
+int gsm_imsi_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data);
diff --git a/src/sim/reader.c b/src/sim/reader.c
new file mode 100644
index 00000000..160f1758
--- /dev/null
+++ b/src/sim/reader.c
@@ -0,0 +1,271 @@
+/* Card reader abstraction for libosmosim */
+/*
+ * (C) 2012 by Harald Welte <laforge@gnumonks.org>
+ *
+ * All Rights Reserved
+ *
+ * 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
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+
+#include <errno.h>
+#include <stdint.h>
+#include <stdlib.h>
+#include <string.h>
+#include <stdio.h>
+
+#include <netinet/in.h>
+
+#include <osmocom/core/msgb.h>
+#include <osmocom/sim/sim.h>
+
+
+#include "sim_int.h"
+
+/* remove the SW from end of the message */
+static int get_sw(struct msgb *resp)
+{
+ int ret;
+
+ if (!msgb_apdu_de(resp) || msgb_apdu_le(resp) < 2)
+ return -EIO;
+
+ ret = msgb_get_u16(resp);
+
+ return ret;
+}
+
+/* According to ISO7816-4 Annex A */
+static int transceive_apdu_t0(struct osim_card_hdl *st, struct msgb *amsg)
+{
+ struct osim_reader_hdl *rh = st->reader;
+ struct msgb *tmsg = msgb_alloc(1024, "TPDU");
+ struct osim_apdu_cmd_hdr *tpduh;
+ uint8_t *cur;
+ uint16_t sw;
+ int rc, num_resp = 0;
+
+ /* create TPDU header from APDU header */
+ tpduh = (struct osim_apdu_cmd_hdr *) msgb_put(tmsg, sizeof(*tpduh));
+ memcpy(tpduh, msgb_apdu_h(amsg), sizeof(*tpduh));
+
+ switch (msgb_apdu_case(amsg)) {
+ case APDU_CASE_1:
+ tpduh->p3 = 0x00;
+ break;
+ case APDU_CASE_2S:
+ tpduh->p3 = msgb_apdu_le(amsg);
+ break;
+ case APDU_CASE_2E:
+ if (msgb_apdu_le(amsg) <= 256) {
+ /* case 2E.1 */
+ tpduh->p3 = msgb_apdu_le(amsg) & 0xff;
+ } else {
+ /* case 2E.2 */
+ tpduh->p3 = 0;
+ msgb_put_u16(tmsg, msgb_apdu_le(amsg));
+ }
+ break;
+ case APDU_CASE_3S:
+ case APDU_CASE_4S:
+ tpduh->p3 = msgb_apdu_lc(amsg);
+ cur = msgb_put(tmsg, tpduh->p3);
+ memcpy(cur, msgb_apdu_dc(amsg), tpduh->p3);
+ break;
+ case APDU_CASE_3E:
+ case APDU_CASE_4E:
+ if (msgb_apdu_lc(amsg) < 256) {
+ /* Case 3E.1 */
+ tpduh->p3 = msgb_apdu_lc(amsg);
+ } else {
+ /* Case 3E.2 */
+ /* FXIME: Split using ENVELOPE! */
+ return -1;
+ }
+ break;
+ }
+
+transceive_again:
+
+ /* store pointer to start of response */
+ tmsg->l3h = tmsg->tail;
+
+ /* transceive */
+ rc = rh->ops->transceive(st->reader, tmsg);
+ if (rc < 0) {
+ msgb_free(tmsg);
+ return rc;
+ }
+ msgb_apdu_sw(tmsg) = get_sw(tmsg);
+
+ /* increase number of responsese received */
+ num_resp++;
+
+ /* save SW */
+ sw = msgb_apdu_sw(tmsg);
+ printf("sw = 0x%04x\n", sw);
+ msgb_apdu_sw(amsg) = sw;
+
+ switch (msgb_apdu_case(amsg)) {
+ case APDU_CASE_1:
+ case APDU_CASE_3S:
+ /* just copy SW */
+ break;
+ case APDU_CASE_2S:
+case_2s:
+ switch (sw >> 8) {
+ case 0x67: /* Case 2S.2: Le definitely not accepted */
+ break;
+ case 0x6c: /* Case 2S.3: Le not accepted, La indicated */
+ tpduh->p3 = sw & 0xff;
+ /* re-issue the command with La as */
+ goto transceive_again;
+ break;
+ case 0x90:
+ /* Case 2S.1, fall-through */
+ case 0x91: case 0x92: case 0x93: case 0x94: case 0x95:
+ case 0x96: case 0x97: case 0x98: case 0x99: case 0x9a:
+ case 0x9b: case 0x9c: case 0x9d: case 0x9e: case 0x9f:
+ /* Case 2S.4 */
+ /* copy response data over */
+ cur = msgb_put(amsg, msgb_l3len(tmsg));
+ memcpy(cur, tmsg->l3h, msgb_l3len(tmsg));
+ }
+ break;
+ case APDU_CASE_4S:
+ /* FIXME: this is 4S.2 only for 2nd... response: */
+ if (num_resp >= 2)
+ goto case_2s;
+
+ switch (sw >> 8) {
+ case 0x60: case 0x62: case 0x63: case 0x64: case 0x65:
+ case 0x66: case 0x67: case 0x68: case 0x69: case 0x6a:
+ case 0x6b: case 0x6c: case 0x6d: case 0x6e: case 0x6f:
+ /* Case 4S.1: Command not accepted: just copy SW */
+ break;
+ case 0x90:
+ /* case 4S.2: Command accepted */
+ tpduh->ins = 0xC0;
+ tpduh->p1 = tpduh->p2 = 0;
+ tpduh->p3 = msgb_apdu_le(amsg);
+ /* strip off current result */
+ msgb_get(tmsg, msgb_length(tmsg)-sizeof(*tpduh));
+ goto transceive_again;
+ break;
+ case 0x61: /* Case 4S.3: command accepted with info added */
+ case 0x9F: /* FIXME: This is specific to SIM cards */
+ tpduh->ins = 0xC0;
+ tpduh->p1 = tpduh->p2 = 0;
+ tpduh->p3 = OSMO_MIN(msgb_apdu_le(amsg), sw & 0xff);
+ /* strip off current result */
+ msgb_get(tmsg, msgb_length(tmsg)-sizeof(*tpduh));
+ goto transceive_again;
+ break;
+ }
+ /* Case 4S.2: Command accepted: just copy SW */
+ /* Case 4S.4: Just copy SW */
+ break;
+ case APDU_CASE_2E:
+ if (msgb_apdu_le(amsg) <= 256) {
+ /* Case 2E.1: Le <= 256 */
+ goto case_2s;
+ }
+ switch (sw >> 8) {
+ case 0x67:
+ /* Case 2E.2a: wrong length, abort */
+ break;
+ case 0x6c:
+ /* Case 2E.2b: wrong length, La given */
+ tpduh->p3 = sw & 0xff;
+ /* re-issue the command with La as given */
+ goto transceive_again;
+ break;
+ case 0x90:
+ /* Case 2E.2c: */
+ break;
+ case 0x61:
+ /* Case 2E.2d: more data available */
+ /* FIXME: issue yet another GET RESPONSE */
+ break;
+ }
+ break;
+ case APDU_CASE_3E:
+ /* FIXME: handling for ENVELOPE splitting */
+ break;
+ case APDU_CASE_4E:
+ break;
+ }
+
+ msgb_free(tmsg);
+
+ /* compute total length of response data */
+ msgb_apdu_le(amsg) = amsg->tail - msgb_apdu_de(amsg);
+
+ return sw;
+}
+
+/* FIXME: T=1 According to ISO7816-4 Annex B */
+
+int osim_transceive_apdu(struct osim_chan_hdl *st, struct msgb *amsg)
+{
+ switch (st->card->proto) {
+ case OSIM_PROTO_T0:
+ return transceive_apdu_t0(st->card, amsg);
+ default:
+ return -ENOTSUP;
+ }
+}
+
+struct osim_reader_hdl *osim_reader_open(enum osim_reader_driver driver, int idx,
+ const char *name, void *ctx)
+{
+ const struct osim_reader_ops *ops;
+ struct osim_reader_hdl *rh;
+
+ switch (driver) {
+ case OSIM_READER_DRV_PCSC:
+ ops = &pcsc_reader_ops;
+ break;
+ default:
+ return NULL;
+ }
+
+ rh = ops->reader_open(idx, name, ctx);
+ if (!rh)
+ return NULL;
+ rh->ops = ops;
+
+ /* FIXME: for now we only do T=0 on all readers */
+ rh->proto_supported = (1 << OSIM_PROTO_T0);
+
+ return rh;
+}
+
+struct osim_card_hdl *osim_card_open(struct osim_reader_hdl *rh, enum osim_proto proto)
+{
+ struct osim_card_hdl *ch;
+
+ if (!(rh->proto_supported & (1 << proto)))
+ return NULL;
+
+ ch = rh->ops->card_open(rh, proto);
+ if (!ch)
+ return NULL;
+
+ ch->proto = proto;
+
+ return ch;
+}
diff --git a/src/sim/reader_pcsc.c b/src/sim/reader_pcsc.c
new file mode 100644
index 00000000..5e670912
--- /dev/null
+++ b/src/sim/reader_pcsc.c
@@ -0,0 +1,161 @@
+/* PC/SC Card reader backend for libosmosim */
+/*
+ * (C) 2012 by Harald Welte <laforge@gnumonks.org>
+ *
+ * All Rights Reserved
+ *
+ * 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
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+
+#include <string.h>
+#include <stdint.h>
+#include <stdio.h>
+#include <errno.h>
+
+#include <osmocom/core/talloc.h>
+#include <osmocom/sim/sim.h>
+
+#include <PCSC/wintypes.h>
+#include <PCSC/winscard.h>
+
+#include "sim_int.h"
+
+#define PCSC_ERROR(rv, text) \
+if (rv != SCARD_S_SUCCESS) { \
+ fprintf(stderr, text ": %s (0x%lX)\n", pcsc_stringify_error(rv), rv); \
+ goto end; \
+} else { \
+ printf(text ": OK\n\n"); \
+}
+
+
+
+struct pcsc_reader_state {
+ SCARDCONTEXT hContext;
+ SCARDHANDLE hCard;
+ DWORD dwActiveProtocol;
+ const SCARD_IO_REQUEST *pioSendPci;
+ SCARD_IO_REQUEST pioRecvPci;
+ char *name;
+};
+
+static struct osim_reader_hdl *pcsc_reader_open(int num, const char *id, void *ctx)
+{
+ struct osim_reader_hdl *rh;
+ struct pcsc_reader_state *st;
+ LONG rc;
+ LPSTR mszReaders = NULL;
+ DWORD dwReaders;
+ unsigned int num_readers;
+ char *ptr;
+
+ /* FIXME: implement matching on id or num */
+
+ rh = talloc_zero(ctx, struct osim_reader_hdl);
+ st = rh->priv = talloc_zero(rh, struct pcsc_reader_state);
+
+ rc = SCardEstablishContext(SCARD_SCOPE_SYSTEM, NULL, NULL,
+ &st->hContext);
+ PCSC_ERROR(rc, "SCardEstablishContext");
+ if (rc != SCARD_S_SUCCESS)
+ goto end;
+
+ dwReaders = SCARD_AUTOALLOCATE;
+ rc = SCardListReaders(st->hContext, NULL, (LPSTR)&mszReaders, &dwReaders);
+ PCSC_ERROR(rc, "SCardListReaders");
+
+ num_readers = 0;
+ ptr = mszReaders;
+ while (*ptr != '\0') {
+ ptr += strlen(ptr)+1;
+ num_readers++;
+ }
+
+ if (num_readers == 0)
+ goto end;
+
+ st->name = talloc_strdup(rh, mszReaders);
+ st->dwActiveProtocol = -1;
+
+ return rh;
+end:
+ talloc_free(rh);
+ return NULL;
+}
+
+static struct osim_card_hdl *pcsc_card_open(struct osim_reader_hdl *rh,
+ enum osim_proto proto)
+{
+ struct pcsc_reader_state *st = rh->priv;
+ struct osim_card_hdl *card;
+ struct osim_chan_hdl *chan;
+ LONG rc;
+
+ if (proto != OSIM_PROTO_T0)
+ return NULL;
+
+ rc = SCardConnect(st->hContext, st->name, SCARD_SHARE_SHARED,
+ SCARD_PROTOCOL_T0, &st->hCard, &st->dwActiveProtocol);
+ PCSC_ERROR(rc, "SCardConnect");
+
+ st->pioSendPci = SCARD_PCI_T0;
+
+ card = talloc_zero(rh, struct osim_card_hdl);
+ INIT_LLIST_HEAD(&card->channels);
+ card->reader = rh;
+ rh->card = card;
+
+ /* create a default channel */
+ chan = talloc_zero(card, struct osim_chan_hdl);
+ chan->card = card;
+ llist_add(&chan->list, &card->channels);
+
+ return card;
+
+end:
+ return NULL;
+}
+
+
+static int pcsc_transceive(struct osim_reader_hdl *rh, struct msgb *msg)
+{
+ struct pcsc_reader_state *st = rh->priv;
+ DWORD rlen = msgb_tailroom(msg);
+ LONG rc;
+
+ printf("TX: %s\n", osmo_hexdump(msg->data, msg->len));
+
+ rc = SCardTransmit(st->hCard, st->pioSendPci, msg->data, msgb_length(msg),
+ &st->pioRecvPci, msg->tail, &rlen);
+ PCSC_ERROR(rc, "SCardEndTransaction");
+
+ printf("RX: %s\n", osmo_hexdump(msg->tail, rlen));
+ msgb_put(msg, rlen);
+ msgb_apdu_le(msg) = rlen;
+
+ return 0;
+end:
+ return -EIO;
+}
+
+const struct osim_reader_ops pcsc_reader_ops = {
+ .name = "PC/SC",
+ .reader_open = pcsc_reader_open,
+ .card_open = pcsc_card_open,
+ .transceive = pcsc_transceive,
+};
+
diff --git a/src/sim/sim_int.h b/src/sim/sim_int.h
new file mode 100644
index 00000000..c10c5f08
--- /dev/null
+++ b/src/sim/sim_int.h
@@ -0,0 +1,41 @@
+#ifndef _SIM_INT_H
+
+#include <osmocom/sim/sim.h>
+
+struct osim_decoded_element *
+element_alloc(struct osim_decoded_data *dd, const char *name,
+ enum osim_element_type type, enum osim_element_repr repr);
+
+struct osim_decoded_element *
+element_alloc_sub(struct osim_decoded_element *ee, const char *name,
+ enum osim_element_type type, enum osim_element_repr repr);
+
+extern const struct osim_card_sw ts102221_uicc_sw[0];
+
+int default_decode(struct osim_decoded_data *dd,
+ const struct osim_file_desc *desc,
+ int len, uint8_t *data);
+
+void add_filedesc(struct osim_file_desc *root, const struct osim_file_desc *in, int num);
+struct osim_file_desc *alloc_df(void *ctx, uint16_t fid, const char *name);
+struct osim_file_desc *
+add_df_with_ef(struct osim_file_desc *parent,
+ uint16_t fid, const char *name,
+ const struct osim_file_desc *in, int num);
+
+struct osim_file_desc *
+add_adf_with_ef(struct osim_file_desc *parent,
+ const uint8_t *adf_name, uint8_t adf_name_len,
+ const char *name, const struct osim_file_desc *in,
+ int num);
+
+struct osim_reader_ops {
+ const char *name;
+ struct osim_reader_hdl *(*reader_open)(int idx, const char *name, void *ctx);
+ struct osim_card_hdl *(*card_open)(struct osim_reader_hdl *rh, enum osim_proto proto);
+ int (*transceive)(struct osim_reader_hdl *rh, struct msgb *msg);
+};
+
+const struct osim_reader_ops pcsc_reader_ops;
+
+#endif