ofono/drivers/atmodem/sim.c

332 lines
7.1 KiB
C
Raw Normal View History

2009-05-22 11:58:49 +00:00
/*
*
* oFono - Open Source Telephony
*
* Copyright (C) 2008-2009 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
#define _GNU_SOURCE
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <glib.h>
#include <ofono/log.h>
#include "driver.h"
#include "gatchat.h"
#include "gatresult.h"
#include "at.h"
static const char *crsm_prefix[] = { "+CRSM:", NULL };
static const char *cnum_prefix[] = { "+CNUM:", NULL };
static void at_crsm_len_cb(gboolean ok, GAtResult *result, gpointer user_data)
{
struct cb_data *cbd = user_data;
GAtResultIter iter;
ofono_sim_file_len_cb_t cb = cbd->cb;
gint sw1, len;
struct ofono_error error;
dump_response("at_crsm_len_cb", ok, result);
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, "+CRSM:")) {
DECLARE_FAILURE(e);
cb(&e, -1, cbd->data);
return;
}
g_at_result_iter_next_number(&iter, &sw1);
g_at_result_iter_next_number(&iter, &len);
ofono_debug("crsm_len_cb: %i, %i", sw1, len);
if (sw1 != 0x67) {
DECLARE_FAILURE(e);
cb(&e, -1, cbd->data);
return;
}
cb(&error, len, cbd->data);
}
static void at_sim_read_file_len(struct ofono_modem *modem, int fileid,
ofono_sim_file_len_cb_t cb,
void *data)
{
struct at_data *at = ofono_modem_userdata(modem);
struct cb_data *cbd = cb_data_new(modem, cb, data);
char buf[64];
if (!cbd)
goto error;
snprintf(buf, sizeof(buf), "AT+CRSM=176,%i,0,0,0", fileid);
if (g_at_chat_send(at->parser, buf, crsm_prefix,
at_crsm_len_cb, cbd, g_free) > 0)
return;
error:
if (cbd)
g_free(cbd);
{
DECLARE_FAILURE(error);
cb(&error, -1, data);
}
}
static void at_crsm_cb(gboolean ok, GAtResult *result, gpointer user_data)
{
struct cb_data *cbd = user_data;
GAtResultIter iter;
ofono_sim_read_cb_t cb = cbd->cb;
struct ofono_error error;
const guint8 *response;
gint sw1, sw2, len;
dump_response("at_crsm_cb", ok, result);
decode_at_error(&error, g_at_result_final_response(result));
if (!ok) {
cb(&error, NULL, 0, cbd->data);
return;
}
g_at_result_iter_init(&iter, result);
if (!g_at_result_iter_next(&iter, "+CRSM:")) {
DECLARE_FAILURE(e);
cb(&e, NULL, 0, cbd->data);
return;
}
g_at_result_iter_next_number(&iter, &sw1);
g_at_result_iter_next_number(&iter, &sw2);
if (!g_at_result_iter_next_hexstring(&iter, &response, &len) ||
(sw1 != 0x90 && sw1 != 0x91 && sw1 != 0x92) ||
(sw1 == 0x90 && sw2 != 0x00)) {
DECLARE_FAILURE(e);
cb(&e, NULL, 0, cbd->data);
return;
}
ofono_debug("crsm_cb: %02x, %02x, %d", sw1, sw2, len);
cb(&error, response, len, cbd->data);
}
static void at_sim_read_file(struct ofono_modem *modem, int fileid, int start,
int length, ofono_sim_read_cb_t cb,
void *data)
{
struct at_data *at = ofono_modem_userdata(modem);
struct cb_data *cbd = cb_data_new(modem, cb, data);
char buf[64];
if (!cbd)
goto error;
snprintf(buf, sizeof(buf), "AT+CRSM=176,%i,%i,%i,%i", fileid,
start >> 8, start & 0xff, length);
if (g_at_chat_send(at->parser, buf, crsm_prefix,
at_crsm_cb, cbd, g_free) > 0)
return;
error:
if (cbd)
g_free(cbd);
{
DECLARE_FAILURE(error);
cb(&error, NULL, 0, data);
}
}
static void at_cimi_cb(gboolean ok, GAtResult *result, gpointer user_data)
{
struct cb_data *cbd = user_data;
GAtResultIter iter;
ofono_imsi_cb_t cb = cbd->cb;
struct ofono_error error;
const char *imsi;
2009-05-29 23:14:40 +00:00
int i;
2009-05-22 11:58:49 +00:00
dump_response("at_cimi_cb", ok, result);
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);
2009-05-29 23:14:40 +00:00
for (i = 0; i < g_at_result_num_response_lines(result); i++)
g_at_result_iter_next(&iter, NULL);
2009-05-22 11:58:49 +00:00
imsi = g_at_result_iter_raw_line(&iter);
ofono_debug("cimi_cb: %s", imsi);
cb(&error, imsi, cbd->data);
}
static void at_read_imsi(struct ofono_modem *modem, ofono_imsi_cb_t cb,
void *data)
{
struct at_data *at = ofono_modem_userdata(modem);
struct cb_data *cbd = cb_data_new(modem, cb, data);
if (!cbd)
goto error;
2009-05-29 23:14:40 +00:00
if (g_at_chat_send(at->parser, "AT+CIMI", NULL,
2009-05-22 11:58:49 +00:00
at_cimi_cb, cbd, g_free) > 0)
return;
error:
if (cbd)
g_free(cbd);
{
DECLARE_FAILURE(error);
cb(&error, NULL, data);
}
}
static void at_cnum_cb(gboolean ok, GAtResult *result, gpointer user_data)
{
struct cb_data *cbd = user_data;
GAtResultIter iter;
ofono_own_numbers_cb_t cb = cbd->cb;
2009-05-22 11:58:49 +00:00
struct ofono_error error;
struct ofono_own_number *numbers;
2009-05-22 11:58:49 +00:00
int count;
const char *str;
dump_response("at_cnum_cb", ok, result);
decode_at_error(&error, g_at_result_final_response(result));
if (!ok) {
cb(&error, 0, NULL, cbd->data);
2009-05-22 11:58:49 +00:00
return;
}
g_at_result_iter_init(&iter, result);
for (count = 0; g_at_result_iter_next(&iter, "+CNUM:"); count++);
ofono_debug("Got %i elements", count);
numbers = g_try_new0(struct ofono_own_number, count);
2009-05-22 11:58:49 +00:00
if (!numbers) {
DECLARE_FAILURE(e);
cb(&e, 0, NULL, cbd->data);
2009-05-22 11:58:49 +00:00
return;
}
g_at_result_iter_init(&iter, result);
count = 0;
while (g_at_result_iter_next(&iter, "+CNUM")) {
/* Skip alnum */
2009-05-22 11:58:49 +00:00
g_at_result_iter_skip_next(&iter);
if (!g_at_result_iter_next_string(&iter, &str))
continue;
g_strlcpy(numbers[count].phone_number.number,
str[0] == '+' ? str+1 : str,
2009-05-22 11:58:49 +00:00
OFONO_MAX_PHONE_NUMBER_LENGTH);
g_at_result_iter_next_number(&iter,
&numbers[count].phone_number.type);
numbers[count].speed = -1;
numbers[count].service = -1;
numbers[count].itc = -1;
g_at_result_iter_skip_next(&iter);
g_at_result_iter_next_number(&iter, &numbers[count].service);
g_at_result_iter_next_number(&iter, &numbers[count].itc);
count++;
2009-05-22 11:58:49 +00:00
}
cb(&error, count, numbers, cbd->data);
2009-05-22 11:58:49 +00:00
g_free(numbers);
}
static void at_read_msisdn(struct ofono_modem *modem, ofono_own_numbers_cb_t cb,
void *data)
2009-05-22 11:58:49 +00:00
{
struct at_data *at = ofono_modem_userdata(modem);
struct cb_data *cbd = cb_data_new(modem, cb, data);
if (!cbd)
goto error;
if (g_at_chat_send(at->parser, "AT+CNUM", cnum_prefix,
at_cnum_cb, cbd, g_free) > 0)
return;
error:
if (cbd)
g_free(cbd);
{
DECLARE_FAILURE(error);
cb(&error, 0, NULL, data);
2009-05-22 11:58:49 +00:00
}
}
static struct ofono_sim_ops ops = {
.read_file_len = at_sim_read_file_len,
.read_file = at_sim_read_file,
.read_imsi = at_read_imsi,
.read_own_numbers = at_read_msisdn,
};
void at_sim_init(struct ofono_modem *modem)
{
ofono_sim_manager_register(modem, &ops);
}
void at_sim_exit(struct ofono_modem *modem)
{
ofono_sim_manager_unregister(modem);
}