/* Copyright (c) 2008-2016, The Linux Foundation. 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 and * only 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. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef CONFIG_DIAG_OVER_USB #include #endif #include "diagmem.h" #include "diagchar.h" #include "diagfwd.h" #include "diagfwd_peripheral.h" #include "diagfwd_cntl.h" #include "diagchar_hdlc.h" #include "diag_dci.h" #include "diag_masks.h" #include "diag_usb.h" #include "diag_mux.h" #define STM_CMD_VERSION_OFFSET 4 #define STM_CMD_MASK_OFFSET 5 #define STM_CMD_DATA_OFFSET 6 #define STM_CMD_NUM_BYTES 7 #define STM_RSP_SUPPORTED_INDEX 7 #define STM_RSP_STATUS_INDEX 8 #define STM_RSP_NUM_BYTES 9 static int timestamp_switch; module_param(timestamp_switch, int, 0644); int wrap_enabled; uint16_t wrap_count; static struct diag_hdlc_decode_type *hdlc_decode; #define DIAG_NUM_COMMON_CMD 1 static uint8_t common_cmds[DIAG_NUM_COMMON_CMD] = { DIAG_CMD_LOG_ON_DMND }; static uint8_t hdlc_timer_in_progress; /* Determine if this device uses a device tree */ #ifdef CONFIG_OF static int has_device_tree(void) { struct device_node *node; node = of_find_node_by_path("/"); if (node) { of_node_put(node); return 1; } return 0; } #else static int has_device_tree(void) { return 0; } #endif int chk_config_get_id(void) { if (driver->use_device_tree) { if (of_machine_is_compatible("qcom,msm8974")) return 4083; else if (of_machine_is_compatible("qcom,apq8974")) return 4090; else return 0; } else return 0; } /* * This will return TRUE for targets which support apps only mode and hence SSR. * This applies to 8960 and newer targets. */ int chk_apps_only(void) { if (driver->use_device_tree) return 1; return 0; } /* * This will return TRUE for targets which support apps as master. * Thus, SW DLOAD and Mode Reset are supported on apps processor. * This applies to 8960 and newer targets. */ int chk_apps_master(void) { if (driver->use_device_tree) return 1; else return 0; } int chk_polling_response(void) { if (!(driver->polling_reg_flag) && chk_apps_master()) /* * If the apps processor is master and no other processor * has registered to respond for polling */ return 1; else if (!(driver->diagfwd_cntl[PERIPHERAL_MODEM] && driver->diagfwd_cntl[PERIPHERAL_MODEM]->ch_open) && (driver->feature[PERIPHERAL_MODEM].rcvd_feature_mask)) /* * If the apps processor is not the master and the modem * is not up or we did not receive the feature masks from Modem */ return 1; else return 0; } /* * This function should be called if you feel that the logging process may * need to be woken up. For instance, if the logging mode is MEMORY_DEVICE MODE * and while trying to read data from data channel there are no buffers * available to read the data into, then this function should be called to * determine if the logging process needs to be woken up. */ void chk_logging_wakeup(void) { int i; int j; int pid = 0; for (j = 0; j < NUM_MD_SESSIONS; j++) { if (!driver->md_session_map[j]) continue; pid = driver->md_session_map[j]->pid; /* Find the index of the logging process */ for (i = 0; i < driver->num_clients; i++) { if (driver->client_map[i].pid != pid) continue; if (driver->data_ready[i] & USER_SPACE_DATA_TYPE) continue; /* * At very high logging rates a race condition can * occur where the buffers containing the data from * a channel are all in use, but the data_ready flag * is cleared. In this case, the buffers never have * their data read/logged. Detect and remedy this * situation. */ driver->data_ready[i] |= USER_SPACE_DATA_TYPE; pr_debug("diag: Force wakeup of logging process\n"); wake_up_interruptible(&driver->wait_q); break; } /* * Diag Memory Device is in normal. Check only for the first * index as all the indices point to the same session * structure. */ if (driver->md_session_mode == DIAG_MD_NORMAL && j == 0) break; } } static void pack_rsp_and_send(unsigned char *buf, int len) { int err; int retry_count = 0; uint32_t write_len = 0; unsigned long flags; unsigned char *rsp_ptr = driver->encoded_rsp_buf; struct diag_pkt_frame_t header; if (!rsp_ptr || !buf) return; if (len > DIAG_MAX_RSP_SIZE || len < 0) { pr_err("diag: In %s, invalid len %d, permissible len %d\n", __func__, len, DIAG_MAX_RSP_SIZE); return; } /* * Keep trying till we get the buffer back. It should probably * take one or two iterations. When this loops till UINT_MAX, it * means we did not get a write complete for the previous * response. */ while (retry_count < UINT_MAX) { if (!driver->rsp_buf_busy) break; /* * Wait for sometime and try again. The value 10000 was chosen * empirically as an optimum value for USB to complete a write */ usleep_range(10000, 10100); retry_count++; /* * There can be a race conditon that clears the data ready flag * for responses. Make sure we don't miss previous wakeups for * draining responses when we are in Memory Device Mode. */ if (driver->logging_mode == DIAG_MEMORY_DEVICE_MODE) chk_logging_wakeup(); } if (driver->rsp_buf_busy) { pr_err("diag: unable to get hold of response buffer\n"); return; } driver->rsp_buf_busy = 1; header.start = CONTROL_CHAR; header.version = 1; header.length = len; memcpy(rsp_ptr, &header, sizeof(header)); write_len += sizeof(header); memcpy(rsp_ptr + write_len, buf, len); write_len += len; *(uint8_t *)(rsp_ptr + write_len) = CONTROL_CHAR; write_len += sizeof(uint8_t); err = diag_mux_write(DIAG_LOCAL_PROC, rsp_ptr, write_len, driver->rsp_buf_ctxt); if (err) { pr_err("diag: In %s, unable to write to mux, err: %d\n", __func__, err); spin_lock_irqsave(&driver->rsp_buf_busy_lock, flags); driver->rsp_buf_busy = 0; spin_unlock_irqrestore(&driver->rsp_buf_busy_lock, flags); } } static void encode_rsp_and_send(unsigned char *buf, int len) { struct diag_send_desc_type send = { NULL, NULL, DIAG_STATE_START, 0 }; struct diag_hdlc_dest_type enc = { NULL, NULL, 0 }; unsigned char *rsp_ptr = driver->encoded_rsp_buf; int err, retry_count = 0; unsigned long flags; if (!rsp_ptr || !buf) return; if (len > DIAG_MAX_RSP_SIZE || len < 0) { pr_err("diag: In %s, invalid len %d, permissible len %d\n", __func__, len, DIAG_MAX_RSP_SIZE); return; } /* * Keep trying till we get the buffer back. It should probably * take one or two iterations. When this loops till UINT_MAX, it * means we did not get a write complete for the previous * response. */ while (retry_count < UINT_MAX) { if (!driver->rsp_buf_busy) break; /* * Wait for sometime and try again. The value 10000 was chosen * empirically as an optimum value for USB to complete a write */ usleep_range(10000, 10100); retry_count++; /* * There can be a race conditon that clears the data ready flag * for responses. Make sure we don't miss previous wakeups for * draining responses when we are in Memory Device Mode. */ if (driver->logging_mode == DIAG_MEMORY_DEVICE_MODE) chk_logging_wakeup(); } if (driver->rsp_buf_busy) { pr_err("diag: unable to get hold of response buffer\n"); return; } spin_lock_irqsave(&driver->rsp_buf_busy_lock, flags); driver->rsp_buf_busy = 1; spin_unlock_irqrestore(&driver->rsp_buf_busy_lock, flags); send.state = DIAG_STATE_START; send.pkt = buf; send.last = (void *)(buf + len - 1); send.terminate = 1; enc.dest = rsp_ptr; enc.dest_last = (void *)(rsp_ptr + DIAG_MAX_HDLC_BUF_SIZE - 1); diag_hdlc_encode(&send, &enc); driver->encoded_rsp_len = (int)(enc.dest - (void *)rsp_ptr); err = diag_mux_write(DIAG_LOCAL_PROC, rsp_ptr, driver->encoded_rsp_len, driver->rsp_buf_ctxt); if (err) { pr_err("diag: In %s, Unable to write to device, err: %d\n", __func__, err); spin_lock_irqsave(&driver->rsp_buf_busy_lock, flags); driver->rsp_buf_busy = 0; spin_unlock_irqrestore(&driver->rsp_buf_busy_lock, flags); } memset(buf, '\0', DIAG_MAX_RSP_SIZE); } void diag_send_rsp(unsigned char *buf, int len) { struct diag_md_session_t *session_info = NULL; uint8_t hdlc_disabled; session_info = diag_md_session_get_peripheral(APPS_DATA); if (session_info) hdlc_disabled = session_info->hdlc_disabled; else hdlc_disabled = driver->hdlc_disabled; if (hdlc_disabled) pack_rsp_and_send(buf, len); else encode_rsp_and_send(buf, len); } void diag_update_pkt_buffer(unsigned char *buf, uint32_t len, int type) { unsigned char *ptr = NULL; unsigned char *temp = buf; int *in_busy = NULL; uint32_t *length = NULL; uint32_t max_len = 0; if (!buf || len == 0) { pr_err("diag: In %s, Invalid ptr %pK and length %d\n", __func__, buf, len); return; } switch (type) { case PKT_TYPE: ptr = driver->apps_req_buf; length = &driver->apps_req_buf_len; max_len = DIAG_MAX_REQ_SIZE; in_busy = &driver->in_busy_pktdata; break; case DCI_PKT_TYPE: ptr = driver->dci_pkt_buf; length = &driver->dci_pkt_length; max_len = DCI_BUF_SIZE; in_busy = &driver->in_busy_dcipktdata; break; default: pr_err("diag: Invalid type %d in %s\n", type, __func__); return; } mutex_lock(&driver->diagchar_mutex); if (CHK_OVERFLOW(ptr, ptr, ptr + max_len, len)) { memcpy(ptr, temp , len); *length = len; *in_busy = 1; } else { pr_alert("diag: In %s, no space for response packet, len: %d, type: %d\n", __func__, len, type); } mutex_unlock(&driver->diagchar_mutex); } void diag_update_userspace_clients(unsigned int type) { int i; mutex_lock(&driver->diagchar_mutex); for (i = 0; i < driver->num_clients; i++) if (driver->client_map[i].pid != 0) driver->data_ready[i] |= type; wake_up_interruptible(&driver->wait_q); mutex_unlock(&driver->diagchar_mutex); } void diag_update_md_clients(unsigned int type) { int i, j; mutex_lock(&driver->diagchar_mutex); for (i = 0; i < NUM_MD_SESSIONS; i++) { if (driver->md_session_map[i] != NULL) for (j = 0; j < driver->num_clients; j++) { if (driver->client_map[j].pid != 0 && driver->client_map[j].pid == driver->md_session_map[i]->pid) { driver->data_ready[j] |= type; break; } } } wake_up_interruptible(&driver->wait_q); mutex_unlock(&driver->diagchar_mutex); } void diag_update_sleeping_process(int process_id, int data_type) { int i; mutex_lock(&driver->diagchar_mutex); for (i = 0; i < driver->num_clients; i++) if (driver->client_map[i].pid == process_id) { driver->data_ready[i] |= data_type; break; } wake_up_interruptible(&driver->wait_q); mutex_unlock(&driver->diagchar_mutex); } static int diag_send_data(struct diag_cmd_reg_t *entry, unsigned char *buf, int len) { if (!entry) return -EIO; if (entry->proc == APPS_DATA) { diag_update_pkt_buffer(buf, len, PKT_TYPE); diag_update_sleeping_process(entry->pid, PKT_TYPE); return 0; } return diagfwd_write(entry->proc, TYPE_CMD, buf, len); } void diag_process_stm_mask(uint8_t cmd, uint8_t data_mask, int data_type) { int status = 0; if (data_type >= PERIPHERAL_MODEM && data_type <= PERIPHERAL_SENSORS) { if (driver->feature[data_type].stm_support) { status = diag_send_stm_state(data_type, cmd); if (status == 1) driver->stm_state[data_type] = cmd; } driver->stm_state_requested[data_type] = cmd; } else if (data_type == APPS_DATA) { driver->stm_state[data_type] = cmd; driver->stm_state_requested[data_type] = cmd; } } int diag_process_stm_cmd(unsigned char *buf, unsigned char *dest_buf) { uint8_t version, mask, cmd; uint8_t rsp_supported = 0; uint8_t rsp_status = 0; int i; if (!buf || !dest_buf) { pr_err("diag: Invalid pointers buf: %pK, dest_buf %pK in %s\n", buf, dest_buf, __func__); return -EIO; } version = *(buf + STM_CMD_VERSION_OFFSET); mask = *(buf + STM_CMD_MASK_OFFSET); cmd = *(buf + STM_CMD_DATA_OFFSET); /* * Check if command is valid. If the command is asking for * status, then the processor mask field is to be ignored. */ if ((version != 2) || (cmd > STATUS_STM) || ((cmd != STATUS_STM) && ((mask == 0) || (0 != (mask >> 4))))) { /* Command is invalid. Send bad param message response */ dest_buf[0] = BAD_PARAM_RESPONSE_MESSAGE; for (i = 0; i < STM_CMD_NUM_BYTES; i++) dest_buf[i+1] = *(buf + i); return STM_CMD_NUM_BYTES+1; } else if (cmd != STATUS_STM) { if (mask & DIAG_STM_MODEM) diag_process_stm_mask(cmd, DIAG_STM_MODEM, PERIPHERAL_MODEM); if (mask & DIAG_STM_LPASS) diag_process_stm_mask(cmd, DIAG_STM_LPASS, PERIPHERAL_LPASS); if (mask & DIAG_STM_WCNSS) diag_process_stm_mask(cmd, DIAG_STM_WCNSS, PERIPHERAL_WCNSS); if (mask & DIAG_STM_SENSORS) diag_process_stm_mask(cmd, DIAG_STM_SENSORS, PERIPHERAL_SENSORS); if (mask & DIAG_STM_APPS) diag_process_stm_mask(cmd, DIAG_STM_APPS, APPS_DATA); } for (i = 0; i < STM_CMD_NUM_BYTES; i++) dest_buf[i] = *(buf + i); /* Set mask denoting which peripherals support STM */ if (driver->feature[PERIPHERAL_MODEM].stm_support) rsp_supported |= DIAG_STM_MODEM; if (driver->feature[PERIPHERAL_LPASS].stm_support) rsp_supported |= DIAG_STM_LPASS; if (driver->feature[PERIPHERAL_WCNSS].stm_support) rsp_supported |= DIAG_STM_WCNSS; if (driver->feature[PERIPHERAL_SENSORS].stm_support) rsp_supported |= DIAG_STM_SENSORS; rsp_supported |= DIAG_STM_APPS; /* Set mask denoting STM state/status for each peripheral/APSS */ if (driver->stm_state[PERIPHERAL_MODEM]) rsp_status |= DIAG_STM_MODEM; if (driver->stm_state[PERIPHERAL_LPASS]) rsp_status |= DIAG_STM_LPASS; if (driver->stm_state[PERIPHERAL_WCNSS]) rsp_status |= DIAG_STM_WCNSS; if (driver->stm_state[PERIPHERAL_SENSORS]) rsp_status |= DIAG_STM_SENSORS; if (driver->stm_state[APPS_DATA]) rsp_status |= DIAG_STM_APPS; dest_buf[STM_RSP_SUPPORTED_INDEX] = rsp_supported; dest_buf[STM_RSP_STATUS_INDEX] = rsp_status; return STM_RSP_NUM_BYTES; } int diag_process_time_sync_query_cmd(unsigned char *src_buf, int src_len, unsigned char *dest_buf, int dest_len) { int write_len = 0; struct diag_cmd_time_sync_query_req_t *req = NULL; struct diag_cmd_time_sync_query_rsp_t rsp; if (!src_buf || !dest_buf || src_len <= 0 || dest_len <= 0) { pr_err("diag: Invalid input in %s, src_buf: %pK, src_len: %d, dest_buf: %pK, dest_len: %d", __func__, src_buf, src_len, dest_buf, dest_len); return -EINVAL; } req = (struct diag_cmd_time_sync_query_req_t *)src_buf; rsp.header.cmd_code = req->header.cmd_code; rsp.header.subsys_id = req->header.subsys_id; rsp.header.subsys_cmd_code = req->header.subsys_cmd_code; rsp.version = req->version; rsp.time_api = driver->uses_time_api; memcpy(dest_buf, &rsp, sizeof(rsp)); write_len = sizeof(rsp); return write_len; } int diag_process_time_sync_switch_cmd(unsigned char *src_buf, int src_len, unsigned char *dest_buf, int dest_len) { uint8_t peripheral, status = 0; struct diag_cmd_time_sync_switch_req_t *req = NULL; struct diag_cmd_time_sync_switch_rsp_t rsp; struct diag_ctrl_msg_time_sync time_sync_msg; int msg_size = sizeof(struct diag_ctrl_msg_time_sync); int err = 0, write_len = 0; if (!src_buf || !dest_buf || src_len <= 0 || dest_len <= 0) { pr_err("diag: Invalid input in %s, src_buf: %pK, src_len: %d, dest_buf: %pK, dest_len: %d", __func__, src_buf, src_len, dest_buf, dest_len); return -EINVAL; } req = (struct diag_cmd_time_sync_switch_req_t *)src_buf; rsp.header.cmd_code = req->header.cmd_code; rsp.header.subsys_id = req->header.subsys_id; rsp.header.subsys_cmd_code = req->header.subsys_cmd_code; rsp.version = req->version; rsp.time_api = req->time_api; if ((req->version > 1) || (req->time_api > 1) || (req->persist_time > 0)) { dest_buf[0] = BAD_PARAM_RESPONSE_MESSAGE; rsp.time_api_status = 0; rsp.persist_time_status = PERSIST_TIME_NOT_SUPPORTED; memcpy(dest_buf + 1, &rsp, sizeof(rsp)); write_len = sizeof(rsp) + 1; timestamp_switch = 0; return write_len; } time_sync_msg.ctrl_pkt_id = DIAG_CTRL_MSG_TIME_SYNC_PKT; time_sync_msg.ctrl_pkt_data_len = 5; time_sync_msg.version = 1; time_sync_msg.time_api = req->time_api; for (peripheral = 0; peripheral < NUM_PERIPHERALS; peripheral++) { err = diagfwd_write(peripheral, TYPE_CNTL, &time_sync_msg, msg_size); if (err && err != -ENODEV) { pr_err("diag: In %s, unable to write to peripheral: %d, type: %d, len: %d, err: %d\n", __func__, peripheral, TYPE_CNTL, msg_size, err); status |= (1 << peripheral); } } driver->time_sync_enabled = 1; driver->uses_time_api = req->time_api; switch (req->time_api) { case 0: timestamp_switch = 0; break; case 1: timestamp_switch = 1; break; default: timestamp_switch = 0; break; } rsp.time_api_status = status; rsp.persist_time_status = PERSIST_TIME_NOT_SUPPORTED; memcpy(dest_buf, &rsp, sizeof(rsp)); write_len = sizeof(rsp); return write_len; } int diag_cmd_log_on_demand(unsigned char *src_buf, int src_len, unsigned char *dest_buf, int dest_len) { int write_len = 0; struct diag_log_on_demand_rsp_t header; if (!driver->diagfwd_cntl[PERIPHERAL_MODEM] || !driver->diagfwd_cntl[PERIPHERAL_MODEM]->ch_open || !driver->log_on_demand_support) return 0; if (!src_buf || !dest_buf || src_len <= 0 || dest_len <= 0) { pr_err("diag: Invalid input in %s, src_buf: %pK, src_len: %d, dest_buf: %pK, dest_len: %d", __func__, src_buf, src_len, dest_buf, dest_len); return -EINVAL; } header.cmd_code = DIAG_CMD_LOG_ON_DMND; header.log_code = *(uint16_t *)(src_buf + 1); header.status = 1; memcpy(dest_buf, &header, sizeof(struct diag_log_on_demand_rsp_t)); write_len += sizeof(struct diag_log_on_demand_rsp_t); return write_len; } int diag_cmd_get_mobile_id(unsigned char *src_buf, int src_len, unsigned char *dest_buf, int dest_len) { int write_len = 0; struct diag_pkt_header_t *header = NULL; struct diag_cmd_ext_mobile_rsp_t rsp = {0}; if (!src_buf || src_len != sizeof(*header) || !dest_buf || dest_len < sizeof(rsp)) return -EIO; header = (struct diag_pkt_header_t *)src_buf; rsp.header.cmd_code = header->cmd_code; rsp.header.subsys_id = header->subsys_id; rsp.header.subsys_cmd_code = header->subsys_cmd_code; rsp.version = 2; rsp.padding[0] = 0; rsp.padding[1] = 0; rsp.padding[2] = 0; rsp.family = 0; memcpy(dest_buf, &rsp, sizeof(rsp)); write_len += sizeof(rsp); return write_len; } int diag_check_common_cmd(struct diag_pkt_header_t *header) { int i; if (!header) return -EIO; for (i = 0; i < DIAG_NUM_COMMON_CMD; i++) { if (header->cmd_code == common_cmds[i]) return 1; } return 0; } static int diag_cmd_chk_stats(unsigned char *src_buf, int src_len, unsigned char *dest_buf, int dest_len) { int payload = 0; int write_len = 0; struct diag_pkt_header_t *header = NULL; struct diag_cmd_stats_rsp_t rsp; if (!src_buf || src_len < sizeof(struct diag_pkt_header_t) || !dest_buf || dest_len < sizeof(rsp)) return -EINVAL; header = (struct diag_pkt_header_t *)src_buf; if (header->cmd_code != DIAG_CMD_DIAG_SUBSYS || header->subsys_id != DIAG_SS_DIAG) return -EINVAL; switch (header->subsys_cmd_code) { case DIAG_CMD_OP_GET_MSG_ALLOC: payload = driver->msg_stats.alloc_count; break; case DIAG_CMD_OP_GET_MSG_DROP: payload = driver->msg_stats.drop_count; break; case DIAG_CMD_OP_RESET_MSG_STATS: diag_record_stats(DATA_TYPE_F3, PKT_RESET); break; case DIAG_CMD_OP_GET_LOG_ALLOC: payload = driver->log_stats.alloc_count; break; case DIAG_CMD_OP_GET_LOG_DROP: payload = driver->log_stats.drop_count; break; case DIAG_CMD_OP_RESET_LOG_STATS: diag_record_stats(DATA_TYPE_LOG, PKT_RESET); break; case DIAG_CMD_OP_GET_EVENT_ALLOC: payload = driver->event_stats.alloc_count; break; case DIAG_CMD_OP_GET_EVENT_DROP: payload = driver->event_stats.drop_count; break; case DIAG_CMD_OP_RESET_EVENT_STATS: diag_record_stats(DATA_TYPE_EVENT, PKT_RESET); break; default: return -EINVAL; } memcpy(&rsp.header, header, sizeof(struct diag_pkt_header_t)); rsp.payload = payload; write_len = sizeof(rsp); memcpy(dest_buf, &rsp, sizeof(rsp)); return write_len; } static int diag_cmd_disable_hdlc(unsigned char *src_buf, int src_len, unsigned char *dest_buf, int dest_len) { struct diag_pkt_header_t *header = NULL; struct diag_cmd_hdlc_disable_rsp_t rsp; int write_len = 0; if (!src_buf || src_len < sizeof(*header) || !dest_buf || dest_len < sizeof(rsp)) { return -EIO; } header = (struct diag_pkt_header_t *)src_buf; if (header->cmd_code != DIAG_CMD_DIAG_SUBSYS || header->subsys_id != DIAG_SS_DIAG || header->subsys_cmd_code != DIAG_CMD_OP_HDLC_DISABLE) { return -EINVAL; } memcpy(&rsp.header, header, sizeof(struct diag_pkt_header_t)); rsp.framing_version = 1; rsp.result = 0; write_len = sizeof(rsp); memcpy(dest_buf, &rsp, sizeof(rsp)); return write_len; } void diag_send_error_rsp(unsigned char *buf, int len) { /* -1 to accomodate the first byte 0x13 */ if (len > (DIAG_MAX_RSP_SIZE - 1)) { pr_err("diag: cannot send err rsp, huge length: %d\n", len); return; } *(uint8_t *)driver->apps_rsp_buf = DIAG_CMD_ERROR; memcpy((driver->apps_rsp_buf + sizeof(uint8_t)), buf, len); diag_send_rsp(driver->apps_rsp_buf, len + 1); } int diag_process_apps_pkt(unsigned char *buf, int len, struct diag_md_session_t *info) { int i; int mask_ret; int write_len = 0; unsigned char *temp = NULL; struct diag_cmd_reg_entry_t entry; struct diag_cmd_reg_entry_t *temp_entry = NULL; struct diag_cmd_reg_t *reg_item = NULL; if (!buf) return -EIO; /* Check if the command is a supported mask command */ mask_ret = diag_process_apps_masks(buf, len, info); if (mask_ret > 0) { diag_send_rsp(driver->apps_rsp_buf, mask_ret); return 0; } temp = buf; entry.cmd_code = (uint16_t)(*(uint8_t *)temp); temp += sizeof(uint8_t); entry.subsys_id = (uint16_t)(*(uint8_t *)temp); temp += sizeof(uint8_t); entry.cmd_code_hi = (uint16_t)(*(uint16_t *)temp); entry.cmd_code_lo = (uint16_t)(*(uint16_t *)temp); temp += sizeof(uint16_t); pr_debug("diag: In %s, received cmd %02x %02x %02x\n", __func__, entry.cmd_code, entry.subsys_id, entry.cmd_code_hi); if (*buf == DIAG_CMD_LOG_ON_DMND && driver->log_on_demand_support && driver->feature[PERIPHERAL_MODEM].rcvd_feature_mask) { write_len = diag_cmd_log_on_demand(buf, len, driver->apps_rsp_buf, DIAG_MAX_RSP_SIZE); if (write_len > 0) diag_send_rsp(driver->apps_rsp_buf, write_len); return 0; } mutex_lock(&driver->cmd_reg_mutex); temp_entry = diag_cmd_search(&entry, ALL_PROC); if (temp_entry) { reg_item = container_of(temp_entry, struct diag_cmd_reg_t, entry); if (info) { if (MD_PERIPHERAL_MASK(reg_item->proc) & info->peripheral_mask) write_len = diag_send_data(reg_item, buf, len); } else write_len = diag_send_data(reg_item, buf, len); mutex_unlock(&driver->cmd_reg_mutex); return write_len; } mutex_unlock(&driver->cmd_reg_mutex); #if defined(CONFIG_DIAG_OVER_USB) /* Check for the command/respond msg for the maximum packet length */ if ((*buf == 0x4b) && (*(buf+1) == 0x12) && (*(uint16_t *)(buf+2) == 0x0055)) { for (i = 0; i < 4; i++) *(driver->apps_rsp_buf+i) = *(buf+i); *(uint32_t *)(driver->apps_rsp_buf+4) = DIAG_MAX_REQ_SIZE; diag_send_rsp(driver->apps_rsp_buf, 8); return 0; } else if ((*buf == 0x4b) && (*(buf+1) == 0x12) && (*(uint16_t *)(buf+2) == DIAG_DIAG_STM)) { len = diag_process_stm_cmd(buf, driver->apps_rsp_buf); if (len > 0) { diag_send_rsp(driver->apps_rsp_buf, len); return 0; } return len; } /* Check for time sync query command */ else if ((*buf == DIAG_CMD_DIAG_SUBSYS) && (*(buf+1) == DIAG_SS_DIAG) && (*(uint16_t *)(buf+2) == DIAG_GET_TIME_API)) { write_len = diag_process_time_sync_query_cmd(buf, len, driver->apps_rsp_buf, DIAG_MAX_RSP_SIZE); if (write_len > 0) diag_send_rsp(driver->apps_rsp_buf, write_len); return 0; } /* Check for time sync switch command */ else if ((*buf == DIAG_CMD_DIAG_SUBSYS) && (*(buf+1) == DIAG_SS_DIAG) && (*(uint16_t *)(buf+2) == DIAG_SET_TIME_API)) { write_len = diag_process_time_sync_switch_cmd(buf, len, driver->apps_rsp_buf, DIAG_MAX_RSP_SIZE); if (write_len > 0) diag_send_rsp(driver->apps_rsp_buf, write_len); return 0; } /* Check for download command */ else if ((chk_apps_master()) && (*buf == 0x3A)) { /* send response back */ driver->apps_rsp_buf[0] = *buf; diag_send_rsp(driver->apps_rsp_buf, 1); msleep(5000); kernel_restart(NULL); /* Not required, represents that command isnt sent to modem */ return 0; } /* Check for polling for Apps only DIAG */ else if ((*buf == 0x4b) && (*(buf+1) == 0x32) && (*(buf+2) == 0x03)) { /* If no one has registered for polling */ if (chk_polling_response()) { /* Respond to polling for Apps only DIAG */ for (i = 0; i < 3; i++) driver->apps_rsp_buf[i] = *(buf+i); for (i = 0; i < 13; i++) driver->apps_rsp_buf[i+3] = 0; diag_send_rsp(driver->apps_rsp_buf, 16); return 0; } } /* Return the Delayed Response Wrap Status */ else if ((*buf == 0x4b) && (*(buf+1) == 0x32) && (*(buf+2) == 0x04) && (*(buf+3) == 0x0)) { memcpy(driver->apps_rsp_buf, buf, 4); driver->apps_rsp_buf[4] = wrap_enabled; diag_send_rsp(driver->apps_rsp_buf, 5); return 0; } /* Wrap the Delayed Rsp ID */ else if ((*buf == 0x4b) && (*(buf+1) == 0x32) && (*(buf+2) == 0x05) && (*(buf+3) == 0x0)) { wrap_enabled = true; memcpy(driver->apps_rsp_buf, buf, 4); driver->apps_rsp_buf[4] = wrap_count; diag_send_rsp(driver->apps_rsp_buf, 6); return 0; } /* Mobile ID Rsp */ else if ((*buf == DIAG_CMD_DIAG_SUBSYS) && (*(buf+1) == DIAG_SS_PARAMS) && (*(buf+2) == DIAG_EXT_MOBILE_ID) && (*(buf+3) == 0x0)) { write_len = diag_cmd_get_mobile_id(buf, len, driver->apps_rsp_buf, DIAG_MAX_RSP_SIZE); if (write_len > 0) { diag_send_rsp(driver->apps_rsp_buf, write_len); return 0; } } /* * If the apps processor is master and no other * processor has registered for polling command. * If modem is not up and we have not received feature * mask update from modem, in that case APPS should * respond for 0X7C command */ else if (chk_apps_master() && !(driver->polling_reg_flag) && !(driver->diagfwd_cntl[PERIPHERAL_MODEM]->ch_open) && !(driver->feature[PERIPHERAL_MODEM].rcvd_feature_mask)) { /* respond to 0x0 command */ if (*buf == 0x00) { for (i = 0; i < 55; i++) driver->apps_rsp_buf[i] = 0; diag_send_rsp(driver->apps_rsp_buf, 55); return 0; } /* respond to 0x7c command */ else if (*buf == 0x7c) { driver->apps_rsp_buf[0] = 0x7c; for (i = 1; i < 8; i++) driver->apps_rsp_buf[i] = 0; /* Tools ID for APQ 8060 */ *(int *)(driver->apps_rsp_buf + 8) = chk_config_get_id(); *(unsigned char *)(driver->apps_rsp_buf + 12) = '\0'; *(unsigned char *)(driver->apps_rsp_buf + 13) = '\0'; diag_send_rsp(driver->apps_rsp_buf, 14); return 0; } } write_len = diag_cmd_chk_stats(buf, len, driver->apps_rsp_buf, DIAG_MAX_RSP_SIZE); if (write_len > 0) { diag_send_rsp(driver->apps_rsp_buf, write_len); return 0; } write_len = diag_cmd_disable_hdlc(buf, len, driver->apps_rsp_buf, DIAG_MAX_RSP_SIZE); if (write_len > 0) { /* * This mutex lock is necessary since we need to drain all the * pending buffers from peripherals which may be HDLC encoded * before disabling HDLC encoding on Apps processor. */ mutex_lock(&driver->hdlc_disable_mutex); diag_send_rsp(driver->apps_rsp_buf, write_len); /* * Set the value of hdlc_disabled after sending the response to * the tools. This is required since the tools is expecting a * HDLC encoded reponse for this request. */ pr_debug("diag: In %s, disabling HDLC encoding\n", __func__); if (info) info->hdlc_disabled = 1; else driver->hdlc_disabled = 1; diag_update_md_clients(HDLC_SUPPORT_TYPE); mutex_unlock(&driver->hdlc_disable_mutex); return 0; } #endif /* We have now come to the end of the function. */ if (chk_apps_only()) diag_send_error_rsp(buf, len); return 0; } void diag_process_hdlc_pkt(void *data, unsigned len, struct diag_md_session_t *info) { int err = 0; int ret = 0; if (len > DIAG_MAX_HDLC_BUF_SIZE) { pr_err("diag: In %s, invalid length: %d\n", __func__, len); return; } mutex_lock(&driver->diag_hdlc_mutex); pr_debug("diag: In %s, received packet of length: %d, req_buf_len: %d\n", __func__, len, driver->hdlc_buf_len); if (driver->hdlc_buf_len >= DIAG_MAX_REQ_SIZE) { pr_err("diag: In %s, request length is more than supported len. Dropping packet.\n", __func__); goto fail; } hdlc_decode->dest_ptr = driver->hdlc_buf + driver->hdlc_buf_len; hdlc_decode->dest_size = DIAG_MAX_HDLC_BUF_SIZE - driver->hdlc_buf_len; hdlc_decode->src_ptr = data; hdlc_decode->src_size = len; hdlc_decode->src_idx = 0; hdlc_decode->dest_idx = 0; ret = diag_hdlc_decode(hdlc_decode); /* * driver->hdlc_buf is of size DIAG_MAX_HDLC_BUF_SIZE. But the decoded * packet should be within DIAG_MAX_REQ_SIZE. */ if (driver->hdlc_buf_len + hdlc_decode->dest_idx <= DIAG_MAX_REQ_SIZE) { driver->hdlc_buf_len += hdlc_decode->dest_idx; } else { pr_err_ratelimited("diag: In %s, Dropping packet. pkt_size: %d, max: %d\n", __func__, driver->hdlc_buf_len + hdlc_decode->dest_idx, DIAG_MAX_REQ_SIZE); goto fail; } if (ret == HDLC_COMPLETE) { err = crc_check(driver->hdlc_buf, driver->hdlc_buf_len); if (err) { /* CRC check failed. */ pr_err_ratelimited("diag: In %s, bad CRC. Dropping packet\n", __func__); goto fail; } driver->hdlc_buf_len -= HDLC_FOOTER_LEN; if (driver->hdlc_buf_len < 1) { pr_err_ratelimited("diag: In %s, message is too short, len: %d, dest len: %d\n", __func__, driver->hdlc_buf_len, hdlc_decode->dest_idx); goto fail; } err = diag_process_apps_pkt(driver->hdlc_buf, driver->hdlc_buf_len, info); if (err < 0) goto fail; } else { goto end; } driver->hdlc_buf_len = 0; mutex_unlock(&driver->diag_hdlc_mutex); return; fail: /* * Tools needs to get a response in order to start its * recovery algorithm. Send an error response if the * packet is not in expected format. */ diag_send_error_rsp(driver->hdlc_buf, driver->hdlc_buf_len); driver->hdlc_buf_len = 0; end: mutex_unlock(&driver->diag_hdlc_mutex); } static int diagfwd_mux_open(int id, int mode) { uint8_t i; unsigned long flags; switch (mode) { case DIAG_USB_MODE: driver->usb_connected = 1; break; case DIAG_MEMORY_DEVICE_MODE: break; default: return -EINVAL; } if (driver->rsp_buf_busy) { /* * When a client switches from callback mode to USB mode * explicitly, there can be a situation when the last response * is not drained to the user space application. Reset the * in_busy flag in this case. */ spin_lock_irqsave(&driver->rsp_buf_busy_lock, flags); driver->rsp_buf_busy = 0; spin_unlock_irqrestore(&driver->rsp_buf_busy_lock, flags); } for (i = 0; i < NUM_PERIPHERALS; i++) { diagfwd_open(i, TYPE_DATA); diagfwd_open(i, TYPE_CMD); } queue_work(driver->diag_real_time_wq, &driver->diag_real_time_work); return 0; } static int diagfwd_mux_close(int id, int mode) { switch (mode) { case DIAG_USB_MODE: driver->usb_connected = 0; break; case DIAG_MEMORY_DEVICE_MODE: break; default: return -EINVAL; } if ((mode == DIAG_USB_MODE && driver->logging_mode == DIAG_MEMORY_DEVICE_MODE) || (mode == DIAG_MEMORY_DEVICE_MODE && driver->logging_mode == DIAG_USB_MODE)) { /* * This case indicates that the USB is removed * but there is a client running in background * with Memory Device mode. */ } else { /* * With clearing of masks on ODL exit and * USB disconnection, closing of the channel is * not needed.This enables read and drop of stale packets. */ /* Re enable HDLC encoding */ pr_debug("diag: In %s, re-enabling HDLC encoding\n", __func__); mutex_lock(&driver->hdlc_disable_mutex); if (driver->md_session_mode == DIAG_MD_NONE) driver->hdlc_disabled = 0; mutex_unlock(&driver->hdlc_disable_mutex); queue_work(driver->diag_wq, &(driver->update_user_clients)); } queue_work(driver->diag_real_time_wq, &driver->diag_real_time_work); return 0; } static uint8_t hdlc_reset; static void hdlc_reset_timer_start(struct diag_md_session_t *info) { if (!hdlc_timer_in_progress) { hdlc_timer_in_progress = 1; if (info) mod_timer(&info->hdlc_reset_timer, jiffies + msecs_to_jiffies(200)); else mod_timer(&driver->hdlc_reset_timer, jiffies + msecs_to_jiffies(200)); } } static void hdlc_reset_timer_func(unsigned long data) { pr_debug("diag: In %s, re-enabling HDLC encoding\n", __func__); if (hdlc_reset) { driver->hdlc_disabled = 0; queue_work(driver->diag_wq, &(driver->update_user_clients)); } hdlc_timer_in_progress = 0; } void diag_md_hdlc_reset_timer_func(unsigned long pid) { struct diag_md_session_t *session_info = NULL; pr_debug("diag: In %s, re-enabling HDLC encoding\n", __func__); if (hdlc_reset) { session_info = diag_md_session_get_pid(pid); if (session_info) session_info->hdlc_disabled = 0; queue_work(driver->diag_wq, &(driver->update_md_clients)); } hdlc_timer_in_progress = 0; } static void diag_hdlc_start_recovery(unsigned char *buf, int len, struct diag_md_session_t *info) { int i; static uint32_t bad_byte_counter; unsigned char *start_ptr = NULL; struct diag_pkt_frame_t *actual_pkt = NULL; hdlc_reset = 1; hdlc_reset_timer_start(info); actual_pkt = (struct diag_pkt_frame_t *)buf; for (i = 0; i < len; i++) { if (actual_pkt->start == CONTROL_CHAR && actual_pkt->version == 1 && actual_pkt->length < len && (*(uint8_t *)(buf + sizeof(struct diag_pkt_frame_t) + actual_pkt->length) == CONTROL_CHAR)) { start_ptr = &buf[i]; break; } bad_byte_counter++; if (bad_byte_counter > (DIAG_MAX_REQ_SIZE + sizeof(struct diag_pkt_frame_t) + 1)) { bad_byte_counter = 0; pr_err("diag: In %s, re-enabling HDLC encoding\n", __func__); mutex_lock(&driver->hdlc_disable_mutex); if (info) info->hdlc_disabled = 0; else driver->hdlc_disabled = 0; mutex_unlock(&driver->hdlc_disable_mutex); diag_update_md_clients(HDLC_SUPPORT_TYPE); return; } } if (start_ptr) { /* Discard any partial packet reads */ driver->incoming_pkt.processing = 0; diag_process_non_hdlc_pkt(start_ptr, len - i, info); } } void diag_process_non_hdlc_pkt(unsigned char *buf, int len, struct diag_md_session_t *info) { int err = 0; uint16_t pkt_len = 0; uint32_t read_bytes = 0; const uint32_t header_len = sizeof(struct diag_pkt_frame_t); struct diag_pkt_frame_t *actual_pkt = NULL; unsigned char *data_ptr = NULL; struct diag_partial_pkt_t *partial_pkt = &driver->incoming_pkt; if (!buf || len <= 0) return; if (!partial_pkt->processing) goto start; if (partial_pkt->remaining > len) { if ((partial_pkt->read_len + len) > partial_pkt->capacity) { pr_err("diag: Invalid length %d, %d received in %s\n", partial_pkt->read_len, len, __func__); goto end; } memcpy(partial_pkt->data + partial_pkt->read_len, buf, len); read_bytes += len; buf += read_bytes; partial_pkt->read_len += len; partial_pkt->remaining -= len; } else { if ((partial_pkt->read_len + partial_pkt->remaining) > partial_pkt->capacity) { pr_err("diag: Invalid length during partial read %d, %d received in %s\n", partial_pkt->read_len, partial_pkt->remaining, __func__); goto end; } memcpy(partial_pkt->data + partial_pkt->read_len, buf, partial_pkt->remaining); read_bytes += partial_pkt->remaining; buf += read_bytes; partial_pkt->read_len += partial_pkt->remaining; partial_pkt->remaining = 0; } if (partial_pkt->remaining == 0) { actual_pkt = (struct diag_pkt_frame_t *)(partial_pkt->data); data_ptr = partial_pkt->data + header_len; if (*(uint8_t *)(data_ptr + actual_pkt->length) != CONTROL_CHAR) diag_hdlc_start_recovery(buf, len, info); err = diag_process_apps_pkt(data_ptr, actual_pkt->length, info); if (err) { pr_err("diag: In %s, unable to process incoming data packet, err: %d\n", __func__, err); goto end; } partial_pkt->read_len = 0; partial_pkt->total_len = 0; partial_pkt->processing = 0; goto start; } goto end; start: while (read_bytes < len) { actual_pkt = (struct diag_pkt_frame_t *)buf; pkt_len = actual_pkt->length; if (actual_pkt->start != CONTROL_CHAR) { diag_hdlc_start_recovery(buf, len, info); diag_send_error_rsp(buf, len); goto end; } if (pkt_len + header_len > partial_pkt->capacity) { pr_err("diag: In %s, incoming data is too large for the request buffer %d\n", __func__, pkt_len); diag_hdlc_start_recovery(buf, len, info); break; } if ((pkt_len + header_len) > (len - read_bytes)) { partial_pkt->read_len = len - read_bytes; partial_pkt->total_len = pkt_len + header_len; partial_pkt->remaining = partial_pkt->total_len - partial_pkt->read_len; partial_pkt->processing = 1; memcpy(partial_pkt->data, buf, partial_pkt->read_len); break; } data_ptr = buf + header_len; if (*(uint8_t *)(data_ptr + actual_pkt->length) != CONTROL_CHAR) diag_hdlc_start_recovery(buf, len, info); else hdlc_reset = 0; err = diag_process_apps_pkt(data_ptr, actual_pkt->length, info); if (err) break; read_bytes += header_len + pkt_len + 1; buf += header_len + pkt_len + 1; /* advance to next pkt */ } end: return; } static int diagfwd_mux_read_done(unsigned char *buf, int len, int ctxt) { if (!buf || len <= 0) return -EINVAL; if (!driver->hdlc_disabled) diag_process_hdlc_pkt(buf, len, NULL); else diag_process_non_hdlc_pkt(buf, len, NULL); diag_mux_queue_read(ctxt); return 0; } static int diagfwd_mux_write_done(unsigned char *buf, int len, int buf_ctxt, int ctxt) { unsigned long flags; int peripheral = -1; int type = -1; int num = -1; if (!buf || len < 0) return -EINVAL; peripheral = GET_BUF_PERIPHERAL(buf_ctxt); type = GET_BUF_TYPE(buf_ctxt); num = GET_BUF_NUM(buf_ctxt); switch (type) { case TYPE_DATA: if (peripheral >= 0 && peripheral < NUM_PERIPHERALS) { diagfwd_write_done(peripheral, type, num); diag_ws_on_copy(DIAG_WS_MUX); } else if (peripheral == APPS_DATA) { diagmem_free(driver, (unsigned char *)buf, POOL_TYPE_HDLC); buf = NULL; } else { pr_err_ratelimited("diag: Invalid peripheral %d in %s, type: %d\n", peripheral, __func__, type); } break; case TYPE_CMD: if (peripheral >= 0 && peripheral < NUM_PERIPHERALS) { diagfwd_write_done(peripheral, type, num); } else if (peripheral == APPS_DATA) { spin_lock_irqsave(&driver->rsp_buf_busy_lock, flags); driver->rsp_buf_busy = 0; driver->encoded_rsp_len = 0; spin_unlock_irqrestore(&driver->rsp_buf_busy_lock, flags); } else { pr_err_ratelimited("diag: Invalid peripheral %d in %s, type: %d\n", peripheral, __func__, type); } break; default: pr_err_ratelimited("diag: Incorrect data type %d, buf_ctxt: %d in %s\n", type, buf_ctxt, __func__); break; } return 0; } static struct diag_mux_ops diagfwd_mux_ops = { .open = diagfwd_mux_open, .close = diagfwd_mux_close, .read_done = diagfwd_mux_read_done, .write_done = diagfwd_mux_write_done }; int diagfwd_init(void) { int ret; int i; wrap_enabled = 0; wrap_count = 0; driver->use_device_tree = has_device_tree(); for (i = 0; i < DIAG_NUM_PROC; i++) driver->real_time_mode[i] = 1; driver->supports_separate_cmdrsp = 1; driver->supports_apps_hdlc_encoding = 1; mutex_init(&driver->diag_hdlc_mutex); mutex_init(&driver->diag_cntl_mutex); mutex_init(&driver->mode_lock); driver->encoded_rsp_buf = kzalloc(DIAG_MAX_HDLC_BUF_SIZE + APF_DIAG_PADDING, GFP_KERNEL); if (!driver->encoded_rsp_buf) goto err; kmemleak_not_leak(driver->encoded_rsp_buf); hdlc_decode = kzalloc(sizeof(struct diag_hdlc_decode_type), GFP_KERNEL); if (!hdlc_decode) goto err; setup_timer(&driver->hdlc_reset_timer, hdlc_reset_timer_func, 0); kmemleak_not_leak(hdlc_decode); driver->encoded_rsp_len = 0; driver->rsp_buf_busy = 0; spin_lock_init(&driver->rsp_buf_busy_lock); driver->user_space_data_busy = 0; driver->hdlc_buf_len = 0; INIT_LIST_HEAD(&driver->cmd_reg_list); driver->cmd_reg_count = 0; mutex_init(&driver->cmd_reg_mutex); for (i = 0; i < NUM_PERIPHERALS; i++) { driver->feature[i].separate_cmd_rsp = 0; driver->feature[i].stm_support = DISABLE_STM; driver->feature[i].rcvd_feature_mask = 0; driver->feature[i].peripheral_buffering = 0; driver->feature[i].encode_hdlc = 0; driver->feature[i].mask_centralization = 0; driver->feature[i].log_on_demand = 0; driver->feature[i].sent_feature_mask = 0; driver->buffering_mode[i].peripheral = i; driver->buffering_mode[i].mode = DIAG_BUFFERING_MODE_STREAMING; driver->buffering_mode[i].high_wm_val = DEFAULT_HIGH_WM_VAL; driver->buffering_mode[i].low_wm_val = DEFAULT_LOW_WM_VAL; } for (i = 0; i < NUM_STM_PROCESSORS; i++) { driver->stm_state_requested[i] = DISABLE_STM; driver->stm_state[i] = DISABLE_STM; } if (driver->hdlc_buf == NULL) { driver->hdlc_buf = kzalloc(DIAG_MAX_HDLC_BUF_SIZE, GFP_KERNEL); if (!driver->hdlc_buf) goto err; kmemleak_not_leak(driver->hdlc_buf); } if (driver->user_space_data_buf == NULL) driver->user_space_data_buf = kzalloc(USER_SPACE_DATA, GFP_KERNEL); if (driver->user_space_data_buf == NULL) goto err; kmemleak_not_leak(driver->user_space_data_buf); if (driver->client_map == NULL && (driver->client_map = kzalloc ((driver->num_clients) * sizeof(struct diag_client_map), GFP_KERNEL)) == NULL) goto err; kmemleak_not_leak(driver->client_map); if (driver->data_ready == NULL && (driver->data_ready = kzalloc(driver->num_clients * sizeof(int) , GFP_KERNEL)) == NULL) goto err; kmemleak_not_leak(driver->data_ready); if (driver->apps_req_buf == NULL) { driver->apps_req_buf = kzalloc(DIAG_MAX_REQ_SIZE, GFP_KERNEL); if (!driver->apps_req_buf) goto err; kmemleak_not_leak(driver->apps_req_buf); } if (driver->dci_pkt_buf == NULL) { driver->dci_pkt_buf = kzalloc(DCI_BUF_SIZE, GFP_KERNEL); if (!driver->dci_pkt_buf) goto err; kmemleak_not_leak(driver->dci_pkt_buf); } if (driver->apps_rsp_buf == NULL) { driver->apps_rsp_buf = kzalloc(DIAG_MAX_RSP_SIZE, GFP_KERNEL); if (driver->apps_rsp_buf == NULL) goto err; kmemleak_not_leak(driver->apps_rsp_buf); } driver->diag_wq = create_singlethread_workqueue("diag_wq"); if (!driver->diag_wq) goto err; ret = diag_mux_register(DIAG_LOCAL_PROC, DIAG_LOCAL_PROC, &diagfwd_mux_ops); if (ret) { pr_err("diag: Unable to register with USB, err: %d\n", ret); goto err; } return 0; err: pr_err("diag: In %s, couldn't initialize diag\n", __func__); diag_usb_exit(DIAG_USB_LOCAL); kfree(driver->encoded_rsp_buf); kfree(driver->hdlc_buf); kfree(driver->client_map); kfree(driver->data_ready); kfree(driver->apps_req_buf); kfree(driver->dci_pkt_buf); kfree(driver->apps_rsp_buf); kfree(hdlc_decode); kfree(driver->user_space_data_buf); if (driver->diag_wq) destroy_workqueue(driver->diag_wq); return -ENOMEM; } void diagfwd_exit(void) { kfree(driver->encoded_rsp_buf); kfree(driver->hdlc_buf); kfree(hdlc_decode); kfree(driver->client_map); kfree(driver->data_ready); kfree(driver->apps_req_buf); kfree(driver->dci_pkt_buf); kfree(driver->apps_rsp_buf); kfree(driver->user_space_data_buf); destroy_workqueue(driver->diag_wq); }