2 * PacketService Control Module
4 * Copyright (c) 2012 Samsung Electronics Co., Ltd. All rights reserved.
6 * Contact: DongHoo Park <donghoo.park@samsung.com>
8 * Licensed under the Apache License, Version 2.0 (the "License");
9 * you may not use this file except in compliance with the License.
10 * You may obtain a copy of the License at
12 * http://www.apache.org/licenses/LICENSE-2.0
14 * Unless required by applicable law or agreed to in writing, software
15 * distributed under the License is distributed on an "AS IS" BASIS,
16 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17 * See the License for the specific language governing permissions and
18 * limitations under the License.
22 #include "ps-service.h"
27 #include <core_object.h>
30 #define PROP_DEFAULT FALSE
31 #define PROP_DEFAULT_STR NULL
32 #define BOOL2STRING(a) ((a==TRUE) ? ("TRUE"):("FALSE"))
34 #define TIMEOUT_DEFAULT 5
35 #define TIMEOUT_MAX 1280
37 guint connection_timeout;
49 PROP_SERVICE_CO_NETWORK,
54 SIG_SERVICE_CONTEXT_ADDED,
55 SIG_SERVICE_CONTEXT_REMOVED,
56 SIG_SERVICE_PROPERTY_CHANGED,
60 static guint32 signals[SIG_SERVICE_LAST] = {0,};
62 struct PsServiceClass {
66 void (*context_added)(PsService *service, gchar *context_path);
67 void (*context_removed)(PsService *service, gchar *context_path);
68 void (*property_changed)(PsService *service, GHashTable *service_property);
76 DBusGConnection *conn;
78 CoreObject *co_network;
83 enum telephony_network_access_technology act;
88 G_DEFINE_TYPE(PsService, ps_service, G_TYPE_OBJECT);
90 /*Function Declaration*/
91 static void __ps_service_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec);
92 static void __ps_service_set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec);
94 gboolean ps_iface_service_get_properties(PsService *service, DBusGMethodInvocation *context);
95 gboolean ps_iface_service_get_contexts(PsService *service, DBusGMethodInvocation *context);
97 static void __ps_service_emit_property_changed_signal(PsService *service);
98 static void __ps_service_emit_context_added_signal(PsService *service, gpointer context);
99 static void __ps_service_emit_context_removed_signal(PsService *service, gpointer context);
101 static void __remove_context(gpointer data);
102 static char *__ps_service_act2string(enum telephony_network_access_technology act);
103 static gboolean __ps_service_check_connection_option(gpointer service);
104 static gboolean __ps_service_connetion_timeout_handler(gpointer user_data);
106 #include "ps-iface-service-glue.h"
108 static void ps_service_init(PsService *service)
110 dbg("service initialize");
112 service->path = PROP_DEFAULT_STR;
114 service->conn = NULL;
115 service->p_modem = NULL;
116 service->co_network = NULL;
117 service->co_ps = NULL;
119 service->ps_attached = PROP_DEFAULT;
120 service->roaming = PROP_DEFAULT;
121 service->act = NETWORK_ACT_UNKNOWN;
123 service->contexts = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, __remove_context);
127 static void ps_service_class_init(PsServiceClass *klass)
129 GObjectClass *object_class = G_OBJECT_CLASS(klass);
134 object_class->get_property = __ps_service_get_property;
135 object_class->set_property = __ps_service_set_property;
138 dbus_g_object_type_install_info(PS_TYPE_SERVICE, &dbus_glib_ps_iface_service_object_info);
141 g_object_class_install_property(
144 g_param_spec_string("path", "PATH", "Technology Path", NULL,
145 G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
147 g_object_class_install_property(
149 PROP_SERVICE_P_MODEM,
150 g_param_spec_pointer("p_modem", "MODEM", "Parent Modem Object",
151 G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
153 g_object_class_install_property(
156 g_param_spec_pointer("plg", "PLUGIN", "Plug in Object",
157 G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
159 g_object_class_install_property(
161 PROP_SERVICE_CO_NETWORK,
162 g_param_spec_pointer("co_network", "COREOBJECT NETWORK", "CoreObject of Network",
163 G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
165 g_object_class_install_property(
168 g_param_spec_pointer("co_ps", "COREOBJECTPS", "CoreObject of PS",
169 G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
171 g_object_class_install_property(
174 g_param_spec_boxed("conn", "CONNECTION", "DBus connection", DBUS_TYPE_G_CONNECTION,
175 G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
178 signals[SIG_SERVICE_CONTEXT_ADDED] = g_signal_new("context-added", G_OBJECT_CLASS_TYPE(klass),
179 G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET(PsServiceClass, context_added), NULL, NULL,
180 g_cclosure_marshal_VOID__BOXED, G_TYPE_NONE, 1, DBUS_TYPE_G_STRING_STRING_HASHTABLE);
182 signals[SIG_SERVICE_CONTEXT_REMOVED] = g_signal_new("context-removed",
183 G_OBJECT_CLASS_TYPE(klass), G_SIGNAL_RUN_LAST,
184 G_STRUCT_OFFSET(PsServiceClass, context_removed), NULL, NULL,
185 g_cclosure_marshal_VOID__STRING, G_TYPE_NONE, 1, DBUS_TYPE_G_OBJECT_PATH);
187 signals[SIG_SERVICE_PROPERTY_CHANGED] = g_signal_new("property-changed",
188 G_OBJECT_CLASS_TYPE(klass), G_SIGNAL_RUN_LAST,
189 G_STRUCT_OFFSET(PsServiceClass, property_changed), NULL, NULL,
190 g_cclosure_marshal_VOID__BOXED, G_TYPE_NONE, 1, DBUS_TYPE_G_STRING_STRING_HASHTABLE);
195 static void __ps_service_get_property(GObject *object, guint prop_id, GValue *value,
201 static void __ps_service_set_property(GObject *object, guint prop_id, const GValue *value,
204 PsService *service = PS_SERVICE(object);
207 case PROP_SERVICE_PATH: {
208 if (service->path) g_free(service->path);
209 service->path = g_value_dup_string(value);
210 msg("service(%p) set path(%s)", service, service->path);
213 case PROP_SERVICE_P_MODEM: {
214 service->p_modem = g_value_get_pointer(value);
215 msg("service(%p) set modem(%p)", service, service->p_modem);
218 case PROP_SERVICE_PLUGIN: {
219 service->plg = g_value_get_pointer(value);
220 msg("service(%p) set plg(%p)", service, service->plg);
223 case PROP_SERVICE_CO_NETWORK: {
224 service->co_network = g_value_get_pointer(value);
225 msg("service(%p) set co_network(%p)", service, service->co_network);
228 case PROP_SERVICE_CO_PS: {
229 service->co_ps = g_value_get_pointer(value);
230 msg("service(%p) set co_ps(%p)", service, service->co_ps);
233 case PROP_SERVICE_CONN: {
234 service->conn = g_value_get_boxed(value);
235 msg("service(%p) set conn(%p)", service, service->conn);
239 G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
246 gboolean ps_iface_service_get_properties(PsService *service, DBusGMethodInvocation *context)
250 GHashTable *properties = NULL;
252 dbg("get service properties");
254 properties = g_hash_table_new(g_str_hash, g_str_equal);
255 rv = _ps_service_get_properties(service, properties);
257 g_set_error(&error, PS_ERROR, PS_ERR_INTERNAL, "fail to get properties service(%p)",
259 dbus_g_method_return_error(context, error);
260 g_hash_table_destroy(properties);
264 dbus_g_method_return(context, properties);
265 g_hash_table_destroy(properties);
269 gboolean ps_iface_service_get_contexts(PsService *service, DBusGMethodInvocation *context)
274 GHashTable *contexts;
276 dbg("service get contexts interface");
278 if (service->contexts == NULL) {
279 g_set_error(&error, PS_ERROR, PS_ERR_INTERNAL, "service(%p) does not have contexts",
281 dbus_g_method_return_error(context, error);
285 contexts = g_hash_table_new_full(g_direct_hash, g_str_equal, g_free,
286 (GDestroyNotify) g_hash_table_destroy);
288 g_hash_table_iter_init(&iter, service->contexts);
289 while (g_hash_table_iter_next(&iter, &key, &value) == TRUE) {
292 GHashTable *properties = NULL;
294 properties = g_hash_table_new(g_str_hash, g_str_equal);
295 rv = _ps_context_get_properties(value, properties);
297 g_set_error(&error, PS_ERROR, PS_ERR_INTERNAL, "fail to get properties context(%p)",
299 dbus_g_method_return_error(context, error);
300 g_hash_table_destroy(properties);
301 g_hash_table_destroy(contexts);
305 path = _ps_context_ref_path(value);
306 g_hash_table_insert(contexts, g_strdup(path), properties);
307 dbg("service (%p) inserted into hash", value);
310 dbus_g_method_return(context, contexts);
311 g_hash_table_destroy(contexts);
316 static void __ps_service_emit_property_changed_signal(PsService *service)
318 GHashTable *properties = NULL;
320 properties = g_hash_table_new(g_str_hash, g_str_equal);
321 _ps_service_get_properties(service, properties);
322 g_signal_emit(service, signals[SIG_SERVICE_PROPERTY_CHANGED], 0, properties);
323 dbg("service (%p) emit property changed signal", service);
324 g_hash_table_destroy(properties);
329 static void __ps_service_emit_context_added_signal(PsService *service, gpointer context)
331 GHashTable *properties = NULL;
333 properties = g_hash_table_new(g_str_hash, g_str_equal);
334 _ps_context_get_properties(context, properties);
335 g_signal_emit(service, signals[SIG_SERVICE_CONTEXT_ADDED], 0, properties);
336 dbg("service (%p) emit the context(%p) added signal", service, context);
337 g_hash_table_destroy(properties);
341 static void __ps_service_emit_context_removed_signal(PsService *service, gpointer context)
343 g_signal_emit(service, signals[SIG_SERVICE_CONTEXT_REMOVED], 0, _ps_context_ref_path(context));
344 dbg("service (%p) emit the context(%p) removed signal", service, context);
348 static void __remove_context(gpointer data)
353 static char *__ps_service_act2string(enum telephony_network_access_technology act)
356 case NETWORK_ACT_GSM:
357 case NETWORK_ACT_GPRS:
358 case NETWORK_ACT_EGPRS:
359 case NETWORK_ACT_UMTS:
360 case NETWORK_ACT_GSM_UTRAN:
362 case NETWORK_ACT_IS95A:
363 case NETWORK_ACT_IS95B:
364 case NETWORK_ACT_CDMA_1X:
365 case NETWORK_ACT_EVDO_REV0:
366 case NETWORK_ACT_CDMA_1X_EVDO_REV0:
367 case NETWORK_ACT_EVDO_REVA:
368 case NETWORK_ACT_CDMA_1X_EVDO_REVA:
369 case NETWORK_ACT_EVDV:
371 case NETWORK_ACT_LTE:
373 case NETWORK_ACT_UNKNOWN:
381 static gboolean __ps_service_check_connection_option(gpointer object)
383 gboolean b_connect = TRUE;
384 PsService *service = object;
386 if(service->roaming){
387 b_connect &=_ps_modem_get_data_roaming_allowed(service->p_modem);
390 b_connect &= _ps_modem_get_power(service->p_modem);
391 b_connect &= _ps_modem_get_sim_init(service->p_modem);
392 b_connect &= _ps_modem_get_data_allowed(service->p_modem);
393 b_connect &= !_ps_modem_get_flght_mode(service->p_modem);
394 b_connect &= service->ps_attached;
399 static gboolean __ps_service_connetion_timeout_handler(gpointer context)
402 PsService *service = NULL;
404 service = _ps_context_ref_service(context);
405 rv = _ps_service_activate_context(service, context);
406 dbg("return rv(%d)", rv);
411 gpointer _ps_service_create_service(DBusGConnection *conn, TcorePlugin *p, gpointer p_modem,
412 CoreObject *co_network, CoreObject *co_ps, gchar* path)
417 GError *error = NULL;
419 dbg("service object create");
420 g_return_val_if_fail(conn != NULL, NULL);
421 g_return_val_if_fail(p_modem != NULL, NULL);
423 proxy = dbus_g_proxy_new_for_name(conn, "org.freedesktop.DBus", "/org/freedesktop/DBus",
424 "org.freedesktop.DBus");
426 if (!dbus_g_proxy_call(proxy, "RequestName", &error, G_TYPE_STRING, PS_DBUS_SERVICE,
427 G_TYPE_UINT, 0, G_TYPE_INVALID, G_TYPE_UINT, &rv, G_TYPE_INVALID)) {
428 err("Failed to acquire context(%s) error(%s)", PS_DBUS_SERVICE, error->message);
432 object = g_object_new(PS_TYPE_SERVICE, "conn", conn, "plg", p, "p_modem", p_modem, "co_network",
433 co_network, "co_ps", co_ps, "path", path, NULL);
435 _ps_hook_co_network_event(object);
436 _ps_get_co_network_values(object);
437 _ps_hook_co_ps_event(object);
439 dbus_g_connection_register_g_object(conn, path, object);
440 msg("service(%p) register dbus path(%s)", object, path);
445 gboolean _ps_service_ref_context(gpointer object, gpointer context)
448 gchar *s_path = NULL;
449 PsService *service = object;
451 dbg("service refer to context");
452 g_return_val_if_fail(service != NULL, FALSE);
454 s_path = _ps_context_ref_path(context);
455 tmp = g_hash_table_lookup(service->contexts, s_path);
457 dbg("context(%p) already existed", tmp);
461 _ps_context_set_service(context, service);
462 tcore_ps_add_context(service->co_ps, (CoreObject *) _ps_context_ref_co_context(context));
463 g_hash_table_insert(service->contexts, g_strdup(s_path), context);
465 dbg("context(%p) insert to hash", context);
466 __ps_service_emit_context_added_signal(service, context);
468 //_ps_service_connect_default_context(service);
472 gboolean _ps_service_ref_contexts(gpointer object, GHashTable *contexts, gchar *operator)
476 PsService *service = object;
478 dbg("service refer to contexts");
479 g_return_val_if_fail(service != NULL, FALSE);
481 g_hash_table_iter_init(&iter, contexts);
482 while (g_hash_table_iter_next(&iter, &key, &value) == TRUE) {
483 gchar *s_path = NULL;
486 s_path = _ps_context_ref_path(value);
487 tmp = g_hash_table_lookup(service->contexts, s_path);
489 dbg("context(%p) already existed", tmp);
493 _ps_context_set_service(value, service);
494 tcore_ps_add_context(service->co_ps, (CoreObject *) _ps_context_ref_co_context(value));
495 g_hash_table_insert(service->contexts, g_strdup(s_path), value);
497 dbg("context(%p) insert to hash", value);
498 __ps_service_emit_context_added_signal(service, value);
501 _ps_update_cellular_state_key(service);
502 _ps_service_connect_default_context(service);
506 gboolean _ps_service_unref_context(gpointer object, gpointer context)
508 PsService *service = object;
510 dbg("service unref contexts");
511 g_return_val_if_fail(service != NULL, FALSE);
512 g_return_val_if_fail(context != NULL, FALSE);
514 dbg("remove context(%p) from service(%p)", context, service);
515 tcore_ps_remove_context(service->co_ps, (CoreObject *) _ps_context_ref_co_context(context));
516 g_hash_table_remove(service->contexts, _ps_context_ref_path(context));
517 __ps_service_emit_context_removed_signal(service, context);
522 gboolean _ps_service_get_properties(gpointer object, GHashTable *properties)
524 PsService *service = object;
526 dbg("get service properties");
527 g_return_val_if_fail(service != NULL, FALSE);
528 g_return_val_if_fail(properties != NULL, FALSE);
530 g_hash_table_insert(properties, "path", g_strdup(service->path));
531 g_hash_table_insert(properties, "ps_attached", BOOL2STRING(service->ps_attached));
532 g_hash_table_insert(properties, "roaming", BOOL2STRING(service->roaming));
533 g_hash_table_insert(properties, "act", __ps_service_act2string(service->act));
538 gchar* _ps_service_ref_path(gpointer object)
540 PsService *service = object;
541 g_return_val_if_fail(service != NULL, NULL);
543 return service->path;
546 gpointer _ps_service_ref_plugin(gpointer object)
548 PsService *service = object;
549 g_return_val_if_fail(service != NULL, NULL);
554 gpointer _ps_service_ref_co_network(gpointer object)
556 PsService *service = object;
557 g_return_val_if_fail(service != NULL, NULL);
559 return service->co_network;
562 gpointer _ps_service_ref_co_ps(gpointer object)
564 PsService *service = object;
565 g_return_val_if_fail(service != NULL, NULL);
567 return service->co_ps;
570 gboolean _ps_service_set_context_info(gpointer object, struct tnoti_ps_pdp_ipconfiguration *devinfo)
572 GSList* contexts = NULL;
573 PsService *service = object;
575 dbg("set context info");
576 g_return_val_if_fail(service != NULL, FALSE);
578 contexts = tcore_ps_ref_context_by_id(service->co_ps, devinfo->context_id);
580 CoreObject *co_context = NULL;
582 co_context = contexts->data;
583 if (!co_context) continue;
585 tcore_context_set_devinfo(co_context, devinfo);
587 } while ((contexts = g_slist_next(contexts)));
592 int _ps_service_activate_context(gpointer object, gpointer context)
594 PsService *service = object;
595 CoreObject *co_context = NULL;
596 gboolean b_connect = TRUE;
598 dbg("activate context");
599 g_return_val_if_fail(service != NULL, FALSE);
601 co_context = (CoreObject *)_ps_context_ref_co_context(context);
603 b_connect = __ps_service_check_connection_option(service);
605 return TCORE_RETURN_EPERM;
607 return tcore_ps_activate_context(service->co_ps, co_context, NULL);
610 gboolean _ps_service_deactivate_context(gpointer object, gpointer context)
612 PsService *service = object;
613 CoreObject *co_context = NULL;
615 dbg("deactivate context");
616 g_return_val_if_fail(service != NULL, FALSE);
618 co_context = (CoreObject *)_ps_context_ref_co_context(context);
620 return tcore_ps_deactivate_context(service->co_ps, co_context, NULL);
623 void _ps_service_connection_timer(gpointer object, gpointer context)
625 gboolean f_awo = FALSE;
627 f_awo = _ps_context_get_alwayson_enable(context);
631 timer_src = g_timeout_add_seconds(connection_timeout, __ps_service_connetion_timeout_handler, context);
633 dbg("cellular service timer started timer src(%d), timeout(%d)", timer_src, connection_timeout);
634 connection_timeout = connection_timeout*2;
635 if(connection_timeout > TIMEOUT_MAX)
636 connection_timeout = TIMEOUT_MAX;
641 void _ps_service_reset_connection_timer(gpointer context)
643 gboolean f_awo = FALSE;
645 f_awo = _ps_context_get_alwayson_enable(context);
649 connection_timeout = TIMEOUT_DEFAULT;
651 if (timer_src != 0) {
652 dbg("remove connection retry timer (%d)", timer_src);
653 g_source_remove(timer_src);
660 void _ps_service_reset_contexts(gpointer object)
664 PsService *service = object;
666 dbg("service disconnect all contexts");
667 g_return_if_fail(service != NULL);
669 g_hash_table_iter_init(&iter, service->contexts);
670 while (g_hash_table_iter_next(&iter, &key, &value) == TRUE) {
672 CoreObject *context = NULL;
674 _ps_service_reset_connection_timer(value);
676 context = _ps_context_ref_co_context(value);
677 state = tcore_context_get_state(context);
678 if(state == CONTEXT_STATE_DEACTIVATED){
682 tcore_ps_clear_context_id(service->co_ps, context);
683 _ps_context_set_connected(value, FALSE);
689 void _ps_service_disconnect_contexts(gpointer object)
693 PsService *service = object;
695 dbg("service disconnect all contexts");
696 g_return_if_fail(service != NULL);
698 g_hash_table_iter_init(&iter, service->contexts);
699 while (g_hash_table_iter_next(&iter, &key, &value) == TRUE) {
700 _ps_service_reset_connection_timer(value);
701 _ps_service_deactivate_context(service, value);
707 void _ps_service_connect_default_context(gpointer object)
711 PsService *service = object;
713 dbg("service connect default context");
714 g_return_if_fail(service != NULL);
716 g_hash_table_iter_init(&iter, service->contexts);
717 while (g_hash_table_iter_next(&iter, &key, &value) == TRUE) {
718 gboolean f_awo = FALSE;
719 f_awo = _ps_context_get_alwayson_enable(value);
721 if(f_awo && (service->act > NETWORK_ACT_UNKNOWN && service->act < NETWORK_ACT_LTE)){
723 _ps_service_reset_connection_timer(value);
724 rv = _ps_service_activate_context(service, value);
725 dbg("return rv(%d)", rv);
733 gboolean _ps_service_processing_network_event(gpointer object, gboolean ps_attached, gboolean roaming)
735 PsService *service = object;
736 g_return_val_if_fail(service != NULL, FALSE);
738 _ps_update_cellular_state_key(service);
739 if(service->ps_attached == ps_attached && service->roaming == roaming)
742 _ps_service_set_ps_attached(service, ps_attached);
743 _ps_service_set_roaming(service, roaming);
745 if(service->ps_attached)
746 _ps_service_connect_default_context(service);
750 gboolean _ps_service_set_connected(gpointer object, int context_id, gboolean enabled)
754 PsService *service = NULL;
756 service = (PsService *) object;
757 g_hash_table_iter_init(&iter, service->contexts);
758 while (g_hash_table_iter_next(&iter, &key, &value) == TRUE) {
759 CoreObject *context = NULL;
762 context = _ps_context_ref_co_context(value);
763 tmp_cid = tcore_context_get_id(context);
765 if (tmp_cid != context_id) continue;
768 tcore_ps_clear_context_id(service->co_ps, context);
770 _ps_context_set_connected(value, enabled);
776 gboolean _ps_service_set_ps_attached(gpointer object, gboolean value)
778 PsService *service = object;
779 g_return_val_if_fail(service != NULL, FALSE);
781 service->ps_attached = value;
782 dbg("service(%p) ps_attached(%d)", service, service->ps_attached);
787 gboolean _ps_service_get_roaming(gpointer object)
789 PsService *service = object;
790 g_return_val_if_fail(service != NULL, FALSE);
792 return service->roaming;
795 gboolean _ps_service_set_roaming(gpointer object, gboolean value)
797 PsService *service = object;
798 g_return_val_if_fail(service != NULL, FALSE);
800 service->roaming = value;
801 dbg("service(%p) roaming(%d)", service, service->roaming);
802 __ps_service_emit_property_changed_signal(service);
807 gboolean _ps_service_set_access_technology(gpointer object,
808 enum telephony_network_access_technology value)
810 PsService *service = object;
811 g_return_val_if_fail(service != NULL, FALSE);
813 service->act = value;
814 dbg("service(%p) Access Technology(%d)", service, service->act);
816 if(service->act > NETWORK_ACT_UNKNOWN && service->act < NETWORK_ACT_LTE){
817 _ps_update_cellular_state_key(service);
818 _ps_service_connect_default_context(service);
824 enum telephony_ps_state _ps_service_check_cellular_state(gpointer object)
826 gboolean state = FALSE;
827 PsService *service = object;
828 g_return_val_if_fail(service != NULL, TELEPHONY_PS_NO_SERVICE);
830 state = _ps_modem_get_power(service->p_modem);
832 return TELEPHONY_PS_NO_SERVICE;
835 state = _ps_modem_get_sim_init(service->p_modem);
837 return TELEPHONY_PS_NO_SERVICE;
840 if(!service->ps_attached){
841 return TELEPHONY_PS_NO_SERVICE;
844 state = _ps_modem_get_flght_mode(service->p_modem);
846 return TELEPHONY_PS_FLIGHT_MODE;
849 state = _ps_modem_get_data_roaming_allowed(service->p_modem);
850 if(service->roaming && !state){
851 return TELEPHONY_PS_ROAMING_OFF;
854 state = _ps_modem_get_data_allowed(service->p_modem);
856 return TELEPHONY_PS_3G_OFF;
859 return TELEPHONY_PS_ON;