Permalink
Cannot retrieve contributors at this time
Join GitHub today
GitHub is home to over 50 million developers working together to host and review code, manage projects, and build software together.
Sign up| /* | |
| * | |
| * oFono - Open Source Telephony | |
| * | |
| * Copyright (C) 2008-2011 Intel Corporation. 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 version 2 as | |
| * published by the Free Software Foundation. | |
| * | |
| * 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 St, Fifth Floor, Boston, MA 02110-1301 USA | |
| * | |
| */ | |
| #ifdef HAVE_CONFIG_H | |
| #include <config.h> | |
| #endif | |
| #include <glib.h> | |
| #include <errno.h> | |
| #include <gdbus.h> | |
| #include "ofono.h" | |
| #define OFONO_ERROR_INTERFACE "org.ofono.Error" | |
| static DBusConnection *g_connection; | |
| struct error_mapping_entry { | |
| int error; | |
| DBusMessage *(*ofono_error_func)(DBusMessage *); | |
| }; | |
| static const struct error_mapping_entry cme_errors_mapping[] = { | |
| { 3, __ofono_error_not_allowed }, | |
| { 4, __ofono_error_not_supported }, | |
| { 16, __ofono_error_incorrect_password }, | |
| { 30, __ofono_error_not_registered }, | |
| { 31, __ofono_error_timed_out }, | |
| { 32, __ofono_error_access_denied }, | |
| { 50, __ofono_error_invalid_args }, | |
| { } | |
| }; | |
| static const struct error_mapping_entry errno_errors_mapping[] = { | |
| { EACCES, __ofono_error_access_denied }, | |
| { EOPNOTSUPP, __ofono_error_not_supported }, | |
| { ENOSYS, __ofono_error_not_implemented }, | |
| { ETIMEDOUT, __ofono_error_timed_out }, | |
| { EINPROGRESS, __ofono_error_busy }, | |
| { } | |
| }; | |
| static void append_variant(DBusMessageIter *iter, | |
| int type, const void *value) | |
| { | |
| char sig[2]; | |
| DBusMessageIter valueiter; | |
| sig[0] = type; | |
| sig[1] = 0; | |
| dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT, | |
| sig, &valueiter); | |
| dbus_message_iter_append_basic(&valueiter, type, value); | |
| dbus_message_iter_close_container(iter, &valueiter); | |
| } | |
| void ofono_dbus_dict_append(DBusMessageIter *dict, | |
| const char *key, int type, const void *value) | |
| { | |
| DBusMessageIter keyiter; | |
| if (type == DBUS_TYPE_STRING) { | |
| const char *str = *((const char **) value); | |
| if (str == NULL) | |
| return; | |
| } | |
| dbus_message_iter_open_container(dict, DBUS_TYPE_DICT_ENTRY, | |
| NULL, &keyiter); | |
| dbus_message_iter_append_basic(&keyiter, DBUS_TYPE_STRING, &key); | |
| append_variant(&keyiter, type, value); | |
| dbus_message_iter_close_container(dict, &keyiter); | |
| } | |
| static void append_array_variant(DBusMessageIter *iter, int type, | |
| const void *val) | |
| { | |
| DBusMessageIter variant, array; | |
| char typesig[2]; | |
| char arraysig[3]; | |
| const char **str_array = *(const char ***) val; | |
| int i; | |
| arraysig[0] = DBUS_TYPE_ARRAY; | |
| arraysig[1] = typesig[0] = type; | |
| arraysig[2] = typesig[1] = '\0'; | |
| dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT, | |
| arraysig, &variant); | |
| dbus_message_iter_open_container(&variant, DBUS_TYPE_ARRAY, | |
| typesig, &array); | |
| for (i = 0; str_array[i]; i++) | |
| dbus_message_iter_append_basic(&array, type, | |
| &(str_array[i])); | |
| dbus_message_iter_close_container(&variant, &array); | |
| dbus_message_iter_close_container(iter, &variant); | |
| } | |
| void ofono_dbus_dict_append_array(DBusMessageIter *dict, const char *key, | |
| int type, const void *val) | |
| { | |
| DBusMessageIter entry; | |
| dbus_message_iter_open_container(dict, DBUS_TYPE_DICT_ENTRY, | |
| NULL, &entry); | |
| dbus_message_iter_append_basic(&entry, DBUS_TYPE_STRING, &key); | |
| append_array_variant(&entry, type, val); | |
| dbus_message_iter_close_container(dict, &entry); | |
| } | |
| static void append_dict_variant(DBusMessageIter *iter, int type, | |
| const void *val) | |
| { | |
| DBusMessageIter variant, array, entry; | |
| char typesig[5]; | |
| char arraysig[6]; | |
| const void **val_array = *(const void ***) val; | |
| int i; | |
| arraysig[0] = DBUS_TYPE_ARRAY; | |
| arraysig[1] = typesig[0] = DBUS_DICT_ENTRY_BEGIN_CHAR; | |
| arraysig[2] = typesig[1] = DBUS_TYPE_STRING; | |
| arraysig[3] = typesig[2] = type; | |
| arraysig[4] = typesig[3] = DBUS_DICT_ENTRY_END_CHAR; | |
| arraysig[5] = typesig[4] = '\0'; | |
| dbus_message_iter_open_container(iter, DBUS_TYPE_VARIANT, | |
| arraysig, &variant); | |
| dbus_message_iter_open_container(&variant, DBUS_TYPE_ARRAY, | |
| typesig, &array); | |
| for (i = 0; val_array[i]; i += 2) { | |
| dbus_message_iter_open_container(&array, DBUS_TYPE_DICT_ENTRY, | |
| NULL, &entry); | |
| dbus_message_iter_append_basic(&entry, DBUS_TYPE_STRING, | |
| &(val_array[i + 0])); | |
| /* | |
| * D-Bus expects a char** or uint8* depending on the type | |
| * given. Since we are dealing with an array through a void** | |
| * (and thus val_array[i] is a pointer) we need to | |
| * differentiate DBUS_TYPE_STRING from the others. The other | |
| * option would be the user to pass the exact type to this | |
| * function, instead of a pointer to it. However in this case | |
| * a cast from type to void* would be needed, which is not | |
| * good. | |
| */ | |
| if (type == DBUS_TYPE_STRING) { | |
| dbus_message_iter_append_basic(&entry, type, | |
| &(val_array[i + 1])); | |
| } else { | |
| dbus_message_iter_append_basic(&entry, type, | |
| val_array[i + 1]); | |
| } | |
| dbus_message_iter_close_container(&array, &entry); | |
| } | |
| dbus_message_iter_close_container(&variant, &array); | |
| dbus_message_iter_close_container(iter, &variant); | |
| } | |
| void ofono_dbus_dict_append_dict(DBusMessageIter *dict, const char *key, | |
| int type, const void *val) | |
| { | |
| DBusMessageIter entry; | |
| dbus_message_iter_open_container(dict, DBUS_TYPE_DICT_ENTRY, | |
| NULL, &entry); | |
| dbus_message_iter_append_basic(&entry, DBUS_TYPE_STRING, &key); | |
| append_dict_variant(&entry, type, val); | |
| dbus_message_iter_close_container(dict, &entry); | |
| } | |
| int ofono_dbus_signal_property_changed(DBusConnection *conn, | |
| const char *path, | |
| const char *interface, | |
| const char *name, | |
| int type, const void *value) | |
| { | |
| DBusMessage *signal; | |
| DBusMessageIter iter; | |
| signal = dbus_message_new_signal(path, interface, "PropertyChanged"); | |
| if (signal == NULL) { | |
| ofono_error("Unable to allocate new %s.PropertyChanged signal", | |
| interface); | |
| return -1; | |
| } | |
| dbus_message_iter_init_append(signal, &iter); | |
| dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &name); | |
| append_variant(&iter, type, value); | |
| return g_dbus_send_message(conn, signal); | |
| } | |
| int ofono_dbus_signal_array_property_changed(DBusConnection *conn, | |
| const char *path, | |
| const char *interface, | |
| const char *name, | |
| int type, const void *value) | |
| { | |
| DBusMessage *signal; | |
| DBusMessageIter iter; | |
| signal = dbus_message_new_signal(path, interface, "PropertyChanged"); | |
| if (signal == NULL) { | |
| ofono_error("Unable to allocate new %s.PropertyChanged signal", | |
| interface); | |
| return -1; | |
| } | |
| dbus_message_iter_init_append(signal, &iter); | |
| dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &name); | |
| append_array_variant(&iter, type, value); | |
| return g_dbus_send_message(conn, signal); | |
| } | |
| int ofono_dbus_signal_dict_property_changed(DBusConnection *conn, | |
| const char *path, | |
| const char *interface, | |
| const char *name, | |
| int type, const void *value) | |
| { | |
| DBusMessage *signal; | |
| DBusMessageIter iter; | |
| signal = dbus_message_new_signal(path, interface, "PropertyChanged"); | |
| if (signal == NULL) { | |
| ofono_error("Unable to allocate new %s.PropertyChanged signal", | |
| interface); | |
| return -1; | |
| } | |
| dbus_message_iter_init_append(signal, &iter); | |
| dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &name); | |
| append_dict_variant(&iter, type, value); | |
| return g_dbus_send_message(conn, signal); | |
| } | |
| DBusMessage *__ofono_error_invalid_args(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE | |
| ".InvalidArguments", | |
| "Invalid arguments in method call"); | |
| } | |
| DBusMessage *__ofono_error_invalid_format(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE | |
| ".InvalidFormat", | |
| "Argument format is not recognized"); | |
| } | |
| DBusMessage *__ofono_error_not_implemented(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE | |
| ".NotImplemented", | |
| "Implementation not provided"); | |
| } | |
| DBusMessage *__ofono_error_failed(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".Failed", | |
| "Operation failed"); | |
| } | |
| DBusMessage *__ofono_error_busy(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".InProgress", | |
| "Operation already in progress"); | |
| } | |
| DBusMessage *__ofono_error_not_found(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".NotFound", | |
| "Object is not found or not valid for this operation"); | |
| } | |
| DBusMessage *__ofono_error_not_active(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".NotActive", | |
| "Operation is not active or in progress"); | |
| } | |
| DBusMessage *__ofono_error_not_supported(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE | |
| ".NotSupported", | |
| "Operation is not supported by the" | |
| " network / modem"); | |
| } | |
| DBusMessage *__ofono_error_not_available(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE | |
| ".NotAvailable", | |
| "Operation currently not available"); | |
| } | |
| DBusMessage *__ofono_error_timed_out(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".Timedout", | |
| "Operation failure due to timeout"); | |
| } | |
| DBusMessage *__ofono_error_sim_not_ready(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".SimNotReady", | |
| "SIM is not ready or not inserted"); | |
| } | |
| DBusMessage *__ofono_error_in_use(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".InUse", | |
| "The resource is currently in use"); | |
| } | |
| DBusMessage *__ofono_error_not_attached(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".NotAttached", | |
| "GPRS is not attached"); | |
| } | |
| DBusMessage *__ofono_error_attach_in_progress(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, | |
| OFONO_ERROR_INTERFACE ".AttachInProgress", | |
| "GPRS Attach is in progress"); | |
| } | |
| DBusMessage *__ofono_error_not_registered(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, | |
| OFONO_ERROR_INTERFACE ".NotRegistered", | |
| "Modem is not registered to the network"); | |
| } | |
| DBusMessage *__ofono_error_canceled(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".Canceled", | |
| "Operation has been canceled"); | |
| } | |
| DBusMessage *__ofono_error_access_denied(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".AccessDenied", | |
| "Operation not permitted"); | |
| } | |
| DBusMessage *__ofono_error_emergency_active(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, | |
| OFONO_ERROR_INTERFACE ".EmergencyActive", | |
| "Emergency mode active"); | |
| } | |
| DBusMessage *__ofono_error_incorrect_password(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, | |
| OFONO_ERROR_INTERFACE ".IncorrectPassword", | |
| "Password is incorrect"); | |
| } | |
| DBusMessage *__ofono_error_not_allowed(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".NotAllowed", | |
| "Operation is not allowed"); | |
| } | |
| DBusMessage *__ofono_error_not_recognized(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE ".NotRecognized", | |
| "String not recognized as USSD/SS"); | |
| } | |
| DBusMessage *__ofono_error_network_terminated(DBusMessage *msg) | |
| { | |
| return g_dbus_create_error(msg, OFONO_ERROR_INTERFACE | |
| ".Terminated", | |
| "Operation was terminated by the" | |
| " network"); | |
| } | |
| static DBusMessage *__ofono_map_error(const struct error_mapping_entry *map, | |
| int error, DBusMessage *msg) | |
| { | |
| const struct error_mapping_entry *e; | |
| for (e = map; e->ofono_error_func; e++) | |
| if (e->error == error) | |
| return e->ofono_error_func(msg); | |
| return __ofono_error_failed(msg); | |
| } | |
| DBusMessage *__ofono_error_from_error(const struct ofono_error *error, | |
| DBusMessage *msg) | |
| { | |
| switch (error->type) { | |
| case OFONO_ERROR_TYPE_CME: | |
| return __ofono_map_error(cme_errors_mapping, error->error, msg); | |
| case OFONO_ERROR_TYPE_CMS: | |
| return __ofono_error_failed(msg); | |
| case OFONO_ERROR_TYPE_CEER: | |
| return __ofono_error_failed(msg); | |
| case OFONO_ERROR_TYPE_ERRNO: | |
| return __ofono_map_error(errno_errors_mapping, | |
| ABS(error->error), msg); | |
| default: | |
| return __ofono_error_failed(msg); | |
| } | |
| return __ofono_error_failed(msg); | |
| } | |
| void __ofono_dbus_pending_reply(DBusMessage **msg, DBusMessage *reply) | |
| { | |
| DBusConnection *conn = ofono_dbus_get_connection(); | |
| g_dbus_send_message(conn, reply); | |
| dbus_message_unref(*msg); | |
| *msg = NULL; | |
| } | |
| DBusConnection *ofono_dbus_get_connection(void) | |
| { | |
| return g_connection; | |
| } | |
| static void dbus_gsm_set_connection(DBusConnection *conn) | |
| { | |
| if (conn && g_connection != NULL) | |
| ofono_error("Setting a connection when it is not NULL"); | |
| g_connection = conn; | |
| } | |
| int __ofono_dbus_init(DBusConnection *conn) | |
| { | |
| dbus_gsm_set_connection(conn); | |
| return 0; | |
| } | |
| void __ofono_dbus_cleanup(void) | |
| { | |
| DBusConnection *conn = ofono_dbus_get_connection(); | |
| if (conn == NULL || !dbus_connection_get_is_connected(conn)) | |
| return; | |
| dbus_gsm_set_connection(NULL); | |
| } |