2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2015-11-11 05:29:01 +03:00
/*
* Generic OPP debugfs interface
*
* Copyright ( C ) 2015 - 2016 Viresh Kumar < viresh . kumar @ linaro . org >
*/
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
# include <linux/debugfs.h>
# include <linux/device.h>
# include <linux/err.h>
2022-02-10 12:07:53 +03:00
# include <linux/of.h>
2015-11-11 05:29:01 +03:00
# include <linux/init.h>
# include <linux/limits.h>
2016-12-01 13:58:19 +03:00
# include <linux/slab.h>
2015-11-11 05:29:01 +03:00
# include "opp.h"
static struct dentry * rootdir ;
static void opp_set_dev_name ( const struct device * dev , char * name )
{
if ( dev - > parent )
snprintf ( name , NAME_MAX , " %s-%s " , dev_name ( dev - > parent ) ,
dev_name ( dev ) ) ;
else
snprintf ( name , NAME_MAX , " %s " , dev_name ( dev ) ) ;
}
void opp_debug_remove_one ( struct dev_pm_opp * opp )
{
debugfs_remove_recursive ( opp - > dentry ) ;
}
2020-05-18 14:25:32 +03:00
static ssize_t bw_name_read ( struct file * fp , char __user * userbuf ,
size_t count , loff_t * ppos )
{
struct icc_path * path = fp - > private_data ;
char buf [ 64 ] ;
int i ;
i = scnprintf ( buf , sizeof ( buf ) , " %.62s \n " , icc_get_name ( path ) ) ;
return simple_read_from_buffer ( userbuf , count , ppos , buf , i ) ;
}
static const struct file_operations bw_name_fops = {
. open = simple_open ,
. read = bw_name_read ,
. llseek = default_llseek ,
} ;
static void opp_debug_create_bw ( struct dev_pm_opp * opp ,
struct opp_table * opp_table ,
struct dentry * pdentry )
{
struct dentry * d ;
char name [ 11 ] ;
int i ;
for ( i = 0 ; i < opp_table - > path_count ; i + + ) {
snprintf ( name , sizeof ( name ) , " icc-path-%.1d " , i ) ;
/* Create per-path directory */
d = debugfs_create_dir ( name , pdentry ) ;
debugfs_create_file ( " name " , S_IRUGO , d , opp_table - > paths [ i ] ,
& bw_name_fops ) ;
debugfs_create_u32 ( " peak_bw " , S_IRUGO , d ,
& opp - > bandwidth [ i ] . peak ) ;
debugfs_create_u32 ( " avg_bw " , S_IRUGO , d ,
& opp - > bandwidth [ i ] . avg ) ;
}
}
2022-06-10 09:32:04 +03:00
static void opp_debug_create_clks ( struct dev_pm_opp * opp ,
struct opp_table * opp_table ,
struct dentry * pdentry )
{
char name [ 12 ] ;
int i ;
if ( opp_table - > clk_count = = 1 ) {
debugfs_create_ulong ( " rate_hz " , S_IRUGO , pdentry , & opp - > rates [ 0 ] ) ;
return ;
}
for ( i = 0 ; i < opp_table - > clk_count ; i + + ) {
snprintf ( name , sizeof ( name ) , " rate_hz_%d " , i ) ;
debugfs_create_ulong ( name , S_IRUGO , pdentry , & opp - > rates [ i ] ) ;
}
}
2019-01-22 18:21:17 +03:00
static void opp_debug_create_supplies ( struct dev_pm_opp * opp ,
2016-12-01 13:58:19 +03:00
struct opp_table * opp_table ,
struct dentry * pdentry )
{
struct dentry * d ;
2017-05-23 07:02:13 +03:00
int i ;
2016-12-01 13:58:19 +03:00
2017-05-23 07:02:13 +03:00
for ( i = 0 ; i < opp_table - > regulator_count ; i + + ) {
2017-09-21 08:45:36 +03:00
char name [ 15 ] ;
snprintf ( name , sizeof ( name ) , " supply-%d " , i ) ;
2016-12-01 13:58:19 +03:00
/* Create per-opp directory */
d = debugfs_create_dir ( name , pdentry ) ;
2019-01-22 18:21:17 +03:00
debugfs_create_ulong ( " u_volt_target " , S_IRUGO , d ,
& opp - > supplies [ i ] . u_volt ) ;
2016-12-01 13:58:19 +03:00
2019-01-22 18:21:17 +03:00
debugfs_create_ulong ( " u_volt_min " , S_IRUGO , d ,
& opp - > supplies [ i ] . u_volt_min ) ;
2016-12-01 13:58:19 +03:00
2019-01-22 18:21:17 +03:00
debugfs_create_ulong ( " u_volt_max " , S_IRUGO , d ,
& opp - > supplies [ i ] . u_volt_max ) ;
2016-12-01 13:58:19 +03:00
2019-01-22 18:21:17 +03:00
debugfs_create_ulong ( " u_amp " , S_IRUGO , d ,
& opp - > supplies [ i ] . u_amp ) ;
2022-03-02 14:29:14 +03:00
debugfs_create_ulong ( " u_watt " , S_IRUGO , d ,
& opp - > supplies [ i ] . u_watt ) ;
2017-05-23 07:02:13 +03:00
}
2016-12-01 13:58:19 +03:00
}
2019-01-22 18:21:17 +03:00
void opp_debug_create_one ( struct dev_pm_opp * opp , struct opp_table * opp_table )
2015-11-11 05:29:01 +03:00
{
2016-02-16 11:47:53 +03:00
struct dentry * pdentry = opp_table - > dentry ;
2015-11-11 05:29:01 +03:00
struct dentry * d ;
2018-04-06 12:05:45 +03:00
unsigned long id ;
2015-11-11 05:29:01 +03:00
char name [ 25 ] ; /* 20 chars for 64 bit value + 5 (opp:\0) */
2018-04-06 12:05:45 +03:00
/*
* Get directory name for OPP .
*
* - Normally rate is unique to each OPP , use it to get unique opp - name .
2022-06-10 09:32:04 +03:00
* - For some devices rate isn ' t available or there are multiple , use
* index instead for them .
2018-04-06 12:05:45 +03:00
*/
2022-06-10 09:32:04 +03:00
if ( likely ( opp_table - > clk_count = = 1 & & opp - > rates [ 0 ] ) )
id = opp - > rates [ 0 ] ;
2018-04-06 12:05:45 +03:00
else
id = _get_opp_count ( opp_table ) ;
snprintf ( name , sizeof ( name ) , " opp:%lu " , id ) ;
2015-11-11 05:29:01 +03:00
/* Create per-opp directory */
d = debugfs_create_dir ( name , pdentry ) ;
2017-10-11 10:24:14 +03:00
2019-01-22 18:21:17 +03:00
debugfs_create_bool ( " available " , S_IRUGO , d , & opp - > available ) ;
debugfs_create_bool ( " dynamic " , S_IRUGO , d , & opp - > dynamic ) ;
debugfs_create_bool ( " turbo " , S_IRUGO , d , & opp - > turbo ) ;
debugfs_create_bool ( " suspend " , S_IRUGO , d , & opp - > suspend ) ;
debugfs_create_u32 ( " performance_state " , S_IRUGO , d , & opp - > pstate ) ;
2022-02-10 12:07:53 +03:00
debugfs_create_u32 ( " level " , S_IRUGO , d , & opp - > level ) ;
2019-01-22 18:21:17 +03:00
debugfs_create_ulong ( " clock_latency_ns " , S_IRUGO , d ,
& opp - > clock_latency_ns ) ;
2015-11-11 05:29:01 +03:00
2022-02-10 12:07:53 +03:00
opp - > of_name = of_node_full_name ( opp - > np ) ;
debugfs_create_str ( " of_name " , S_IRUGO , d , ( char * * ) & opp - > of_name ) ;
2022-06-10 09:32:04 +03:00
opp_debug_create_clks ( opp , opp_table , d ) ;
2019-01-22 18:21:17 +03:00
opp_debug_create_supplies ( opp , opp_table , d ) ;
2020-05-18 14:25:32 +03:00
opp_debug_create_bw ( opp , opp_table , d ) ;
2015-11-11 05:29:01 +03:00
opp - > dentry = d ;
}
2019-01-22 18:21:17 +03:00
static void opp_list_debug_create_dir ( struct opp_device * opp_dev ,
struct opp_table * opp_table )
2015-11-11 05:29:01 +03:00
{
2016-02-16 11:47:53 +03:00
const struct device * dev = opp_dev - > dev ;
2015-11-11 05:29:01 +03:00
struct dentry * d ;
2016-02-16 11:47:53 +03:00
opp_set_dev_name ( dev , opp_table - > dentry_name ) ;
2015-11-11 05:29:01 +03:00
/* Create device specific directory */
2016-02-16 11:47:53 +03:00
d = debugfs_create_dir ( opp_table - > dentry_name , rootdir ) ;
2015-11-11 05:29:01 +03:00
2016-02-16 11:47:53 +03:00
opp_dev - > dentry = d ;
opp_table - > dentry = d ;
2015-11-11 05:29:01 +03:00
}
2019-01-22 18:21:17 +03:00
static void opp_list_debug_create_link ( struct opp_device * opp_dev ,
struct opp_table * opp_table )
2015-11-11 05:29:01 +03:00
{
char name [ NAME_MAX ] ;
2016-02-16 11:47:53 +03:00
opp_set_dev_name ( opp_dev - > dev , name ) ;
2015-11-11 05:29:01 +03:00
/* Create device specific directory link */
2019-01-22 18:21:17 +03:00
opp_dev - > dentry = debugfs_create_symlink ( name , rootdir ,
opp_table - > dentry_name ) ;
2015-11-11 05:29:01 +03:00
}
/**
* opp_debug_register - add a device opp node to the debugfs ' opp ' directory
2016-02-16 11:47:53 +03:00
* @ opp_dev : opp - dev pointer for device
* @ opp_table : the device - opp being added
2015-11-11 05:29:01 +03:00
*
* Dynamically adds device specific directory in debugfs ' opp ' directory . If the
* device - opp is shared with other devices , then links will be created for all
* devices except the first .
*/
2019-01-22 18:21:17 +03:00
void opp_debug_register ( struct opp_device * opp_dev , struct opp_table * opp_table )
2015-11-11 05:29:01 +03:00
{
2016-02-16 11:47:53 +03:00
if ( opp_table - > dentry )
2019-01-22 18:21:17 +03:00
opp_list_debug_create_link ( opp_dev , opp_table ) ;
else
opp_list_debug_create_dir ( opp_dev , opp_table ) ;
2015-11-11 05:29:01 +03:00
}
2016-02-16 11:47:53 +03:00
static void opp_migrate_dentry ( struct opp_device * opp_dev ,
struct opp_table * opp_table )
2015-11-11 05:29:01 +03:00
{
2022-03-31 11:30:18 +03:00
struct opp_device * new_dev = NULL , * iter ;
2015-11-11 05:29:01 +03:00
const struct device * dev ;
struct dentry * dentry ;
2016-02-16 11:47:53 +03:00
/* Look for next opp-dev */
2022-03-31 11:30:18 +03:00
list_for_each_entry ( iter , & opp_table - > dev_list , node )
if ( iter ! = opp_dev ) {
new_dev = iter ;
2015-11-11 05:29:01 +03:00
break ;
2022-03-31 11:30:18 +03:00
}
BUG_ON ( ! new_dev ) ;
2015-11-11 05:29:01 +03:00
/* new_dev is guaranteed to be valid here */
dev = new_dev - > dev ;
debugfs_remove_recursive ( new_dev - > dentry ) ;
2016-02-16 11:47:53 +03:00
opp_set_dev_name ( dev , opp_table - > dentry_name ) ;
2015-11-11 05:29:01 +03:00
2016-02-16 11:47:53 +03:00
dentry = debugfs_rename ( rootdir , opp_dev - > dentry , rootdir ,
opp_table - > dentry_name ) ;
2015-11-11 05:29:01 +03:00
if ( ! dentry ) {
dev_err ( dev , " %s: Failed to rename link from: %s to %s \n " ,
2016-02-16 11:47:53 +03:00
__func__ , dev_name ( opp_dev - > dev ) , dev_name ( dev ) ) ;
2015-11-11 05:29:01 +03:00
return ;
}
new_dev - > dentry = dentry ;
2016-02-16 11:47:53 +03:00
opp_table - > dentry = dentry ;
2015-11-11 05:29:01 +03:00
}
/**
* opp_debug_unregister - remove a device opp node from debugfs opp directory
2016-02-16 11:47:53 +03:00
* @ opp_dev : opp - dev pointer for device
* @ opp_table : the device - opp being removed
2015-11-11 05:29:01 +03:00
*
* Dynamically removes device specific directory from debugfs ' opp ' directory .
*/
2016-02-16 11:47:53 +03:00
void opp_debug_unregister ( struct opp_device * opp_dev ,
struct opp_table * opp_table )
2015-11-11 05:29:01 +03:00
{
2016-02-16 11:47:53 +03:00
if ( opp_dev - > dentry = = opp_table - > dentry ) {
2015-11-11 05:29:01 +03:00
/* Move the real dentry object under another device */
2016-02-16 11:47:53 +03:00
if ( ! list_is_singular ( & opp_table - > dev_list ) ) {
opp_migrate_dentry ( opp_dev , opp_table ) ;
2015-11-11 05:29:01 +03:00
goto out ;
}
2016-02-16 11:47:53 +03:00
opp_table - > dentry = NULL ;
2015-11-11 05:29:01 +03:00
}
2016-02-16 11:47:53 +03:00
debugfs_remove_recursive ( opp_dev - > dentry ) ;
2015-11-11 05:29:01 +03:00
out :
2016-02-16 11:47:53 +03:00
opp_dev - > dentry = NULL ;
2015-11-11 05:29:01 +03:00
}
static int __init opp_debug_init ( void )
{
/* Create /sys/kernel/debug/opp directory */
rootdir = debugfs_create_dir ( " opp " , NULL ) ;
return 0 ;
}
core_initcall ( opp_debug_init ) ;