1
0
mirror of https://github.com/physwizz/a155-U-u1.git synced 2025-10-28 16:15:46 +00:00
Files
a155-U-u1/kernel-5.10/drivers/samsung/debug/sec_debug_log.c
physwizz 99537be4e2 first
2024-03-11 06:53:12 +11:00

340 lines
8.7 KiB
C

/*
* Copyright (c) 2022 Samsung Electronics Co., Ltd.
* http://www.samsung.com
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
#include <linux/module.h>
#include <linux/console.h>
#include <linux/kernel.h>
#include <linux/slab.h>
#include <linux/stat.h>
#include <linux/uaccess.h>
#include <linux/proc_fs.h>
#include <linux/memblock.h>
#include <linux/sched/clock.h>
#include <linux/of.h>
#include <linux/of_reserved_mem.h>
#include <linux/vmalloc.h>
#include <linux/io.h>
#include <linux/platform_device.h>
#include <linux/sec_ext.h>
#include <linux/sec_debug.h>
#include <linux/notifier.h>
#include <trace/hooks/debug.h>
static u32 klog_rmem_base;
static u32 klog_rmem_size;
static char *secdbg_klog_buf;
static unsigned int secdbg_klog_size;
static unsigned int *secdbg_klog_buf_pos;
static unsigned int *secdbg_klog_head;
void *secdbg_log_rmem_set_vmap(phys_addr_t base, phys_addr_t size)
{
pgprot_t prot = __pgprot(PROT_NORMAL_NC);
int i, page_size;
struct page *page, **pages;
void *vaddr;
page_size = size / PAGE_SIZE;
pages = kzalloc(sizeof(struct page *) * page_size, GFP_KERNEL);
page = phys_to_page(base);
for (i = 0; i < page_size; i++)
pages[i] = page++;
vaddr = vmap(pages, page_size, VM_NO_GUARD | VM_MAP, prot);
kfree(pages);
pr_notice("%s, base=%llx, size=%llx, vaddr=%llx\n", __func__, base, size, vaddr);
return vaddr;
}
EXPORT_SYMBOL(secdbg_log_rmem_set_vmap);
struct reserved_mem *secdbg_log_get_rmem(const char *compatible)
{
struct reserved_mem *rmem;
struct device_node *rmem_np;
pr_info("%s: start to get %s\n", __func__, compatible);
rmem_np = of_find_compatible_node(NULL, NULL, compatible);
if (!rmem_np) {
pr_info("%s: no such reserved mem compatable with %s\n", __func__,
compatible);
return 0;
}
rmem = of_reserved_mem_lookup(rmem_np);
if (!rmem) {
pr_info("%s: no such reserved mem compatable with %s\n", __func__,
compatible);
return 0;
} else if (!rmem->base || !rmem->size) {
pr_info("%s: wrong base(0x%llx) or size(0x%llx)\n",
__func__, rmem->base, rmem->size);
return 0;
}
pr_info("%s: found (base=%llx, size=%llx)\n", __func__,
(unsigned long long)rmem->base, (unsigned long long)rmem->size);
return rmem;
}
EXPORT_SYMBOL(secdbg_log_get_rmem);
static u32 secdbg_lastklog_base;
static u32 secdbg_lastklog_size;
static char *last_kmsg_buffer;
static unsigned int last_kmsg_size;
static void secdbg_save_lastklog(unsigned int lastkbase, unsigned int lastksize)
{
/* provide previous log as last_kmsg */
unsigned int pos = *secdbg_klog_buf_pos;
last_kmsg_buffer = phys_to_virt(lastkbase);
if (last_kmsg_buffer) {
if (*secdbg_klog_head == 1) {
last_kmsg_size = secdbg_klog_size;
memcpy(last_kmsg_buffer, &secdbg_klog_buf[pos], secdbg_klog_size - pos);
memcpy(&last_kmsg_buffer[secdbg_klog_size - pos], secdbg_klog_buf, pos);
} else {
last_kmsg_size = pos;
memcpy(last_kmsg_buffer, secdbg_klog_buf, lastksize);
}
pr_info("%s: saved old log at %u@%p\n",
__func__, last_kmsg_size, last_kmsg_buffer);
} else
pr_err("%s: failed saving old log %u@%p\n",
__func__, last_kmsg_size, last_kmsg_buffer);
}
static ssize_t sec_last_kmsg_read(struct file *file, char __user *buf,
size_t len, loff_t *offset)
{
loff_t pos = *offset;
ssize_t count;
if (pos >= last_kmsg_size)
return 0;
count = min(len, (size_t) (last_kmsg_size - pos));
if (copy_to_user(buf, last_kmsg_buffer + pos, count))
return -EFAULT;
*offset += count;
return count;
}
static const struct proc_ops last_kmsg_file_ops = {
.proc_read = sec_last_kmsg_read,
};
static int init_sec_last_kmsg_proc(void)
{
struct proc_dir_entry *entry;
if (last_kmsg_buffer == NULL)
return 0;
entry = proc_create("last_kmsg", S_IFREG | 0444,
NULL, &last_kmsg_file_ops);
if (!entry) {
pr_err("%s: failed to create proc entry\n", __func__);
return 0;
}
proc_set_size(entry, last_kmsg_size);
return 0;
}
static void secdbg_lastkernel_log_init(void)
{
struct reserved_mem *rmem;
rmem = secdbg_log_get_rmem("samsung,secdbg-last-klog");
secdbg_lastklog_base = rmem->base;
secdbg_lastklog_size = rmem->size;
pr_notice("%s: 0x%llx - 0x%llx (0x%llx)\n",
"secdbg-last-klog", (unsigned long long)rmem->base,
(unsigned long long)rmem->base + (unsigned long long)rmem->size,
(unsigned long long)rmem->size);
}
static ssize_t sec_log_read(struct file *file, char __user *buf,
size_t len, loff_t *offset)
{
loff_t pos = *offset;
ssize_t count;
if (pos >= klog_rmem_size)
return 0;
count = min(len, (size_t) (klog_rmem_size - pos));
if (copy_to_user(buf, secdbg_klog_buf + pos, count))
return -EFAULT;
*offset += count;
return count;
}
static const struct proc_ops sec_log_file_ops = {
.proc_read = sec_log_read,
};
static int init_sec_log_proc(void)
{
struct proc_dir_entry *entry;
if (secdbg_klog_buf == NULL)
return 0;
entry = proc_create("sec_log", S_IFREG | 0444,
NULL, &sec_log_file_ops);
if (!entry) {
pr_err("%s: failed to create proc entry\n", __func__);
return 0;
}
proc_set_size(entry, klog_rmem_size);
return 0;
}
static inline void secdbg_hook_kernel_log(const char *text, size_t size)
{
unsigned int pos = *secdbg_klog_buf_pos;
if (likely((unsigned int)size + pos <= secdbg_klog_size))
memcpy(&secdbg_klog_buf[pos], text, (unsigned int)size);
else {
unsigned int first = secdbg_klog_size - pos;
unsigned int second = (unsigned int)size - first;
memcpy(&secdbg_klog_buf[pos], text, first);
memcpy(&secdbg_klog_buf[0], text + first, second);
*secdbg_klog_head = 1;
}
(*secdbg_klog_buf_pos) += (unsigned int)size;
/* Check overflow */
if (unlikely(*secdbg_klog_buf_pos >= secdbg_klog_size))
*secdbg_klog_buf_pos -= secdbg_klog_size;
}
static void (*hook_init_log)(const char *str, size_t size);
void register_hook_init_log(void (*func)(const char *str, size_t size))
{
if (!func)
return;
hook_init_log = func;
pr_info("%s done!\n", __func__);
}
EXPORT_SYMBOL_GPL(register_hook_init_log);
static void secdbg_klog_console_write(struct console *con, const char *s,
unsigned c)
{
secdbg_hook_kernel_log(s, c);
if ((hook_init_log) && task_pid_nr(current) == 1)
hook_init_log(s, c);
}
static struct console secdbg_klog_console = {
.name = "secdbg_klog",
.write = secdbg_klog_console_write,
.flags = CON_PRINTBUFFER | CON_ENABLED | CON_ANYTIME,
.index = -1,
};
static int secdbg_klog_buf_init(void)
{
unsigned int *klog_magic;
secdbg_klog_buf = secdbg_log_rmem_set_vmap(klog_rmem_base, klog_rmem_size);
secdbg_klog_size = klog_rmem_size - (sizeof(*secdbg_klog_head) + sizeof(*secdbg_klog_buf_pos) + sizeof(*klog_magic));
secdbg_klog_head = (unsigned int *)(secdbg_klog_buf + secdbg_klog_size);
secdbg_klog_buf_pos = (unsigned int *)(secdbg_klog_buf + secdbg_klog_size + sizeof(*secdbg_klog_head));
klog_magic = (unsigned int *)(secdbg_klog_buf + secdbg_klog_size + sizeof(*secdbg_klog_head) + sizeof(*secdbg_klog_buf_pos));
pr_info("%s: *secdbg_klog_head:%u, *klog_magic:%x, *secdbg_klog_buf_pos:%u, secdbg_klog_buf:0x%p, secdbg_klog_size:0x%x\n",
__func__, *secdbg_klog_head, *klog_magic, *secdbg_klog_buf_pos, secdbg_klog_buf, secdbg_klog_size);
if (*klog_magic != LOG_MAGIC) {
pr_info("%s: no old log found\n", __func__);
*secdbg_klog_head = 0;
*secdbg_klog_buf_pos = 0;
*klog_magic = LOG_MAGIC;
}
else if ((reset_reason == RR_K) ||
(reset_reason == RR_D) ||
(reset_reason == RR_P) ||
(reset_reason == RR_C) ||
(reset_reason == RR_M)) {
last_kmsg_size = secdbg_lastklog_size;
last_kmsg_buffer = phys_to_virt(secdbg_lastklog_base);
pr_info("%s: previous klog on last_kmsg buffer (stored by lk_aee)\n", __func__);
} else {
secdbg_save_lastklog(secdbg_lastklog_base, secdbg_lastklog_size);
}
return 0;
}
static void secdbg_kernel_log_init(void)
{
struct reserved_mem *rmem;
rmem = secdbg_log_get_rmem("samsung,secdbg-kernel-log");
klog_rmem_base = rmem->base;
klog_rmem_size = rmem->size;
pr_notice("%s: 0x%llx - 0x%llx (0x%llx)\n",
"secdbg-kernel-log", (unsigned long long)rmem->base,
(unsigned long long)rmem->base + (unsigned long long)rmem->size,
(unsigned long long)rmem->size);
secdbg_klog_buf_init();
}
/*
* Log module loading order
*
* 0. sec_debug_init() @ core_initcall
* 1. secdbg_init_log_init() @ postcore_initcall
* 2. secdbg_lastkernel_log_init() @ arch_initcall
* 3. secdbg_kernel_log_init() @ arch_initcall
* 4. secdbg_pmsg_log_init() @ device_initcall
*/
int sec_debug_log_init(void)
{
pr_info("%s: start\n", __func__);
secdbg_lastkernel_log_init();
secdbg_kernel_log_init();
init_sec_log_proc();
init_sec_last_kmsg_proc();
register_console(&secdbg_klog_console);
console_suspend_enabled = false;
return 0;
}
EXPORT_SYMBOL(sec_debug_log_init);