blob: 558e362a6b2231d161a17a3a453c705fb3e415b1 [file] [log] [blame]
/* Copyright (c) 2014-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 <linux/slab.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/device.h>
#include <linux/err.h>
#include <linux/sched.h>
#include <linux/ratelimit.h>
#include <linux/workqueue.h>
#include <linux/diagchar.h>
#include <linux/delay.h>
#include <linux/kmemleak.h>
#include <linux/uaccess.h>
#include "diagchar.h"
#include "diag_memorydevice.h"
#include "diagfwd_bridge.h"
#include "diag_mux.h"
#include "diagmem.h"
#include "diagfwd.h"
#include "diagfwd_peripheral.h"
struct diag_md_info diag_md[NUM_DIAG_MD_DEV] = {
{
.id = DIAG_MD_LOCAL,
.ctx = 0,
.mempool = POOL_TYPE_MUX_APPS,
.num_tbl_entries = 0,
.tbl = NULL,
.ops = NULL,
},
#ifdef CONFIG_DIAGFWD_BRIDGE_CODE
{
.id = DIAG_MD_MDM,
.ctx = 0,
.mempool = POOL_TYPE_MDM_MUX,
.num_tbl_entries = 0,
.tbl = NULL,
.ops = NULL,
},
{
.id = DIAG_MD_MDM2,
.ctx = 0,
.mempool = POOL_TYPE_MDM2_MUX,
.num_tbl_entries = 0,
.tbl = NULL,
.ops = NULL,
},
{
.id = DIAG_MD_SMUX,
.ctx = 0,
.mempool = POOL_TYPE_QSC_MUX,
.num_tbl_entries = 0,
.tbl = NULL,
.ops = NULL,
}
#endif
};
int diag_md_register(int id, int ctx, struct diag_mux_ops *ops)
{
if (id < 0 || id >= NUM_DIAG_MD_DEV || !ops)
return -EINVAL;
diag_md[id].ops = ops;
diag_md[id].ctx = ctx;
return 0;
}
void diag_md_open_all(void)
{
int i;
struct diag_md_info *ch = NULL;
for (i = 0; i < NUM_DIAG_MD_DEV; i++) {
ch = &diag_md[i];
if (ch->ops && ch->ops->open)
ch->ops->open(ch->ctx, DIAG_MEMORY_DEVICE_MODE);
}
}
void diag_md_close_all(void)
{
int i, j;
unsigned long flags;
struct diag_md_info *ch = NULL;
struct diag_buf_tbl_t *entry = NULL;
for (i = 0; i < NUM_DIAG_MD_DEV; i++) {
ch = &diag_md[i];
if (ch->ops && ch->ops->close)
ch->ops->close(ch->ctx, DIAG_MEMORY_DEVICE_MODE);
/*
* When we close the Memory device mode, make sure we flush the
* internal buffers in the table so that there are no stale
* entries.
*/
spin_lock_irqsave(&ch->lock, flags);
for (j = 0; j < ch->num_tbl_entries; j++) {
entry = &ch->tbl[j];
if (entry->len <= 0)
continue;
if (ch->ops && ch->ops->write_done)
ch->ops->write_done(entry->buf, entry->len,
entry->ctx,
DIAG_MEMORY_DEVICE_MODE);
entry->buf = NULL;
entry->len = 0;
entry->ctx = 0;
}
spin_unlock_irqrestore(&ch->lock, flags);
}
diag_ws_reset(DIAG_WS_MUX);
}
int diag_md_write(int id, unsigned char *buf, int len, int ctx)
{
int i;
uint8_t found = 0;
unsigned long flags;
struct diag_md_info *ch = NULL;
uint8_t peripheral;
struct diag_md_session_t *session_info = NULL;
if (id < 0 || id >= NUM_DIAG_MD_DEV || id >= DIAG_NUM_PROC)
return -EINVAL;
if (!buf || len < 0)
return -EINVAL;
peripheral = GET_BUF_PERIPHERAL(ctx);
if (peripheral > NUM_PERIPHERALS)
return -EINVAL;
session_info = diag_md_session_get_peripheral(peripheral);
if (!session_info)
return -EIO;
ch = &diag_md[id];
spin_lock_irqsave(&ch->lock, flags);
for (i = 0; i < ch->num_tbl_entries && !found; i++) {
if (ch->tbl[i].buf != buf)
continue;
found = 1;
pr_err_ratelimited("diag: trying to write the same buffer buf: %pK, ctxt: %d len: %d at i: %d back to the table, proc: %d, mode: %d\n",
buf, ctx, ch->tbl[i].len,
i, id, driver->logging_mode);
}
spin_unlock_irqrestore(&ch->lock, flags);
if (found)
return -ENOMEM;
spin_lock_irqsave(&ch->lock, flags);
for (i = 0; i < ch->num_tbl_entries && !found; i++) {
if (ch->tbl[i].len == 0) {
ch->tbl[i].buf = buf;
ch->tbl[i].len = len;
ch->tbl[i].ctx = ctx;
found = 1;
diag_ws_on_read(DIAG_WS_MUX, len);
}
}
spin_unlock_irqrestore(&ch->lock, flags);
if (!found) {
pr_err_ratelimited("diag: Unable to find an empty space in table, please reduce logging rate, proc: %d\n",
id);
return -ENOMEM;
}
found = 0;
for (i = 0; i < driver->num_clients && !found; i++) {
if ((driver->client_map[i].pid !=
session_info->pid) ||
(driver->client_map[i].pid == 0))
continue;
found = 1;
driver->data_ready[i] |= USER_SPACE_DATA_TYPE;
pr_debug("diag: wake up logging process\n");
wake_up_interruptible(&driver->wait_q);
}
if (!found)
return -EINVAL;
return 0;
}
int diag_md_copy_to_user(char __user *buf, int *pret, size_t buf_size,
struct diag_md_session_t *info)
{
int i, j;
int err = 0;
int ret = *pret;
int num_data = 0;
int remote_token;
unsigned long flags;
struct diag_md_info *ch = NULL;
struct diag_buf_tbl_t *entry = NULL;
uint8_t drain_again = 0;
uint8_t peripheral = 0;
struct diag_md_session_t *session_info = NULL;
for (i = 0; i < NUM_DIAG_MD_DEV && !err; i++) {
ch = &diag_md[i];
for (j = 0; j < ch->num_tbl_entries && !err; j++) {
entry = &ch->tbl[j];
if (entry->len <= 0)
continue;
peripheral = GET_BUF_PERIPHERAL(entry->ctx);
/* Account for Apps data as well */
if (peripheral > NUM_PERIPHERALS)
goto drop_data;
session_info =
diag_md_session_get_peripheral(peripheral);
if (session_info && info &&
(session_info->pid != info->pid))
continue;
if ((info && (info->peripheral_mask &
MD_PERIPHERAL_MASK(peripheral)) == 0))
goto drop_data;
/*
* If the data is from remote processor, copy the remote
* token first
*/
if (i > 0) {
if ((ret + (3 * sizeof(int)) + entry->len) >=
buf_size) {
drain_again = 1;
break;
}
} else {
if ((ret + (2 * sizeof(int)) + entry->len) >=
buf_size) {
drain_again = 1;
break;
}
}
if (i > 0) {
remote_token = diag_get_remote(i);
err = copy_to_user(buf + ret, &remote_token,
sizeof(int));
if (err)
goto drop_data;
ret += sizeof(int);
}
/* Copy the length of data being passed */
err = copy_to_user(buf + ret, (void *)&(entry->len),
sizeof(int));
if (err)
goto drop_data;
ret += sizeof(int);
/* Copy the actual data being passed */
err = copy_to_user(buf + ret, (void *)entry->buf,
entry->len);
if (err)
goto drop_data;
ret += entry->len;
/*
* The data is now copied to the user space client,
* Notify that the write is complete and delete its
* entry from the table
*/
num_data++;
drop_data:
spin_lock_irqsave(&ch->lock, flags);
if (ch->ops && ch->ops->write_done)
ch->ops->write_done(entry->buf, entry->len,
entry->ctx,
DIAG_MEMORY_DEVICE_MODE);
diag_ws_on_copy(DIAG_WS_MUX);
entry->buf = NULL;
entry->len = 0;
entry->ctx = 0;
spin_unlock_irqrestore(&ch->lock, flags);
}
}
*pret = ret;
err = copy_to_user(buf + sizeof(int), (void *)&num_data, sizeof(int));
diag_ws_on_copy_complete(DIAG_WS_MUX);
if (drain_again)
chk_logging_wakeup();
return err;
}
int diag_md_close_peripheral(int id, uint8_t peripheral)
{
int i;
uint8_t found = 0;
unsigned long flags;
struct diag_md_info *ch = NULL;
struct diag_buf_tbl_t *entry = NULL;
if (id < 0 || id >= NUM_DIAG_MD_DEV || id >= DIAG_NUM_PROC)
return -EINVAL;
ch = &diag_md[id];
spin_lock_irqsave(&ch->lock, flags);
for (i = 0; i < ch->num_tbl_entries && !found; i++) {
entry = &ch->tbl[i];
if (GET_BUF_PERIPHERAL(entry->ctx) != peripheral)
continue;
found = 1;
if (ch->ops && ch->ops->write_done) {
ch->ops->write_done(entry->buf, entry->len,
entry->ctx,
DIAG_MEMORY_DEVICE_MODE);
entry->buf = NULL;
entry->len = 0;
entry->ctx = 0;
}
}
spin_unlock_irqrestore(&ch->lock, flags);
return 0;
}
int diag_md_init(void)
{
int i, j;
struct diag_md_info *ch = NULL;
for (i = 0; i < NUM_DIAG_MD_DEV; i++) {
ch = &diag_md[i];
ch->num_tbl_entries = diag_mempools[ch->mempool].poolsize;
ch->tbl = kzalloc(ch->num_tbl_entries *
sizeof(struct diag_buf_tbl_t),
GFP_KERNEL);
if (!ch->tbl)
goto fail;
for (j = 0; j < ch->num_tbl_entries; j++) {
ch->tbl[j].buf = NULL;
ch->tbl[j].len = 0;
ch->tbl[j].ctx = 0;
spin_lock_init(&(ch->lock));
}
}
return 0;
fail:
diag_md_exit();
return -ENOMEM;
}
void diag_md_exit(void)
{
int i;
struct diag_md_info *ch = NULL;
for (i = 0; i < NUM_DIAG_MD_DEV; i++) {
ch = &diag_md[i];
kfree(ch->tbl);
ch->num_tbl_entries = 0;
ch->ops = NULL;
}
}