2015-11-11 07:59:01 +05:30
/*
* Generic OPP debugfs interface
*
* Copyright ( C ) 2015 - 2016 Viresh Kumar < viresh . kumar @ linaro . org >
*
* 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 .
*/
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
# include <linux/debugfs.h>
# include <linux/device.h>
# include <linux/err.h>
# include <linux/init.h>
# include <linux/limits.h>
# 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 ) ;
}
2016-02-16 14:17:53 +05:30
int opp_debug_create_one ( struct dev_pm_opp * opp , struct opp_table * opp_table )
2015-11-11 07:59:01 +05:30
{
2016-02-16 14:17:53 +05:30
struct dentry * pdentry = opp_table - > dentry ;
2015-11-11 07:59:01 +05:30
struct dentry * d ;
char name [ 25 ] ; /* 20 chars for 64 bit value + 5 (opp:\0) */
/* Rate is unique to each OPP, use it to give opp-name */
snprintf ( name , sizeof ( name ) , " opp:%lu " , opp - > rate ) ;
/* Create per-opp directory */
d = debugfs_create_dir ( name , pdentry ) ;
if ( ! d )
return - ENOMEM ;
if ( ! debugfs_create_bool ( " available " , S_IRUGO , d , & opp - > available ) )
return - ENOMEM ;
if ( ! debugfs_create_bool ( " dynamic " , S_IRUGO , d , & opp - > dynamic ) )
return - ENOMEM ;
if ( ! debugfs_create_bool ( " turbo " , S_IRUGO , d , & opp - > turbo ) )
return - ENOMEM ;
if ( ! debugfs_create_bool ( " suspend " , S_IRUGO , d , & opp - > suspend ) )
return - ENOMEM ;
if ( ! debugfs_create_ulong ( " rate_hz " , S_IRUGO , d , & opp - > rate ) )
return - ENOMEM ;
2016-12-01 16:28:17 +05:30
if ( ! debugfs_create_ulong ( " u_volt_target " , S_IRUGO , d , & opp - > supply . u_volt ) )
2015-11-11 07:59:01 +05:30
return - ENOMEM ;
2016-12-01 16:28:17 +05:30
if ( ! debugfs_create_ulong ( " u_volt_min " , S_IRUGO , d , & opp - > supply . u_volt_min ) )
2015-11-11 07:59:01 +05:30
return - ENOMEM ;
2016-12-01 16:28:17 +05:30
if ( ! debugfs_create_ulong ( " u_volt_max " , S_IRUGO , d , & opp - > supply . u_volt_max ) )
2015-11-11 07:59:01 +05:30
return - ENOMEM ;
2016-12-01 16:28:17 +05:30
if ( ! debugfs_create_ulong ( " u_amp " , S_IRUGO , d , & opp - > supply . u_amp ) )
2015-11-11 07:59:01 +05:30
return - ENOMEM ;
if ( ! debugfs_create_ulong ( " clock_latency_ns " , S_IRUGO , d ,
& opp - > clock_latency_ns ) )
return - ENOMEM ;
opp - > dentry = d ;
return 0 ;
}
2016-02-16 14:17:53 +05:30
static int opp_list_debug_create_dir ( struct opp_device * opp_dev ,
struct opp_table * opp_table )
2015-11-11 07:59:01 +05:30
{
2016-02-16 14:17:53 +05:30
const struct device * dev = opp_dev - > dev ;
2015-11-11 07:59:01 +05:30
struct dentry * d ;
2016-02-16 14:17:53 +05:30
opp_set_dev_name ( dev , opp_table - > dentry_name ) ;
2015-11-11 07:59:01 +05:30
/* Create device specific directory */
2016-02-16 14:17:53 +05:30
d = debugfs_create_dir ( opp_table - > dentry_name , rootdir ) ;
2015-11-11 07:59:01 +05:30
if ( ! d ) {
dev_err ( dev , " %s: Failed to create debugfs dir \n " , __func__ ) ;
return - ENOMEM ;
}
2016-02-16 14:17:53 +05:30
opp_dev - > dentry = d ;
opp_table - > dentry = d ;
2015-11-11 07:59:01 +05:30
return 0 ;
}
2016-02-16 14:17:53 +05:30
static int opp_list_debug_create_link ( struct opp_device * opp_dev ,
struct opp_table * opp_table )
2015-11-11 07:59:01 +05:30
{
2016-02-16 14:17:53 +05:30
const struct device * dev = opp_dev - > dev ;
2015-11-11 07:59:01 +05:30
char name [ NAME_MAX ] ;
struct dentry * d ;
2016-02-16 14:17:53 +05:30
opp_set_dev_name ( opp_dev - > dev , name ) ;
2015-11-11 07:59:01 +05:30
/* Create device specific directory link */
2016-02-16 14:17:53 +05:30
d = debugfs_create_symlink ( name , rootdir , opp_table - > dentry_name ) ;
2015-11-11 07:59:01 +05:30
if ( ! d ) {
dev_err ( dev , " %s: Failed to create link \n " , __func__ ) ;
return - ENOMEM ;
}
2016-02-16 14:17:53 +05:30
opp_dev - > dentry = d ;
2015-11-11 07:59:01 +05:30
return 0 ;
}
/**
* opp_debug_register - add a device opp node to the debugfs ' opp ' directory
2016-02-16 14:17:53 +05:30
* @ opp_dev : opp - dev pointer for device
* @ opp_table : the device - opp being added
2015-11-11 07:59:01 +05:30
*
* 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 .
*
* Return : 0 on success , otherwise negative error .
*/
2016-02-16 14:17:53 +05:30
int opp_debug_register ( struct opp_device * opp_dev , struct opp_table * opp_table )
2015-11-11 07:59:01 +05:30
{
if ( ! rootdir ) {
pr_debug ( " %s: Uninitialized rootdir \n " , __func__ ) ;
return - EINVAL ;
}
2016-02-16 14:17:53 +05:30
if ( opp_table - > dentry )
return opp_list_debug_create_link ( opp_dev , opp_table ) ;
2015-11-11 07:59:01 +05:30
2016-02-16 14:17:53 +05:30
return opp_list_debug_create_dir ( opp_dev , opp_table ) ;
2015-11-11 07:59:01 +05:30
}
2016-02-16 14:17:53 +05:30
static void opp_migrate_dentry ( struct opp_device * opp_dev ,
struct opp_table * opp_table )
2015-11-11 07:59:01 +05:30
{
2016-02-16 14:17:53 +05:30
struct opp_device * new_dev ;
2015-11-11 07:59:01 +05:30
const struct device * dev ;
struct dentry * dentry ;
2016-02-16 14:17:53 +05:30
/* Look for next opp-dev */
list_for_each_entry ( new_dev , & opp_table - > dev_list , node )
if ( new_dev ! = opp_dev )
2015-11-11 07:59:01 +05:30
break ;
/* new_dev is guaranteed to be valid here */
dev = new_dev - > dev ;
debugfs_remove_recursive ( new_dev - > dentry ) ;
2016-02-16 14:17:53 +05:30
opp_set_dev_name ( dev , opp_table - > dentry_name ) ;
2015-11-11 07:59:01 +05:30
2016-02-16 14:17:53 +05:30
dentry = debugfs_rename ( rootdir , opp_dev - > dentry , rootdir ,
opp_table - > dentry_name ) ;
2015-11-11 07:59:01 +05:30
if ( ! dentry ) {
dev_err ( dev , " %s: Failed to rename link from: %s to %s \n " ,
2016-02-16 14:17:53 +05:30
__func__ , dev_name ( opp_dev - > dev ) , dev_name ( dev ) ) ;
2015-11-11 07:59:01 +05:30
return ;
}
new_dev - > dentry = dentry ;
2016-02-16 14:17:53 +05:30
opp_table - > dentry = dentry ;
2015-11-11 07:59:01 +05:30
}
/**
* opp_debug_unregister - remove a device opp node from debugfs opp directory
2016-02-16 14:17:53 +05:30
* @ opp_dev : opp - dev pointer for device
* @ opp_table : the device - opp being removed
2015-11-11 07:59:01 +05:30
*
* Dynamically removes device specific directory from debugfs ' opp ' directory .
*/
2016-02-16 14:17:53 +05:30
void opp_debug_unregister ( struct opp_device * opp_dev ,
struct opp_table * opp_table )
2015-11-11 07:59:01 +05:30
{
2016-02-16 14:17:53 +05:30
if ( opp_dev - > dentry = = opp_table - > dentry ) {
2015-11-11 07:59:01 +05:30
/* Move the real dentry object under another device */
2016-02-16 14:17:53 +05:30
if ( ! list_is_singular ( & opp_table - > dev_list ) ) {
opp_migrate_dentry ( opp_dev , opp_table ) ;
2015-11-11 07:59:01 +05:30
goto out ;
}
2016-02-16 14:17:53 +05:30
opp_table - > dentry = NULL ;
2015-11-11 07:59:01 +05:30
}
2016-02-16 14:17:53 +05:30
debugfs_remove_recursive ( opp_dev - > dentry ) ;
2015-11-11 07:59:01 +05:30
out :
2016-02-16 14:17:53 +05:30
opp_dev - > dentry = NULL ;
2015-11-11 07:59:01 +05:30
}
static int __init opp_debug_init ( void )
{
/* Create /sys/kernel/debug/opp directory */
rootdir = debugfs_create_dir ( " opp " , NULL ) ;
if ( ! rootdir ) {
pr_err ( " %s: Failed to create root directory \n " , __func__ ) ;
return - ENOMEM ;
}
return 0 ;
}
core_initcall ( opp_debug_init ) ;