/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * Copyright (C) 2010 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 #include #include "commands.h" #include "errors.h" #include "dm-commands.h" #include "nv-items.h" #include "result-private.h" #include "utils.h" /**********************************************************************/ static u_int8_t cdma_prev_to_qcdm (u_int8_t cdma) { switch (cdma) { case CDMA_PREV_IS_95: return QCDM_CDMA_PREV_IS_95; case CDMA_PREV_IS_95A: return QCDM_CDMA_PREV_IS_95A; case CDMA_PREV_IS_95A_TSB74: return QCDM_CDMA_PREV_IS_95A_TSB74; case CDMA_PREV_IS_95B_PHASE1: return QCDM_CDMA_PREV_IS_95B_PHASE1; case CDMA_PREV_IS_95B_PHASE2: return QCDM_CDMA_PREV_IS_95B_PHASE2; case CDMA_PREV_IS2000_REL0: return QCDM_CDMA_PREV_IS2000_REL0; case CDMA_PREV_IS2000_RELA: return QCDM_CDMA_PREV_IS2000_RELA; default: break; } return QCDM_CDMA_PREV_UNKNOWN; } static u_int8_t cdma_band_class_to_qcdm (u_int8_t cdma) { switch (cdma) { case CDMA_BAND_CLASS_0_CELLULAR_800: return QCDM_CDMA_BAND_CLASS_0_CELLULAR_800; case CDMA_BAND_CLASS_1_PCS: return QCDM_CDMA_BAND_CLASS_1_PCS; case CDMA_BAND_CLASS_2_TACS: return QCDM_CDMA_BAND_CLASS_2_TACS; case CDMA_BAND_CLASS_3_JTACS: return QCDM_CDMA_BAND_CLASS_3_JTACS; case CDMA_BAND_CLASS_4_KOREAN_PCS: return QCDM_CDMA_BAND_CLASS_4_KOREAN_PCS; case CDMA_BAND_CLASS_5_NMT450: return QCDM_CDMA_BAND_CLASS_5_NMT450; case CDMA_BAND_CLASS_6_IMT2000: return QCDM_CDMA_BAND_CLASS_6_IMT2000; case CDMA_BAND_CLASS_7_CELLULAR_700: return QCDM_CDMA_BAND_CLASS_7_CELLULAR_700; case CDMA_BAND_CLASS_8_1800: return QCDM_CDMA_BAND_CLASS_8_1800; case CDMA_BAND_CLASS_9_900: return QCDM_CDMA_BAND_CLASS_9_900; case CDMA_BAND_CLASS_10_SECONDARY_800: return QCDM_CDMA_BAND_CLASS_10_SECONDARY_800; case CDMA_BAND_CLASS_11_PAMR_400: return QCDM_CDMA_BAND_CLASS_11_PAMR_400; case CDMA_BAND_CLASS_12_PAMR_800: return QCDM_CDMA_BAND_CLASS_12_PAMR_800; case CDMA_BAND_CLASS_13_IMT2000_2500: return QCDM_CDMA_BAND_CLASS_13_IMT2000_2500; case CDMA_BAND_CLASS_14_US_PCS_1900: return QCDM_CDMA_BAND_CLASS_14_US_PCS_1900; case CDMA_BAND_CLASS_15_AWS: return QCDM_CDMA_BAND_CLASS_15_AWS; case CDMA_BAND_CLASS_16_US_2500: return QCDM_CDMA_BAND_CLASS_16_US_2500; case CDMA_BAND_CLASS_17_US_FLO_2500: return QCDM_CDMA_BAND_CLASS_17_US_FLO_2500; case CDMA_BAND_CLASS_18_US_PS_700: return QCDM_CDMA_BAND_CLASS_18_US_PS_700; case CDMA_BAND_CLASS_19_US_LOWER_700: return QCDM_CDMA_BAND_CLASS_19_US_LOWER_700; default: break; } return QCDM_CDMA_BAND_CLASS_UNKNOWN; } /**********************************************************************/ /* * utils_bin2hexstr * * Convert a byte-array into a hexadecimal string. * * Code originally by Alex Larsson and * copyright Red Hat, Inc. under terms of the LGPL. * */ static char * bin2hexstr (const u_int8_t *bytes, int len) { static char hex_digits[] = "0123456789abcdef"; char *result; int i; size_t buflen = (len * 2) + 1; qcdm_return_val_if_fail (bytes != NULL, NULL); qcdm_return_val_if_fail (len > 0, NULL); qcdm_return_val_if_fail (len < 4096, NULL); /* Arbitrary limit */ result = calloc (1, buflen); if (result == NULL) return NULL; for (i = 0; i < len; i++) { result[2*i] = hex_digits[(bytes[i] >> 4) & 0xf]; result[2*i+1] = hex_digits[bytes[i] & 0xf]; } result[buflen - 1] = '\0'; return result; } /**********************************************************************/ static qcdmbool check_command (const char *buf, size_t len, u_int8_t cmd, size_t min_len, int *out_error) { if (len < 1) { qcdm_err (0, "DM command response malformed (must be at least 1 byte in length)"); if (out_error) *out_error = -QCDM_ERROR_RESPONSE_MALFORMED; return FALSE; } switch (buf[0]) { case DIAG_CMD_BAD_CMD: qcdm_err (0, "DM command %d unknown or unimplemented by the device", cmd); if (out_error) *out_error = -QCDM_ERROR_RESPONSE_BAD_COMMAND; return FALSE; case DIAG_CMD_BAD_PARM: qcdm_err (0, "DM command %d contained invalid parameter", cmd); if (out_error) *out_error = -QCDM_ERROR_RESPONSE_BAD_PARAMETER; return FALSE; case DIAG_CMD_BAD_LEN: qcdm_err (0, "DM command %d was the wrong size", cmd); if (out_error) *out_error = -QCDM_ERROR_RESPONSE_BAD_LENGTH; return FALSE; case DIAG_CMD_BAD_DEV: qcdm_err (0, "DM command %d was not accepted by the device", cmd); if (out_error) *out_error = -QCDM_ERROR_RESPONSE_NOT_ACCEPTED; return FALSE; case DIAG_CMD_BAD_MODE: qcdm_err (0, "DM command %d not allowed in the current device mode", cmd); if (out_error) *out_error = -QCDM_ERROR_RESPONSE_BAD_MODE; return FALSE; case DIAG_CMD_BAD_SPC_MODE: qcdm_err (0, "DM command %d not allowed because the Service Programming Code is locked", cmd); if (out_error) *out_error = -QCDM_ERROR_SPC_LOCKED; return FALSE; default: break; } if (buf[0] != cmd) { qcdm_err (0, "Unexpected DM command response (expected %d, got %d)", cmd, buf[0]); if (out_error) *out_error = -QCDM_ERROR_RESPONSE_UNEXPECTED; return FALSE; } if (len < min_len) { qcdm_err (0, "DM command %d response not long enough (got %zu, expected " "at least %zu).", cmd, len, min_len); if (out_error) *out_error = -QCDM_ERROR_RESPONSE_BAD_LENGTH; return FALSE; } return TRUE; } static int nv_status_to_qcdm_error (u_int16_t status) { switch (status) { case DIAG_NV_STATUS_OK: return QCDM_SUCCESS; case DIAG_NV_STATUS_BUSY: return -QCDM_ERROR_NV_ERROR_BUSY; case DIAG_NV_STATUS_BAD_COMMAND: return -QCDM_ERROR_NV_ERROR_BAD_COMMAND; case DIAG_NV_STATUS_MEMORY_FULL: return -QCDM_ERROR_NV_ERROR_MEMORY_FULL; case DIAG_NV_STATUS_FAILED: return -QCDM_ERROR_NV_ERROR_FAILED; case DIAG_NV_STATUS_INACTIVE: return -QCDM_ERROR_NV_ERROR_INACTIVE; case DIAG_NV_STATUS_BAD_PARAMETER: return -QCDM_ERROR_NV_ERROR_BAD_PARAMETER; case DIAG_NV_STATUS_READ_ONLY: return -QCDM_ERROR_NV_ERROR_READ_ONLY; default: return -QCDM_ERROR_NVCMD_FAILED; } } static qcdmbool check_nv_cmd (DMCmdNVReadWrite *cmd, u_int16_t nv_item, int *out_error) { u_int16_t cmd_item; qcdm_return_val_if_fail (cmd != NULL, FALSE); qcdm_return_val_if_fail ((cmd->code == DIAG_CMD_NV_READ) || (cmd->code == DIAG_CMD_NV_WRITE), FALSE); /* NV read/write have a status byte at the end */ if (cmd->status != 0) { qcdm_err (0, "The NV operation failed (status 0x%X).", le16toh (cmd->status)); if (out_error) *out_error = nv_status_to_qcdm_error (le16toh (cmd->status)); return FALSE; } cmd_item = le16toh (cmd->nv_item); if (cmd_item != nv_item) { qcdm_err (0, "Unexpected DM NV command response (expected item %d, got " "item %d)", nv_item, cmd_item); if (out_error) *out_error = -QCDM_ERROR_RESPONSE_UNEXPECTED; return FALSE; } return TRUE; } /**********************************************************************/ size_t qcdm_cmd_version_info_new (char *buf, size_t len) { char cmdbuf[3]; DMCmdHeader *cmd = (DMCmdHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_VERSION_INFO; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_version_info_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdVersionInfoRsp *rsp = (DMCmdVersionInfoRsp *) buf; char tmp[12]; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_VERSION_INFO, sizeof (DMCmdVersionInfoRsp), out_error)) return NULL; result = qcdm_result_new (); memset (tmp, 0, sizeof (tmp)); qcdm_assert (sizeof (rsp->comp_date) <= sizeof (tmp)); memcpy (tmp, rsp->comp_date, sizeof (rsp->comp_date)); qcdm_result_add_string (result, QCDM_CMD_VERSION_INFO_ITEM_COMP_DATE, tmp); memset (tmp, 0, sizeof (tmp)); qcdm_assert (sizeof (rsp->comp_time) <= sizeof (tmp)); memcpy (tmp, rsp->comp_time, sizeof (rsp->comp_time)); qcdm_result_add_string (result, QCDM_CMD_VERSION_INFO_ITEM_COMP_TIME, tmp); memset (tmp, 0, sizeof (tmp)); qcdm_assert (sizeof (rsp->rel_date) <= sizeof (tmp)); memcpy (tmp, rsp->rel_date, sizeof (rsp->rel_date)); qcdm_result_add_string (result, QCDM_CMD_VERSION_INFO_ITEM_RELEASE_DATE, tmp); memset (tmp, 0, sizeof (tmp)); qcdm_assert (sizeof (rsp->rel_time) <= sizeof (tmp)); memcpy (tmp, rsp->rel_time, sizeof (rsp->rel_time)); qcdm_result_add_string (result, QCDM_CMD_VERSION_INFO_ITEM_RELEASE_TIME, tmp); memset (tmp, 0, sizeof (tmp)); qcdm_assert (sizeof (rsp->model) <= sizeof (tmp)); memcpy (tmp, rsp->model, sizeof (rsp->model)); qcdm_result_add_string (result, QCDM_CMD_VERSION_INFO_ITEM_MODEL, tmp); return result; } /**********************************************************************/ size_t qcdm_cmd_esn_new (char *buf, size_t len) { char cmdbuf[3]; DMCmdHeader *cmd = (DMCmdHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_ESN; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_esn_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdEsnRsp *rsp = (DMCmdEsnRsp *) buf; char *tmp; u_int8_t swapped[4]; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_ESN, sizeof (DMCmdEsnRsp), out_error)) return NULL; /* Convert the ESN from binary to a hex string; it's LE so we have to * swap it to get the correct ordering. */ swapped[0] = rsp->esn[3]; swapped[1] = rsp->esn[2]; swapped[2] = rsp->esn[1]; swapped[3] = rsp->esn[0]; tmp = bin2hexstr (&swapped[0], sizeof (swapped)); if (tmp != NULL) { result = qcdm_result_new (); qcdm_result_add_string (result, QCDM_CMD_ESN_ITEM_ESN, tmp); free (tmp); } return result; } /**********************************************************************/ size_t qcdm_cmd_cdma_status_new (char *buf, size_t len) { char cmdbuf[3]; DMCmdHeader *cmd = (DMCmdHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_STATUS; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_cdma_status_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdStatusRsp *rsp = (DMCmdStatusRsp *) buf; char *tmp; u_int8_t swapped[4]; u_int32_t tmp_num; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_STATUS, sizeof (DMCmdStatusRsp), out_error)) return NULL; result = qcdm_result_new (); /* Convert the ESN from binary to a hex string; it's LE so we have to * swap it to get the correct ordering. */ swapped[0] = rsp->esn[3]; swapped[1] = rsp->esn[2]; swapped[2] = rsp->esn[1]; swapped[3] = rsp->esn[0]; tmp = bin2hexstr (&swapped[0], sizeof (swapped)); qcdm_result_add_string (result, QCDM_CMD_CDMA_STATUS_ITEM_ESN, tmp); free (tmp); tmp_num = (u_int32_t) le16toh (rsp->rf_mode); qcdm_result_add_u32 (result, QCDM_CMD_CDMA_STATUS_ITEM_RF_MODE, tmp_num); tmp_num = (u_int32_t) le16toh (rsp->cdma_rx_state); qcdm_result_add_u32 (result, QCDM_CMD_CDMA_STATUS_ITEM_RX_STATE, tmp_num); tmp_num = (u_int32_t) le16toh (rsp->entry_reason); qcdm_result_add_u32 (result, QCDM_CMD_CDMA_STATUS_ITEM_ENTRY_REASON, tmp_num); tmp_num = (u_int32_t) le16toh (rsp->curr_chan); qcdm_result_add_u32 (result, QCDM_CMD_CDMA_STATUS_ITEM_CURRENT_CHANNEL, tmp_num); qcdm_result_add_u8 (result, QCDM_CMD_CDMA_STATUS_ITEM_CODE_CHANNEL, rsp->cdma_code_chan); tmp_num = (u_int32_t) le16toh (rsp->pilot_base); qcdm_result_add_u32 (result, QCDM_CMD_CDMA_STATUS_ITEM_PILOT_BASE, tmp_num); tmp_num = (u_int32_t) le16toh (rsp->sid); qcdm_result_add_u32 (result, QCDM_CMD_CDMA_STATUS_ITEM_SID, tmp_num); tmp_num = (u_int32_t) le16toh (rsp->nid); qcdm_result_add_u32 (result, QCDM_CMD_CDMA_STATUS_ITEM_NID, tmp_num); return result; } /**********************************************************************/ size_t qcdm_cmd_sw_version_new (char *buf, size_t len) { char cmdbuf[3]; DMCmdHeader *cmd = (DMCmdHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_SW_VERSION; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_sw_version_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdSwVersionRsp *rsp = (DMCmdSwVersionRsp *) buf; char tmp[32]; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_SW_VERSION, sizeof (*rsp), out_error)) return NULL; result = qcdm_result_new (); memset (tmp, 0, sizeof (tmp)); qcdm_assert (sizeof (rsp->version) <= sizeof (tmp)); memcpy (tmp, rsp->version, sizeof (rsp->version)); qcdm_result_add_string (result, QCDM_CMD_SW_VERSION_ITEM_VERSION, tmp); memset (tmp, 0, sizeof (tmp)); qcdm_assert (sizeof (rsp->comp_date) <= sizeof (tmp)); memcpy (tmp, rsp->comp_date, sizeof (rsp->comp_date)); qcdm_result_add_string (result, QCDM_CMD_SW_VERSION_ITEM_COMP_DATE, tmp); memset (tmp, 0, sizeof (tmp)); qcdm_assert (sizeof (rsp->comp_time) <= sizeof (tmp)); memcpy (tmp, rsp->comp_time, sizeof (rsp->comp_time)); qcdm_result_add_string (result, QCDM_CMD_SW_VERSION_ITEM_COMP_TIME, tmp); return result; } /**********************************************************************/ size_t qcdm_cmd_status_snapshot_new (char *buf, size_t len) { char cmdbuf[3]; DMCmdHeader *cmd = (DMCmdHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_STATUS_SNAPSHOT; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } static u_int8_t snapshot_state_to_qcdm (u_int8_t cdma_state) { /* CDMA_STATUS_SNAPSHOT_STATE_* -> QCDM_STATUS_SNAPSHOT_STATE_* */ return cdma_state + 1; } QcdmResult * qcdm_cmd_status_snapshot_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdStatusSnapshotRsp *rsp = (DMCmdStatusSnapshotRsp *) buf; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_STATUS_SNAPSHOT, sizeof (*rsp), out_error)) return NULL; result = qcdm_result_new (); qcdm_result_add_u8 (result, QCDM_CMD_STATUS_SNAPSHOT_ITEM_BAND_CLASS, cdma_band_class_to_qcdm (rsp->band_class)); qcdm_result_add_u8 (result, QCDM_CMD_STATUS_SNAPSHOT_ITEM_BASE_STATION_PREV, cdma_prev_to_qcdm (rsp->prev)); qcdm_result_add_u8 (result, QCDM_CMD_STATUS_SNAPSHOT_ITEM_MOBILE_PREV, cdma_prev_to_qcdm (rsp->mob_prev)); qcdm_result_add_u8 (result, QCDM_CMD_STATUS_SNAPSHOT_ITEM_PREV_IN_USE, cdma_prev_to_qcdm (rsp->prev_in_use)); qcdm_result_add_u8 (result, QCDM_CMD_STATUS_SNAPSHOT_ITEM_STATE, snapshot_state_to_qcdm (rsp->state & 0xF)); return result; } /**********************************************************************/ size_t qcdm_cmd_pilot_sets_new (char *buf, size_t len) { char cmdbuf[3]; DMCmdHeader *cmd = (DMCmdHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_PILOT_SETS; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } #define PILOT_SETS_CMD_ACTIVE_SET "active-set" #define PILOT_SETS_CMD_CANDIDATE_SET "candidate-set" #define PILOT_SETS_CMD_NEIGHBOR_SET "neighbor-set" static const char * set_num_to_str (u_int32_t num) { if (num == QCDM_CMD_PILOT_SETS_TYPE_ACTIVE) return PILOT_SETS_CMD_ACTIVE_SET; if (num == QCDM_CMD_PILOT_SETS_TYPE_CANDIDATE) return PILOT_SETS_CMD_CANDIDATE_SET; if (num == QCDM_CMD_PILOT_SETS_TYPE_NEIGHBOR) return PILOT_SETS_CMD_NEIGHBOR_SET; return NULL; } QcdmResult * qcdm_cmd_pilot_sets_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdPilotSetsRsp *rsp = (DMCmdPilotSetsRsp *) buf; size_t sets_len; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_PILOT_SETS, sizeof (DMCmdPilotSetsRsp), out_error)) return NULL; result = qcdm_result_new (); sets_len = rsp->active_count * sizeof (DMCmdPilotSetsSet); if (sets_len > 0) { qcdm_result_add_u8_array (result, PILOT_SETS_CMD_ACTIVE_SET, (const u_int8_t *) &rsp->sets[0], sets_len); } sets_len = rsp->candidate_count * sizeof (DMCmdPilotSetsSet); if (sets_len > 0) { qcdm_result_add_u8_array (result, PILOT_SETS_CMD_ACTIVE_SET, (const u_int8_t *) &rsp->sets[rsp->active_count], sets_len); } sets_len = rsp->neighbor_count * sizeof (DMCmdPilotSetsSet); if (sets_len > 0) { qcdm_result_add_u8_array (result, PILOT_SETS_CMD_ACTIVE_SET, (const u_int8_t *) &rsp->sets[rsp->active_count + rsp->candidate_count], sets_len); } return result; } qcdmbool qcdm_cmd_pilot_sets_result_get_num (QcdmResult *result, u_int32_t set_type, u_int32_t *out_num) { const char *set_name; const u_int8_t *array = NULL; size_t array_len = 0; qcdm_return_val_if_fail (result != NULL, FALSE); set_name = set_num_to_str (set_type); qcdm_return_val_if_fail (set_name != NULL, FALSE); if (!qcdm_result_get_u8_array (result, set_name, &array, &array_len)) return FALSE; *out_num = array_len / sizeof (DMCmdPilotSetsSet); return TRUE; } qcdmbool qcdm_cmd_pilot_sets_result_get_pilot (QcdmResult *result, u_int32_t set_type, u_int32_t num, u_int32_t *out_pn_offset, u_int32_t *out_ecio, float *out_db) { const char *set_name; DMCmdPilotSetsSet *set; const u_int8_t *array = NULL; size_t array_len = 0; qcdm_return_val_if_fail (result != NULL, FALSE); set_name = set_num_to_str (set_type); qcdm_return_val_if_fail (set_name != NULL, FALSE); if (!qcdm_result_get_u8_array (result, set_name, &array, &array_len)) return FALSE; qcdm_return_val_if_fail (num < array_len / sizeof (DMCmdPilotSetsSet), FALSE); set = (DMCmdPilotSetsSet *) &array[num * sizeof (DMCmdPilotSetsSet)]; *out_pn_offset = set->pn_offset; *out_ecio = set->ecio; /* EC/IO is in units of -0.5 dB per the specs */ *out_db = (float) set->ecio * -0.5; return TRUE; } /**********************************************************************/ size_t qcdm_cmd_nv_get_mdn_new (char *buf, size_t len, u_int8_t profile) { char cmdbuf[sizeof (DMCmdNVReadWrite) + 2]; DMCmdNVReadWrite *cmd = (DMCmdNVReadWrite *) &cmdbuf[0]; DMNVItemMdn *req; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_NV_READ; cmd->nv_item = htole16 (DIAG_NV_DIR_NUMBER); req = (DMNVItemMdn *) &cmd->data[0]; req->profile = profile; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_nv_get_mdn_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdNVReadWrite *rsp = (DMCmdNVReadWrite *) buf; DMNVItemMdn *mdn; char tmp[11]; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_NV_READ, sizeof (DMCmdNVReadWrite), out_error)) return NULL; if (!check_nv_cmd (rsp, DIAG_NV_DIR_NUMBER, out_error)) return NULL; mdn = (DMNVItemMdn *) &rsp->data[0]; result = qcdm_result_new (); qcdm_result_add_u8 (result, QCDM_CMD_NV_GET_MDN_ITEM_PROFILE, mdn->profile); memset (tmp, 0, sizeof (tmp)); qcdm_assert (sizeof (mdn->mdn) <= sizeof (tmp)); memcpy (tmp, mdn->mdn, sizeof (mdn->mdn)); qcdm_result_add_string (result, QCDM_CMD_NV_GET_MDN_ITEM_MDN, tmp); return result; } /**********************************************************************/ static qcdmbool roam_pref_validate (u_int8_t dm) { if ( dm == DIAG_NV_ROAM_PREF_HOME_ONLY || dm == DIAG_NV_ROAM_PREF_ROAM_ONLY || dm == DIAG_NV_ROAM_PREF_AUTO) return TRUE; return FALSE; } size_t qcdm_cmd_nv_get_roam_pref_new (char *buf, size_t len, u_int8_t profile) { char cmdbuf[sizeof (DMCmdNVReadWrite) + 2]; DMCmdNVReadWrite *cmd = (DMCmdNVReadWrite *) &cmdbuf[0]; DMNVItemRoamPref *req; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_NV_READ; cmd->nv_item = htole16 (DIAG_NV_ROAM_PREF); req = (DMNVItemRoamPref *) &cmd->data[0]; req->profile = profile; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_nv_get_roam_pref_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdNVReadWrite *rsp = (DMCmdNVReadWrite *) buf; DMNVItemRoamPref *roam; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_NV_READ, sizeof (DMCmdNVReadWrite), out_error)) return NULL; if (!check_nv_cmd (rsp, DIAG_NV_ROAM_PREF, out_error)) return NULL; roam = (DMNVItemRoamPref *) &rsp->data[0]; if (!roam_pref_validate (roam->roam_pref)) { qcdm_err (0, "Unknown roam preference 0x%X", roam->roam_pref); return NULL; } result = qcdm_result_new (); qcdm_result_add_u8 (result, QCDM_CMD_NV_GET_ROAM_PREF_ITEM_PROFILE, roam->profile); qcdm_result_add_u8 (result, QCDM_CMD_NV_GET_ROAM_PREF_ITEM_ROAM_PREF, roam->roam_pref); return result; } size_t qcdm_cmd_nv_set_roam_pref_new (char *buf, size_t len, u_int8_t profile, u_int8_t roam_pref) { char cmdbuf[sizeof (DMCmdNVReadWrite) + 2]; DMCmdNVReadWrite *cmd = (DMCmdNVReadWrite *) &cmdbuf[0]; DMNVItemRoamPref *req; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); if (!roam_pref_validate (roam_pref)) { qcdm_err (0, "Invalid roam preference %d", roam_pref); return 0; } memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_NV_WRITE; cmd->nv_item = htole16 (DIAG_NV_ROAM_PREF); req = (DMNVItemRoamPref *) &cmd->data[0]; req->profile = profile; req->roam_pref = roam_pref; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_nv_set_roam_pref_result (const char *buf, size_t len, int *out_error) { qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_NV_WRITE, sizeof (DMCmdNVReadWrite), out_error)) return NULL; if (!check_nv_cmd ((DMCmdNVReadWrite *) buf, DIAG_NV_ROAM_PREF, out_error)) return NULL; return qcdm_result_new (); } /**********************************************************************/ static qcdmbool mode_pref_validate (u_int8_t dm) { switch (dm) { case DIAG_NV_MODE_PREF_DIGITAL: case DIAG_NV_MODE_PREF_DIGITAL_ONLY: case DIAG_NV_MODE_PREF_AUTO: case DIAG_NV_MODE_PREF_1X_ONLY: case DIAG_NV_MODE_PREF_HDR_ONLY: case DIAG_NV_MODE_PREF_1X_HDR_ONLY: case DIAG_NV_MODE_PREF_LTE_ONLY: case DIAG_NV_MODE_PREF_1X_HDR_LTE_ONLY: return TRUE; default: return FALSE; } } size_t qcdm_cmd_nv_get_mode_pref_new (char *buf, size_t len, u_int8_t profile) { char cmdbuf[sizeof (DMCmdNVReadWrite) + 2]; DMCmdNVReadWrite *cmd = (DMCmdNVReadWrite *) &cmdbuf[0]; DMNVItemModePref *req; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_NV_READ; cmd->nv_item = htole16 (DIAG_NV_MODE_PREF); req = (DMNVItemModePref *) &cmd->data[0]; req->profile = profile; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_nv_get_mode_pref_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdNVReadWrite *rsp = (DMCmdNVReadWrite *) buf; DMNVItemModePref *mode; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_NV_READ, sizeof (DMCmdNVReadWrite), out_error)) return NULL; if (!check_nv_cmd (rsp, DIAG_NV_MODE_PREF, out_error)) return NULL; mode = (DMNVItemModePref *) &rsp->data[0]; if (!mode_pref_validate (mode->mode_pref)) qcdm_warn (0, "Unknown mode preference 0x%X", mode->mode_pref); result = qcdm_result_new (); qcdm_result_add_u8 (result, QCDM_CMD_NV_GET_MODE_PREF_ITEM_PROFILE, mode->profile); qcdm_result_add_u8 (result, QCDM_CMD_NV_GET_MODE_PREF_ITEM_MODE_PREF, mode->mode_pref); return result; } size_t qcdm_cmd_nv_set_mode_pref_new (char *buf, size_t len, u_int8_t profile, u_int8_t mode_pref) { char cmdbuf[sizeof (DMCmdNVReadWrite) + 2]; DMCmdNVReadWrite *cmd = (DMCmdNVReadWrite *) &cmdbuf[0]; DMNVItemModePref *req; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); if (!mode_pref_validate (mode_pref)) { qcdm_err (0, "Invalid mode preference %d", mode_pref); return 0; } memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_NV_WRITE; cmd->nv_item = htole16 (DIAG_NV_MODE_PREF); req = (DMNVItemModePref *) &cmd->data[0]; req->profile = profile; req->mode_pref = mode_pref; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_nv_set_mode_pref_result (const char *buf, size_t len, int *out_error) { qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_NV_WRITE, sizeof (DMCmdNVReadWrite), out_error)) return NULL; if (!check_nv_cmd ((DMCmdNVReadWrite *) buf, DIAG_NV_MODE_PREF, out_error)) return NULL; return qcdm_result_new (); } /**********************************************************************/ static qcdmbool hdr_rev_pref_validate (u_int8_t dm) { if ( dm == DIAG_NV_HDR_REV_PREF_0 || dm == DIAG_NV_HDR_REV_PREF_A || dm == DIAG_NV_HDR_REV_PREF_EHRPD) return TRUE; return FALSE; } size_t qcdm_cmd_nv_get_hdr_rev_pref_new (char *buf, size_t len) { char cmdbuf[sizeof (DMCmdNVReadWrite) + 2]; DMCmdNVReadWrite *cmd = (DMCmdNVReadWrite *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_NV_READ; cmd->nv_item = htole16 (DIAG_NV_HDR_REV_PREF); return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_nv_get_hdr_rev_pref_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdNVReadWrite *rsp = (DMCmdNVReadWrite *) buf; DMNVItemHdrRevPref *rev; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_NV_READ, sizeof (DMCmdNVReadWrite), out_error)) return NULL; if (!check_nv_cmd (rsp, DIAG_NV_HDR_REV_PREF, out_error)) return NULL; rev = (DMNVItemHdrRevPref *) &rsp->data[0]; if (!hdr_rev_pref_validate (rev->rev_pref)) { qcdm_err (0, "Unknown HDR revision preference 0x%X", rev->rev_pref); return NULL; } result = qcdm_result_new (); qcdm_result_add_u8 (result, QCDM_CMD_NV_GET_HDR_REV_PREF_ITEM_REV_PREF, rev->rev_pref); return result; } size_t qcdm_cmd_nv_set_hdr_rev_pref_new (char *buf, size_t len, u_int8_t rev_pref) { char cmdbuf[sizeof (DMCmdNVReadWrite) + 2]; DMCmdNVReadWrite *cmd = (DMCmdNVReadWrite *) &cmdbuf[0]; DMNVItemHdrRevPref *req; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); if (!hdr_rev_pref_validate (rev_pref)) { qcdm_err (0, "Invalid HDR revision preference %d", rev_pref); return 0; } memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_NV_WRITE; cmd->nv_item = htole16 (DIAG_NV_HDR_REV_PREF); req = (DMNVItemHdrRevPref *) &cmd->data[0]; req->rev_pref = rev_pref; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_nv_set_hdr_rev_pref_result (const char *buf, size_t len, int *out_error) { qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_NV_WRITE, sizeof (DMCmdNVReadWrite), out_error)) return NULL; if (!check_nv_cmd ((DMCmdNVReadWrite *) buf, DIAG_NV_HDR_REV_PREF, out_error)) return NULL; return qcdm_result_new (); } /**********************************************************************/ size_t qcdm_cmd_cm_subsys_state_info_new (char *buf, size_t len) { char cmdbuf[sizeof (DMCmdSubsysHeader) + 2]; DMCmdSubsysHeader *cmd = (DMCmdSubsysHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_SUBSYS; cmd->subsys_id = DIAG_SUBSYS_CM; cmd->subsys_cmd = htole16 (DIAG_SUBSYS_CM_STATE_INFO); return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_cm_subsys_state_info_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdSubsysCMStateInfoRsp *rsp = (DMCmdSubsysCMStateInfoRsp *) buf; u_int32_t tmp_num; u_int32_t roam_pref; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_SUBSYS, sizeof (DMCmdSubsysCMStateInfoRsp), out_error)) return NULL; roam_pref = (u_int32_t) le32toh (rsp->roam_pref); if (!roam_pref_validate (roam_pref)) { qcdm_err (0, "Unknown roam preference 0x%X", roam_pref); return NULL; } result = qcdm_result_new (); tmp_num = (u_int32_t) le32toh (rsp->call_state); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_CALL_STATE, tmp_num); tmp_num = (u_int32_t) le32toh (rsp->oper_mode); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_OPERATING_MODE, tmp_num); tmp_num = (u_int32_t) le32toh (rsp->system_mode); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_SYSTEM_MODE, tmp_num); tmp_num = (u_int32_t) le32toh (rsp->mode_pref); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_MODE_PREF, tmp_num); tmp_num = (u_int32_t) le32toh (rsp->band_pref); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_BAND_PREF, tmp_num); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_ROAM_PREF, roam_pref); tmp_num = (u_int32_t) le32toh (rsp->srv_domain_pref); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_SERVICE_DOMAIN_PREF, tmp_num); tmp_num = (u_int32_t) le32toh (rsp->acq_order_pref); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_ACQ_ORDER_PREF, tmp_num); tmp_num = (u_int32_t) le32toh (rsp->hybrid_pref); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_HYBRID_PREF, tmp_num); tmp_num = (u_int32_t) le32toh (rsp->network_sel_mode_pref); qcdm_result_add_u32 (result, QCDM_CMD_CM_SUBSYS_STATE_INFO_ITEM_NETWORK_SELECTION_PREF, tmp_num); return result; } /**********************************************************************/ size_t qcdm_cmd_hdr_subsys_state_info_new (char *buf, size_t len) { char cmdbuf[sizeof (DMCmdSubsysHeader) + 2]; DMCmdSubsysHeader *cmd = (DMCmdSubsysHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_SUBSYS; cmd->subsys_id = DIAG_SUBSYS_HDR; cmd->subsys_cmd = htole16 (DIAG_SUBSYS_HDR_STATE_INFO); return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_hdr_subsys_state_info_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdSubsysHDRStateInfoRsp *rsp = (DMCmdSubsysHDRStateInfoRsp *) buf; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_SUBSYS, sizeof (DMCmdSubsysHDRStateInfoRsp), out_error)) return NULL; result = qcdm_result_new (); qcdm_result_add_u8 (result, QCDM_CMD_HDR_SUBSYS_STATE_INFO_ITEM_AT_STATE, rsp->at_state); qcdm_result_add_u8 (result, QCDM_CMD_HDR_SUBSYS_STATE_INFO_ITEM_SESSION_STATE, rsp->session_state); qcdm_result_add_u8 (result, QCDM_CMD_HDR_SUBSYS_STATE_INFO_ITEM_ALMP_STATE, rsp->almp_state); qcdm_result_add_u8 (result, QCDM_CMD_HDR_SUBSYS_STATE_INFO_ITEM_INIT_STATE, rsp->init_state); qcdm_result_add_u8 (result, QCDM_CMD_HDR_SUBSYS_STATE_INFO_ITEM_IDLE_STATE, rsp->idle_state); qcdm_result_add_u8 (result, QCDM_CMD_HDR_SUBSYS_STATE_INFO_ITEM_CONNECTED_STATE, rsp->connected_state); qcdm_result_add_u8 (result, QCDM_CMD_HDR_SUBSYS_STATE_INFO_ITEM_ROUTE_UPDATE_STATE, rsp->route_update_state); qcdm_result_add_u8 (result, QCDM_CMD_HDR_SUBSYS_STATE_INFO_ITEM_OVERHEAD_MSG_STATE, rsp->overhead_msg_state); qcdm_result_add_u8 (result, QCDM_CMD_HDR_SUBSYS_STATE_INFO_ITEM_HDR_HYBRID_MODE, rsp->hdr_hybrid_mode); return result; } /**********************************************************************/ size_t qcdm_cmd_ext_logmask_new (char *buf, size_t len, u_int32_t items[], u_int16_t maxlog) { char cmdbuf[sizeof (DMCmdExtLogMask) + 2]; DMCmdExtLogMask *cmd = (DMCmdExtLogMask *) &cmdbuf[0]; u_int16_t highest = 0; size_t total = 3; u_int32_t i; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_EXT_LOGMASK; if (items) { for (i = 0; items[i] > 0; i++) { qcdm_warn_if_fail (items[i] > 0); qcdm_warn_if_fail (items[i] < 4095); cmd->mask[items[i] / 8] |= 1 << items[i] % 8; if (items[i] > highest) highest = items[i]; } } qcdm_return_val_if_fail (highest <= maxlog, 0); cmd->len = htole16 (maxlog); total += maxlog / 8; if (maxlog && maxlog % 8) total++; return dm_encapsulate_buffer (cmdbuf, total, sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_ext_logmask_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdExtLogMask *rsp = (DMCmdExtLogMask *) buf; u_int32_t masklen = 0, maxlog = 0; size_t minlen = 0; qcdm_return_val_if_fail (buf != NULL, NULL); /* Ensure size is at least enough for the command header */ if (len < 1) { qcdm_err (0, "DM command %d response not long enough (got %zu, expected " "at least %d).", DIAG_CMD_EXT_LOGMASK, len, 3); return FALSE; } /* Result of a 'set' operation will be only 1 byte in size; result of * a 'get' operation (ie setting len to 0x0000 in the request) will be * the size of the header (3) plus the max log length. */ if (len == 1) minlen = 1; else { /* Ensure size is equal to max # of log items + 3 */ maxlog = le16toh (rsp->len); masklen = maxlog / 8; if (maxlog % 8) masklen++; if (len < (masklen + 3)) { qcdm_err (0, "DM command %d response not long enough (got %zu, expected " "at least %d).", DIAG_CMD_EXT_LOGMASK, len, masklen + 3); return FALSE; } minlen = masklen + 3; } if (!check_command (buf, len, DIAG_CMD_EXT_LOGMASK, minlen, out_error)) return NULL; result = qcdm_result_new (); if (minlen != 4) qcdm_result_add_u32 (result, QCDM_CMD_EXT_LOGMASK_ITEM_MAX_ITEMS, maxlog); return result; } qcdmbool qcmd_cmd_ext_logmask_result_get_item (QcdmResult *result, u_int16_t item) { return FALSE; } /**********************************************************************/ size_t qcdm_cmd_event_report_new (char *buf, size_t len, qcdmbool start) { char cmdbuf[4]; DMCmdEventReport *cmd = (DMCmdEventReport *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_EVENT_REPORT; cmd->on = start ? 1 : 0; return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_event_report_result (const char *buf, size_t len, int *out_error) { qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_EVENT_REPORT, sizeof (DMCmdEventReport), out_error)) return NULL; return qcdm_result_new (); } /**********************************************************************/ size_t qcdm_cmd_zte_subsys_status_new (char *buf, size_t len) { char cmdbuf[sizeof (DMCmdSubsysHeader) + 2]; DMCmdSubsysHeader *cmd = (DMCmdSubsysHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_SUBSYS; cmd->subsys_id = DIAG_SUBSYS_ZTE; cmd->subsys_cmd = htole16 (DIAG_SUBSYS_ZTE_STATUS); return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_zte_subsys_status_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdSubsysZteStatusRsp *rsp = (DMCmdSubsysZteStatusRsp *) buf; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_SUBSYS, sizeof (DMCmdSubsysZteStatusRsp), out_error)) return NULL; result = qcdm_result_new (); qcdm_result_add_u8 (result, QCDM_CMD_ZTE_SUBSYS_STATUS_ITEM_SIGNAL_INDICATOR, rsp->signal_ind); return result; } /**********************************************************************/ size_t qcdm_cmd_nw_subsys_modem_snapshot_cdma_new (char *buf, size_t len, u_int8_t chipset) { char cmdbuf[sizeof (DMCmdSubsysNwSnapshotReq) + 2]; DMCmdSubsysNwSnapshotReq *cmd = (DMCmdSubsysNwSnapshotReq *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); /* Validate chipset */ if (chipset != QCDM_NW_CHIPSET_6500 && chipset != QCDM_NW_CHIPSET_6800) { qcdm_err (0, "Unknown Novatel chipset 0x%X", chipset); return 0; } memset (cmd, 0, sizeof (*cmd)); cmd->hdr.code = DIAG_CMD_SUBSYS; switch (chipset) { case QCDM_NW_CHIPSET_6500: cmd->hdr.subsys_id = DIAG_SUBSYS_NW_CONTROL_6500; break; case QCDM_NW_CHIPSET_6800: cmd->hdr.subsys_id = DIAG_SUBSYS_NW_CONTROL_6800; break; default: qcdm_assert_not_reached (); } cmd->hdr.subsys_cmd = htole16 (DIAG_SUBSYS_NW_CONTROL_MODEM_SNAPSHOT); cmd->technology = DIAG_SUBSYS_NW_CONTROL_MODEM_SNAPSHOT_TECH_CDMA_EVDO; cmd->snapshot_mask = htole32 (0xFFFF); return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_nw_subsys_modem_snapshot_cdma_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdSubsysNwSnapshotRsp *rsp = (DMCmdSubsysNwSnapshotRsp *) buf; DMCmdSubsysNwSnapshotCdma *cdma = (DMCmdSubsysNwSnapshotCdma *) &rsp->data; u_int32_t num; u_int8_t num8; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_SUBSYS, sizeof (DMCmdSubsysNwSnapshotRsp), out_error)) return NULL; /* FIXME: check response_code when we know what it means */ result = qcdm_result_new (); num = le32toh (cdma->rssi); qcdm_result_add_u32 (result, QCDM_CMD_NW_SUBSYS_MODEM_SNAPSHOT_CDMA_ITEM_RSSI, num); num8 = cdma_prev_to_qcdm (cdma->prev); qcdm_result_add_u8 (result, QCDM_CMD_NW_SUBSYS_MODEM_SNAPSHOT_CDMA_ITEM_PREV, num8); num8 = cdma_band_class_to_qcdm (cdma->band_class); qcdm_result_add_u8 (result, QCDM_CMD_NW_SUBSYS_MODEM_SNAPSHOT_CDMA_ITEM_BAND_CLASS, num8); qcdm_result_add_u8 (result, QCDM_CMD_NW_SUBSYS_MODEM_SNAPSHOT_CDMA_ITEM_ERI, cdma->eri); num8 = QCDM_HDR_REV_UNKNOWN; switch (cdma->hdr_rev) { case 0: num8 = QCDM_HDR_REV_0; break; case 1: num8 = QCDM_HDR_REV_A; break; default: break; } qcdm_result_add_u8 (result, QCDM_CMD_NW_SUBSYS_MODEM_SNAPSHOT_CDMA_ITEM_HDR_REV, num8); return result; } /**********************************************************************/ static size_t qcdm_cmd_log_config_new (char *buf, size_t len, u_int32_t op, u_int32_t equip_id, u_int16_t items[]) { DMCmdLogConfig *cmd; u_int16_t highest = 0; u_int32_t items_len = 0; size_t cmdsize = 0, cmdbufsize; u_int32_t i; u_int16_t log_code; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail ((equip_id & 0xFFF0) == 0, 0); /* Find number of log items */ if (items) { while (items_len < 4095 && items[items_len]) { /* Find highest log item so we can size the items mask */ log_code = items[items_len] & 0x0FFF; if (log_code > highest) highest = log_code; items_len++; } } cmdsize = sizeof (DMCmdLogConfig) + ((highest + 7) / 8); cmdbufsize = cmdsize + DIAG_TRAILER_LEN; qcdm_return_val_if_fail (len >= cmdsize, 0); cmd = calloc (1, cmdbufsize); cmd->code = DIAG_CMD_LOG_CONFIG; cmd->op = htole32 (op); cmd->equipid = htole32 (equip_id); if (items) { /* Set up the bitmask of log items */ for (i = 0; i < items_len; i++) { log_code = items[i] & 0x0FFF; /* Strip off equip ID */ cmd->mask[log_code / 8] |= 1 << log_code % 8; } cmd->num_items = htole32 (highest); } return dm_encapsulate_buffer ((char *) cmd, cmdsize, cmdbufsize, buf, len); } size_t qcdm_cmd_log_config_get_mask_new (char *buf, size_t len, u_int32_t equip_id) { return qcdm_cmd_log_config_new (buf, len, DIAG_CMD_LOG_CONFIG_OP_GET_MASK, equip_id, NULL); } static int check_log_config_respose (const char *buf, size_t len, u_int32_t op) { DMCmdLogConfigRsp *rsp = (DMCmdLogConfigRsp *) buf; size_t minlen = 16; /* minimum valid resposne */ int err; /* Ensure size is at least enough for the command header */ if (len < 1) { qcdm_err (0, "DIAG_CMD_LOG_CONFIG response not long enough (got %zu, " "expected at least %d).", len, 3); return -QCDM_ERROR_RESPONSE_BAD_LENGTH; } if (rsp->code == DIAG_CMD_LOG_CONFIG) { u_int32_t rspop; if (len < 16) { /* At least enough for code + op + result + equipid */ qcdm_err (0, "DIAG_CMD_LOG_CONFIG response not long enough (got %zu, " "expected at least %d).", len, 16); return -QCDM_ERROR_RESPONSE_BAD_LENGTH; } rspop = le32toh (rsp->op); if (rspop != op) { qcdm_err (0, "DIAG_CMD_LOG_CONFIG response operation mismatch (got " "op %u, expected %u)", rspop, op); return -QCDM_ERROR_RESPONSE_BAD_COMMAND; } /* check for success */ if (le32toh (rsp->result) != 0) { qcdm_err (0, "DIAG_CMD_LOG_CONFIG response failed with result %u.", le32toh (rsp->result)); return -QCDM_ERROR_RESPONSE_FAILED; } switch (rspop) { case DIAG_CMD_LOG_CONFIG_OP_GET_RANGE: minlen += 16; /* get_range_items */ break; case DIAG_CMD_LOG_CONFIG_OP_SET_MASK: case DIAG_CMD_LOG_CONFIG_OP_GET_MASK: if (len < 16) { qcdm_err (0, "DIAG_CMD_LOG_CONFIG response not long enough " "(got %zu, expected at least %d).", len, 16); return -QCDM_ERROR_RESPONSE_BAD_LENGTH; } minlen += 4; /* num_items */ minlen += (le32toh (rsp->u.get_set_items.num_items) + 7) / 8; break; default: qcdm_err (0, "Unknown DIAG_CMD_LOG_CONFIG response operation %d", rspop); return -QCDM_ERROR_RESPONSE_UNEXPECTED; } } if (!check_command (buf, len, DIAG_CMD_LOG_CONFIG, minlen, &err)) return err; return 0; } #define LOG_CODE_SET(mask, code) (mask[code / 8] & (1 << (code % 8))) static QcdmResult * log_config_get_set_result (const char *buf, size_t len, u_int32_t op, int *out_error) { QcdmResult *result = NULL; DMCmdLogConfigRsp *rsp = (DMCmdLogConfigRsp *) buf; int err; u_int32_t num_items; u_int32_t equipid; qcdm_return_val_if_fail (buf != NULL, NULL); err = check_log_config_respose (buf, len, op); if (err) { if (out_error) *out_error = err; return NULL; } result = qcdm_result_new (); equipid = le32toh (rsp->equipid); qcdm_result_add_u32 (result, QCDM_CMD_LOG_CONFIG_MASK_ITEM_EQUIP_ID, equipid); num_items = le32toh (rsp->u.get_set_items.num_items); qcdm_result_add_u32 (result, QCDM_CMD_LOG_CONFIG_MASK_ITEM_NUM_ITEMS, num_items); if (num_items > 0) { u_int32_t i, num_result_items = 0, count = 0; u_int16_t *items; /* First pass to find out how many are actually enabled */ for (i = 0; i < num_items; i++) { /* Check if the bit corresponding to this log item is set */ if (LOG_CODE_SET (rsp->u.get_set_items.mask, i)) num_result_items++; } if (num_result_items) { items = malloc (sizeof (*items) * num_result_items); for (i = 0; i < num_items; i++) { if (LOG_CODE_SET (rsp->u.get_set_items.mask, i)) items[count++] = (equipid << 12) | (i & 0x0FFF); } qcdm_result_add_u16_array (result, QCDM_CMD_LOG_CONFIG_MASK_ITEM_ITEMS, items, count); free (items); } } return result; } QcdmResult * qcdm_cmd_log_config_get_mask_result (const char *buf, size_t len, int *out_error) { return log_config_get_set_result (buf, len, DIAG_CMD_LOG_CONFIG_OP_GET_MASK, out_error); } size_t qcdm_cmd_log_config_set_mask_new (char *buf, size_t len, u_int32_t equip_id, u_int16_t items[]) { return qcdm_cmd_log_config_new (buf, len, DIAG_CMD_LOG_CONFIG_OP_SET_MASK, equip_id, items); } QcdmResult * qcdm_cmd_log_config_set_mask_result (const char *buf, size_t len, int *out_error) { return log_config_get_set_result (buf, len, DIAG_CMD_LOG_CONFIG_OP_SET_MASK, out_error); } qcdmbool qcmd_cmd_log_config_mask_result_code_set (QcdmResult *result, u_int32_t equipid, u_int16_t log_code) { const u_int16_t *items = NULL; size_t len = 0; u_int32_t i, tmp; qcdm_return_val_if_fail (result != NULL, FALSE); if (qcdm_result_get_u32 (result, QCDM_CMD_LOG_CONFIG_MASK_ITEM_EQUIP_ID, &tmp) != 0) return FALSE; qcdm_return_val_if_fail (equipid != tmp, FALSE); if (qcdm_result_get_u16_array (result, QCDM_CMD_LOG_CONFIG_MASK_ITEM_ITEMS, &items, &len)) { for (i = 0; i < len; i++) { if ((items[i] & 0x0FFF) == (log_code & 0x0FFF)) return TRUE; } } return FALSE; } /**********************************************************************/ static char bcd_chars[] = "0123456789\0\0\0\0\0\0"; static qcdmbool imxi_bcd_to_string (u_int8_t bytes[8], size_t len, char *buf, size_t buflen) { char *p; u_int32_t i; if (bytes[0] == 0) return TRUE; qcdm_return_val_if_fail (len == 8, FALSE); qcdm_return_val_if_fail (buf != NULL, FALSE); qcdm_return_val_if_fail (buflen > len, FALSE); p = buf; for (i = 0 ; i < len; i++) { /* IMxI are 15 chars long, so the lower 4-bits of the first * byte of the IMxI is skipped. Not sure what it does. */ if (i > 0) { *p = bcd_chars[bytes[i] & 0xf]; if (!*p) return FALSE; p++; } *p = bcd_chars[(bytes[i] >> 4) & 0xf]; if (!*p) return FALSE; p++; } *p++ = '\0'; return TRUE; } size_t qcdm_cmd_wcdma_subsys_state_info_new (char *buf, size_t len) { char cmdbuf[sizeof (DMCmdSubsysHeader) + 2]; DMCmdSubsysHeader *cmd = (DMCmdSubsysHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_SUBSYS; cmd->subsys_id = DIAG_SUBSYS_WCDMA; cmd->subsys_cmd = htole16 (DIAG_SUBSYS_WCDMA_STATE_INFO); return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_wcdma_subsys_state_info_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdSubsysWcdmaStateInfoRsp *rsp = (DMCmdSubsysWcdmaStateInfoRsp *) buf; char imxi[10]; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_SUBSYS, sizeof (DMCmdSubsysWcdmaStateInfoRsp), out_error)) return NULL; result = qcdm_result_new (); qcdm_result_add_u8 (result, QCDM_CMD_WCDMA_SUBSYS_STATE_INFO_ITEM_L1_STATE, rsp->l1_state); memset (imxi, 0, sizeof (imxi)); if (imxi_bcd_to_string (rsp->imei, rsp->imei_len, imxi, sizeof (imxi))) qcdm_result_add_string (result, QCDM_CMD_WCDMA_SUBSYS_STATE_INFO_ITEM_IMEI, imxi); memset (imxi, 0, sizeof (imxi)); if (imxi_bcd_to_string (rsp->imsi, rsp->imsi_len, imxi, sizeof (imxi))) qcdm_result_add_string (result, QCDM_CMD_WCDMA_SUBSYS_STATE_INFO_ITEM_IMSI, imxi); return result; } /**********************************************************************/ size_t qcdm_cmd_gsm_subsys_state_info_new (char *buf, size_t len) { char cmdbuf[sizeof (DMCmdSubsysHeader) + 2]; DMCmdSubsysHeader *cmd = (DMCmdSubsysHeader *) &cmdbuf[0]; qcdm_return_val_if_fail (buf != NULL, 0); qcdm_return_val_if_fail (len >= sizeof (*cmd) + DIAG_TRAILER_LEN, 0); memset (cmd, 0, sizeof (*cmd)); cmd->code = DIAG_CMD_SUBSYS; cmd->subsys_id = DIAG_SUBSYS_GSM; cmd->subsys_cmd = htole16 (DIAG_SUBSYS_GSM_STATE_INFO); return dm_encapsulate_buffer (cmdbuf, sizeof (*cmd), sizeof (cmdbuf), buf, len); } QcdmResult * qcdm_cmd_gsm_subsys_state_info_result (const char *buf, size_t len, int *out_error) { QcdmResult *result = NULL; DMCmdSubsysGsmStateInfoRsp *rsp = (DMCmdSubsysGsmStateInfoRsp *) buf; char imxi[10]; u_int32_t mcc = 0, mnc = 0; u_int8_t mnc3; qcdm_return_val_if_fail (buf != NULL, NULL); if (!check_command (buf, len, DIAG_CMD_SUBSYS, sizeof (DMCmdSubsysGsmStateInfoRsp), out_error)) return NULL; result = qcdm_result_new (); memset (imxi, 0, sizeof (imxi)); if (imxi_bcd_to_string (rsp->imei, rsp->imei_len, imxi, sizeof (imxi))) qcdm_result_add_string (result, QCDM_CMD_GSM_SUBSYS_STATE_INFO_ITEM_IMEI, imxi); memset (imxi, 0, sizeof (imxi)); if (imxi_bcd_to_string (rsp->imsi, rsp->imsi_len, imxi, sizeof (imxi))) qcdm_result_add_string (result, QCDM_CMD_GSM_SUBSYS_STATE_INFO_ITEM_IMSI, imxi); qcdm_result_add_u8 (result, QCDM_CMD_GSM_SUBSYS_STATE_INFO_ITEM_CM_CALL_STATE, rsp->cm_call_state); qcdm_result_add_u8 (result, QCDM_CMD_GSM_SUBSYS_STATE_INFO_ITEM_CM_OP_MODE, rsp->cm_opmode); qcdm_result_add_u8 (result, QCDM_CMD_GSM_SUBSYS_STATE_INFO_ITEM_CM_SYS_MODE, rsp->cm_sysmode); /* MCC/MNC, LAC, and CI don't seem to be valid when the modem is not in GSM mode */ if ( rsp->cm_sysmode == QCDM_CMD_CM_SUBSYS_STATE_INFO_SYSTEM_MODE_GSM || rsp->cm_sysmode == QCDM_CMD_CM_SUBSYS_STATE_INFO_SYSTEM_MODE_GW) { /* Quick convert BCD LAI into MCC/MNC/LAC */ mcc = (rsp->lai[0] & 0xF) * 100; mcc += ((rsp->lai[0] >> 4) & 0xF) * 10; mcc += rsp->lai[1] & 0xF; qcdm_result_add_u32 (result, QCDM_CMD_GSM_SUBSYS_STATE_INFO_ITEM_LAI_MCC, mcc); mnc = (rsp->lai[2] & 0XF) * 100; mnc += ((rsp->lai[2] >> 4) & 0xF) * 10; mnc3 = (rsp->lai[1] >> 4) & 0xF; if (mnc3 != 0xF) mnc += mnc3; qcdm_result_add_u32 (result, QCDM_CMD_GSM_SUBSYS_STATE_INFO_ITEM_LAI_MNC, mnc); qcdm_result_add_u32 (result, QCDM_CMD_GSM_SUBSYS_STATE_INFO_ITEM_LAI_LAC, le16toh (*(u_int16_t *)(&rsp->lai[3]))); qcdm_result_add_u32 (result, QCDM_CMD_GSM_SUBSYS_STATE_INFO_ITEM_CELLID, le16toh (rsp->cellid)); } return result; } /**********************************************************************/