/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * Copyright (C) 2011 Red Hat, Inc. * * This program is free software: you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License 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, see . */ #include #include "commands.h" #include "errors.h" #include "result-private.h" #include "utils.h" #include "protocol.h" /**********************************************************************/ static int check_command (const char *buf, gsize len, u_int8_t cmd, size_t min_len) { if (len < 1) { wmc_err (0, "Zero-length response"); return -WMC_ERROR_RESPONSE_BAD_LENGTH; } if ((u_int8_t) buf[0] != WMC_CMD_MARKER) { wmc_err (0, "Missing WMC command marker (expected 0x%02X, got 0x%02X)", WMC_CMD_MARKER, (u_int8_t) buf[0]); return -WMC_ERROR_RESPONSE_UNEXPECTED; } if ((u_int8_t) buf[1] != cmd) { wmc_err (0, "Unexpected WMC command response (expected 0x%02X, got 0x%02X)", (u_int8_t) cmd, (u_int8_t) buf[1]); return -WMC_ERROR_RESPONSE_UNEXPECTED; } if (len < min_len) { wmc_err (0, "WMC command %d response not long enough (got %zu, expected " "at least %zu).", cmd, len, min_len); return -WMC_ERROR_RESPONSE_BAD_LENGTH; } return 0; } /**********************************************************************/ /** * wmc_cmd_init_new: * @buf: buffer in which to store constructed command * @buflen: size of @buf * @wmc2: if %TRUE add additional data that later-model devices (UML290) want * * Returns: size of the constructed command on success, or 0 on failure */ size_t wmc_cmd_init_new (char *buf, size_t buflen, int wmc2) { wmc_return_val_if_fail (buf != NULL, 0); if (wmc2) { WmcCmdInit2 *cmd = (WmcCmdInit2 *) buf; const char data[] = { 0xda, 0x07, 0x0c, 0x00, 0x1e, 0x00, 0x09, 0x00, 0x39, 0x00, 0x18, 0x00, 0x04, 0x00 }; wmc_return_val_if_fail (buflen >= sizeof (*cmd), 0); memset (cmd, 0, sizeof (*cmd)); cmd->hdr.marker = WMC_CMD_MARKER; cmd->hdr.cmd = WMC_CMD_INIT; memcpy (cmd->_unknown1, data, sizeof (data)); return sizeof (*cmd); } else { WmcCmdHeader *cmd = (WmcCmdHeader *) buf; wmc_return_val_if_fail (buflen >= sizeof (*cmd), 0); memset (cmd, 0, sizeof (*cmd)); cmd->marker = WMC_CMD_MARKER; cmd->cmd = WMC_CMD_INIT; return sizeof (*cmd); } } WmcResult * wmc_cmd_init_result (const char *buf, gsize buflen, int wmc2) { g_return_val_if_fail (buf != NULL, NULL); if (wmc2) { if (check_command (buf, buflen, WMC_CMD_INIT, sizeof (WmcCmdInit2Rsp)) < 0) return NULL; } else { if (check_command (buf, buflen, WMC_CMD_INIT, sizeof (WmcCmdHeader)) < 0) return NULL; } return wmc_result_new (); } /**********************************************************************/ size_t wmc_cmd_device_info_new (char *buf, size_t buflen) { WmcCmdHeader *cmd = (WmcCmdHeader *) buf; wmc_return_val_if_fail (buf != NULL, 0); wmc_return_val_if_fail (buflen >= sizeof (*cmd), 0); memset (cmd, 0, sizeof (*cmd)); cmd->marker = WMC_CMD_MARKER; cmd->cmd = WMC_CMD_DEVICE_INFO; return sizeof (*cmd); } WmcResult * wmc_cmd_device_info_result (const char *buf, gsize buflen) { WmcResult *r = NULL; WmcCmdDeviceInfoRsp *rsp = (WmcCmdDeviceInfoRsp *) buf; WmcCmdDeviceInfo2Rsp *rsp2 = (WmcCmdDeviceInfo2Rsp *) buf; char tmp[65]; g_return_val_if_fail (buf != NULL, NULL); if (check_command (buf, buflen, WMC_CMD_DEVICE_INFO, sizeof (WmcCmdDeviceInfo2Rsp)) < 0) { rsp2 = NULL; if (check_command (buf, buflen, WMC_CMD_DEVICE_INFO, sizeof (WmcCmdDeviceInfoRsp)) < 0) return NULL; } r = wmc_result_new (); /* Manf */ memset (tmp, 0, sizeof (tmp)); g_assert (sizeof (rsp->manf) <= sizeof (tmp)); memcpy (tmp, rsp->manf, sizeof (rsp->manf)); wmc_result_add_string (r, WMC_CMD_DEVICE_INFO_ITEM_MANUFACTURER, tmp); /* Model */ memset (tmp, 0, sizeof (tmp)); g_assert (sizeof (rsp->model) <= sizeof (tmp)); memcpy (tmp, rsp->model, sizeof (rsp->model)); wmc_result_add_string (r, WMC_CMD_DEVICE_INFO_ITEM_MODEL, tmp); /* Firmware revision */ memset (tmp, 0, sizeof (tmp)); g_assert (sizeof (rsp->fwrev) <= sizeof (tmp)); memcpy (tmp, rsp->fwrev, sizeof (rsp->fwrev)); wmc_result_add_string (r, WMC_CMD_DEVICE_INFO_ITEM_FW_REVISION, tmp); /* Hardware revision */ memset (tmp, 0, sizeof (tmp)); g_assert (sizeof (rsp->hwrev) <= sizeof (tmp)); memcpy (tmp, rsp->hwrev, sizeof (rsp->hwrev)); wmc_result_add_string (r, WMC_CMD_DEVICE_INFO_ITEM_HW_REVISION, tmp); if (rsp2) { /* IMEI */ memset (tmp, 0, sizeof (tmp)); g_assert (sizeof (rsp2->imei) <= sizeof (tmp)); memcpy (tmp, rsp2->imei, sizeof (rsp2->imei)); wmc_result_add_string (r, WMC_CMD_DEVICE_INFO_ITEM_IMEI, tmp); /* IMSI */ memset (tmp, 0, sizeof (tmp)); g_assert (sizeof (rsp2->iccid) <= sizeof (tmp)); memcpy (tmp, rsp2->iccid, sizeof (rsp2->iccid)); wmc_result_add_string (r, WMC_CMD_DEVICE_INFO_ITEM_ICCID, tmp); /* MCC */ memset (tmp, 0, sizeof (tmp)); g_assert (sizeof (rsp2->mcc) <= sizeof (tmp)); memcpy (tmp, rsp2->mcc, sizeof (rsp2->mcc)); wmc_result_add_string (r, WMC_CMD_DEVICE_INFO_ITEM_MCC, tmp); /* MNC */ memset (tmp, 0, sizeof (tmp)); g_assert (sizeof (rsp2->mnc) <= sizeof (tmp)); memcpy (tmp, rsp2->mnc, sizeof (rsp2->mnc)); wmc_result_add_string (r, WMC_CMD_DEVICE_INFO_ITEM_MNC, tmp); } return r; } /**********************************************************************/ size_t wmc_cmd_status_new (char *buf, size_t buflen) { WmcCmdHeader *cmd = (WmcCmdHeader *) buf; wmc_return_val_if_fail (buf != NULL, 0); wmc_return_val_if_fail (buflen >= sizeof (*cmd), 0); memset (cmd, 0, sizeof (*cmd)); cmd->marker = WMC_CMD_MARKER; cmd->cmd = WMC_CMD_STATUS; return sizeof (*cmd); } static u_int8_t sanitize_dbm (u_int8_t in_dbm) { /* 0x7D (-125 dBm) really means no signal */ return in_dbm >= 0x7D ? 0 : in_dbm; } WmcResult * wmc_cmd_status_result (const char *buf, gsize buflen) { WmcResult *r = NULL; WmcCmdStatusRsp *rsp = (WmcCmdStatusRsp *) buf; WmcCmdStatus2Rsp *rsp2 = (WmcCmdStatus2Rsp *) buf; char tmp[65]; g_return_val_if_fail (buf != NULL, NULL); if (check_command (buf, buflen, WMC_CMD_STATUS, sizeof (WmcCmdStatus2Rsp)) < 0) { rsp2 = NULL; if (check_command (buf, buflen, WMC_CMD_STATUS, sizeof (WmcCmdStatusRsp)) < 0) return NULL; } r = wmc_result_new (); wmc_result_add_u8 (r, WMC_CMD_STATUS_ITEM_CDMA_DBM, sanitize_dbm (rsp->cdma1x_dbm)); if (rsp2) { wmc_result_add_u8 (r, WMC_CMD_STATUS_ITEM_HDR_DBM, sanitize_dbm (rsp2->hdr_dbm)); wmc_result_add_u8 (r, WMC_CMD_STATUS_ITEM_LTE_DBM, sanitize_dbm (rsp2->lte_dbm)); memset (tmp, 0, sizeof (tmp)); if (sanitize_dbm (rsp2->lte_dbm)) { /* LTE operator name */ g_assert (sizeof (rsp2->lte_opname) <= sizeof (tmp)); memcpy (tmp, rsp2->lte_opname, sizeof (rsp2->lte_opname)); wmc_result_add_string (r, WMC_CMD_STATUS_ITEM_OPNAME, tmp); } else if (sanitize_dbm (rsp2->hdr_dbm) || sanitize_dbm (rsp2->cdma1x_dbm)) { /* CDMA2000 operator name */ g_assert (sizeof (rsp2->cdma_opname) <= sizeof (tmp)); memcpy (tmp, rsp2->cdma_opname, sizeof (rsp2->cdma_opname)); wmc_result_add_string (r, WMC_CMD_STATUS_ITEM_OPNAME, tmp); } } return r; } /**********************************************************************/