|
|
|
/* Copyright (c) 2016, 2019-2020 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/kernel.h>
|
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/io.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/debugfs.h>
|
|
|
|
#include <linux/fs.h>
|
|
|
|
#include <linux/time.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/export.h>
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/mm.h>
|
|
|
|
#include <asm/arch_timer.h>
|
|
|
|
#include <soc/qcom/boot_stats.h>
|
|
|
|
|
|
|
|
#define MAX_STRING_LEN 256
|
|
|
|
#define BOOT_MARKER_MAX_LEN 50
|
|
|
|
#define MSM_ARCH_TIMER_FREQ 19200000
|
|
|
|
#define BOOTKPI_BUF_SIZE (2 * PAGE_SIZE)
|
|
|
|
|
|
|
|
struct boot_marker {
|
|
|
|
char marker_name[BOOT_MARKER_MAX_LEN];
|
|
|
|
unsigned long long int timer_value;
|
|
|
|
struct list_head list;
|
|
|
|
spinlock_t slock;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct dentry *dent_bkpi, *dent_bkpi_status, *dent_mpm_timer;
|
|
|
|
static struct boot_marker boot_marker_list;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Caller is expected to hold the list spinlock.
|
|
|
|
*/
|
|
|
|
static void delete_boot_marker(const char *name)
|
|
|
|
{
|
|
|
|
struct boot_marker *marker;
|
|
|
|
struct boot_marker *temp_addr;
|
|
|
|
|
|
|
|
list_for_each_entry_safe(marker, temp_addr, &boot_marker_list.list,
|
|
|
|
list) {
|
|
|
|
if (strnstr(marker->marker_name, name,
|
|
|
|
strlen(marker->marker_name))) {
|
|
|
|
list_del(&marker->list);
|
|
|
|
kfree(marker);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Caller is expected to hold the list spinlock.
|
|
|
|
*/
|
|
|
|
static void _create_boot_marker(const char *name,
|
|
|
|
unsigned long long int timer_value)
|
|
|
|
{
|
|
|
|
struct boot_marker *new_boot_marker;
|
|
|
|
|
|
|
|
pr_debug("%-41s:%llu.%03llu seconds\n", name,
|
|
|
|
timer_value/TIMER_KHZ,
|
|
|
|
((timer_value % TIMER_KHZ)
|
|
|
|
* 1000) / TIMER_KHZ);
|
|
|
|
|
|
|
|
new_boot_marker = kmalloc(sizeof(*new_boot_marker), GFP_ATOMIC);
|
|
|
|
if (!new_boot_marker)
|
|
|
|
return;
|
|
|
|
|
|
|
|
strlcpy(new_boot_marker->marker_name, name,
|
|
|
|
sizeof(new_boot_marker->marker_name));
|
|
|
|
new_boot_marker->timer_value = timer_value;
|
|
|
|
|
|
|
|
list_add_tail(&(new_boot_marker->list), &(boot_marker_list.list));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Update existing boot marker. Delete existing boot marker and add it
|
|
|
|
* to the tail of boot marker list (to keep timestamp in order). Used to
|
|
|
|
* avoid duplicate boot markers.
|
|
|
|
*/
|
|
|
|
void update_marker(const char *name)
|
|
|
|
{
|
|
|
|
struct boot_marker *marker;
|
|
|
|
struct boot_marker *temp_addr;
|
|
|
|
|
|
|
|
unsigned long long timer_value = msm_timer_get_sclk_ticks();
|
|
|
|
|
|
|
|
spin_lock(&boot_marker_list.slock);
|
|
|
|
list_for_each_entry_safe(marker, temp_addr, &boot_marker_list.list,
|
|
|
|
list) {
|
|
|
|
if (strnstr(marker->marker_name, name,
|
|
|
|
strlen(marker->marker_name))) {
|
|
|
|
delete_boot_marker(marker->marker_name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_create_boot_marker(name, timer_value);
|
|
|
|
spin_unlock(&boot_marker_list.slock);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(update_marker);
|
|
|
|
|
|
|
|
static void set_bootloader_stats(bool hibernation_restore)
|
|
|
|
{
|
|
|
|
if (IS_ERR_OR_NULL(boot_stats)) {
|
|
|
|
pr_err("boot_marker: imem not initialized!\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_lock(&boot_marker_list.slock);
|
|
|
|
_create_boot_marker("M - APPSBL Start - ",
|
|
|
|
readl_relaxed(&boot_stats->bootloader_start));
|
|
|
|
if (!hibernation_restore) {
|
|
|
|
_create_boot_marker("D - APPSBL Kernel Load Start - ",
|
|
|
|
readl_relaxed(&boot_stats->load_kernel_start));
|
|
|
|
_create_boot_marker("D - APPSBL Kernel Load End - ",
|
|
|
|
readl_relaxed(&boot_stats->load_kernel_done));
|
|
|
|
_create_boot_marker("D - APPSBL Kernel Load Time - ",
|
|
|
|
readl_relaxed(&boot_stats->load_kernel_done) -
|
|
|
|
readl_relaxed(&boot_stats->load_kernel_start));
|
|
|
|
_create_boot_marker("D - APPSBL Kernel Auth Time - ",
|
|
|
|
readl_relaxed(&boot_stats->bootloader_chksum_done) -
|
|
|
|
readl_relaxed(&boot_stats->bootloader_chksum_start));
|
|
|
|
} else {
|
|
|
|
_create_boot_marker("D - APPSBL Hibernation Image Load Start -",
|
|
|
|
readl_relaxed(&boot_stats->load_kernel_start));
|
|
|
|
_create_boot_marker("D - APPSBL Hibernation Image Load End - ",
|
|
|
|
readl_relaxed(&boot_stats->load_kernel_done));
|
|
|
|
}
|
|
|
|
_create_boot_marker("M - APPSBL End - ",
|
|
|
|
readl_relaxed(&boot_stats->bootloader_end));
|
|
|
|
spin_unlock(&boot_marker_list.slock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void boot_marker_cleanup(void)
|
|
|
|
{
|
|
|
|
struct boot_marker *marker;
|
|
|
|
struct boot_marker *temp_addr;
|
|
|
|
|
|
|
|
spin_lock(&boot_marker_list.slock);
|
|
|
|
list_for_each_entry_safe(marker, temp_addr, &boot_marker_list.list,
|
|
|
|
list) {
|
|
|
|
list_del(&marker->list);
|
|
|
|
kfree(marker);
|
|
|
|
}
|
|
|
|
spin_unlock(&boot_marker_list.slock);
|
|
|
|
}
|
|
|
|
|
|
|
|
void place_marker(const char *name)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_HIBERNATION
|
|
|
|
if (!strcmp(name, "M - Image Kernel Start")) {
|
|
|
|
/* In restore phase, remove Cold Boot KPIs */
|
|
|
|
boot_marker_cleanup();
|
|
|
|
set_bootloader_stats(true);
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_HIBERNATION */
|
|
|
|
spin_lock(&boot_marker_list.slock);
|
|
|
|
_create_boot_marker((char *)name, msm_timer_get_sclk_ticks());
|
|
|
|
spin_unlock(&boot_marker_list.slock);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(place_marker);
|
|
|
|
|
|
|
|
static inline u64 get_time_in_msec(u64 counter)
|
|
|
|
{
|
|
|
|
counter *= MSEC_PER_SEC;
|
|
|
|
do_div(counter, MSM_ARCH_TIMER_FREQ);
|
|
|
|
return counter;
|
|
|
|
}
|
|
|
|
|
|
|
|
void measure_wake_up_time(void)
|
|
|
|
{
|
|
|
|
u64 wake_up_time, deep_sleep_exit_time, current_time;
|
|
|
|
char wakeup_marker[50] = {0,};
|
|
|
|
|
|
|
|
current_time = arch_counter_get_cntvct();
|
|
|
|
deep_sleep_exit_time = get_sleep_exit_time();
|
|
|
|
if (deep_sleep_exit_time) {
|
|
|
|
wake_up_time = current_time - deep_sleep_exit_time;
|
|
|
|
wake_up_time = get_time_in_msec(wake_up_time);
|
|
|
|
pr_debug("Current= %llu, wakeup=%llu, kpi=%llu msec\n",
|
|
|
|
current_time, deep_sleep_exit_time, wake_up_time);
|
|
|
|
snprintf(wakeup_marker, sizeof(wakeup_marker),
|
|
|
|
"M - STR Wakeup : %llu ms", wake_up_time);
|
|
|
|
spin_lock(&boot_marker_list.slock);
|
|
|
|
delete_boot_marker("M - STR Wakeup");
|
|
|
|
spin_unlock(&boot_marker_list.slock);
|
|
|
|
place_marker(wakeup_marker);
|
|
|
|
} else {
|
|
|
|
spin_lock(&boot_marker_list.slock);
|
|
|
|
delete_boot_marker("M - STR Wakeup");
|
|
|
|
spin_unlock(&boot_marker_list.slock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(measure_wake_up_time);
|
|
|
|
|
|
|
|
static ssize_t bootkpi_reader(struct file *fp, char __user *user_buffer,
|
|
|
|
size_t count, loff_t *position)
|
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
char *buf;
|
|
|
|
int temp = 0;
|
|
|
|
struct boot_marker *marker;
|
|
|
|
|
|
|
|
buf = kmalloc(BOOTKPI_BUF_SIZE, GFP_KERNEL);
|
|
|
|
if (!buf)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
spin_lock(&boot_marker_list.slock);
|
|
|
|
list_for_each_entry(marker, &boot_marker_list.list, list) {
|
|
|
|
WARN_ON((BOOTKPI_BUF_SIZE - temp) <= 0);
|
|
|
|
temp += scnprintf(buf + temp, BOOTKPI_BUF_SIZE - temp,
|
|
|
|
"%-41s:%llu.%03llu seconds\n",
|
|
|
|
marker->marker_name,
|
|
|
|
marker->timer_value/TIMER_KHZ,
|
|
|
|
(((marker->timer_value % TIMER_KHZ)
|
|
|
|
* 1000) / TIMER_KHZ));
|
|
|
|
}
|
|
|
|
spin_unlock(&boot_marker_list.slock);
|
|
|
|
rc = simple_read_from_buffer(user_buffer, count, position, buf, temp);
|
|
|
|
kfree(buf);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t bootkpi_writer(struct file *fp, const char __user *user_buffer,
|
|
|
|
size_t count, loff_t *position)
|
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
char buf[MAX_STRING_LEN];
|
|
|
|
|
|
|
|
if (count > MAX_STRING_LEN)
|
|
|
|
return -EINVAL;
|
|
|
|
rc = simple_write_to_buffer(buf,
|
|
|
|
sizeof(buf) - 1, position, user_buffer, count);
|
|
|
|
if (rc < 0)
|
|
|
|
return rc;
|
|
|
|
buf[rc] = '\0';
|
|
|
|
place_marker(buf);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int bootkpi_open(struct inode *inode, struct file *file)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct file_operations fops_bkpi = {
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.open = bootkpi_open,
|
|
|
|
.read = bootkpi_reader,
|
|
|
|
.write = bootkpi_writer,
|
|
|
|
};
|
|
|
|
|
|
|
|
static ssize_t mpm_timer_read(struct file *fp, char __user *user_buffer,
|
|
|
|
size_t count, loff_t *position)
|
|
|
|
{
|
|
|
|
unsigned long long int timer_value;
|
|
|
|
int rc = 0;
|
|
|
|
char buf[100];
|
|
|
|
int temp = 0;
|
|
|
|
|
|
|
|
timer_value = msm_timer_get_sclk_ticks();
|
|
|
|
|
|
|
|
temp = scnprintf(buf, sizeof(buf), "%llu.%03llu seconds\n",
|
|
|
|
timer_value/TIMER_KHZ,
|
|
|
|
(((timer_value % TIMER_KHZ) * 1000) / TIMER_KHZ));
|
|
|
|
|
|
|
|
rc = simple_read_from_buffer(user_buffer, count, position, buf, temp);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mpm_timer_open(struct inode *inode, struct file *file)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mpm_timer_mmap(struct file *file, struct vm_area_struct *vma)
|
|
|
|
{
|
|
|
|
phys_addr_t addr = msm_timer_get_pa();
|
|
|
|
|
|
|
|
if (vma->vm_flags & VM_WRITE)
|
|
|
|
return -EPERM;
|
|
|
|
|
|
|
|
vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
|
|
|
|
return vm_iomap_memory(vma, addr, PAGE_SIZE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct file_operations fops_mpm_timer = {
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.open = mpm_timer_open,
|
|
|
|
.read = mpm_timer_read,
|
|
|
|
.mmap = mpm_timer_mmap,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __init init_bootkpi(void)
|
|
|
|
{
|
|
|
|
dent_bkpi = debugfs_create_dir("bootkpi", NULL);
|
|
|
|
if (IS_ERR_OR_NULL(dent_bkpi))
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
dent_bkpi_status = debugfs_create_file_unsafe("kpi_values",
|
|
|
|
0666, dent_bkpi, NULL, &fops_bkpi);
|
|
|
|
if (IS_ERR_OR_NULL(dent_bkpi_status)) {
|
|
|
|
debugfs_remove(dent_bkpi);
|
|
|
|
dent_bkpi = NULL;
|
|
|
|
pr_err("boot_marker: Could not create 'kpi_values' debugfs file\n");
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
dent_mpm_timer = debugfs_create_file("mpm_timer",
|
|
|
|
0444, dent_bkpi, NULL, &fops_mpm_timer);
|
|
|
|
if (IS_ERR_OR_NULL(dent_mpm_timer)) {
|
|
|
|
debugfs_remove(dent_bkpi_status);
|
|
|
|
dent_bkpi_status = NULL;
|
|
|
|
debugfs_remove(dent_bkpi);
|
|
|
|
dent_bkpi = NULL;
|
|
|
|
pr_err("boot_marker: Could not create 'mpm_timer' debugfs file\n");
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
debugfs_create_dir("bootloader_log", dent_bkpi);
|
|
|
|
|
|
|
|
INIT_LIST_HEAD(&boot_marker_list.list);
|
|
|
|
spin_lock_init(&boot_marker_list.slock);
|
|
|
|
set_bootloader_stats(false);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
subsys_initcall(init_bootkpi);
|
|
|
|
|
|
|
|
static void __exit exit_bootkpi(void)
|
|
|
|
{
|
|
|
|
debugfs_remove_recursive(dent_bkpi);
|
|
|
|
boot_marker_cleanup();
|
|
|
|
boot_stats_exit();
|
|
|
|
}
|
|
|
|
module_exit(exit_bootkpi);
|
|
|
|
|
|
|
|
MODULE_DESCRIPTION("MSM boot key performance indicators");
|
|
|
|
MODULE_LICENSE("GPL v2");
|