android_kernel_samsung_msm8976/drivers/platform/msm/mhi/mhi_sys.c

332 lines
9.0 KiB
C

/* Copyright (c) 2014-2017, 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/module.h>
#include <linux/device.h>
#include <linux/uaccess.h>
#include <linux/debugfs.h>
#include <linux/slab.h>
#include "mhi_sys.h"
enum MHI_DEBUG_LEVEL mhi_msg_lvl = MHI_MSG_ERROR;
enum MHI_DEBUG_LEVEL mhi_ipc_log_lvl = MHI_MSG_VERBOSE;
unsigned int mhi_log_override;
module_param(mhi_msg_lvl , uint, S_IRUGO | S_IWUSR);
MODULE_PARM_DESC(mhi_msg_lvl, "dbg lvl");
module_param(mhi_ipc_log_lvl, uint, S_IRUGO | S_IWUSR);
MODULE_PARM_DESC(mhi_ipc_log_lvl, "dbg lvl");
module_param(mhi_log_override , uint, S_IRUGO | S_IWUSR);
MODULE_PARM_DESC(mhi_log_override, "dbg class");
static ssize_t mhi_dbgfs_chan_read(struct file *fp, char __user *buf,
size_t count, loff_t *offp)
{
int amnt_copied = 0;
struct mhi_chan_ctxt *chan_ctxt;
struct mhi_device_ctxt *mhi_dev_ctxt =
&mhi_devices.device_list[0].mhi_ctxt;
uintptr_t v_wp_index;
uintptr_t v_rp_index;
int valid_chan = 0;
struct mhi_chan_ctxt *cc_list;
struct mhi_client_handle *client_handle;
if (NULL == mhi_dev_ctxt)
return -EIO;
cc_list = mhi_dev_ctxt->mhi_ctrl_seg->mhi_cc_list;
*offp = (u32)(*offp) % MHI_MAX_CHANNELS;
while (!valid_chan) {
client_handle = mhi_dev_ctxt->client_handle_list[*offp];
if (*offp == (MHI_MAX_CHANNELS - 1))
msleep(1000);
if (!VALID_CHAN_NR(*offp) ||
!cc_list[*offp].mhi_trb_ring_base_addr ||
!client_handle) {
*offp += 1;
*offp = (u32)(*offp) % MHI_MAX_CHANNELS;
continue;
}
valid_chan = 1;
}
chan_ctxt = &cc_list[*offp];
get_element_index(&mhi_dev_ctxt->mhi_local_chan_ctxt[*offp],
mhi_dev_ctxt->mhi_local_chan_ctxt[*offp].rp,
&v_rp_index);
get_element_index(&mhi_dev_ctxt->mhi_local_chan_ctxt[*offp],
mhi_dev_ctxt->mhi_local_chan_ctxt[*offp].wp,
&v_wp_index);
amnt_copied =
scnprintf(mhi_dev_ctxt->chan_info,
MHI_LOG_SIZE,
"%s0x%x %s %d %s 0x%x %s 0x%llx %s %p %s %p %s %lu %s %p %s %lu %s %d %s %d\n",
"chan:",
(unsigned int)*offp,
"pkts from dev:",
mhi_dev_ctxt->counters.chan_pkts_xferd[*offp],
"state:",
chan_ctxt->mhi_chan_state,
"p_base:",
chan_ctxt->mhi_trb_ring_base_addr,
"v_base:",
mhi_dev_ctxt->mhi_local_chan_ctxt[*offp].base,
"v_wp:",
mhi_dev_ctxt->mhi_local_chan_ctxt[*offp].wp,
"index:",
v_wp_index,
"v_rp:",
mhi_dev_ctxt->mhi_local_chan_ctxt[*offp].rp,
"index:",
v_rp_index,
"pkts_queued",
get_nr_avail_ring_elements(
&mhi_dev_ctxt->mhi_local_chan_ctxt[*offp]),
"/",
client_handle->chan_info.max_desc);
*offp += 1;
if (amnt_copied < count)
return amnt_copied -
copy_to_user(buf, mhi_dev_ctxt->chan_info, amnt_copied);
else
return -ENOMEM;
}
static const struct file_operations mhi_dbgfs_chan_fops = {
.read = mhi_dbgfs_chan_read,
.write = NULL,
};
static ssize_t mhi_dbgfs_ev_read(struct file *fp, char __user *buf,
size_t count, loff_t *offp)
{
int amnt_copied = 0;
int event_ring_index = 0;
struct mhi_event_ctxt *ev_ctxt;
uintptr_t v_wp_index;
uintptr_t v_rp_index;
uintptr_t device_p_rp_index;
struct mhi_device_ctxt *mhi_dev_ctxt =
&mhi_devices.device_list[0].mhi_ctxt;
if (NULL == mhi_dev_ctxt)
return -EIO;
*offp = (u32)(*offp) % mhi_dev_ctxt->mmio_info.nr_event_rings;
event_ring_index = *offp;
ev_ctxt = &mhi_dev_ctxt->mhi_ctrl_seg->mhi_ec_list[event_ring_index];
if (*offp == (mhi_dev_ctxt->mmio_info.nr_event_rings - 1))
msleep(1000);
get_element_index(&mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index],
mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index].rp,
&v_rp_index);
get_element_index(&mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index],
mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index].wp,
&v_wp_index);
get_element_index(&mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index],
mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index].wp,
&v_wp_index);
get_element_index(&mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index],
(void *)dma_to_virt(NULL, ev_ctxt->mhi_event_read_ptr),
&device_p_rp_index);
amnt_copied =
scnprintf(mhi_dev_ctxt->chan_info,
MHI_LOG_SIZE,
"%s 0x%d %s %02x %s 0x%08x %s 0x%08x %s 0x%llx %s %llx %s %lu %s %p %s %p %s %lu %s %p %s %lu\n",
"Event Context ",
(unsigned int)event_ring_index,
"Intmod_T",
MHI_GET_EV_CTXT(EVENT_CTXT_INTMODT, ev_ctxt),
"MSI Vector",
ev_ctxt->mhi_msi_vector,
"MSI RX Count",
mhi_dev_ctxt->counters.msi_counter[*offp],
"p_base:",
ev_ctxt->mhi_event_ring_base_addr,
"p_rp:",
ev_ctxt->mhi_event_read_ptr,
"index:",
device_p_rp_index,
"v_base:",
mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index].base,
"v_wp:",
mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index].wp,
"index:",
v_wp_index,
"v_rp:",
mhi_dev_ctxt->mhi_local_event_ctxt[event_ring_index].rp,
"index:",
v_rp_index);
*offp += 1;
if (amnt_copied < count)
return amnt_copied -
copy_to_user(buf, mhi_dev_ctxt->chan_info, amnt_copied);
else
return -ENOMEM;
}
static const struct file_operations mhi_dbgfs_ev_fops = {
.read = mhi_dbgfs_ev_read,
.write = NULL,
};
static ssize_t mhi_dbgfs_state_read(struct file *fp, char __user *buf,
size_t count, loff_t *offp)
{
int amnt_copied = 0;
struct mhi_device_ctxt *mhi_dev_ctxt =
&mhi_devices.device_list[0].mhi_ctxt;
if (NULL == mhi_dev_ctxt)
return -EIO;
msleep(100);
amnt_copied =
scnprintf(mhi_dev_ctxt->chan_info,
MHI_LOG_SIZE,
"%s %u %s %d %s %d %s %d %s %d %s %d %s %d %s %d %s %d %s %d %s %d, %s, %d, %s %d\n",
"Our State:",
mhi_dev_ctxt->mhi_state,
"M0->M1:",
mhi_dev_ctxt->counters.m0_m1,
"M0<-M1:",
mhi_dev_ctxt->counters.m1_m0,
"M1->M2:",
mhi_dev_ctxt->counters.m1_m2,
"M0<-M2:",
mhi_dev_ctxt->counters.m2_m0,
"M0->M3:",
mhi_dev_ctxt->counters.m0_m3,
"M0<-M3:",
mhi_dev_ctxt->counters.m3_m0,
"M3_ev_TO:",
mhi_dev_ctxt->counters.m3_event_timeouts,
"M0_ev_TO:",
mhi_dev_ctxt->counters.m0_event_timeouts,
"MSI_d:",
mhi_dev_ctxt->counters.msi_disable_cntr,
"MSI_e:",
mhi_dev_ctxt->counters.msi_enable_cntr,
"outstanding_acks:",
atomic_read(&mhi_dev_ctxt->counters.outbound_acks),
"LPM:",
mhi_dev_ctxt->enable_lpm);
if (amnt_copied < count)
return amnt_copied - copy_to_user(buf,
mhi_dev_ctxt->chan_info, amnt_copied);
else
return -ENOMEM;
}
static const struct file_operations mhi_dbgfs_state_fops = {
.read = mhi_dbgfs_state_read,
.write = NULL,
};
inline void *mhi_get_virt_addr(struct mhi_meminfo *meminfo)
{
return (void *)meminfo->va_aligned;
}
inline u64 mhi_get_memregion_len(struct mhi_meminfo *meminfo)
{
return meminfo->size;
}
enum MHI_STATUS mhi_mallocmemregion(struct mhi_meminfo *meminfo, size_t size)
{
meminfo->va_unaligned = (uintptr_t)dma_alloc_coherent(
meminfo->dev,
size,
(dma_addr_t *)&(meminfo->pa_unaligned),
GFP_KERNEL);
if (!meminfo->va_unaligned)
return MHI_STATUS_ERROR;
meminfo->va_aligned = meminfo->va_unaligned;
meminfo->pa_aligned = meminfo->pa_unaligned;
meminfo->size = size;
if ((meminfo->pa_unaligned + size) >= MHI_DATA_SEG_WINDOW_END_ADDR)
return MHI_STATUS_ERROR;
if (0 == meminfo->va_unaligned)
return MHI_STATUS_ERROR;
mb();
return MHI_STATUS_SUCCESS;
}
void mhi_freememregion(struct mhi_meminfo *meminfo)
{
mb();
dma_free_coherent(meminfo->dev,
meminfo->size,
(dma_addr_t *)&meminfo->pa_unaligned,
GFP_KERNEL);
meminfo->va_aligned = 0;
meminfo->pa_aligned = 0;
meminfo->va_unaligned = 0;
meminfo->pa_unaligned = 0;
}
int mhi_init_debugfs(struct mhi_device_ctxt *mhi_dev_ctxt)
{
struct dentry *mhi_chan_stats;
struct dentry *mhi_state_stats;
struct dentry *mhi_ev_stats;
mhi_dev_ctxt->mhi_parent_folder =
debugfs_create_dir("mhi", NULL);
if (mhi_dev_ctxt->mhi_parent_folder == NULL) {
mhi_log(MHI_MSG_INFO, "Failed to create debugfs parent dir.\n");
return -EIO;
}
mhi_chan_stats = debugfs_create_file("mhi_chan_stats",
0444,
mhi_dev_ctxt->mhi_parent_folder,
mhi_dev_ctxt,
&mhi_dbgfs_chan_fops);
if (mhi_chan_stats == NULL)
return -ENOMEM;
mhi_ev_stats = debugfs_create_file("mhi_ev_stats",
0444,
mhi_dev_ctxt->mhi_parent_folder,
mhi_dev_ctxt,
&mhi_dbgfs_ev_fops);
if (mhi_ev_stats == NULL)
goto clean_chan;
mhi_state_stats = debugfs_create_file("mhi_state_stats",
0444,
mhi_dev_ctxt->mhi_parent_folder,
mhi_dev_ctxt,
&mhi_dbgfs_state_fops);
if (mhi_state_stats == NULL)
goto clean_ev_stats;
mhi_dev_ctxt->chan_info = kmalloc(MHI_LOG_SIZE, GFP_KERNEL);
if (mhi_dev_ctxt->chan_info == NULL)
goto clean_all;
return 0;
clean_all:
debugfs_remove(mhi_state_stats);
clean_ev_stats:
debugfs_remove(mhi_ev_stats);
clean_chan:
debugfs_remove(mhi_chan_stats);
debugfs_remove(mhi_dev_ctxt->mhi_parent_folder);
return -ENOMEM;
}