tegrakernel/kernel/nvidia/drivers/platform/tegra/tegra186_aondbg.c

783 lines
19 KiB
C

/*
* Copyright (c) 2015-2016, NVIDIA CORPORATION. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms and conditions of the GNU General Public License,
* version 2, as published by the Free Software Foundation.
*
* This program is distributed in the hope 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 eeceived a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <linux/interrupt.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/debugfs.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/of_reserved_mem.h>
#include <linux/platform_device.h>
#include <linux/slab.h>
#include <linux/tegra-aon.h>
#include <linux/mailbox_client.h>
#include <linux/uaccess.h>
#include <linux/completion.h>
#include <linux/jiffies.h>
#include <linux/tegra-firmwares.h>
#include <asm/cacheflush.h>
#include "aon-ivc-dbg-messages.h"
#define AON_REQUEST_MASK 0xF
#define AON_REQUESTS_TOTAL (AON_REQUEST_TYPE_MAX + 1)
#define AON_PSTATES_TOTAL 7
#define AON_PSTATE_MASK (0x7F << AON_REQUESTS_TOTAL)
#define AON_STATE(req) ((req & AON_PSTATE_MASK) >> AON_REQUESTS_TOTAL)
#define TX_BLOCK_PERIOD 20
#define AON_ROOT 0
#define AON_PM 1
#define AON_MODS 2
#define AON_STATS 3
#define IVC_DBG_CH_FRAME_SIZE 128
#define MODS_DEFAULT_VAL 0xFFFF
#define PSTATE_NAME_MAX_LEN 15
static struct device *aondev;
struct tegra_aondbg {
struct device *dev;
struct mbox_client cl;
struct mbox_chan *mbox;
struct dentry *aon_root;
};
struct aon_dbgfs_node {
char *name;
u32 id;
u8 pdr_id;
mode_t mode;
struct completion *wait_on;
const struct file_operations *fops;
char data[IVC_DBG_CH_FRAME_SIZE];
};
struct dbgfs_dir {
const char *name;
struct dentry *dir;
struct dbgfs_dir *parent;
};
static struct dbgfs_dir aon_dbgfs_dirs[] = {
{.name = "aon", .parent = NULL},
{.name = "aon_pm", .parent = &aon_dbgfs_dirs[AON_ROOT]},
{.name = "aon_mods", .parent = &aon_dbgfs_dirs[AON_ROOT]},
{.name = "stats", .parent = &aon_dbgfs_dirs[AON_PM]}
};
static const char *const pstates[] = {"idle", "standby", "dormant", "deepidle",
"deepstandby", "deepdormant", "active"};
static u32 thresholds[AON_PSTATES_TOTAL];
static u32 mods_result = MODS_DEFAULT_VAL;
static unsigned int completion_timeout = 50;
static DEFINE_MUTEX(aon_mutex);
static DEFINE_SPINLOCK(mods);
static DEFINE_SPINLOCK(completion);
static struct aon_dbgfs_node aon_nodes[];
static void set_mods_result(u32 result)
{
spin_lock(&mods);
mods_result = result;
spin_unlock(&mods);
}
static unsigned int get_mods_result(void)
{
u32 val;
spin_lock(&mods);
val = mods_result;
spin_unlock(&mods);
return val;
}
static void set_completion_timeout(unsigned int timeout)
{
spin_lock(&completion);
completion_timeout = timeout;
spin_unlock(&completion);
}
static unsigned int get_completion_timeout(void)
{
unsigned int val;
spin_lock(&completion);
val = completion_timeout;
spin_unlock(&completion);
return val;
}
static struct aon_dbg_response *aon_create_ivc_dbg_req(u32 request,
u32 flag, u32 data)
{
struct tegra_aondbg *aondbg;
struct aon_dbg_request req;
struct aon_dbg_response *resp;
struct tegra_aon_mbox_msg msg;
int ret = 0;
uint16_t pstate = 0;
unsigned int timeout = 0;
if (aondev == NULL)
return (void *)-EPROBE_DEFER;
aondbg = dev_get_drvdata(aondev);
if (aondbg == NULL) {
dev_err(aondev, "Control data for IPC with SPE is NULL\n");
return (void *)-EINVAL;
}
pstate = (request > AON_REQUEST_TYPE_MAX) ?
__builtin_ctz(AON_STATE(request)) : 0;
req.req_type = request & AON_REQUEST_MASK;
switch (req.req_type) {
case AON_PM_TARGET_POWER_STATE:
req.data.pm_xfer.type.pstate.flag = flag;
req.data.pm_xfer.type.pstate.state = (flag) ? data : 0;
break;
case AON_PM_THRESHOLD:
req.data.pm_xfer.type.threshold.flag = flag;
req.data.pm_xfer.type.threshold.state = pstate;
req.data.pm_xfer.type.threshold.val = data;
break;
case AON_PM_WAKE_TIMEOUT:
req.data.pm_xfer.type.wake_tout.flag = flag;
req.data.pm_xfer.type.wake_tout.timeout = (flag) ? data : 0;
break;
case AON_PM_DISABLE_PLLAON:
req.data.pm_xfer.type.disable_pllaon.flag = flag;
req.data.pm_xfer.type.disable_pllaon.disable = data;
break;
case AON_MODS_LOOPS_TEST:
req.data.mods_xfer.loops = data;
break;
case AON_PM_VDD_RTC_RETENTION:
req.data.pm_xfer.type.retention.flag = flag;
req.data.pm_xfer.type.retention.enable = (flag) ? data : 0;
break;
case AON_PM_SC8_COUNT:
case AON_MODS_CRC:
case AON_PM_STATE_TIME:
case AON_PM_STATE_COUNT:
case AON_PING_TEST:
break;
default:
dev_err(aondev, "Invalid aon dbg request\n");
return ERR_PTR(-EINVAL);
}
msg.length = sizeof(struct aon_dbg_request);
msg.data = (void *)&req;
ret = mbox_send_message(aondbg->mbox, (void *)&msg);
if (ret < 0) {
dev_err(aondev, "mbox_send_message failed\n");
return ERR_PTR(ret);
}
timeout = get_completion_timeout();
ret = wait_for_completion_timeout(aon_nodes[req.req_type].wait_on,
msecs_to_jiffies(timeout));
if (!ret) {
dev_err(aondev, "No response\n");
return ERR_PTR(-ETIMEDOUT);
}
resp = (void *)aon_nodes[req.req_type].data;
if (resp->resp_type > AON_REQUEST_TYPE_MAX) {
dev_err(aondev, "Invalid aon dbg response\n");
return ERR_PTR(-EIO);
}
if (resp->status != AON_DBG_STATUS_OK) {
dev_err(aondev, "Request failed\n");
return ERR_PTR(-EIO);
}
return resp;
}
int tegra_aon_get_pllaon_state(void)
{
int ret = -EINVAL;
u64 val;
struct aon_dbg_response *resp = NULL;
mutex_lock(&aon_mutex);
resp = aon_create_ivc_dbg_req(AON_PM_DISABLE_PLLAON, READ, 0);
if (IS_ERR(resp)) {
ret = PTR_ERR(resp);
mutex_unlock(&aon_mutex);
return ret;
}
if (resp->resp_type == AON_PM_DISABLE_PLLAON) {
val = resp->data.pm_xfer.type.disable_pllaon.disable;
ret = (val == 1) ? 1 : 0;
}
mutex_unlock(&aon_mutex);
return ret;
}
EXPORT_SYMBOL(tegra_aon_get_pllaon_state);
int tegra_aon_set_pllaon_state(bool enable)
{
int ret = 0;
u64 val = enable ? 0 : 1;
struct aon_dbg_response *resp = NULL;
mutex_lock(&aon_mutex);
resp = aon_create_ivc_dbg_req(AON_PM_DISABLE_PLLAON, WRITE, val);
if (IS_ERR(resp))
ret = PTR_ERR(resp);
mutex_unlock(&aon_mutex);
return ret;
}
EXPORT_SYMBOL(tegra_aon_set_pllaon_state);
static ssize_t aon_pm_target_power_state_write(struct file *file,
const char __user *user_buf, size_t count, loff_t *ppos)
{
char buf[PSTATE_NAME_MAX_LEN];
char *name;
struct aon_dbg_response *resp;
int i;
if (count >= sizeof(buf))
return -EINVAL;
if (strncpy_from_user(buf, user_buf, count) <= 0)
return -EFAULT;
buf[count] = '\0';
name = strim(buf);
for (i = 0; i < ARRAY_SIZE(pstates); i++) {
if (!strcmp(name, pstates[i]))
break;
}
if (i >= ARRAY_SIZE(pstates))
return -EINVAL;
mutex_lock(&aon_mutex);
resp = aon_create_ivc_dbg_req(AON_PM_TARGET_POWER_STATE,
WRITE, i);
mutex_unlock(&aon_mutex);
if (IS_ERR(resp))
return PTR_ERR(resp);
return count;
}
static int aon_pm_target_power_state_show(struct seq_file *file, void *data)
{
struct aon_dbg_response *resp;
int pstate_index;
mutex_lock(&aon_mutex);
resp = aon_create_ivc_dbg_req(*(u32 *)file->private, READ, 0);
if (IS_ERR(resp)) {
mutex_unlock(&aon_mutex);
return PTR_ERR(resp);
}
pstate_index = resp->data.pm_xfer.type.pstate.state;
if (pstate_index < AON_PSTATES_TOTAL) {
seq_printf(file, "%s\n",
pstates[pstate_index]);
} else {
dev_err(aondev, "invalid pstate in response\n");
}
mutex_unlock(&aon_mutex);
return 0;
}
static int aon_pm_target_power_state_open(struct inode *inode,
struct file *file)
{
return single_open(file, aon_pm_target_power_state_show,
inode->i_private);
}
static const struct file_operations aon_pm_target_power_state_fops = {
.open = aon_pm_target_power_state_open,
.llseek = seq_lseek,
.read = seq_read,
.write = aon_pm_target_power_state_write,
.release = single_release
};
static int aon_pm_show(void *data, u64 *val)
{
struct aon_dbg_response *resp;
mutex_lock(&aon_mutex);
resp = aon_create_ivc_dbg_req(*(u32 *)data, READ, 0);
if (IS_ERR(resp)) {
mutex_unlock(&aon_mutex);
return PTR_ERR(resp);
}
switch (resp->resp_type) {
case AON_PM_WAKE_TIMEOUT:
*val = resp->data.pm_xfer.type.wake_tout.timeout;
break;
case AON_PM_THRESHOLD:
*val = resp->data.pm_xfer.type.threshold.val;
break;
case AON_PM_VDD_RTC_RETENTION:
*val = resp->data.pm_xfer.type.retention.enable;
break;
case AON_PM_DISABLE_PLLAON:
*val = resp->data.pm_xfer.type.disable_pllaon.disable;
break;
case AON_PM_SC8_COUNT:
*val = resp->data.pm_xfer.type.sc8_count.count;
break;
default:
dev_err(aondev, "Invalid pm response\n");
break;
}
mutex_unlock(&aon_mutex);
return 0;
}
static int aon_pm_store(void *data, u64 val)
{
struct aon_dbg_response *resp;
mutex_lock(&aon_mutex);
resp = aon_create_ivc_dbg_req(*(u32 *)data, WRITE, val);
mutex_unlock(&aon_mutex);
if (IS_ERR(resp))
return PTR_ERR(resp);
return 0;
}
DEFINE_SIMPLE_ATTRIBUTE(aon_pm_threshold_fops, aon_pm_show,
aon_pm_store, "%lld\n");
DEFINE_SIMPLE_ATTRIBUTE(aon_pm_wake_timeout_fops, aon_pm_show,
aon_pm_store, "%lld\n");
DEFINE_SIMPLE_ATTRIBUTE(aon_pm_disable_pllaon_fops, aon_pm_show,
aon_pm_store, "%lld\n");
DEFINE_SIMPLE_ATTRIBUTE(aon_pm_retention_fops, aon_pm_show,
aon_pm_store, "%lld\n");
DEFINE_SIMPLE_ATTRIBUTE(aon_pm_sc8_count_fops, aon_pm_show,
NULL, "%lld\n");
static int aon_pm_state_times_show(struct seq_file *file, void *data)
{
struct aon_dbg_response *resp;
int i, len;
u64 *arr;
mutex_lock(&aon_mutex);
resp = aon_create_ivc_dbg_req(*(u32 *)file->private, READ, 0);
if (IS_ERR(resp)) {
mutex_unlock(&aon_mutex);
return PTR_ERR(resp);
}
len = ARRAY_SIZE(resp->data.pm_xfer.type.state_times.state_durations);
arr = resp->data.pm_xfer.type.state_times.state_durations;
for (i = 0; i < len; i++)
seq_printf(file, "%s: %llu\n", pstates[i], arr[i]);
mutex_unlock(&aon_mutex);
return 0;
}
static int aon_pm_state_times_open(struct inode *inode, struct file *file)
{
return single_open(file, aon_pm_state_times_show, inode->i_private);
}
static const struct file_operations aon_pm_state_times_fops = {
.open = aon_pm_state_times_open,
.read = seq_read,
.llseek = seq_lseek,
.release = single_release
};
static int aon_pm_state_counts_show(struct seq_file *file, void *data)
{
struct aon_dbg_response *resp;
int i, len;
u32 *arr;
mutex_lock(&aon_mutex);
resp = aon_create_ivc_dbg_req(*(u32 *)file->private, READ, 0);
if (IS_ERR(resp)) {
mutex_unlock(&aon_mutex);
return PTR_ERR(resp);
}
len = ARRAY_SIZE(resp->data.pm_xfer.type.state_counts.state_entry_counts);
arr = resp->data.pm_xfer.type.state_counts.state_entry_counts;
for (i = 0; i < len; i++)
seq_printf(file, "%s: %u\n", pstates[i], arr[i]);
mutex_unlock(&aon_mutex);
return 0;
}
static int aon_pm_state_counts_open(struct inode *inode, struct file *file)
{
return single_open(file, aon_pm_state_counts_show, inode->i_private);
}
static const struct file_operations aon_pm_state_counts_fops = {
.open = aon_pm_state_counts_open,
.read = seq_read,
.llseek = seq_lseek,
.release = single_release
};
static ssize_t __aon_do_ping(u32 context, char **data)
{
struct aon_dbg_response *resp;
int ret = 0;
*data = NULL;
resp = aon_create_ivc_dbg_req(context, READ, 0);
if (IS_ERR(resp))
ret = PTR_ERR(resp);
else
*data = resp->data.ping_xfer.data;
return ret;
}
static int aon_ping_show(struct seq_file *file, void *param)
{
char *data;
int ret;
mutex_lock(&aon_mutex);
ret = __aon_do_ping(*(u32 *)file->private, &data);
if (ret >= 0)
seq_printf(file, "%s\n", data);
mutex_unlock(&aon_mutex);
return ret;
}
static ssize_t aon_version_show(struct device *dev, char *buf, size_t size)
{
char *data;
int ret = 0;
mutex_lock(&aon_mutex);
ret = __aon_do_ping(AON_PING_TEST, &data);
if (ret < 0)
ret = snprintf(buf, size, "error retrieving version: %d", ret);
else
ret = snprintf(buf, size, "%s", data ? data : "unavailable");
mutex_unlock(&aon_mutex);
return ret;
}
static int aon_ping_open(struct inode *inode, struct file *file)
{
return single_open(file, aon_ping_show, inode->i_private);
}
static const struct file_operations aon_ping_fops = {
.open = aon_ping_open,
.read = seq_read,
.llseek = seq_lseek,
.release = single_release
};
static int aon_mods_loops_store(void *data, u64 val)
{
struct aon_dbg_response *resp;
int ret = 0;
mutex_lock(&aon_mutex);
set_mods_result(MODS_DEFAULT_VAL);
resp = aon_create_ivc_dbg_req(*(u32 *)data, WRITE, val);
if (IS_ERR(resp))
ret = PTR_ERR(resp);
else
set_mods_result(resp->status);
mutex_unlock(&aon_mutex);
return ret;
}
DEFINE_SIMPLE_ATTRIBUTE(aon_mods_loops_fops, NULL,
aon_mods_loops_store, "%lld\n");
static int aon_mods_result_show(void *data, u64 *val)
{
*val = get_mods_result();
return 0;
}
DEFINE_SIMPLE_ATTRIBUTE(aon_mods_result_fops, aon_mods_result_show,
NULL, "%lld\n");
static int aon_mods_crc_show(void *data, u64 *val)
{
struct aon_dbg_response *resp;
int ret = 0;
mutex_lock(&aon_mutex);
resp = aon_create_ivc_dbg_req(*(u32 *)data, READ, 0);
if (IS_ERR(resp))
ret = PTR_ERR(resp);
else
*val = resp->data.crc_xfer.crc;
mutex_unlock(&aon_mutex);
return ret;
}
DEFINE_SIMPLE_ATTRIBUTE(aon_mods_crc_fops, aon_mods_crc_show,
NULL, "%llx\n");
static int aon_timeout_show(void *data, u64 *val)
{
*val = get_completion_timeout();
return 0;
}
static int aon_timeout_store(void *data, u64 val)
{
set_completion_timeout(val);
return 0;
}
DEFINE_SIMPLE_ATTRIBUTE(aon_timeout_fops, aon_timeout_show,
aon_timeout_store, "%lld\n");
static struct aon_dbgfs_node aon_nodes[] = {
{.name = "target_power_state", .id = AON_PM_TARGET_POWER_STATE,
.pdr_id = AON_PM, .mode = S_IRUGO | S_IWUSR,
.fops = &aon_pm_target_power_state_fops,},
{.name = "threshold", .id = AON_PM_THRESHOLD, .pdr_id = AON_PM,
.mode = S_IRUGO | S_IWUSR,
.fops = &aon_pm_threshold_fops,},
{.name = "wake_timeout", .id = AON_PM_WAKE_TIMEOUT, .pdr_id = AON_PM,
.mode = S_IRUGO | S_IWUSR,
.fops = &aon_pm_wake_timeout_fops,},
{.name = "disable_pllaon", .id = AON_PM_DISABLE_PLLAON, .pdr_id = AON_PM,
.mode = S_IRUGO | S_IWUSR,
.fops = &aon_pm_disable_pllaon_fops,},
{.name = "state_times", .id = AON_PM_STATE_TIME,
.pdr_id = AON_STATS, .mode = S_IRUGO,
.fops = &aon_pm_state_times_fops,},
{.name = "state_counts", .id = AON_PM_STATE_COUNT,
.pdr_id = AON_STATS, .mode = S_IRUGO,
.fops = &aon_pm_state_counts_fops,},
{.name = "loops", .id = AON_MODS_LOOPS_TEST, .pdr_id = AON_MODS,
.mode = S_IWUSR, .fops = &aon_mods_loops_fops, },
{.name = "result", .id = AON_MODS_RESULT, .pdr_id = AON_MODS,
.mode = S_IRUGO , .fops = &aon_mods_result_fops,},
{.name = "crc", .id = AON_MODS_CRC, .pdr_id = AON_MODS,
.mode = S_IRUGO , .fops = &aon_mods_crc_fops,},
{.name = "ping", .id = AON_PING_TEST, .pdr_id = AON_ROOT,
.mode = S_IRUGO | S_IWUSR, .fops = &aon_ping_fops,},
{.name = "enable_retention", .id = AON_PM_VDD_RTC_RETENTION,
.pdr_id = AON_PM, .mode = S_IRUGO | S_IWUSR,
.fops = &aon_pm_retention_fops,},
{.name = "sc8_count", .id = AON_PM_SC8_COUNT,
.pdr_id = AON_STATS, .mode = S_IRUGO,
.fops = &aon_pm_sc8_count_fops,},
{.name = "completion_timeout", .pdr_id = AON_ROOT,
.mode = S_IRUGO | S_IWUSR, .fops = &aon_timeout_fops,},
};
static void tegra_aondbg_recv_msg(struct mbox_client *cl, void *rx_msg)
{
struct tegra_aon_mbox_msg *msg;
struct aon_dbg_response *resp;
msg = (struct tegra_aon_mbox_msg *)rx_msg;
resp = (void *)msg->data;
if (resp->resp_type > AON_REQUEST_TYPE_MAX) {
dev_err(aondev, "Multiple request types in 1 response\n");
return;
}
memcpy(aon_nodes[resp->resp_type].data, msg->data,
IVC_DBG_CH_FRAME_SIZE);
complete(aon_nodes[resp->resp_type].wait_on);
}
static int aon_pm_dbg_init(struct tegra_aondbg *aon)
{
struct dentry *d;
struct dentry *td;
int i, j;
d = debugfs_create_dir(aon_dbgfs_dirs[0].name, NULL);
if (IS_ERR_OR_NULL(d))
goto clean;
aon_dbgfs_dirs[0].dir = d;
aon->aon_root = d;
for (i = 1; i < ARRAY_SIZE(aon_dbgfs_dirs); i++) {
d = debugfs_create_dir(aon_dbgfs_dirs[i].name,
aon_dbgfs_dirs[i].parent->dir);
if (IS_ERR_OR_NULL(d))
goto clean;
aon_dbgfs_dirs[i].dir = d;
}
for (i = 0; i < ARRAY_SIZE(aon_nodes); i++) {
if (strcmp(aon_nodes[i].name, "threshold")) {
d = debugfs_create_file(aon_nodes[i].name,
aon_nodes[i].mode,
aon_dbgfs_dirs[aon_nodes[i].pdr_id].dir,
&aon_nodes[i].id, aon_nodes[i].fops);
if (IS_ERR_OR_NULL(d))
goto clean;
continue;
}
d = debugfs_create_dir(aon_nodes[i].name,
aon_dbgfs_dirs[aon_nodes[i].pdr_id].dir);
if (IS_ERR_OR_NULL(d))
goto clean;
/*
* We only need to loop till off by one as we don't need
* active state as a separate node in threshold directory.
*/
for (j = 0; j < ARRAY_SIZE(pstates) - 1; j++) {
thresholds[j] = aon_nodes[i].id |
BIT(j + AON_REQUESTS_TOTAL);
td = debugfs_create_file(pstates[j], aon_nodes[i].mode,
d, &thresholds[j], aon_nodes[i].fops);
if (IS_ERR_OR_NULL(d)) {
d = td;
goto clean;
}
}
}
return 0;
clean:
debugfs_remove_recursive(aon->aon_root);
return PTR_ERR(d);
}
#define NV(p) "nvidia," p
static int tegra_aondbg_probe(struct platform_device *pdev)
{
struct tegra_aondbg *aondbg;
struct device *dev = &pdev->dev;
struct device_node *np = dev->of_node;
int i;
int ret;
aondev = &pdev->dev;
dev_dbg(dev, "aondbg driver probe()\n");
if (!np) {
dev_err(dev, "tegra-aondbg: DT data required.\n");
return -EINVAL;
}
aondbg = devm_kzalloc(&pdev->dev, sizeof(*aondbg), GFP_KERNEL);
if (!aondbg)
return -ENOMEM;
dev_set_drvdata(&pdev->dev, aondbg);
aondbg->dev = &pdev->dev;
aondbg->cl.dev = &pdev->dev;
aondbg->cl.tx_block = true;
aondbg->cl.tx_tout = TX_BLOCK_PERIOD;
aondbg->cl.knows_txdone = false;
aondbg->cl.rx_callback = tegra_aondbg_recv_msg;
aondbg->mbox = mbox_request_channel(&aondbg->cl, 0);
if (IS_ERR(aondbg->mbox)) {
ret = PTR_ERR(aondbg->mbox);
if (ret != -EPROBE_DEFER)
dev_warn(&pdev->dev,
"can't get mailbox channel (%d)\n", ret);
return ret;
}
dev_dbg(dev, "aondbg->mbox = %p\n", aondbg->mbox);
for (i = 0; i < ARRAY_SIZE(aon_nodes); i++) {
aon_nodes[i].wait_on = devm_kzalloc(&pdev->dev,
sizeof(struct completion), GFP_KERNEL);
if (!aon_nodes[i].wait_on) {
dev_err(dev, "out of memory.\n");
return -ENOMEM;
}
init_completion(aon_nodes[i].wait_on);
}
ret = aon_pm_dbg_init(aondbg);
if (ret) {
dev_err(dev, "failed to create debugfs nodes.\n");
return ret;
}
dev_info(dev, "aondbg driver probe() OK\n");
devm_tegrafw_register(&pdev->dev, "spe", TFW_NORMAL,
aon_version_show, NULL);
return 0;
}
static int tegra_aondbg_remove(struct platform_device *pdev)
{
struct tegra_aondbg *aondbg;
aondbg = dev_get_drvdata(&pdev->dev);
mbox_free_channel(aondbg->mbox);
debugfs_remove_recursive(aondbg->aon_root);
return 0;
}
static const struct of_device_id tegra_aondbg_of_match[] = {
{
.compatible = "nvidia,tegra186-aondbg",
},
{},
};
MODULE_DEVICE_TABLE(of, tegra_aondbg_of_match);
static struct platform_driver tegra_aondbg_driver = {
.driver = {
.name = "tegra186-aondbg",
.owner = THIS_MODULE,
.of_match_table = of_match_ptr(tegra_aondbg_of_match),
},
.probe = tegra_aondbg_probe,
.remove = tegra_aondbg_remove,
};
module_platform_driver(tegra_aondbg_driver);
MODULE_DESCRIPTION("AON DBG driver");
MODULE_AUTHOR("NVIDIA");
MODULE_LICENSE("GPL v2");