2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2013-01-31 16:54:38 -08:00
/*
2009-07-30 23:23:42 +01:00
* Copyright ( c ) 2009 Simtec Electronics
* http : //armlinux.simtec.co.uk/
* Ben Dooks < ben @ simtec . co . uk >
*
* S3C24XX CPU Frequency scaling - debugfs status support
*/
2016-04-05 13:28:25 -07:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2009-07-30 23:23:42 +01:00
# include <linux/init.h>
2011-11-17 01:14:38 +09:00
# include <linux/export.h>
2009-07-30 23:23:42 +01:00
# include <linux/interrupt.h>
# include <linux/ioport.h>
# include <linux/cpufreq.h>
# include <linux/debugfs.h>
# include <linux/seq_file.h>
# include <linux/err.h>
2020-08-06 20:20:52 +02:00
# include <linux/soc/samsung/s3c-cpufreq-core.h>
2009-07-30 23:23:42 +01:00
static struct dentry * dbgfs_root ;
static struct dentry * dbgfs_file_io ;
static struct dentry * dbgfs_file_info ;
static struct dentry * dbgfs_file_board ;
# define print_ns(x) ((x) / 10), ((x) % 10)
static void show_max ( struct seq_file * seq , struct s3c_freq * f )
{
seq_printf ( seq , " MAX: F=%lu, H=%lu, P=%lu, A=%lu \n " ,
f - > fclk , f - > hclk , f - > pclk , f - > armclk ) ;
}
static int board_show ( struct seq_file * seq , void * p )
{
struct s3c_cpufreq_config * cfg ;
struct s3c_cpufreq_board * brd ;
cfg = s3c_cpufreq_getconfig ( ) ;
if ( ! cfg ) {
seq_printf ( seq , " no configuration registered \n " ) ;
return 0 ;
}
brd = cfg - > board ;
if ( ! brd ) {
seq_printf ( seq , " no board definition set? \n " ) ;
return 0 ;
}
seq_printf ( seq , " SDRAM refresh %u ns \n " , brd - > refresh ) ;
seq_printf ( seq , " auto_io=%u \n " , brd - > auto_io ) ;
seq_printf ( seq , " need_io=%u \n " , brd - > need_io ) ;
show_max ( seq , & brd - > max ) ;
return 0 ;
}
2018-11-07 11:13:46 -05:00
DEFINE_SHOW_ATTRIBUTE ( board ) ;
2009-07-30 23:23:42 +01:00
static int info_show ( struct seq_file * seq , void * p )
{
struct s3c_cpufreq_config * cfg ;
cfg = s3c_cpufreq_getconfig ( ) ;
if ( ! cfg ) {
seq_printf ( seq , " no configuration registered \n " ) ;
return 0 ;
}
seq_printf ( seq , " FCLK %ld Hz \n " , cfg - > freq . fclk ) ;
seq_printf ( seq , " HCLK %ld Hz (%lu.%lu ns) \n " ,
cfg - > freq . hclk , print_ns ( cfg - > freq . hclk_tns ) ) ;
seq_printf ( seq , " PCLK %ld Hz \n " , cfg - > freq . hclk ) ;
seq_printf ( seq , " ARMCLK %ld Hz \n " , cfg - > freq . armclk ) ;
seq_printf ( seq , " \n " ) ;
show_max ( seq , & cfg - > max ) ;
seq_printf ( seq , " Divisors: P=%d, H=%d, A=%d, dvs=%s \n " ,
cfg - > divs . h_divisor , cfg - > divs . p_divisor ,
cfg - > divs . arm_divisor , cfg - > divs . dvs ? " on " : " off " ) ;
seq_printf ( seq , " \n " ) ;
seq_printf ( seq , " lock_pll=%u \n " , cfg - > lock_pll ) ;
return 0 ;
}
2018-11-07 11:13:46 -05:00
DEFINE_SHOW_ATTRIBUTE ( info ) ;
2009-07-30 23:23:42 +01:00
static int io_show ( struct seq_file * seq , void * p )
{
void ( * show_bank ) ( struct seq_file * , struct s3c_cpufreq_config * , union s3c_iobank * ) ;
struct s3c_cpufreq_config * cfg ;
struct s3c_iotimings * iot ;
union s3c_iobank * iob ;
int bank ;
cfg = s3c_cpufreq_getconfig ( ) ;
if ( ! cfg ) {
seq_printf ( seq , " no configuration registered \n " ) ;
return 0 ;
}
show_bank = cfg - > info - > debug_io_show ;
if ( ! show_bank ) {
seq_printf ( seq , " no code to show bank timing \n " ) ;
return 0 ;
}
iot = s3c_cpufreq_getiotimings ( ) ;
if ( ! iot ) {
seq_printf ( seq , " no io timings registered \n " ) ;
return 0 ;
}
seq_printf ( seq , " hclk period is %lu.%lu ns \n " , print_ns ( cfg - > freq . hclk_tns ) ) ;
for ( bank = 0 ; bank < MAX_BANKS ; bank + + ) {
iob = & iot - > bank [ bank ] ;
seq_printf ( seq , " bank %d: " , bank ) ;
if ( ! iob - > io_2410 ) {
seq_printf ( seq , " nothing set \n " ) ;
continue ;
}
show_bank ( seq , cfg , iob ) ;
}
return 0 ;
}
2018-11-07 11:13:46 -05:00
DEFINE_SHOW_ATTRIBUTE ( io ) ;
2009-07-30 23:23:42 +01:00
static int __init s3c_freq_debugfs_init ( void )
{
dbgfs_root = debugfs_create_dir ( " s3c-cpufreq " , NULL ) ;
if ( IS_ERR ( dbgfs_root ) ) {
2016-04-05 13:28:24 -07:00
pr_err ( " %s: error creating debugfs root \n " , __func__ ) ;
2009-07-30 23:23:42 +01:00
return PTR_ERR ( dbgfs_root ) ;
}
dbgfs_file_io = debugfs_create_file ( " io-timing " , S_IRUGO , dbgfs_root ,
2018-11-07 11:13:46 -05:00
NULL , & io_fops ) ;
2009-07-30 23:23:42 +01:00
dbgfs_file_info = debugfs_create_file ( " info " , S_IRUGO , dbgfs_root ,
2018-11-07 11:13:46 -05:00
NULL , & info_fops ) ;
2009-07-30 23:23:42 +01:00
dbgfs_file_board = debugfs_create_file ( " board " , S_IRUGO , dbgfs_root ,
2018-11-07 11:13:46 -05:00
NULL , & board_fops ) ;
2009-07-30 23:23:42 +01:00
return 0 ;
}
late_initcall ( s3c_freq_debugfs_init ) ;