blob: 9b4f9dafd24e6eca2186fd4b5c41db763b596790 [file] [log] [blame]
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* 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:
*
* Copyright (C) 2012 Lanedo GmbH <aleksander@lanedo.com>
*/
#include <string.h>
#include <ctype.h>
#include <stdlib.h>
#include "mm-common-helpers.h"
#include "mm-errors-types.h"
#include "mm-location-gps-nmea.h"
/**
* SECTION: mm-location-gps-nmea
* @title: MMLocationGpsNmea
* @short_description: Helper object to handle NMEA-based GPS location information.
*
* The #MMLocationGpsNmea is an object handling the location information of the
* modem when this is reported by GPS.
*
* This object is retrieved with either mm_modem_location_get_gps_nmea(),
* mm_modem_location_get_gps_nmea_sync(), mm_modem_location_get_full() or
* mm_modem_location_get_full_sync().
*/
G_DEFINE_TYPE (MMLocationGpsNmea, mm_location_gps_nmea, G_TYPE_OBJECT);
struct _MMLocationGpsNmeaPrivate {
GHashTable *traces;
GRegex *sequence_regex;
};
/*****************************************************************************/
static gboolean
check_append_or_replace (MMLocationGpsNmea *self,
const gchar *trace)
{
/* By default, replace */
gboolean append_or_replace = FALSE;
GMatchInfo *match_info = NULL;
if (G_UNLIKELY (!self->priv->sequence_regex))
self->priv->sequence_regex = g_regex_new ("\\$GPGSV,(\\d),(\\d).*",
G_REGEX_RAW | G_REGEX_OPTIMIZE,
0,
NULL);
if (g_regex_match (self->priv->sequence_regex, trace, 0, &match_info)) {
guint index;
/* If we don't have the first element of a sequence, append */
if (mm_get_uint_from_match_info (match_info, 2, &index) && index != 1)
append_or_replace = TRUE;
}
g_match_info_free (match_info);
return append_or_replace;
}
static gboolean
location_gps_nmea_take_trace (MMLocationGpsNmea *self,
gchar *trace)
{
gchar *i;
gchar *trace_type;
i = strchr (trace, ',');
if (!i || i == trace) {
g_free (trace);
return FALSE;
}
trace_type = g_malloc (i - trace + 1);
memcpy (trace_type, trace, i - trace);
trace_type[i - trace] = '\0';
/* Some traces are part of a SEQUENCE; so we need to decide whether we
* completely replace the previous trace, or we append the new one to
* the already existing list */
if (check_append_or_replace (self, trace)) {
/* Append */
const gchar *previous;
previous = g_hash_table_lookup (self->priv->traces, trace_type);
if (previous) {
gchar *sequence;
/* Skip the trace if we already have it there */
if (strstr (previous, trace)) {
g_free (trace_type);
g_free (trace);
return TRUE;
}
sequence = g_strdup_printf ("%s%s%s",
previous,
g_str_has_suffix (previous, "\r\n") ? "" : "\r\n",
trace);
g_free (trace);
trace = sequence;
}
}
g_hash_table_replace (self->priv->traces,
trace_type,
trace);
return TRUE;
}
gboolean
mm_location_gps_nmea_add_trace (MMLocationGpsNmea *self,
const gchar *trace)
{
return location_gps_nmea_take_trace (self, g_strdup (trace));
}
/*****************************************************************************/
/**
* mm_location_gps_nmea_get_trace:
* @self: a #MMLocationGpsNmea.
* @trace_type: specific NMEA trace type to gather.
*
* Gets the last cached value of the specific @trace_type given.
*
* Returns: the NMEA trace, or %NULL if not available. Do not free the returned value, it is owned by @self.
*/
const gchar *
mm_location_gps_nmea_get_trace (MMLocationGpsNmea *self,
const gchar *trace_type)
{
return (const gchar *)g_hash_table_lookup (self->priv->traces, trace_type);
}
/*****************************************************************************/
static void
build_full_foreach (const gchar *trace_type,
const gchar *trace,
GString **built)
{
if ((*built)->len == 0 || g_str_has_suffix ((*built)->str, "\r\n"))
g_string_append (*built, trace);
else
g_string_append_printf (*built, "\r\n%s", trace);
}
/**
* mm_location_gps_nmea_build_full:
* @self: a #MMLocationGpsNmea.
*
* Gets a compilation of all cached traces.
*
* Returns: (transfer full): a string containing all traces, or #NULL if none available. The returned value should be freed with g_free().
*/
gchar *
mm_location_gps_nmea_build_full (MMLocationGpsNmea *self)
{
GString *built;
built = g_string_new ("");
g_hash_table_foreach (self->priv->traces,
(GHFunc)build_full_foreach,
&built);
return g_string_free (built, FALSE);
}
/*****************************************************************************/
GVariant *
mm_location_gps_nmea_get_string_variant (MMLocationGpsNmea *self)
{
GVariant *variant = NULL;
gchar *built;
g_return_val_if_fail (MM_IS_LOCATION_GPS_NMEA (self), NULL);
built = mm_location_gps_nmea_build_full (self);
variant = g_variant_new_string (built);
g_free (built);
return variant;
}
/*****************************************************************************/
MMLocationGpsNmea *
mm_location_gps_nmea_new_from_string_variant (GVariant *string,
GError **error)
{
MMLocationGpsNmea *self = NULL;
gchar **split;
guint i;
if (!g_variant_is_of_type (string, G_VARIANT_TYPE_STRING)) {
g_set_error (error,
MM_CORE_ERROR,
MM_CORE_ERROR_INVALID_ARGS,
"Cannot create GPS NMEA location from string: "
"invalid variant type received");
return NULL;
}
split = g_strsplit (g_variant_get_string (string, NULL), "\r\n", -1);
if (!split) {
g_set_error (error,
MM_CORE_ERROR,
MM_CORE_ERROR_INVALID_ARGS,
"Invalid GPS NMEA location string: '%s'",
g_variant_get_string (string, NULL));
return NULL;
}
/* Create new location object */
self = mm_location_gps_nmea_new ();
for (i = 0; split[i]; i++) {
location_gps_nmea_take_trace (self, split[i]);
}
/* Note that the strings in the array of strings were already taken
* or freed */
g_free (split);
return self;
}
/*****************************************************************************/
MMLocationGpsNmea *
mm_location_gps_nmea_new (void)
{
return (MM_LOCATION_GPS_NMEA (
g_object_new (MM_TYPE_LOCATION_GPS_NMEA, NULL)));
}
static void
mm_location_gps_nmea_init (MMLocationGpsNmea *self)
{
self->priv = G_TYPE_INSTANCE_GET_PRIVATE ((self),
MM_TYPE_LOCATION_GPS_NMEA,
MMLocationGpsNmeaPrivate);
self->priv->traces = g_hash_table_new_full (g_str_hash,
g_str_equal,
g_free,
g_free);
}
static void
finalize (GObject *object)
{
MMLocationGpsNmea *self = MM_LOCATION_GPS_NMEA (object);
g_hash_table_destroy (self->priv->traces);
if (self->priv->sequence_regex)
g_regex_unref (self->priv->sequence_regex);
G_OBJECT_CLASS (mm_location_gps_nmea_parent_class)->finalize (object);
}
static void
mm_location_gps_nmea_class_init (MMLocationGpsNmeaClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
g_type_class_add_private (object_class, sizeof (MMLocationGpsNmeaPrivate));
object_class->finalize = finalize;
}