mirror of git://git.sysmocom.de/ofono
246 lines
5.0 KiB
C
246 lines
5.0 KiB
C
/*
|
|
*
|
|
* oFono - Open Source Telephony
|
|
*
|
|
* Copyright (C) 2008-2010 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 <gatchat.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#define OFONO_API_SUBJECT_TO_CHANGE
|
|
#include <ofono/log.h>
|
|
#include <ofono/types.h>
|
|
|
|
#include "atutil.h"
|
|
|
|
void dump_response(const char *func, gboolean ok, GAtResult *result)
|
|
{
|
|
GSList *l;
|
|
|
|
ofono_debug("%s got result: %d", func, ok);
|
|
ofono_debug("Final response: %s", result->final_or_pdu);
|
|
|
|
for (l = result->lines; l; l = l->next)
|
|
ofono_debug("Response line: %s", (char *) l->data);
|
|
}
|
|
|
|
void decode_at_error(struct ofono_error *error, const char *final)
|
|
{
|
|
if (!strcmp(final, "OK")) {
|
|
error->type = OFONO_ERROR_TYPE_NO_ERROR;
|
|
error->error = 0;
|
|
} else {
|
|
error->type = OFONO_ERROR_TYPE_FAILURE;
|
|
error->error = 0;
|
|
}
|
|
}
|
|
|
|
gint at_util_call_compare_by_status(gconstpointer a, gconstpointer b)
|
|
{
|
|
const struct ofono_call *call = a;
|
|
int status = GPOINTER_TO_INT(b);
|
|
|
|
if (status != call->status)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
gint at_util_call_compare_by_phone_number(gconstpointer a, gconstpointer b)
|
|
{
|
|
const struct ofono_call *call = a;
|
|
const struct ofono_phone_number *pb = b;
|
|
|
|
return memcmp(&call->phone_number, pb,
|
|
sizeof(struct ofono_phone_number));
|
|
}
|
|
|
|
gint at_util_call_compare(gconstpointer a, gconstpointer b)
|
|
{
|
|
const struct ofono_call *ca = a;
|
|
const struct ofono_call *cb = b;
|
|
|
|
if (ca->id < cb->id)
|
|
return -1;
|
|
|
|
if (ca->id > cb->id)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
GSList *at_util_parse_clcc(GAtResult *result)
|
|
{
|
|
GAtResultIter iter;
|
|
GSList *l = NULL;
|
|
int id, dir, status, type;
|
|
ofono_bool_t mpty;
|
|
struct ofono_call *call;
|
|
|
|
g_at_result_iter_init(&iter, result);
|
|
|
|
while (g_at_result_iter_next(&iter, "+CLCC:")) {
|
|
const char *str = "";
|
|
int number_type = 129;
|
|
|
|
if (!g_at_result_iter_next_number(&iter, &id))
|
|
continue;
|
|
|
|
if (!g_at_result_iter_next_number(&iter, &dir))
|
|
continue;
|
|
|
|
if (!g_at_result_iter_next_number(&iter, &status))
|
|
continue;
|
|
|
|
if (!g_at_result_iter_next_number(&iter, &type))
|
|
continue;
|
|
|
|
if (!g_at_result_iter_next_number(&iter, &mpty))
|
|
continue;
|
|
|
|
if (g_at_result_iter_next_string(&iter, &str))
|
|
g_at_result_iter_next_number(&iter, &number_type);
|
|
|
|
call = g_try_new0(struct ofono_call, 1);
|
|
|
|
if (!call)
|
|
break;
|
|
|
|
call->id = id;
|
|
call->direction = dir;
|
|
call->status = status;
|
|
call->type = type;
|
|
call->mpty = mpty;
|
|
strncpy(call->phone_number.number, str,
|
|
OFONO_MAX_PHONE_NUMBER_LENGTH);
|
|
call->phone_number.type = number_type;
|
|
|
|
if (strlen(call->phone_number.number) > 0)
|
|
call->clip_validity = 0;
|
|
else
|
|
call->clip_validity = 2;
|
|
|
|
l = g_slist_insert_sorted(l, call, at_util_call_compare);
|
|
}
|
|
|
|
return l;
|
|
}
|
|
|
|
gboolean at_util_parse_reg_unsolicited(GAtResult *result, const char *prefix,
|
|
int *status,
|
|
int *lac, int *ci, int *tech)
|
|
{
|
|
GAtResultIter iter;
|
|
int s;
|
|
int l = -1, c = -1, t = -1;
|
|
const char *str;
|
|
|
|
g_at_result_iter_init(&iter, result);
|
|
|
|
if (g_at_result_iter_next(&iter, prefix) == FALSE)
|
|
return FALSE;
|
|
|
|
if (g_at_result_iter_next_number(&iter, &s) == FALSE)
|
|
return FALSE;
|
|
|
|
if (g_at_result_iter_next_string(&iter, &str) == TRUE)
|
|
l = strtol(str, NULL, 16);
|
|
else
|
|
goto out;
|
|
|
|
if (g_at_result_iter_next_string(&iter, &str) == TRUE)
|
|
c = strtol(str, NULL, 16);
|
|
else
|
|
goto out;
|
|
|
|
g_at_result_iter_next_number(&iter, &t);
|
|
|
|
out:
|
|
if (status)
|
|
*status = s;
|
|
|
|
if (lac)
|
|
*lac = l;
|
|
|
|
if (ci)
|
|
*ci = c;
|
|
|
|
if (tech)
|
|
*tech = t;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean at_util_parse_reg(GAtResult *result, const char *prefix,
|
|
int *mode, int *status,
|
|
int *lac, int *ci, int *tech)
|
|
{
|
|
GAtResultIter iter;
|
|
int m, s;
|
|
int l = -1, c = -1, t = -1;
|
|
const char *str;
|
|
|
|
g_at_result_iter_init(&iter, result);
|
|
|
|
while (g_at_result_iter_next(&iter, prefix)) {
|
|
g_at_result_iter_next_number(&iter, &m);
|
|
|
|
/* Sometimes we get an unsolicited CREG/CGREG here, skip it */
|
|
if (g_at_result_iter_next_number(&iter, &s) == FALSE)
|
|
continue;
|
|
|
|
if (g_at_result_iter_next_string(&iter, &str) == TRUE)
|
|
l = strtol(str, NULL, 16);
|
|
else
|
|
goto out;
|
|
|
|
if (g_at_result_iter_next_string(&iter, &str) == TRUE)
|
|
c = strtol(str, NULL, 16);
|
|
else
|
|
goto out;
|
|
|
|
g_at_result_iter_next_number(&iter, &t);
|
|
|
|
out:
|
|
if (mode)
|
|
*mode = m;
|
|
|
|
if (status)
|
|
*status = s;
|
|
|
|
if (lac)
|
|
*lac = l;
|
|
|
|
if (ci)
|
|
*ci = c;
|
|
|
|
if (tech)
|
|
*tech = t;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|