/* Copyright (c) 2015, 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 "diagchar.h" #include "diagchar_hdlc.h" #include "diagfwd_peripheral.h" #include "diagfwd_cntl.h" #include "diag_masks.h" #include "diag_dci.h" #include "diagfwd.h" #include "diagfwd_smd.h" #include "diagfwd_socket.h" #include "diag_mux.h" #include "diag_ipc_logging.h" struct data_header { uint8_t control_char; uint8_t version; uint16_t length; }; static struct diagfwd_info *early_init_info[NUM_TRANSPORT]; static void diagfwd_queue_read(struct diagfwd_info *fwd_info); static void diagfwd_buffers_exit(struct diagfwd_info *fwd_info); static void diagfwd_cntl_open(struct diagfwd_info *fwd_info); static void diagfwd_cntl_close(struct diagfwd_info *fwd_info); static void diagfwd_dci_open(struct diagfwd_info *fwd_info); static void diagfwd_dci_close(struct diagfwd_info *fwd_info); static void diagfwd_data_read_done(struct diagfwd_info *fwd_info, unsigned char *buf, int len); static void diagfwd_cntl_read_done(struct diagfwd_info *fwd_info, unsigned char *buf, int len); static void diagfwd_dci_read_done(struct diagfwd_info *fwd_info, unsigned char *buf, int len); struct diagfwd_info peripheral_info[NUM_TYPES][NUM_PERIPHERALS]; static struct diag_channel_ops data_ch_ops = { .open = NULL, .close = NULL, .read_done = diagfwd_data_read_done }; static struct diag_channel_ops cntl_ch_ops = { .open = diagfwd_cntl_open, .close = diagfwd_cntl_close, .read_done = diagfwd_cntl_read_done }; static struct diag_channel_ops dci_ch_ops = { .open = diagfwd_dci_open, .close = diagfwd_dci_close, .read_done = diagfwd_dci_read_done }; static void diagfwd_cntl_open(struct diagfwd_info *fwd_info) { if (!fwd_info) return; diag_cntl_channel_open(fwd_info); } static void diagfwd_cntl_close(struct diagfwd_info *fwd_info) { if (!fwd_info) return; diag_cntl_channel_close(fwd_info); } static void diagfwd_dci_open(struct diagfwd_info *fwd_info) { if (!fwd_info) return; diag_dci_notify_client(PERIPHERAL_MASK(fwd_info->peripheral), DIAG_STATUS_OPEN, DCI_LOCAL_PROC); } static void diagfwd_dci_close(struct diagfwd_info *fwd_info) { if (!fwd_info) return; diag_dci_notify_client(PERIPHERAL_MASK(fwd_info->peripheral), DIAG_STATUS_CLOSED, DCI_LOCAL_PROC); } static int diag_add_hdlc_encoding(unsigned char *dest_buf, int *dest_len, 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 }; struct data_header *header; int header_size = sizeof(struct data_header); uint8_t *end_control_char = NULL; uint8_t *payload = NULL; uint8_t *temp_buf = NULL; uint8_t *temp_encode_buf = NULL; int src_pkt_len; int encoded_pkt_length; int max_size; int total_processed = 0; int bytes_remaining; int err = 0; uint8_t loop_count = 0; if (!dest_buf || !dest_len || !buf) return -EIO; temp_buf = buf; temp_encode_buf = dest_buf; bytes_remaining = *dest_len; while (total_processed < len) { loop_count++; header = (struct data_header *)temp_buf; /* Perform initial error checking */ if (header->control_char != CONTROL_CHAR || header->version != 1) { err = -EINVAL; break; } if (header->length >= bytes_remaining) break; payload = temp_buf + header_size; end_control_char = payload + header->length; if (*end_control_char != CONTROL_CHAR) { err = -EINVAL; break; } max_size = 2 * header->length + 3; if (bytes_remaining < max_size) { err = -EINVAL; break; } /* Prepare for encoding the data */ send.state = DIAG_STATE_START; send.pkt = payload; send.last = (void *)(payload + header->length - 1); send.terminate = 1; enc.dest = temp_encode_buf; enc.dest_last = (void *)(temp_encode_buf + max_size); enc.crc = 0; diag_hdlc_encode(&send, &enc); /* Prepare for next packet */ src_pkt_len = (header_size + header->length + 1); total_processed += src_pkt_len; temp_buf += src_pkt_len; encoded_pkt_length = (uint8_t *)enc.dest - temp_encode_buf; bytes_remaining -= encoded_pkt_length; temp_encode_buf = enc.dest; } *dest_len = (int)(temp_encode_buf - dest_buf); return err; } static int check_bufsize_for_encoding(struct diagfwd_buf_t *buf, uint32_t len) { uint32_t max_size = 0; unsigned char *temp_buf = NULL; if (!buf || len == 0) return -EINVAL; max_size = (2 * len) + 3; if (max_size > PERIPHERAL_BUF_SZ) { if (max_size > MAX_PERIPHERAL_HDLC_BUF_SZ) { pr_err("diag: In %s, max_size is going beyond limit %d\n", __func__, max_size); max_size = MAX_PERIPHERAL_HDLC_BUF_SZ; } if (buf->len < max_size) { temp_buf = krealloc(buf->data, max_size + APF_DIAG_PADDING, GFP_KERNEL); if (!temp_buf) return -ENOMEM; buf->data = temp_buf; buf->len = max_size; } } return buf->len; } static void diagfwd_data_read_done(struct diagfwd_info *fwd_info, unsigned char *buf, int len) { int err = 0; int write_len = 0; unsigned char *write_buf = NULL; struct diagfwd_buf_t *temp_buf = NULL; struct diag_md_session_t *session_info = NULL; uint8_t hdlc_disabled = 0; if (!fwd_info || !buf || len <= 0) { diag_ws_release(); return; } switch (fwd_info->type) { case TYPE_DATA: case TYPE_CMD: break; default: pr_err_ratelimited("diag: In %s, invalid type %d for peripheral %d\n", __func__, fwd_info->type, fwd_info->peripheral); diag_ws_release(); return; } mutex_lock(&driver->hdlc_disable_mutex); mutex_lock(&fwd_info->data_mutex); session_info = diag_md_session_get_peripheral(fwd_info->peripheral); if (session_info) hdlc_disabled = session_info->hdlc_disabled; else hdlc_disabled = driver->hdlc_disabled; if (!driver->feature[fwd_info->peripheral].encode_hdlc) { if (fwd_info->buf_1 && fwd_info->buf_1->data == buf) { temp_buf = fwd_info->buf_1; write_buf = fwd_info->buf_1->data; } else if (fwd_info->buf_2 && fwd_info->buf_2->data == buf) { temp_buf = fwd_info->buf_2; write_buf = fwd_info->buf_2->data; } else { pr_err("diag: In %s, no match for buffer %p, peripheral %d, type: %d\n", __func__, buf, fwd_info->peripheral, fwd_info->type); goto end; } write_len = len; } else if (hdlc_disabled) { /* The data is raw and and on APPS side HDLC is disabled */ if (fwd_info->buf_1 && fwd_info->buf_1->data_raw == buf) { temp_buf = fwd_info->buf_1; } else if (fwd_info->buf_2 && fwd_info->buf_2->data_raw == buf) { temp_buf = fwd_info->buf_2; } else { pr_err("diag: In %s, no match for non encode buffer %p, peripheral %d, type: %d\n", __func__, buf, fwd_info->peripheral, fwd_info->type); goto end; } if (len > PERIPHERAL_BUF_SZ) { pr_err("diag: In %s, Incoming buffer too large %d, peripheral %d, type: %d\n", __func__, len, fwd_info->peripheral, fwd_info->type); goto end; } write_len = len; write_buf = buf; } else { if (fwd_info->buf_1 && fwd_info->buf_1->data_raw == buf) { temp_buf = fwd_info->buf_1; } else if (fwd_info->buf_2 && fwd_info->buf_2->data_raw == buf) { temp_buf = fwd_info->buf_2; } else { pr_err("diag: In %s, no match for non encode buffer %p, peripheral %d, type: %d\n", __func__, buf, fwd_info->peripheral, fwd_info->type); goto end; } write_len = check_bufsize_for_encoding(temp_buf, len); if (write_len <= 0) { pr_err("diag: error in checking buf for encoding\n"); goto end; } write_buf = temp_buf->data; err = diag_add_hdlc_encoding(write_buf, &write_len, buf, len); if (err) { pr_err("diag: error in adding hdlc encoding\n"); goto end; } } if (write_len > 0) { err = diag_mux_write(DIAG_LOCAL_PROC, write_buf, write_len, temp_buf->ctxt); if (err) { pr_err_ratelimited("diag: In %s, unable to write to mux error: %d\n", __func__, err); goto end; } } mutex_unlock(&fwd_info->data_mutex); mutex_unlock(&driver->hdlc_disable_mutex); diagfwd_queue_read(fwd_info); return; end: diag_ws_release(); mutex_unlock(&fwd_info->data_mutex); mutex_unlock(&driver->hdlc_disable_mutex); if (temp_buf) { diagfwd_write_done(fwd_info->peripheral, fwd_info->type, GET_BUF_NUM(temp_buf->ctxt)); } diagfwd_queue_read(fwd_info); return; } static void diagfwd_cntl_read_done(struct diagfwd_info *fwd_info, unsigned char *buf, int len) { if (!fwd_info) { diag_ws_release(); return; } if (fwd_info->type != TYPE_CNTL) { pr_err("diag: In %s, invalid type %d for peripheral %d\n", __func__, fwd_info->type, fwd_info->peripheral); diag_ws_release(); return; } diag_ws_on_read(DIAG_WS_MUX, len); diag_cntl_process_read_data(fwd_info, buf, len); /* * Control packets are not consumed by the clients. Mimic * consumption by setting and clearing the wakeup source copy_count * explicitly. */ diag_ws_on_copy_fail(DIAG_WS_MUX); /* Reset the buffer in_busy value after processing the data */ if (fwd_info->buf_1) atomic_set(&fwd_info->buf_1->in_busy, 0); diagfwd_queue_read(fwd_info); diagfwd_queue_read(&peripheral_info[TYPE_DATA][fwd_info->peripheral]); diagfwd_queue_read(&peripheral_info[TYPE_CMD][fwd_info->peripheral]); } static void diagfwd_dci_read_done(struct diagfwd_info *fwd_info, unsigned char *buf, int len) { if (!fwd_info) return; switch (fwd_info->type) { case TYPE_DCI: case TYPE_DCI_CMD: break; default: pr_err("diag: In %s, invalid type %d for peripheral %d\n", __func__, fwd_info->type, fwd_info->peripheral); return; } diag_dci_process_peripheral_data(fwd_info, (void *)buf, len); /* Reset the buffer in_busy value after processing the data */ if (fwd_info->buf_1) atomic_set(&fwd_info->buf_1->in_busy, 0); diagfwd_queue_read(fwd_info); } static void diagfwd_reset_buffers(struct diagfwd_info *fwd_info, unsigned char *buf) { if (!fwd_info || !buf) return; if (!driver->feature[fwd_info->peripheral].encode_hdlc) { if (fwd_info->buf_1 && fwd_info->buf_1->data == buf) atomic_set(&fwd_info->buf_1->in_busy, 0); else if (fwd_info->buf_2 && fwd_info->buf_2->data == buf) atomic_set(&fwd_info->buf_2->in_busy, 0); } else { if (fwd_info->buf_1 && fwd_info->buf_1->data_raw == buf) atomic_set(&fwd_info->buf_1->in_busy, 0); else if (fwd_info->buf_2 && fwd_info->buf_2->data_raw == buf) atomic_set(&fwd_info->buf_2->in_busy, 0); } } int diagfwd_peripheral_init(void) { uint8_t peripheral; uint8_t transport; uint8_t type; struct diagfwd_info *fwd_info = NULL; for (transport = 0; transport < NUM_TRANSPORT; transport++) { early_init_info[transport] = kzalloc( sizeof(struct diagfwd_info) * NUM_PERIPHERALS, GFP_KERNEL); if (!early_init_info[transport]) return -ENOMEM; kmemleak_not_leak(early_init_info[transport]); } for (peripheral = 0; peripheral < NUM_PERIPHERALS; peripheral++) { for (transport = 0; transport < NUM_TRANSPORT; transport++) { fwd_info = &early_init_info[transport][peripheral]; fwd_info->peripheral = peripheral; fwd_info->type = TYPE_CNTL; fwd_info->transport = transport; fwd_info->ctxt = NULL; fwd_info->p_ops = NULL; fwd_info->ch_open = 0; fwd_info->inited = 1; fwd_info->read_bytes = 0; fwd_info->write_bytes = 0; spin_lock_init(&fwd_info->buf_lock); mutex_init(&fwd_info->data_mutex); } } for (peripheral = 0; peripheral < NUM_PERIPHERALS; peripheral++) { for (type = 0; type < NUM_TYPES; type++) { fwd_info = &peripheral_info[type][peripheral]; fwd_info->peripheral = peripheral; fwd_info->type = type; fwd_info->ctxt = NULL; fwd_info->p_ops = NULL; fwd_info->ch_open = 0; fwd_info->read_bytes = 0; fwd_info->write_bytes = 0; spin_lock_init(&fwd_info->buf_lock); mutex_init(&fwd_info->data_mutex); /* * This state shouldn't be set for Control channels * during initialization. This is set when the feature * mask is received for the first time. */ if (type != TYPE_CNTL) fwd_info->inited = 1; } driver->diagfwd_data[peripheral] = &peripheral_info[TYPE_DATA][peripheral]; driver->diagfwd_cntl[peripheral] = &peripheral_info[TYPE_CNTL][peripheral]; driver->diagfwd_dci[peripheral] = &peripheral_info[TYPE_DCI][peripheral]; driver->diagfwd_cmd[peripheral] = &peripheral_info[TYPE_CMD][peripheral]; driver->diagfwd_dci_cmd[peripheral] = &peripheral_info[TYPE_DCI_CMD][peripheral]; } diag_smd_init(); if (driver->supports_sockets) diag_socket_init(); return 0; } void diagfwd_peripheral_exit(void) { uint8_t peripheral; uint8_t type; struct diagfwd_info *fwd_info = NULL; diag_smd_exit(); diag_socket_exit(); for (peripheral = 0; peripheral < NUM_PERIPHERALS; peripheral++) { for (type = 0; type < NUM_TYPES; type++) { fwd_info = &peripheral_info[type][peripheral]; fwd_info->ctxt = NULL; fwd_info->p_ops = NULL; fwd_info->ch_open = 0; diagfwd_buffers_exit(fwd_info); } } for (peripheral = 0; peripheral < NUM_PERIPHERALS; peripheral++) { driver->diagfwd_data[peripheral] = NULL; driver->diagfwd_cntl[peripheral] = NULL; driver->diagfwd_dci[peripheral] = NULL; driver->diagfwd_cmd[peripheral] = NULL; driver->diagfwd_dci_cmd[peripheral] = NULL; } kfree(early_init_info); } int diagfwd_cntl_register(uint8_t transport, uint8_t peripheral, void *ctxt, struct diag_peripheral_ops *ops, struct diagfwd_info **fwd_ctxt) { struct diagfwd_info *fwd_info = NULL; if (!ctxt || !ops) return -EIO; if (transport >= NUM_TRANSPORT || peripheral >= NUM_PERIPHERALS) return -EINVAL; fwd_info = &early_init_info[transport][peripheral]; *fwd_ctxt = &early_init_info[transport][peripheral]; fwd_info->ctxt = ctxt; fwd_info->p_ops = ops; fwd_info->c_ops = &cntl_ch_ops; return 0; } int diagfwd_register(uint8_t transport, uint8_t peripheral, uint8_t type, void *ctxt, struct diag_peripheral_ops *ops, struct diagfwd_info **fwd_ctxt) { struct diagfwd_info *fwd_info = NULL; if (peripheral >= NUM_PERIPHERALS || type >= NUM_TYPES || !ctxt || !ops || transport >= NUM_TRANSPORT) { pr_err("diag: In %s, returning error\n", __func__); return -EIO; } fwd_info = &peripheral_info[type][peripheral]; *fwd_ctxt = &peripheral_info[type][peripheral]; fwd_info->ctxt = ctxt; fwd_info->p_ops = ops; fwd_info->transport = transport; fwd_info->ch_open = 0; switch (type) { case TYPE_DATA: case TYPE_CMD: fwd_info->c_ops = &data_ch_ops; break; case TYPE_DCI: case TYPE_DCI_CMD: fwd_info->c_ops = &dci_ch_ops; break; default: pr_err("diag: In %s, invalid type: %d\n", __func__, type); return -EINVAL; } if (atomic_read(&fwd_info->opened) && fwd_info->p_ops && fwd_info->p_ops->open) { /* * The registration can happen late, like in the case of * sockets. fwd_info->opened reflects diag_state. Propogate the * state to the peipherals. */ fwd_info->p_ops->open(fwd_info->ctxt); } return 0; } void diagfwd_deregister(uint8_t peripheral, uint8_t type, void *ctxt) { struct diagfwd_info *fwd_info = NULL; if (peripheral >= NUM_PERIPHERALS || type >= NUM_TYPES || !ctxt) return; fwd_info = &peripheral_info[type][peripheral]; if (fwd_info->ctxt != ctxt) { pr_err("diag: In %s, unable to find a match for p: %d t: %d\n", __func__, peripheral, type); return; } fwd_info->ctxt = NULL; fwd_info->p_ops = NULL; fwd_info->ch_open = 0; diagfwd_buffers_exit(fwd_info); switch (type) { case TYPE_DATA: driver->diagfwd_data[peripheral] = NULL; break; case TYPE_CNTL: driver->diagfwd_cntl[peripheral] = NULL; break; case TYPE_DCI: driver->diagfwd_dci[peripheral] = NULL; break; case TYPE_CMD: driver->diagfwd_cmd[peripheral] = NULL; break; case TYPE_DCI_CMD: driver->diagfwd_dci_cmd[peripheral] = NULL; break; } } void diagfwd_close_transport(uint8_t transport, uint8_t peripheral) { struct diagfwd_info *fwd_info = NULL; struct diagfwd_info *dest_info = NULL; int (*init_fn)(uint8_t) = NULL; void (*invalidate_fn)(void *, struct diagfwd_info *) = NULL; int (*check_channel_state)(void *) = NULL; uint8_t transport_open = 0; if (peripheral >= NUM_PERIPHERALS) return; switch (transport) { case TRANSPORT_SMD: transport_open = TRANSPORT_SOCKET; init_fn = diag_socket_init_peripheral; invalidate_fn = diag_socket_invalidate; check_channel_state = diag_socket_check_state; break; case TRANSPORT_SOCKET: transport_open = TRANSPORT_SMD; init_fn = diag_smd_init_peripheral; invalidate_fn = diag_smd_invalidate; check_channel_state = diag_smd_check_state; break; default: return; } fwd_info = &early_init_info[transport][peripheral]; if (fwd_info->p_ops && fwd_info->p_ops->close) fwd_info->p_ops->close(fwd_info->ctxt); fwd_info = &early_init_info[transport_open][peripheral]; dest_info = &peripheral_info[TYPE_CNTL][peripheral]; dest_info->inited = 1; dest_info->ctxt = fwd_info->ctxt; dest_info->p_ops = fwd_info->p_ops; dest_info->c_ops = fwd_info->c_ops; dest_info->ch_open = fwd_info->ch_open; dest_info->read_bytes = fwd_info->read_bytes; dest_info->write_bytes = fwd_info->write_bytes; dest_info->inited = fwd_info->inited; dest_info->buf_1 = fwd_info->buf_1; dest_info->buf_2 = fwd_info->buf_2; dest_info->transport = fwd_info->transport; invalidate_fn(dest_info->ctxt, dest_info); if (!check_channel_state(dest_info->ctxt)) diagfwd_late_open(dest_info); diagfwd_cntl_open(dest_info); init_fn(peripheral); diagfwd_queue_read(&peripheral_info[TYPE_DATA][peripheral]); diagfwd_queue_read(&peripheral_info[TYPE_CMD][peripheral]); } int diagfwd_write(uint8_t peripheral, uint8_t type, void *buf, int len) { struct diagfwd_info *fwd_info = NULL; int err = 0; uint8_t retry_count = 0; uint8_t max_retries = 3; if (peripheral >= NUM_PERIPHERALS || type >= NUM_TYPES) return -EINVAL; if (type == TYPE_CMD || type == TYPE_DCI_CMD) { if (!driver->feature[peripheral].rcvd_feature_mask || !driver->feature[peripheral].sent_feature_mask) { pr_debug_ratelimited("diag: In %s, feature mask for peripheral: %d not received or sent yet\n", __func__, peripheral); return 0; } if (!driver->feature[peripheral].separate_cmd_rsp) type = (type == TYPE_CMD) ? TYPE_DATA : TYPE_DCI; } fwd_info = &peripheral_info[type][peripheral]; if (!fwd_info->inited || !atomic_read(&fwd_info->opened)) return -ENODEV; if (!(fwd_info->p_ops && fwd_info->p_ops->write && fwd_info->ctxt)) return -EIO; while (retry_count < max_retries) { err = 0; err = fwd_info->p_ops->write(fwd_info->ctxt, buf, len); if (err && err != -ENODEV) { usleep_range(100000, 101000); retry_count++; continue; } break; } if (!err) fwd_info->write_bytes += len; return err; } static void __diag_fwd_open(struct diagfwd_info *fwd_info) { if (!fwd_info) return; atomic_set(&fwd_info->opened, 1); if (!fwd_info->inited) return; if (fwd_info->buf_1) atomic_set(&fwd_info->buf_1->in_busy, 0); if (fwd_info->buf_2) atomic_set(&fwd_info->buf_2->in_busy, 0); if (fwd_info->p_ops && fwd_info->p_ops->open) fwd_info->p_ops->open(fwd_info->ctxt); diagfwd_queue_read(fwd_info); } void diagfwd_early_open(uint8_t peripheral) { uint8_t transport = 0; struct diagfwd_info *fwd_info = NULL; if (peripheral >= NUM_PERIPHERALS) return; for (transport = 0; transport < NUM_TRANSPORT; transport++) { fwd_info = &early_init_info[transport][peripheral]; __diag_fwd_open(fwd_info); } } void diagfwd_open(uint8_t peripheral, uint8_t type) { struct diagfwd_info *fwd_info = NULL; if (peripheral >= NUM_PERIPHERALS || type >= NUM_TYPES) return; fwd_info = &peripheral_info[type][peripheral]; __diag_fwd_open(fwd_info); } void diagfwd_late_open(struct diagfwd_info *fwd_info) { __diag_fwd_open(fwd_info); } void diagfwd_close(uint8_t peripheral, uint8_t type) { struct diagfwd_info *fwd_info = NULL; if (peripheral >= NUM_PERIPHERALS || type >= NUM_TYPES) return; fwd_info = &peripheral_info[type][peripheral]; atomic_set(&fwd_info->opened, 0); if (!fwd_info->inited) return; if (fwd_info->p_ops && fwd_info->p_ops->close) fwd_info->p_ops->close(fwd_info->ctxt); if (fwd_info->buf_1) atomic_set(&fwd_info->buf_1->in_busy, 1); /* * Only Data channels have two buffers. Set both the buffers * to busy on close. */ if (fwd_info->buf_2) atomic_set(&fwd_info->buf_2->in_busy, 1); } int diagfwd_channel_open(struct diagfwd_info *fwd_info) { if (!fwd_info) return -EIO; if (!fwd_info->inited) { pr_debug("diag: In %s, channel is not inited, p: %d, t: %d\n", __func__, fwd_info->peripheral, fwd_info->type); return -EINVAL; } if (fwd_info->ch_open) { pr_debug("diag: In %s, channel is already open, p: %d, t: %d\n", __func__, fwd_info->peripheral, fwd_info->type); return 0; } fwd_info->ch_open = 1; diagfwd_buffers_init(fwd_info); if (fwd_info && fwd_info->c_ops && fwd_info->c_ops->open) fwd_info->c_ops->open(fwd_info); diagfwd_queue_read(fwd_info); DIAG_LOG(DIAG_DEBUG_PERIPHERALS, "p: %d t: %d considered opened\n", fwd_info->peripheral, fwd_info->type); if (atomic_read(&fwd_info->opened)) { if (fwd_info->p_ops && fwd_info->p_ops->open) fwd_info->p_ops->open(fwd_info->ctxt); } return 0; } int diagfwd_channel_close(struct diagfwd_info *fwd_info) { if (!fwd_info) return -EIO; fwd_info->ch_open = 0; if (fwd_info && fwd_info->c_ops && fwd_info->c_ops->close) fwd_info->c_ops->close(fwd_info); if (fwd_info->buf_1 && fwd_info->buf_1->data) atomic_set(&fwd_info->buf_1->in_busy, 0); if (fwd_info->buf_2 && fwd_info->buf_2->data) atomic_set(&fwd_info->buf_2->in_busy, 0); DIAG_LOG(DIAG_DEBUG_PERIPHERALS, "p: %d t: %d considered closed\n", fwd_info->peripheral, fwd_info->type); return 0; } int diagfwd_channel_read_done(struct diagfwd_info *fwd_info, unsigned char *buf, uint32_t len) { if (!fwd_info) { diag_ws_release(); return -EIO; } /* * Diag peripheral layers should send len as 0 if there is any error * in reading data from the transport. Use this information to reset the * in_busy flags. No need to queue read in this case. */ if (len == 0) { diagfwd_reset_buffers(fwd_info, buf); diag_ws_release(); return 0; } if (fwd_info && fwd_info->c_ops && fwd_info->c_ops->read_done) fwd_info->c_ops->read_done(fwd_info, buf, len); fwd_info->read_bytes += len; return 0; } void diagfwd_write_done(uint8_t peripheral, uint8_t type, int ctxt) { struct diagfwd_info *fwd_info = NULL; if (peripheral >= NUM_PERIPHERALS || type >= NUM_TYPES) return; fwd_info = &peripheral_info[type][peripheral]; if (ctxt == 1 && fwd_info->buf_1) atomic_set(&fwd_info->buf_1->in_busy, 0); else if (ctxt == 2 && fwd_info->buf_2) atomic_set(&fwd_info->buf_2->in_busy, 0); else pr_err("diag: In %s, invalid ctxt %d\n", __func__, ctxt); diagfwd_queue_read(fwd_info); } void diagfwd_channel_read(struct diagfwd_info *fwd_info) { int err = 0; uint32_t read_len = 0; unsigned char *read_buf = NULL; struct diagfwd_buf_t *temp_buf = NULL; if (!fwd_info) { diag_ws_release(); return; } if (!fwd_info->inited || !atomic_read(&fwd_info->opened)) { pr_debug("diag: In %s, p: %d, t: %d, inited: %d, opened: %d ch_open: %d\n", __func__, fwd_info->peripheral, fwd_info->type, fwd_info->inited, atomic_read(&fwd_info->opened), fwd_info->ch_open); diag_ws_release(); return; } if (fwd_info->buf_1 && !atomic_read(&fwd_info->buf_1->in_busy)) { temp_buf = fwd_info->buf_1; atomic_set(&temp_buf->in_busy, 1); if (driver->feature[fwd_info->peripheral].encode_hdlc && (fwd_info->type == TYPE_DATA || fwd_info->type == TYPE_CMD)) { read_buf = fwd_info->buf_1->data_raw; read_len = fwd_info->buf_1->len_raw; } else { read_buf = fwd_info->buf_1->data; read_len = fwd_info->buf_1->len; } } else if (fwd_info->buf_2 && !atomic_read(&fwd_info->buf_2->in_busy)) { temp_buf = fwd_info->buf_2; atomic_set(&temp_buf->in_busy, 1); if (driver->feature[fwd_info->peripheral].encode_hdlc && (fwd_info->type == TYPE_DATA || fwd_info->type == TYPE_CMD)) { read_buf = fwd_info->buf_2->data_raw; read_len = fwd_info->buf_2->len_raw; } else { read_buf = fwd_info->buf_2->data; read_len = fwd_info->buf_2->len; } } else { pr_debug("diag: In %s, both buffers are empty for p: %d, t: %d\n", __func__, fwd_info->peripheral, fwd_info->type); } if (!read_buf) { diag_ws_release(); return; } if (!(fwd_info->p_ops && fwd_info->p_ops->read && fwd_info->ctxt)) goto fail_return; DIAG_LOG(DIAG_DEBUG_PERIPHERALS, "issued a read p: %d t: %d buf: %p\n", fwd_info->peripheral, fwd_info->type, read_buf); err = fwd_info->p_ops->read(fwd_info->ctxt, read_buf, read_len); if (err) goto fail_return; return; fail_return: diag_ws_release(); atomic_set(&temp_buf->in_busy, 0); return; } static void diagfwd_queue_read(struct diagfwd_info *fwd_info) { if (!fwd_info) return; if (!fwd_info->inited || !atomic_read(&fwd_info->opened)) { pr_debug("diag: In %s, p: %d, t: %d, inited: %d, opened: %d ch_open: %d\n", __func__, fwd_info->peripheral, fwd_info->type, fwd_info->inited, atomic_read(&fwd_info->opened), fwd_info->ch_open); return; } /* * Don't queue a read on the data and command channels before receiving * the feature mask from the peripheral. We won't know which buffer to * use - HDLC or non HDLC buffer for reading. */ if ((!driver->feature[fwd_info->peripheral].rcvd_feature_mask) && (fwd_info->type != TYPE_CNTL)) { return; } if (fwd_info->p_ops && fwd_info->p_ops->queue_read && fwd_info->ctxt) fwd_info->p_ops->queue_read(fwd_info->ctxt); } void diagfwd_buffers_init(struct diagfwd_info *fwd_info) { unsigned long flags; if (!fwd_info) return; if (!fwd_info->inited) { pr_err("diag: In %s, channel not inited, p: %d, t: %d\n", __func__, fwd_info->peripheral, fwd_info->type); return; } spin_lock_irqsave(&fwd_info->buf_lock, flags); if (!fwd_info->buf_1) { fwd_info->buf_1 = kzalloc(sizeof(struct diagfwd_buf_t), GFP_ATOMIC); if (!fwd_info->buf_1) goto err; kmemleak_not_leak(fwd_info->buf_1); } if (!fwd_info->buf_1->data) { fwd_info->buf_1->data = kzalloc(PERIPHERAL_BUF_SZ + APF_DIAG_PADDING, GFP_ATOMIC); if (!fwd_info->buf_1->data) goto err; fwd_info->buf_1->len = PERIPHERAL_BUF_SZ; kmemleak_not_leak(fwd_info->buf_1->data); fwd_info->buf_1->ctxt = SET_BUF_CTXT(fwd_info->peripheral, fwd_info->type, 1); } if (fwd_info->type == TYPE_DATA) { if (!fwd_info->buf_2) { fwd_info->buf_2 = kzalloc(sizeof(struct diagfwd_buf_t), GFP_ATOMIC); if (!fwd_info->buf_2) goto err; kmemleak_not_leak(fwd_info->buf_2); } if (!fwd_info->buf_2->data) { fwd_info->buf_2->data = kzalloc(PERIPHERAL_BUF_SZ + APF_DIAG_PADDING, GFP_ATOMIC); if (!fwd_info->buf_2->data) goto err; fwd_info->buf_2->len = PERIPHERAL_BUF_SZ; kmemleak_not_leak(fwd_info->buf_2->data); fwd_info->buf_2->ctxt = SET_BUF_CTXT( fwd_info->peripheral, fwd_info->type, 2); } if (driver->supports_apps_hdlc_encoding) { /* In support of hdlc encoding */ if (!fwd_info->buf_1->data_raw) { fwd_info->buf_1->data_raw = kzalloc(PERIPHERAL_BUF_SZ + APF_DIAG_PADDING, GFP_ATOMIC); if (!fwd_info->buf_1->data_raw) goto err; fwd_info->buf_1->len_raw = PERIPHERAL_BUF_SZ; kmemleak_not_leak(fwd_info->buf_1->data_raw); } if (!fwd_info->buf_2->data_raw) { fwd_info->buf_2->data_raw = kzalloc(PERIPHERAL_BUF_SZ + APF_DIAG_PADDING, GFP_ATOMIC); if (!fwd_info->buf_2->data_raw) goto err; fwd_info->buf_2->len_raw = PERIPHERAL_BUF_SZ; kmemleak_not_leak(fwd_info->buf_2->data_raw); } } } if (fwd_info->type == TYPE_CMD && driver->supports_apps_hdlc_encoding) { /* In support of hdlc encoding */ if (!fwd_info->buf_1->data_raw) { fwd_info->buf_1->data_raw = kzalloc(PERIPHERAL_BUF_SZ + APF_DIAG_PADDING, GFP_ATOMIC); if (!fwd_info->buf_1->data_raw) goto err; fwd_info->buf_1->len_raw = PERIPHERAL_BUF_SZ; kmemleak_not_leak(fwd_info->buf_1->data_raw); } } spin_unlock_irqrestore(&fwd_info->buf_lock, flags); return; err: spin_unlock_irqrestore(&fwd_info->buf_lock, flags); diagfwd_buffers_exit(fwd_info); return; } static void diagfwd_buffers_exit(struct diagfwd_info *fwd_info) { unsigned long flags; if (!fwd_info) return; spin_lock_irqsave(&fwd_info->buf_lock, flags); if (fwd_info->buf_1) { kfree(fwd_info->buf_1->data); fwd_info->buf_1->data = NULL; kfree(fwd_info->buf_1->data_raw); fwd_info->buf_1->data_raw = NULL; kfree(fwd_info->buf_1); fwd_info->buf_1 = NULL; } if (fwd_info->buf_2) { kfree(fwd_info->buf_2->data); fwd_info->buf_2->data = NULL; kfree(fwd_info->buf_2->data_raw); fwd_info->buf_2->data_raw = NULL; kfree(fwd_info->buf_2); fwd_info->buf_2 = NULL; } spin_unlock_irqrestore(&fwd_info->buf_lock, flags); }