/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * qmicli -- Command line interface to control QMI devices * * 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 3 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, see . * * Copyright (C) 2012 Aleksander Morgado */ #include "config.h" #include #include #include #include #include #include #include #include "qmicli.h" #include "qmicli-helpers.h" /* Context */ typedef struct { QmiDevice *device; QmiClientUim *client; GCancellable *cancellable; } Context; static Context *ctx; /* Options */ static gboolean read_efspn_flag; static gboolean read_efimsi_flag; static gboolean reset_flag; static gboolean noop_flag; static GOptionEntry entries[] = { { "uim-read-efspn", 0, 0, G_OPTION_ARG_NONE, &read_efspn_flag, "Read the EFspn file", NULL }, { "uim-read-efimsi", 0, 0, G_OPTION_ARG_NONE, &read_efimsi_flag, "Read the EFimsi file", NULL }, { "uim-reset", 0, 0, G_OPTION_ARG_NONE, &reset_flag, "Reset the service state", NULL }, { "uim-noop", 0, 0, G_OPTION_ARG_NONE, &noop_flag, "Just allocate or release a UIM client. Use with `--client-no-release-cid' and/or `--client-cid'", NULL }, { NULL } }; GOptionGroup * qmicli_uim_get_option_group (void) { GOptionGroup *group; group = g_option_group_new ("uim", "UIM options", "Show User Identity Module options", NULL, NULL); g_option_group_add_entries (group, entries); return group; } gboolean qmicli_uim_options_enabled (void) { static guint n_actions = 0; static gboolean checked = FALSE; if (checked) return !!n_actions; n_actions = (read_efspn_flag + read_efimsi_flag + reset_flag + noop_flag); if (n_actions > 1) { g_printerr ("error: too many UIM actions requested\n"); exit (EXIT_FAILURE); } checked = TRUE; return !!n_actions; } static void context_free (Context *context) { if (!context) return; if (context->client) g_object_unref (context->client); g_object_unref (context->cancellable); g_object_unref (context->device); g_slice_free (Context, context); } static void shutdown (gboolean operation_status) { /* Cleanup context and finish async operation */ context_free (ctx); qmicli_async_operation_done (operation_status); } static void reset_ready (QmiClientUim *client, GAsyncResult *res) { QmiMessageUimResetOutput *output; GError *error = NULL; output = qmi_client_uim_reset_finish (client, res, &error); if (!output) { g_printerr ("error: operation failed: %s\n", error->message); g_error_free (error); shutdown (FALSE); return; } if (!qmi_message_uim_reset_output_get_result (output, &error)) { g_printerr ("error: couldn't reset the UIM service: %s\n", error->message); g_error_free (error); qmi_message_uim_reset_output_unref (output); shutdown (FALSE); return; } g_print ("[%s] Successfully performed UIM service reset\n", qmi_device_get_path_display (ctx->device)); qmi_message_uim_reset_output_unref (output); shutdown (TRUE); } static gboolean noop_cb (gpointer unused) { shutdown (TRUE); return FALSE; } static void read_transparent_ready (QmiClientUim *client, GAsyncResult *res) { QmiMessageUimReadTransparentOutput *output; GError *error = NULL; guint8 sw1 = 0; guint8 sw2 = 0; GArray *read_result = NULL; output = qmi_client_uim_read_transparent_finish (client, res, &error); if (!output) { g_printerr ("error: operation failed: %s\n", error->message); g_error_free (error); shutdown (FALSE); return; } if (!qmi_message_uim_read_transparent_output_get_result (output, &error)) { g_printerr ("error: couldn't read transparent file from the UIM: %s\n", error->message); g_error_free (error); /* Card result */ if (qmi_message_uim_read_transparent_output_get_card_result ( output, &sw1, &sw2, NULL)) { g_print ("Card result:\n" "\tSW1: '0x%02x'\n" "\tSW2: '0x%02x'\n", sw1, sw2); } qmi_message_uim_read_transparent_output_unref (output); shutdown (FALSE); return; } g_print ("[%s] Successfully read information from the UIM:\n", qmi_device_get_path_display (ctx->device)); /* Card result */ if (qmi_message_uim_read_transparent_output_get_card_result ( output, &sw1, &sw2, NULL)) { g_print ("Card result:\n" "\tSW1: '0x%02x'\n" "\tSW2: '0x%02x'\n", sw1, sw2); } /* Read result */ if (qmi_message_uim_read_transparent_output_get_read_result ( output, &read_result, NULL)) { gchar *str; str = qmicli_get_raw_data_printable (read_result, 80, "\t"); g_print ("Read result:\n" "%s\n", str); g_free (str); } qmi_message_uim_read_transparent_output_unref (output); shutdown (TRUE); } typedef struct { gchar *name; guint16 path[3]; } SimFile; static const SimFile sim_files[] = { { "EFspn", { 0x3F00, 0x7F20, 0x6F46 } }, { "EFimsi", { 0x3F00, 0x7F20, 0x6F07 } } }; static QmiMessageUimReadTransparentInput * read_transparent_build_input (const gchar *file_name) { QmiMessageUimReadTransparentInput *input; guint16 file_id = 0; GArray *file_path = NULL; guint i; for (i = 0; i < G_N_ELEMENTS (sim_files); i++) { if (g_str_equal (sim_files[i].name, file_name)) break; } g_assert (i != G_N_ELEMENTS (sim_files)); file_path = g_array_sized_new (FALSE, FALSE, sizeof (guint16), 3); g_array_append_val (file_path, sim_files[i].path[0]); g_array_append_val (file_path, sim_files[i].path[1]); file_id = sim_files[i].path[2]; input = qmi_message_uim_read_transparent_input_new (); qmi_message_uim_read_transparent_input_set_session_information ( input, QMI_UIM_SESSION_TYPE_PRIMARY_GW_PROVISIONING, "qmicli", NULL); qmi_message_uim_read_transparent_input_set_file ( input, file_id, file_path, NULL); qmi_message_uim_read_transparent_input_set_read_information (input, 0, 0, NULL); g_array_unref (file_path); return input; } void qmicli_uim_run (QmiDevice *device, QmiClientUim *client, GCancellable *cancellable) { /* Initialize context */ ctx = g_slice_new (Context); ctx->device = g_object_ref (device); ctx->client = g_object_ref (client); ctx->cancellable = g_object_ref (cancellable); /* Request to read EFspn? */ if (read_efspn_flag) { QmiMessageUimReadTransparentInput *input; input = read_transparent_build_input ("EFspn"); g_debug ("Asynchronously reading EFspn..."); qmi_client_uim_read_transparent (ctx->client, input, 10, ctx->cancellable, (GAsyncReadyCallback)read_transparent_ready, NULL); qmi_message_uim_read_transparent_input_unref (input); return; } /* Request to read EFimsi? */ if (read_efimsi_flag) { QmiMessageUimReadTransparentInput *input; input = read_transparent_build_input ("EFimsi"); g_debug ("Asynchronously reading EFimsi..."); qmi_client_uim_read_transparent (ctx->client, input, 10, ctx->cancellable, (GAsyncReadyCallback)read_transparent_ready, NULL); qmi_message_uim_read_transparent_input_unref (input); return; } /* Request to reset UIM service? */ if (reset_flag) { g_debug ("Asynchronously resetting UIM service..."); qmi_client_uim_reset (ctx->client, NULL, 10, ctx->cancellable, (GAsyncReadyCallback)reset_ready, NULL); return; } /* Just client allocate/release? */ if (noop_flag) { g_idle_add (noop_cb, NULL); return; } g_warn_if_reached (); }