blob: 44a5f1c39303d20f227749d1c0344318dfdccb48 [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) 2008 - 2009 Novell, Inc.
* Copyright (C) 2009 - 2010 Red Hat, Inc.
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <ModemManager.h>
#include <mm-errors-types.h>
#include "mm-qcdm-serial-port.h"
#include "libqcdm/src/com.h"
#include "libqcdm/src/utils.h"
#include "libqcdm/src/errors.h"
#include "mm-log.h"
G_DEFINE_TYPE (MMQcdmSerialPort, mm_qcdm_serial_port, MM_TYPE_SERIAL_PORT)
#define MM_QCDM_SERIAL_PORT_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), MM_TYPE_QCDM_SERIAL_PORT, MMQcdmSerialPortPrivate))
typedef struct {
gboolean foo;
} MMQcdmSerialPortPrivate;
/*****************************************************************************/
static gboolean
find_qcdm_start (GByteArray *response, gsize *start)
{
int i, last = -1;
/* Look for 3 bytes and a QCDM frame marker, ie enough data for a valid
* frame. There will usually be three cases here; (1) a QCDM frame
* starting with data and terminated by 0x7E, and (2) a QCDM frame starting
* with 0x7E and ending with 0x7E, and (3) a non-QCDM frame that still
* uses HDLC framing (like Sierra CnS) that starts and ends with 0x7E.
*/
for (i = 0; i < response->len; i++) {
if (response->data[i] == 0x7E) {
if (i > last + 3) {
/* Got a full QCDM frame; 3 non-0x7E bytes and a terminator */
if (start)
*start = last + 1;
return TRUE;
}
/* Save position of the last QCDM frame marker */
last = i;
}
}
return FALSE;
}
static gboolean
parse_response (MMSerialPort *port, GByteArray *response, GError **error)
{
return find_qcdm_start (response, NULL);
}
static gsize
handle_response (MMSerialPort *port,
GByteArray *response,
GError *error,
GCallback callback,
gpointer callback_data)
{
MMQcdmSerialResponseFn response_callback = (MMQcdmSerialResponseFn) callback;
GByteArray *unescaped = NULL;
GError *dm_error = NULL;
gsize used = 0;
gsize start = 0;
gboolean success = FALSE;
qcdmbool more = FALSE;
gsize unescaped_len = 0;
if (error)
goto callback;
/* Get the offset into the buffer of where the QCDM frame starts */
if (!find_qcdm_start (response, &start)) {
g_set_error_literal (&dm_error,
MM_CORE_ERROR, MM_CORE_ERROR_FAILED,
"Failed to parse QCDM packet.");
/* Discard the unparsable data */
used = response->len;
goto callback;
}
/* FIXME: don't munge around with byte array internals */
unescaped = g_byte_array_sized_new (1024);
success = dm_decapsulate_buffer ((const char *) (response->data + start),
response->len - start,
(char *) unescaped->data,
1024,
&unescaped_len,
&used,
&more);
if (!success) {
g_set_error_literal (&dm_error,
MM_CORE_ERROR, MM_CORE_ERROR_FAILED,
"Failed to unescape QCDM packet.");
g_byte_array_free (unescaped, TRUE);
unescaped = NULL;
} else if (more) {
/* Need more data; we shouldn't have gotten here since the parse
* function checks for the end-of-frame marker, but whatever.
*/
return 0;
} else {
/* Successfully decapsulated the DM command */
unescaped->len = (guint) unescaped_len;
}
callback:
response_callback (MM_QCDM_SERIAL_PORT (port),
unescaped,
dm_error ? dm_error : error,
callback_data);
if (unescaped)
g_byte_array_free (unescaped, TRUE);
g_clear_error (&dm_error);
return start + used;
}
/*****************************************************************************/
void
mm_qcdm_serial_port_queue_command (MMQcdmSerialPort *self,
GByteArray *command,
guint32 timeout_seconds,
GCancellable *cancellable,
MMQcdmSerialResponseFn callback,
gpointer user_data)
{
g_return_if_fail (self != NULL);
g_return_if_fail (MM_IS_QCDM_SERIAL_PORT (self));
g_return_if_fail (command != NULL);
/* 'command' is expected to be already CRC-ed and escaped */
mm_serial_port_queue_command (MM_SERIAL_PORT (self),
command,
TRUE,
timeout_seconds,
cancellable,
(MMSerialResponseFn) callback,
user_data);
}
void
mm_qcdm_serial_port_queue_command_cached (MMQcdmSerialPort *self,
GByteArray *command,
guint32 timeout_seconds,
GCancellable *cancellable,
MMQcdmSerialResponseFn callback,
gpointer user_data)
{
g_return_if_fail (self != NULL);
g_return_if_fail (MM_IS_QCDM_SERIAL_PORT (self));
g_return_if_fail (command != NULL);
/* 'command' is expected to be already CRC-ed and escaped */
mm_serial_port_queue_command_cached (MM_SERIAL_PORT (self),
command,
TRUE,
timeout_seconds,
cancellable,
(MMSerialResponseFn) callback,
user_data);
}
static void
debug_log (MMSerialPort *port, const char *prefix, const char *buf, gsize len)
{
static GString *debug = NULL;
const char *s = buf;
if (!debug)
debug = g_string_sized_new (512);
g_string_append (debug, prefix);
while (len--)
g_string_append_printf (debug, " %02x", (guint8) (*s++ & 0xFF));
mm_dbg ("(%s): %s", mm_port_get_device (MM_PORT (port)), debug->str);
g_string_truncate (debug, 0);
}
/*****************************************************************************/
static gboolean
config_fd (MMSerialPort *port, int fd, GError **error)
{
int err;
err = qcdm_port_setup (fd);
if (err != QCDM_SUCCESS) {
g_set_error (error, MM_SERIAL_ERROR, MM_SERIAL_ERROR_OPEN_FAILED,
"Failed to open QCDM port: %d", err);
return FALSE;
}
return TRUE;
}
/*****************************************************************************/
MMQcdmSerialPort *
mm_qcdm_serial_port_new (const char *name)
{
return MM_QCDM_SERIAL_PORT (g_object_new (MM_TYPE_QCDM_SERIAL_PORT,
MM_PORT_DEVICE, name,
MM_PORT_SUBSYS, MM_PORT_SUBSYS_TTY,
MM_PORT_TYPE, MM_PORT_TYPE_QCDM,
NULL));
}
MMQcdmSerialPort *
mm_qcdm_serial_port_new_fd (int fd)
{
MMQcdmSerialPort *port;
char *name;
name = g_strdup_printf ("port%d", fd);
port = MM_QCDM_SERIAL_PORT (g_object_new (MM_TYPE_QCDM_SERIAL_PORT,
MM_PORT_DEVICE, name,
MM_PORT_SUBSYS, MM_PORT_SUBSYS_TTY,
MM_PORT_TYPE, MM_PORT_TYPE_QCDM,
MM_SERIAL_PORT_FD, fd,
NULL));
g_free (name);
return port;
}
static void
mm_qcdm_serial_port_init (MMQcdmSerialPort *self)
{
}
static void
finalize (GObject *object)
{
G_OBJECT_CLASS (mm_qcdm_serial_port_parent_class)->finalize (object);
}
static void
mm_qcdm_serial_port_class_init (MMQcdmSerialPortClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
MMSerialPortClass *port_class = MM_SERIAL_PORT_CLASS (klass);
g_type_class_add_private (object_class, sizeof (MMQcdmSerialPortPrivate));
/* Virtual methods */
object_class->finalize = finalize;
port_class->parse_response = parse_response;
port_class->handle_response = handle_response;
port_class->config_fd = config_fd;
port_class->debug_log = debug_log;
}