/* -*- 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 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, see . * * Copyright (C) 2015-2017 Aleksander Morgado */ #include "config.h" #include #include #include #include #include #include #include #include "qmicli.h" #include "qmicli-helpers.h" #if defined HAVE_QMI_SERVICE_WMS #define VALIDATE_UNKNOWN(str) (str ? str : "unknown") /* Context */ typedef struct { QmiDevice *device; QmiClientWms *client; GCancellable *cancellable; } Context; static Context *ctx; /* Options */ static gboolean get_supported_messages_flag; static gboolean get_routes_flag; static gchar *set_routes_str; static gboolean reset_flag; static gboolean noop_flag; static GOptionEntry entries[] = { #if defined HAVE_QMI_MESSAGE_WMS_GET_SUPPORTED_MESSAGES { "wms-get-supported-messages", 0, 0, G_OPTION_ARG_NONE, &get_supported_messages_flag, "Get supported messages", NULL }, #endif #if defined HAVE_QMI_MESSAGE_WMS_GET_ROUTES { "wms-get-routes", 0, 0, G_OPTION_ARG_NONE, &get_routes_flag, "Get SMS route information", NULL }, #endif #if defined HAVE_QMI_MESSAGE_WMS_SET_ROUTES { "wms-set-routes", 0, 0, G_OPTION_ARG_STRING, &set_routes_str, "Set SMS route information (keys: type, class, storage, receipt-action)", "[\"key=value,...\"]" }, #endif #if defined HAVE_QMI_MESSAGE_WMS_RESET { "wms-reset", 0, 0, G_OPTION_ARG_NONE, &reset_flag, "Reset the service state", NULL }, #endif { "wms-noop", 0, 0, G_OPTION_ARG_NONE, &noop_flag, "Just allocate or release a WMS client. Use with `--client-no-release-cid' and/or `--client-cid'", NULL }, { NULL, 0, 0, 0, NULL, NULL, NULL } }; GOptionGroup * qmicli_wms_get_option_group (void) { GOptionGroup *group; group = g_option_group_new ("wms", "WMS options:", "Show Wireless Messaging Service options", NULL, NULL); g_option_group_add_entries (group, entries); return group; } gboolean qmicli_wms_options_enabled (void) { static guint n_actions = 0; static gboolean checked = FALSE; if (checked) return !!n_actions; n_actions = (get_supported_messages_flag + get_routes_flag + !!set_routes_str + reset_flag + noop_flag); if (n_actions > 1) { g_printerr ("error: too many WMS 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 operation_shutdown (gboolean operation_status) { /* Cleanup context and finish async operation */ context_free (ctx); qmicli_async_operation_done (operation_status, FALSE); } #if defined HAVE_QMI_MESSAGE_WMS_GET_SUPPORTED_MESSAGES static void get_supported_messages_ready (QmiClientWms *client, GAsyncResult *res) { QmiMessageWmsGetSupportedMessagesOutput *output; GError *error = NULL; GArray *bytearray = NULL; gchar *str; output = qmi_client_wms_get_supported_messages_finish (client, res, &error); if (!output) { g_printerr ("error: operation failed: %s\n", error->message); g_error_free (error); operation_shutdown (FALSE); return; } if (!qmi_message_wms_get_supported_messages_output_get_result (output, &error)) { g_printerr ("error: couldn't get supported WMS messages: %s\n", error->message); g_error_free (error); qmi_message_wms_get_supported_messages_output_unref (output); operation_shutdown (FALSE); return; } g_print ("[%s] Successfully got supported WMS messages:\n", qmi_device_get_path_display (ctx->device)); qmi_message_wms_get_supported_messages_output_get_list (output, &bytearray, NULL); str = qmicli_get_supported_messages_list (bytearray ? (const guint8 *)bytearray->data : NULL, bytearray ? bytearray->len : 0); g_print ("%s", str); g_free (str); qmi_message_wms_get_supported_messages_output_unref (output); operation_shutdown (TRUE); } #endif /* HAVE_QMI_MESSAGE_WMS_GET_SUPPORTED_MESSAGES */ #if defined HAVE_QMI_MESSAGE_WMS_GET_ROUTES static void get_routes_ready (QmiClientWms *client, GAsyncResult *res) { g_autoptr(QmiMessageWmsGetRoutesOutput) output = NULL; GError *error = NULL; GArray *route_list; guint i; output = qmi_client_wms_get_routes_finish (client, res, &error); if (!output) { g_printerr ("error: operation failed: %s\n", error->message); g_error_free (error); operation_shutdown (FALSE); return; } if (!qmi_message_wms_get_routes_output_get_result (output, &error)) { g_printerr ("error: couldn't get SMS routes: %s\n", error->message); g_error_free (error); operation_shutdown (FALSE); return; } if (!qmi_message_wms_get_routes_output_get_route_list (output, &route_list, &error)) { g_printerr ("error: got invalid SMS routes: %s\n", error->message); g_error_free (error); operation_shutdown (FALSE); return; } g_print ("[%s] Got %u SMS routes:\n", qmi_device_get_path_display (ctx->device), route_list->len); for (i = 0; i < route_list->len; i++) { QmiMessageWmsGetRoutesOutputRouteListElement *route; route = &g_array_index (route_list, QmiMessageWmsGetRoutesOutputRouteListElement, i); g_print (" Route #%u:\n", i + 1); g_print (" Message Type: %s\n", VALIDATE_UNKNOWN (qmi_wms_message_type_get_string (route->message_type))); g_print (" Message Class: %s\n", VALIDATE_UNKNOWN (qmi_wms_message_class_get_string (route->message_class))); g_print (" Storage Type: %s\n", VALIDATE_UNKNOWN (qmi_wms_storage_type_get_string (route->storage))); g_print (" Receipt Action: %s\n", VALIDATE_UNKNOWN (qmi_wms_receipt_action_get_string (route->receipt_action))); } operation_shutdown (TRUE); } #endif /* HAVE_QMI_MESSAGE_WMS_GET_ROUTES */ #if defined HAVE_QMI_MESSAGE_WMS_SET_ROUTES typedef struct { GArray *route_list; gboolean message_type_set; gboolean message_class_set; gboolean storage_set; gboolean receipt_action_set; } SetRoutesContext; static void set_routes_context_init (SetRoutesContext *routes_ctx) { memset (routes_ctx, 0, sizeof(SetRoutesContext)); routes_ctx->route_list = g_array_new (FALSE, TRUE, sizeof (QmiMessageWmsSetRoutesInputRouteListElement)); } static void set_routes_context_destroy (SetRoutesContext *routes_ctx) { g_array_unref (routes_ctx->route_list); } static gboolean set_route_properties_handle (const gchar *key, const gchar *value, GError **error, gpointer user_data) { SetRoutesContext *routes_ctx = user_data; QmiMessageWmsSetRoutesInputRouteListElement *cur_route; gboolean ret = FALSE; if (!value || !value[0]) { g_set_error (error, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "key '%s' required a value", key); return FALSE; } if (!routes_ctx->message_type_set && !routes_ctx->message_class_set && !routes_ctx->storage_set && !routes_ctx->receipt_action_set) { QmiMessageWmsSetRoutesInputRouteListElement new_elt; memset (&new_elt, 0, sizeof (QmiMessageWmsSetRoutesInputRouteListElement)); g_array_append_val (routes_ctx->route_list, new_elt); } cur_route = &g_array_index (routes_ctx->route_list, QmiMessageWmsSetRoutesInputRouteListElement, routes_ctx->route_list->len - 1); if (g_ascii_strcasecmp (key, "type") == 0 && !routes_ctx->message_type_set) { if (!qmicli_read_wms_message_type_from_string (value, &cur_route->message_type)) { g_set_error (error, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "unknown message type '%s'", value); return FALSE; } routes_ctx->message_type_set = TRUE; ret = TRUE; } else if (g_ascii_strcasecmp (key, "class") == 0 && !routes_ctx->message_class_set) { if (!qmicli_read_wms_message_class_from_string (value, &cur_route->message_class)) { g_set_error (error, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "unknown message class '%s'", value); return FALSE; } routes_ctx->message_class_set = TRUE; ret = TRUE; } else if (g_ascii_strcasecmp (key, "storage") == 0 && !routes_ctx->storage_set) { if (!qmicli_read_wms_storage_type_from_string (value, &cur_route->storage)) { g_set_error (error, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "unknown storage type '%s'", value); return FALSE; } routes_ctx->storage_set = TRUE; ret = TRUE; } else if (g_ascii_strcasecmp (key, "receipt-action") == 0 && !routes_ctx->receipt_action_set) { if (!qmicli_read_wms_receipt_action_from_string (value, &cur_route->receipt_action)) { g_set_error (error, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "unknown receipt action '%s'", value); return FALSE; } routes_ctx->receipt_action_set = TRUE; ret = TRUE; } if (routes_ctx->message_type_set && routes_ctx->message_class_set && routes_ctx->storage_set && routes_ctx->receipt_action_set) { /* We have a complete set of details for this route. Reset the context state. */ routes_ctx->message_type_set = FALSE; routes_ctx->message_class_set = FALSE; routes_ctx->storage_set = FALSE; routes_ctx->receipt_action_set = FALSE; } if (!ret) { g_set_error (error, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "unrecognized or duplicate option '%s'", key); } return ret; } static QmiMessageWmsSetRoutesInput * set_routes_input_create (const gchar *str, GError **error) { g_autoptr(QmiMessageWmsSetRoutesInput) input = NULL; SetRoutesContext routes_ctx; GError *inner_error = NULL; set_routes_context_init (&routes_ctx); if (!qmicli_parse_key_value_string (str, &inner_error, set_route_properties_handle, &routes_ctx)) { g_propagate_prefixed_error (error, inner_error, "couldn't parse input string: "); set_routes_context_destroy (&routes_ctx); return NULL; } if (routes_ctx.route_list->len == 0) { g_set_error_literal (error, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "route list was empty"); set_routes_context_destroy (&routes_ctx); return NULL; } if (routes_ctx.message_type_set || routes_ctx.message_class_set || routes_ctx.storage_set || routes_ctx.receipt_action_set) { g_set_error_literal (error, QMI_CORE_ERROR, QMI_CORE_ERROR_FAILED, "final route was missing one or more options"); set_routes_context_destroy (&routes_ctx); return NULL; } /* Create input */ input = qmi_message_wms_set_routes_input_new (); if (!qmi_message_wms_set_routes_input_set_route_list (input, routes_ctx.route_list, &inner_error)) { g_propagate_error (error, inner_error); set_routes_context_destroy (&routes_ctx); return NULL; } set_routes_context_destroy (&routes_ctx); return g_steal_pointer (&input); } static void set_routes_ready (QmiClientWms *client, GAsyncResult *res) { g_autoptr(QmiMessageWmsSetRoutesOutput) output = NULL; GError *error = NULL; output = qmi_client_wms_set_routes_finish (client, res, &error); if (!output) { g_printerr ("error: operation failed: %s\n", error->message); g_error_free (error); operation_shutdown (FALSE); return; } if (!qmi_message_wms_set_routes_output_get_result (output, &error)) { g_printerr ("error: couldn't set SMS routes: %s\n", error->message); g_error_free (error); operation_shutdown (FALSE); return; } g_print ("[%s] Successfully set SMS routes\n", qmi_device_get_path_display (ctx->device)); operation_shutdown (TRUE); } #endif /* HAVE_QMI_MESSAGE_WMS_SET_ROUTES */ #if defined HAVE_QMI_MESSAGE_WMS_RESET static void reset_ready (QmiClientWms *client, GAsyncResult *res) { QmiMessageWmsResetOutput *output; GError *error = NULL; output = qmi_client_wms_reset_finish (client, res, &error); if (!output) { g_printerr ("error: operation failed: %s\n", error->message); g_error_free (error); operation_shutdown (FALSE); return; } if (!qmi_message_wms_reset_output_get_result (output, &error)) { g_printerr ("error: couldn't reset the WMS service: %s\n", error->message); g_error_free (error); qmi_message_wms_reset_output_unref (output); operation_shutdown (FALSE); return; } g_print ("[%s] Successfully performed WMS service reset\n", qmi_device_get_path_display (ctx->device)); qmi_message_wms_reset_output_unref (output); operation_shutdown (TRUE); } #endif static gboolean noop_cb (gpointer unused) { operation_shutdown (TRUE); return FALSE; } void qmicli_wms_run (QmiDevice *device, QmiClientWms *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); #if defined HAVE_QMI_MESSAGE_WMS_GET_SUPPORTED_MESSAGES if (get_supported_messages_flag) { g_debug ("Asynchronously getting supported WMS messages..."); qmi_client_wms_get_supported_messages (ctx->client, NULL, 10, ctx->cancellable, (GAsyncReadyCallback)get_supported_messages_ready, NULL); return; } #endif #if defined HAVE_QMI_MESSAGE_WMS_GET_ROUTES if (get_routes_flag) { g_debug ("Asynchronously getting SMS routes..."); qmi_client_wms_get_routes (ctx->client, NULL, 10, ctx->cancellable, (GAsyncReadyCallback)get_routes_ready, NULL); return; } #endif #if defined HAVE_QMI_MESSAGE_WMS_SET_ROUTES if (set_routes_str) { g_autoptr(QmiMessageWmsSetRoutesInput) input = NULL; GError *error = NULL; input = set_routes_input_create (set_routes_str, &error); if (!input) { g_printerr ("Failed to set route: %s\n", error->message); g_error_free (error); operation_shutdown (FALSE); return; } g_debug ("Asynchronously setting SMS routes..."); qmi_client_wms_set_routes (ctx->client, input, 10, ctx->cancellable, (GAsyncReadyCallback)set_routes_ready, NULL); return; } #endif #if defined HAVE_QMI_MESSAGE_WMS_RESET if (reset_flag) { g_debug ("Asynchronously resetting WMS service..."); qmi_client_wms_reset (ctx->client, NULL, 10, ctx->cancellable, (GAsyncReadyCallback)reset_ready, NULL); return; } #endif /* Just client allocate/release? */ if (noop_flag) { g_idle_add (noop_cb, NULL); return; } g_warn_if_reached (); } #endif /* HAVE_QMI_SERVICE_WMS */