ofono/drivers/hfpmodem/network-registration.c

364 lines
8.4 KiB
C

/*
*
* oFono - Open Source Telephony
*
* Copyright (C) 2008-2011 Intel Corporation. All rights reserved.
* Copyright (C) 2009 ProFUSION embedded systems. 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
#define _GNU_SOURCE
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <glib.h>
#include <gatchat.h>
#include <gatresult.h>
#include <ofono/log.h>
#include <ofono/modem.h>
#include <ofono/netreg.h>
#include "common.h"
#include "hfpmodem.h"
#include "slc.h"
#define HFP_MAX_OPERATOR_NAME_LENGTH 16
static const char *cops_prefix[] = { "+COPS:", NULL };
static const char *cind_prefix[] = { "+CIND:", NULL };
static const char *none_prefix[] = { NULL };
struct netreg_data {
GAtChat *chat;
unsigned char cind_pos[HFP_INDICATOR_LAST];
int cind_val[HFP_INDICATOR_LAST];
guint register_source;
};
static void cops_cb(gboolean ok, GAtResult *result, gpointer user_data)
{
struct cb_data *cbd = user_data;
ofono_netreg_operator_cb_t cb = cbd->cb;
struct ofono_network_operator op;
GAtResultIter iter;
int format;
const char *name;
struct ofono_error error;
decode_at_error(&error, g_at_result_final_response(result));
if (!ok) {
cb(&error, NULL, cbd->data);
return;
}
g_at_result_iter_init(&iter, result);
if (!g_at_result_iter_next(&iter, "+COPS:"))
goto error;
g_at_result_iter_skip_next(&iter);
ok = g_at_result_iter_next_number(&iter, &format);
if (ok == FALSE || format != 0)
goto error;
if (g_at_result_iter_next_string(&iter, &name) == FALSE)
goto error;
strncpy(op.name, name, HFP_MAX_OPERATOR_NAME_LENGTH);
op.name[HFP_MAX_OPERATOR_NAME_LENGTH] = '\0';
op.mcc[0] = '\0';
op.mnc[0] = '\0';
op.status = 2;
op.tech = -1;
cb(&error, &op, cbd->data);
return;
error:
CALLBACK_WITH_FAILURE(cb, NULL, cbd->data);
}
static void ciev_notify(GAtResult *result, gpointer user_data)
{
struct ofono_netreg *netreg = user_data;
struct netreg_data *nd = ofono_netreg_get_data(netreg);
GAtResultIter iter;
int index, value, status;
g_at_result_iter_init(&iter, result);
if (!g_at_result_iter_next(&iter, "+CIEV:"))
return;
if (!g_at_result_iter_next_number(&iter, &index))
return;
if (!g_at_result_iter_next_number(&iter, &value))
return;
if (index == nd->cind_pos[HFP_INDICATOR_SERVICE]) {
nd->cind_val[HFP_INDICATOR_SERVICE] = value;
if (value)
status = NETWORK_REGISTRATION_STATUS_REGISTERED;
else
status = NETWORK_REGISTRATION_STATUS_NOT_REGISTERED;
ofono_netreg_status_notify(netreg, status, -1, -1, -1);
} else if (index == nd->cind_pos[HFP_INDICATOR_ROAM]) {
nd->cind_val[HFP_INDICATOR_ROAM] = value;
if (value)
status = NETWORK_REGISTRATION_STATUS_ROAMING;
else if (nd->cind_val[HFP_INDICATOR_SERVICE])
status = NETWORK_REGISTRATION_STATUS_REGISTERED;
else
status = NETWORK_REGISTRATION_STATUS_NOT_REGISTERED;
ofono_netreg_status_notify(netreg, status, -1, -1, -1);
} else if (index == nd->cind_pos[HFP_INDICATOR_SIGNAL]) {
nd->cind_val[HFP_INDICATOR_SIGNAL] = value;
ofono_netreg_strength_notify(netreg, value * 20);
}
return;
}
static void signal_strength_cb(gboolean ok, GAtResult *result,
gpointer user_data)
{
struct cb_data *cbd = user_data;
ofono_netreg_strength_cb_t cb = cbd->cb;
struct netreg_data *nd = ofono_netreg_get_data(cbd->user);
GAtResultIter iter;
int index, strength;
struct ofono_error error;
decode_at_error(&error, g_at_result_final_response(result));
if (!ok) {
cb(&error, -1, cbd->data);
return;
}
g_at_result_iter_init(&iter, result);
if (!g_at_result_iter_next(&iter, "+CIND:")) {
CALLBACK_WITH_FAILURE(cb, -1, cbd->data);
return;
}
index = 1;
while (g_at_result_iter_next_number(&iter, &strength)) {
if (index == nd->cind_pos[HFP_INDICATOR_SIGNAL]) {
nd->cind_val[HFP_INDICATOR_SIGNAL] = strength;
break;
}
index++;
}
DBG("signal_strength_cb: %d", strength);
cb(&error, strength * 20, cbd->data);
}
static void registration_status_cb(gboolean ok, GAtResult *result,
gpointer user_data)
{
struct cb_data *cbd = user_data;
ofono_netreg_status_cb_t cb = cbd->cb;
struct netreg_data *nd = ofono_netreg_get_data(cbd->user);
GAtResultIter iter;
int index, value, status;
struct ofono_error error;
decode_at_error(&error, g_at_result_final_response(result));
if (!ok) {
cb(&error, -1, -1, -1, -1, cbd->data);
return;
}
g_at_result_iter_init(&iter, result);
if (!g_at_result_iter_next(&iter, "+CIND:")) {
CALLBACK_WITH_FAILURE(cb, -1, -1, -1, -1, cbd->data);
return;
}
index = 1;
while (g_at_result_iter_next_number(&iter, &value)) {
if (index == nd->cind_pos[HFP_INDICATOR_SERVICE])
nd->cind_val[HFP_INDICATOR_SERVICE] = value;
if (index == nd->cind_pos[HFP_INDICATOR_ROAM])
nd->cind_val[HFP_INDICATOR_ROAM] = value;
index++;
}
if (nd->cind_val[HFP_INDICATOR_SERVICE])
status = NETWORK_REGISTRATION_STATUS_REGISTERED;
else
status = NETWORK_REGISTRATION_STATUS_NOT_REGISTERED;
if (nd->cind_val[HFP_INDICATOR_ROAM])
status = NETWORK_REGISTRATION_STATUS_ROAMING;
cb(&error, status, -1, -1, -1, cbd->data);
}
static void hfp_registration_status(struct ofono_netreg *netreg,
ofono_netreg_status_cb_t cb,
void *data)
{
struct netreg_data *nd = ofono_netreg_get_data(netreg);
struct cb_data *cbd = cb_data_new(cb, data);
gboolean ok;
cbd->user = netreg;
ok = g_at_chat_send(nd->chat, "AT+CIND?", cind_prefix,
registration_status_cb, cbd, g_free);
if (ok)
return;
g_free(cbd);
CALLBACK_WITH_FAILURE(cb, -1, -1, -1, -1, data);
}
static void hfp_current_operator(struct ofono_netreg *netreg,
ofono_netreg_operator_cb_t cb, void *data)
{
struct netreg_data *nd = ofono_netreg_get_data(netreg);
struct cb_data *cbd = cb_data_new(cb, data);
gboolean ok;
cbd->user = netreg;
ok = g_at_chat_send(nd->chat, "AT+COPS=3,0", none_prefix,
NULL, cbd, NULL);
if (ok)
ok = g_at_chat_send(nd->chat, "AT+COPS?", cops_prefix,
cops_cb, cbd, g_free);
if (ok)
return;
g_free(cbd);
CALLBACK_WITH_FAILURE(cb, NULL, data);
}
static void hfp_signal_strength(struct ofono_netreg *netreg,
ofono_netreg_strength_cb_t cb, void *data)
{
struct netreg_data *nd = ofono_netreg_get_data(netreg);
struct cb_data *cbd = cb_data_new(cb, data);
cbd->user = netreg;
if (g_at_chat_send(nd->chat, "AT+CIND?", cind_prefix,
signal_strength_cb, cbd, g_free) > 0)
return;
g_free(cbd);
CALLBACK_WITH_FAILURE(cb, -1, data);
}
static gboolean hfp_netreg_register(gpointer user_data)
{
struct ofono_netreg *netreg = user_data;
struct netreg_data *nd = ofono_netreg_get_data(netreg);
nd->register_source = 0;
g_at_chat_register(nd->chat, "+CIEV:", ciev_notify, FALSE,
netreg, NULL);
ofono_netreg_register(netreg);
return FALSE;
}
static int hfp_netreg_probe(struct ofono_netreg *netreg, unsigned int vendor,
void *user_data)
{
struct hfp_slc_info *info = user_data;
struct netreg_data *nd;
nd = g_new0(struct netreg_data, 1);
nd->chat = g_at_chat_clone(info->chat);
memcpy(nd->cind_pos, info->cind_pos, HFP_INDICATOR_LAST);
memcpy(nd->cind_val, info->cind_val, HFP_INDICATOR_LAST);
ofono_netreg_set_data(netreg, nd);
nd->register_source = g_idle_add(hfp_netreg_register, netreg);
return 0;
}
static void hfp_netreg_remove(struct ofono_netreg *netreg)
{
struct netreg_data *nd = ofono_netreg_get_data(netreg);
if (nd->register_source != 0)
g_source_remove(nd->register_source);
ofono_netreg_set_data(netreg, NULL);
g_at_chat_unref(nd->chat);
g_free(nd);
}
static const struct ofono_netreg_driver driver = {
.name = "hfpmodem",
.probe = hfp_netreg_probe,
.remove = hfp_netreg_remove,
.registration_status = hfp_registration_status,
.current_operator = hfp_current_operator,
.strength = hfp_signal_strength,
};
void hfp_netreg_init(void)
{
ofono_netreg_driver_register(&driver);
}
void hfp_netreg_exit(void)
{
ofono_netreg_driver_unregister(&driver);
}