kernel-brax3-ubuntu-touch/drivers/misc/mediatek/atf/tfa_debug.c
erascape f319b992b1 kernel-5.15: Initial import brax3 UT kernel
* halium configs enabled

Signed-off-by: erascape <erascape@proton.me>
2025-09-23 15:17:10 +00:00

603 lines
15 KiB
C

// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (c) 2021 MediaTek Inc.
*/
#include <linux/arm-smccc.h>
#include <linux/atomic.h>
#include <linux/io.h>
#include <linux/module.h>
#include <linux/of_device.h>
#include <linux/of_fdt.h>
#include <linux/of_platform.h>
#include <linux/of_reserved_mem.h>
#include <linux/poll.h>
#include <linux/proc_fs.h>
#include <linux/sched/clock.h>
#include <linux/slab.h>
#include <linux/soc/mediatek/mtk_sip_svc.h>
#include <linux/uaccess.h>
#define ATF_LOG_RESERVED_MEMORY_KEY "mediatek,atf-log-reserved"
#define DEBUG_BUF_NAME_LEN 16
#define DEBUG_BUF_MAGIC (0xdb2ab1e8)
#define DEBUG_BUF_ENTRY_MAGIC (0xdbee4298)
#define RUNTIME_LOG_NAME "RUNTIME_LOG"
#define DEBUG_BUF_PROC_FOLDER_NAME "atf_log"
#define DEBUG_BUF_ATF_LOGGER_PROC_NAME "atf_log"
#define DEBUG_BUF_ATF_LOGGER_SNAPSHOT_PROC_NAME "runtime_snapshot"
#define DEBUG_BUF_TOTAL_RAW_PROC_NAME "raw_dump_all"
#define MAGIC_RELEASE_LOG 0x5678
#if CONFIG_OF
static const struct of_device_id tfa_debug_of_ids[] = {
{ .compatible = "mediatek,tfa_debug", },
{},
};
#endif
/*
* |--------------------------|
* | debug_buf_table_header |
* |--------------------------|
* | debug_buf_entry 0 |
* |--------------------------|
* |..........................|
* |--------------------------|
* | debug_buf_entry N |
* |--------------------------|
* | debug_buf_hdr 0 |
* |--------------------------|
* | debug buffer body 0 |
* |--------------------------|
* |..........................|
* |--------------------------|
* | debug_buf_hdr N |
* |--------------------------|
* | debug buffer body N |
* |--------------------------|
*/
struct debug_buf_table_header {
uint32_t magic;
/* Structure may change*/
uint32_t version;
/*
* debug_buf_table_header + all debug_buf_entry +
* all debug buffer
*/
uint32_t total_size;
/* sizeof(debug_buf_table_header) */
uint32_t hdr_size;
/* sizeof(debug_buf_entry) */
uint32_t debug_buf_entry_size;
/* number of debug_buf_entry */
uint32_t debug_buf_entry_count;
/*
* Offset to the first debug_buf_entry from head of
* debug_buf_table_header
*/
uint32_t debug_buf_entries_offset;
uint32_t alignment;
};
struct debug_buf_hdr {
uint32_t magic;
char debug_buf_name[DEBUG_BUF_NAME_LEN];
uint32_t debug_buf_size;
uint32_t w_offset;
uint32_t r_offset;
uint32_t flag;
};
struct tfa_debug_entry_info {
struct debug_buf_hdr *buf_hdr_p;
void *body_head;
uint32_t debug_buf_size;
};
struct tfa_debug_info {
struct proc_dir_entry *proc_dir;
phys_addr_t debug_buf_paddr;
phys_addr_t total_size;
void __iomem *vaddr;
/* Debug buffer entry */
struct tfa_debug_entry_info runtime;
atomic_t release_log;
};
struct tfa_debug_instance {
void __iomem *vaddr;
phys_addr_t debug_buf_size;
size_t read_offset;
struct debug_buf_hdr *debug_buf_hdr_p;
struct wait_queue_head waitq;
};
static struct tfa_debug_info info;
static int is_debug_buf_info_valid(void)
{
if (info.debug_buf_paddr == 0)
return 0;
if (info.total_size == 0)
return 0;
if (IS_ERR(info.vaddr))
return 0;
return 1;
}
static int generic_open(struct inode *inode, struct file *file)
{
int ret;
struct tfa_debug_instance *inst_p = NULL;
ret = nonseekable_open(inode, file);
if (unlikely(ret))
return ret;
inst_p = kmalloc(sizeof(struct tfa_debug_instance),
GFP_KERNEL);
if (IS_ERR(inst_p)) {
file->private_data = NULL;
pr_notice("Fail to kmalloc:%d\n", PTR_ERR(inst_p));
return -ENOMEM;
}
init_waitqueue_head(&inst_p->waitq);
file->private_data = inst_p;
pr_info("%s %u process name:%s, pid:%d\n",
__func__, __LINE__, current->comm,
task_pid_nr(current));
return 0;
}
static int raw_open(struct inode *inode, struct file *file)
{
int ret;
struct tfa_debug_instance *inst_p;
ret = generic_open(inode, file);
if (ret) {
pr_notice("[%s] Fail to open proc file\n", __func__);
return ret;
}
inst_p = (struct tfa_debug_instance *)file->private_data;
inst_p->vaddr = info.vaddr;
inst_p->debug_buf_size = info.total_size;
inst_p->read_offset = 0;
return 0;
}
static int raw_release(struct inode *node, struct file *file)
{
pr_info("%s:vaddr:0x%llx\n", __func__, file->private_data);
if (file->private_data) {
struct tfa_debug_instance *inst_p = file->private_data;
kfree(inst_p);
file->private_data = NULL;
}
return 0;
}
static ssize_t raw_read(struct file *file,
char __user *buf, size_t count, loff_t *ppos)
{
struct tfa_debug_instance *inst_p =
(struct tfa_debug_instance *)file->private_data;
size_t remain_len = inst_p->debug_buf_size - inst_p->read_offset;
size_t copy_len;
if (IS_ERR(info.vaddr))
return -ENOMEM;
if (count > remain_len)
copy_len = remain_len;
else
copy_len = count;
if (copy_to_user(buf,
(void *)(inst_p->vaddr + inst_p->read_offset), copy_len))
return -EFAULT;
/* Update the read pos */
inst_p->read_offset += copy_len;
return copy_len;
}
static const struct proc_ops tfa_debug_raw_fops = {
.proc_open = raw_open,
.proc_read = raw_read,
.proc_release = raw_release,
.proc_lseek = no_llseek,
};
static int runtime_snap_open(struct inode *inode, struct file *file)
{
int ret;
struct tfa_debug_instance *inst_p;
ret = generic_open(inode, file);
if (ret) {
pr_notice("[%s] Fail to open proc file\n", __func__);
return ret;
}
inst_p = (struct tfa_debug_instance *)file->private_data;
inst_p->vaddr = info.runtime.body_head;
inst_p->debug_buf_size = info.runtime.debug_buf_size;
inst_p->read_offset = 0;
return 0;
}
static const struct proc_ops tfa_debug_runtime_snap_fops = {
.proc_open = runtime_snap_open,
.proc_read = raw_read,
.proc_release = raw_release,
.proc_lseek = no_llseek,
};
static int is_runtime_empty_for_read(struct file *file)
{
struct tfa_debug_instance *inst_p =
(struct tfa_debug_instance *)file->private_data;
struct debug_buf_hdr *buf_hdr_p;
if (IS_ERR(inst_p))
return 1;
buf_hdr_p = inst_p->debug_buf_hdr_p;
if (IS_ERR(buf_hdr_p))
return 1;
return (inst_p->read_offset == buf_hdr_p->w_offset);
}
static int getc_for_read(struct file *file, char *p)
{
char *body;
struct tfa_debug_instance *inst_p =
(struct tfa_debug_instance *)file->private_data;
if (is_runtime_empty_for_read(file))
return -1;
body = inst_p->vaddr;
if (inst_p->read_offset >= inst_p->debug_buf_size) {
pr_notice("%s read offset is invalid, :%u\n",
__func__, inst_p->read_offset);
inst_p->read_offset = 0;
}
*p = body[inst_p->read_offset];
inst_p->read_offset = (inst_p->read_offset + 1) % inst_p->debug_buf_size;
dmb(ish);
return 0;
}
static ssize_t runtime_log_read(struct file *file,
char __user *buf, size_t count, loff_t *ppos)
{
int error = 0;
size_t copy_len = 0;
char c;
struct tfa_debug_instance *inst_p = file->private_data;
signed long wait_event_ret;
while (1) {
if ((file->f_flags & O_NONBLOCK) &&
is_runtime_empty_for_read(file))
return -EAGAIN;
wait_event_ret = wait_event_timeout(inst_p->waitq,
!is_runtime_empty_for_read(file), HZ);
if (wait_event_ret != 0)
break;
if (atomic_read(&info.release_log))
break;
}
while (copy_len < count) {
if (getc_for_read(file, &c))
break;
error = __put_user(c, buf);
if (error) {
pr_notice("%s __put_user fail:%d\n", __func__, error);
break;
}
buf++;
copy_len++;
dmb(ish);
}
if (!error)
error = copy_len;
return error;
}
static int runtime_log_open(struct inode *inode, struct file *file)
{
int ret;
struct tfa_debug_instance *inst_p;
ret = generic_open(inode, file);
if (ret) {
pr_notice("[%s] Fail to open proc file\n", __func__);
return ret;
}
inst_p = (struct tfa_debug_instance *)file->private_data;
inst_p->vaddr = info.runtime.body_head;
inst_p->debug_buf_size = info.runtime.debug_buf_size;
inst_p->read_offset = 0;
inst_p->debug_buf_hdr_p = info.runtime.buf_hdr_p;
return 0;
}
static ssize_t runtime_log_write(struct file *file,
const char __user *buf, size_t count, loff_t *ppos)
{
unsigned long ret = -1;
unsigned long param = -1;
struct arm_smccc_res res;
if (count < 12) {
/* use kstrtoul_from_user() instead of */
/* copy_from_user() and kstrtoul() */
ret = kstrtoul_from_user(buf, count, 16, &param);
}
pr_info("[%s]param:0x%lx, count:%zu, ret:%ld\n",
__func__, param, count, ret);
if (param == MAGIC_RELEASE_LOG)
atomic_set(&info.release_log, 1);
else
atomic_set(&info.release_log, 0);
if (!ret) {
arm_smccc_smc(MTK_SIP_KERNEL_ATF_DEBUG,
param, 0, 0, 0, 0, 0, 0, &res);
*ppos = count;
} else
return 0;
return count;
}
static unsigned int runtime_log_poll(struct file *file,
poll_table *wait)
{
unsigned int ret = 0;
struct tfa_debug_instance *inst_p = file->private_data;
poll_wait(file, &(inst_p->waitq), wait);
if (!is_runtime_empty_for_read(file))
ret = POLLIN | POLLRDNORM;
return ret;
}
static const struct proc_ops tfa_debug_runtime_fops = {
.proc_open = runtime_log_open,
.proc_read = runtime_log_read,
.proc_write = runtime_log_write,
.proc_poll = runtime_log_poll,
.proc_release = raw_release,
.proc_lseek = no_llseek,
};
static int lookup_reserved_memory(void)
{
struct device_node *tfa_debug_node;
struct reserved_mem *rmem;
tfa_debug_node = of_find_compatible_node(NULL, NULL,
ATF_LOG_RESERVED_MEMORY_KEY);
if (!tfa_debug_node) {
pr_info("compatible:%s not found\n", ATF_LOG_RESERVED_MEMORY_KEY);
return -EINVAL;
}
rmem = of_reserved_mem_lookup(tfa_debug_node);
if (!rmem) {
pr_info("atf_logger reserved mem not found\n");
return -EINVAL;
}
info.debug_buf_paddr = rmem->base;
info.total_size = rmem->size;
pr_info("%s, start: 0x%llx, size: 0x%llx\n",
ATF_LOG_RESERVED_MEMORY_KEY,
info.debug_buf_paddr, info.total_size);
if ((info.debug_buf_paddr == 0) ||
(info.total_size == 0)) {
pr_notice("debug buf physical addr or size is zero:0x%llx 0x%llx\n",
info.debug_buf_paddr, info.total_size);
return -EINVAL;
}
/* remap reserved memory as non-cacheable */
info.vaddr = ioremap(info.debug_buf_paddr, info.total_size);
if (IS_ERR(info.vaddr)) {
pr_notice("Fail to remap debug buf vaddr:%d\n",
PTR_ERR(info.vaddr));
return -ENOMEM;
}
pr_info("debug buf vaddr:0x%llx\n", info.vaddr);
return 0;
}
/* So far we only export runtime log */
static int lookup_runtime_log(void)
{
struct debug_buf_table_header *debug_buf_table_p;
struct debug_buf_hdr *debug_buf_hdr_p;
/* The first debug_buf_hdr */
u32 debug_buf_hdr_offset;
u32 i;
debug_buf_table_p = (struct debug_buf_table_header *)info.vaddr;
if (debug_buf_table_p->magic != DEBUG_BUF_MAGIC) {
pr_notice("The debug buf magic is invalid:0x%x\n", debug_buf_table_p->magic);
return -EINVAL;
}
debug_buf_hdr_offset = debug_buf_table_p->hdr_size +
(debug_buf_table_p->debug_buf_entry_size *
debug_buf_table_p->debug_buf_entry_count);
debug_buf_hdr_p = (struct debug_buf_hdr *)((char *)debug_buf_table_p +
debug_buf_hdr_offset);
for (i = 0 ; i < debug_buf_table_p->debug_buf_entry_count
; i++) {
if (debug_buf_hdr_p->magic != DEBUG_BUF_ENTRY_MAGIC) {
pr_notice("Invalid debug buffer entry header:0x%x\n",
debug_buf_hdr_p->magic);
return -EINVAL;
}
if (strncmp(debug_buf_hdr_p->debug_buf_name,
RUNTIME_LOG_NAME, DEBUG_BUF_NAME_LEN) == 0)
break;
debug_buf_hdr_p = (struct debug_buf_hdr *)
((char *)debug_buf_hdr_p +
debug_buf_hdr_p->debug_buf_size +
sizeof(struct debug_buf_hdr));
}
if (i == debug_buf_table_p->debug_buf_entry_count) {
pr_notice("Fail to find %s\n", RUNTIME_LOG_NAME);
return -ENXIO;
}
info.runtime.buf_hdr_p = debug_buf_hdr_p;
info.runtime.body_head = (void *)((char *)debug_buf_hdr_p +
sizeof(struct debug_buf_hdr));
info.runtime.debug_buf_size = debug_buf_hdr_p->debug_buf_size;
return 0;
}
static int tfa_time_sync_suspend(struct device *dev)
{
struct arm_smccc_res res;
/* Notify tf-a logger to clear indicator */
arm_smccc_smc(MTK_SIP_KERNEL_TIME_SYNC,
0, 0, 1, 0, 0, 0, 0, &res);
return 0;
}
static int tfa_time_sync_resume(struct device *dev)
{
/* Get local_clock and sync to TF-A */
u64 time_to_sync = local_clock();
struct arm_smccc_res res;
pr_info("[%s]time_to_sync:0x%llx\n", __func__, time_to_sync);
/* Separate time_to_sync into two 32 bits args */
arm_smccc_smc(MTK_SIP_KERNEL_TIME_SYNC,
(u32)time_to_sync, (u32)(time_to_sync >> 32), 0, 0,
0, 0, 0, &res);
return 0;
}
static const struct dev_pm_ops tfa_pm_ops = {
.suspend_noirq = tfa_time_sync_suspend,
.resume_noirq = tfa_time_sync_resume,
};
static int __init tfa_debug_probe(struct platform_device *pdev)
{
struct proc_dir_entry *runtime_log_snapshot_proc_file = NULL;
struct proc_dir_entry *runtime_log_proc_file = NULL;
struct proc_dir_entry *debug_raw_proc_file = NULL;
int ret;
pr_notice("%s\n", __func__);
/* Synchronize timestamp in Kernel and ATF */
tfa_time_sync_resume(NULL);
ret = lookup_reserved_memory();
if (ret) {
pr_notice("Fail to look up atf_logger reserved memory:%d\n",
ret);
return ret;
}
info.proc_dir = proc_mkdir(DEBUG_BUF_PROC_FOLDER_NAME, NULL);
if (info.proc_dir == NULL) {
pr_info("tfa_debug proc_mkdir failed\n");
return -ENOMEM;
}
atomic_set(&info.release_log, 0);
if (lookup_runtime_log() == 0) {
runtime_log_snapshot_proc_file =
proc_create(DEBUG_BUF_ATF_LOGGER_SNAPSHOT_PROC_NAME, 0440,
info.proc_dir, &tfa_debug_runtime_snap_fops);
if (runtime_log_snapshot_proc_file == NULL) {
pr_info("Fail to create %s proc file\n",
DEBUG_BUF_ATF_LOGGER_SNAPSHOT_PROC_NAME);
return -ENOMEM;
}
runtime_log_proc_file =
proc_create(DEBUG_BUF_ATF_LOGGER_PROC_NAME, 0440,
info.proc_dir, &tfa_debug_runtime_fops);
if (runtime_log_proc_file == NULL) {
pr_info("Fail to create %s proc file\n",
DEBUG_BUF_ATF_LOGGER_PROC_NAME);
return -ENOMEM;
}
}
debug_raw_proc_file =
proc_create(DEBUG_BUF_TOTAL_RAW_PROC_NAME, 0440,
info.proc_dir, &tfa_debug_raw_fops);
if (debug_raw_proc_file == NULL) {
pr_info("Fail to create %s proc file\n",
DEBUG_BUF_TOTAL_RAW_PROC_NAME);
return -ENOMEM;
}
return 0;
}
static int tfa_debug_remove(struct platform_device *pdev)
{
pr_notice("%s\n", __func__);
if (!is_debug_buf_info_valid())
return -EINVAL;
if (info.proc_dir == NULL)
return -ENOENT;
remove_proc_entry(DEBUG_BUF_ATF_LOGGER_SNAPSHOT_PROC_NAME,
info.proc_dir);
remove_proc_entry(DEBUG_BUF_TOTAL_RAW_PROC_NAME,
info.proc_dir);
remove_proc_entry(DEBUG_BUF_PROC_FOLDER_NAME, NULL);
iounmap(info.vaddr);
return 0;
}
static struct platform_driver tfa_debug_driver_probe = {
.probe = tfa_debug_probe,
.remove = tfa_debug_remove,
.driver = {
.name = "tfa_debug",
.owner = THIS_MODULE,
#if CONFIG_OF
.of_match_table = tfa_debug_of_ids,
#endif
.pm = &tfa_pm_ops,
},
};
int tfa_debug_drv_register(void)
{
int ret;
pr_notice("%s\n", __func__);
ret = platform_driver_register(&tfa_debug_driver_probe);
if (ret)
pr_notice("%s fail, ret 0x%x\n", __func__, ret);
return ret;
}
int tfa_debug_drv_unregister(void)
{
pr_notice("%s\n", __func__);
platform_driver_unregister(&tfa_debug_driver_probe);
return 0;
}