2017-11-07 19:30:07 +03:00
// SPDX-License-Identifier: GPL-2.0
2014-09-30 17:48:25 +04:00
/*
* cacheinfo support - processor cache information via sysfs
*
* Based on arch / x86 / kernel / cpu / intel_cacheinfo . c
* Author : Sudeep Holla < sudeep . holla @ arm . com >
*/
2016-10-28 11:45:30 +03:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2016-10-28 11:45:29 +03:00
# include <linux/acpi.h>
2014-09-30 17:48:25 +04:00
# include <linux/bitops.h>
# include <linux/cacheinfo.h>
# include <linux/compiler.h>
# include <linux/cpu.h>
# include <linux/device.h>
# include <linux/init.h>
2022-07-04 13:15:46 +03:00
# include <linux/of_device.h>
2014-09-30 17:48:25 +04:00
# include <linux/sched.h>
# include <linux/slab.h>
# include <linux/smp.h>
# include <linux/sysfs.h>
/* pointer to per cpu cacheinfo */
static DEFINE_PER_CPU ( struct cpu_cacheinfo , ci_cpu_cacheinfo ) ;
# define ci_cacheinfo(cpu) (&per_cpu(ci_cpu_cacheinfo, cpu))
# define cache_leaves(cpu) (ci_cacheinfo(cpu)->num_leaves)
# define per_cpu_cacheinfo(cpu) (ci_cacheinfo(cpu)->info_list)
2022-07-04 13:15:47 +03:00
# define per_cpu_cacheinfo_idx(cpu, idx) \
( per_cpu_cacheinfo ( cpu ) + ( idx ) )
2014-09-30 17:48:25 +04:00
struct cpu_cacheinfo * get_cpu_cacheinfo ( unsigned int cpu )
{
return ci_cacheinfo ( cpu ) ;
}
static inline bool cache_leaves_are_shared ( struct cacheinfo * this_leaf ,
struct cacheinfo * sib_leaf )
{
2022-07-04 13:15:48 +03:00
/*
* For non DT / ACPI systems , assume unique level 1 caches ,
* system - wide shared caches for all other levels . This will be used
* only if arch specific code has not populated shared_cpu_map
*/
if ( ! ( IS_ENABLED ( CONFIG_OF ) | | IS_ENABLED ( CONFIG_ACPI ) ) )
return ! ( this_leaf - > level = = 1 ) ;
2022-07-04 13:15:51 +03:00
if ( ( sib_leaf - > attributes & CACHE_ID ) & &
( this_leaf - > attributes & CACHE_ID ) )
return sib_leaf - > id = = this_leaf - > id ;
2018-05-12 02:57:58 +03:00
return sib_leaf - > fw_token = = this_leaf - > fw_token ;
2014-09-30 17:48:25 +04:00
}
2016-10-28 11:45:31 +03:00
2022-07-04 13:15:49 +03:00
bool last_level_cache_is_valid ( unsigned int cpu )
{
struct cacheinfo * llc ;
if ( ! cache_leaves ( cpu ) )
return false ;
llc = per_cpu_cacheinfo_idx ( cpu , cache_leaves ( cpu ) - 1 ) ;
2022-07-04 13:15:51 +03:00
return ( llc - > attributes & CACHE_ID ) | | ! ! llc - > fw_token ;
2022-07-04 13:15:49 +03:00
}
bool last_level_cache_is_shared ( unsigned int cpu_x , unsigned int cpu_y )
{
struct cacheinfo * llc_x , * llc_y ;
if ( ! last_level_cache_is_valid ( cpu_x ) | |
! last_level_cache_is_valid ( cpu_y ) )
return false ;
llc_x = per_cpu_cacheinfo_idx ( cpu_x , cache_leaves ( cpu_x ) - 1 ) ;
llc_y = per_cpu_cacheinfo_idx ( cpu_y , cache_leaves ( cpu_y ) - 1 ) ;
return cache_leaves_are_shared ( llc_x , llc_y ) ;
}
2022-07-04 13:15:48 +03:00
# ifdef CONFIG_OF
2016-10-28 11:45:31 +03:00
/* OF properties to query for a given cache type */
struct cache_type_info {
const char * size_prop ;
const char * line_size_props [ 2 ] ;
const char * nr_sets_prop ;
} ;
static const struct cache_type_info cache_type_info [ ] = {
{
. size_prop = " cache-size " ,
. line_size_props = { " cache-line-size " ,
" cache-block-size " , } ,
. nr_sets_prop = " cache-sets " ,
} , {
. size_prop = " i-cache-size " ,
. line_size_props = { " i-cache-line-size " ,
" i-cache-block-size " , } ,
. nr_sets_prop = " i-cache-sets " ,
} , {
. size_prop = " d-cache-size " ,
. line_size_props = { " d-cache-line-size " ,
" d-cache-block-size " , } ,
. nr_sets_prop = " d-cache-sets " ,
} ,
} ;
static inline int get_cacheinfo_idx ( enum cache_type type )
{
if ( type = = CACHE_TYPE_UNIFIED )
return 0 ;
return type ;
}
2018-05-12 02:57:57 +03:00
static void cache_size ( struct cacheinfo * this_leaf , struct device_node * np )
2016-10-28 11:45:31 +03:00
{
const char * propname ;
int ct_idx ;
ct_idx = get_cacheinfo_idx ( this_leaf - > type ) ;
propname = cache_type_info [ ct_idx ] . size_prop ;
2018-12-19 11:16:03 +03:00
of_property_read_u32 ( np , propname , & this_leaf - > size ) ;
2016-10-28 11:45:31 +03:00
}
/* not cache_line_size() because that's a macro in include/linux/cache.h */
2018-05-12 02:57:57 +03:00
static void cache_get_line_size ( struct cacheinfo * this_leaf ,
struct device_node * np )
2016-10-28 11:45:31 +03:00
{
int i , lim , ct_idx ;
ct_idx = get_cacheinfo_idx ( this_leaf - > type ) ;
lim = ARRAY_SIZE ( cache_type_info [ ct_idx ] . line_size_props ) ;
for ( i = 0 ; i < lim ; i + + ) {
2018-07-06 15:50:31 +03:00
int ret ;
u32 line_size ;
2016-10-28 11:45:31 +03:00
const char * propname ;
propname = cache_type_info [ ct_idx ] . line_size_props [ i ] ;
2018-07-06 15:50:31 +03:00
ret = of_property_read_u32 ( np , propname , & line_size ) ;
if ( ! ret ) {
this_leaf - > coherency_line_size = line_size ;
2016-10-28 11:45:31 +03:00
break ;
2018-07-06 15:50:31 +03:00
}
2016-10-28 11:45:31 +03:00
}
}
2018-05-12 02:57:57 +03:00
static void cache_nr_sets ( struct cacheinfo * this_leaf , struct device_node * np )
2016-10-28 11:45:31 +03:00
{
const char * propname ;
int ct_idx ;
ct_idx = get_cacheinfo_idx ( this_leaf - > type ) ;
propname = cache_type_info [ ct_idx ] . nr_sets_prop ;
2018-12-19 11:16:03 +03:00
of_property_read_u32 ( np , propname , & this_leaf - > number_of_sets ) ;
2016-10-28 11:45:31 +03:00
}
static void cache_associativity ( struct cacheinfo * this_leaf )
{
unsigned int line_size = this_leaf - > coherency_line_size ;
unsigned int nr_sets = this_leaf - > number_of_sets ;
unsigned int size = this_leaf - > size ;
/*
* If the cache is fully associative , there is no need to
* check the other properties .
*/
if ( ! ( nr_sets = = 1 ) & & ( nr_sets > 0 & & size > 0 & & line_size > 0 ) )
this_leaf - > ways_of_associativity = ( size / nr_sets ) / line_size ;
}
2018-05-12 02:57:57 +03:00
static bool cache_node_is_unified ( struct cacheinfo * this_leaf ,
struct device_node * np )
2017-11-17 14:56:41 +03:00
{
2018-05-12 02:57:57 +03:00
return of_property_read_bool ( np , " cache-unified " ) ;
2017-11-17 14:56:41 +03:00
}
2018-05-12 02:57:57 +03:00
static void cache_of_set_props ( struct cacheinfo * this_leaf ,
struct device_node * np )
2016-10-28 11:45:31 +03:00
{
2018-05-12 02:57:57 +03:00
/*
* init_cache_level must setup the cache level correctly
* overriding the architecturally specified levels , so
* if type is NONE at this stage , it should be unified
*/
if ( this_leaf - > type = = CACHE_TYPE_NOCACHE & &
cache_node_is_unified ( this_leaf , np ) )
this_leaf - > type = CACHE_TYPE_UNIFIED ;
cache_size ( this_leaf , np ) ;
cache_get_line_size ( this_leaf , np ) ;
cache_nr_sets ( this_leaf , np ) ;
cache_associativity ( this_leaf ) ;
2016-10-28 11:45:31 +03:00
}
2018-05-12 02:57:56 +03:00
static int cache_setup_of_node ( unsigned int cpu )
{
struct device_node * np ;
struct cacheinfo * this_leaf ;
unsigned int index = 0 ;
2022-07-04 13:15:46 +03:00
np = of_cpu_device_node_get ( cpu ) ;
2018-05-12 02:57:56 +03:00
if ( ! np ) {
pr_err ( " Failed to find cpu%d device node \n " , cpu ) ;
return - ENOENT ;
}
while ( index < cache_leaves ( cpu ) ) {
2022-07-04 13:15:47 +03:00
this_leaf = per_cpu_cacheinfo_idx ( cpu , index ) ;
2018-05-12 02:57:56 +03:00
if ( this_leaf - > level ! = 1 )
np = of_find_next_cache_node ( np ) ;
else
np = of_node_get ( np ) ; /* cpu node itself */
if ( ! np )
break ;
2018-05-12 02:57:57 +03:00
cache_of_set_props ( this_leaf , np ) ;
2018-05-12 02:57:58 +03:00
this_leaf - > fw_token = np ;
2018-05-12 02:57:56 +03:00
index + + ;
}
if ( index ! = cache_leaves ( cpu ) ) /* not all OF nodes populated */
return - ENOENT ;
return 0 ;
}
2014-09-30 17:48:25 +04:00
# else
static inline int cache_setup_of_node ( unsigned int cpu ) { return 0 ; }
# endif
2018-05-12 02:58:02 +03:00
int __weak cache_setup_acpi ( unsigned int cpu )
{
return - ENOTSUPP ;
}
2019-05-28 05:16:53 +03:00
unsigned int coherency_max_size ;
2022-07-04 13:15:50 +03:00
static int cache_setup_properties ( unsigned int cpu )
{
int ret = 0 ;
if ( of_have_populated_dt ( ) )
ret = cache_setup_of_node ( cpu ) ;
else if ( ! acpi_disabled )
ret = cache_setup_acpi ( cpu ) ;
return ret ;
}
2014-09-30 17:48:25 +04:00
static int cache_shared_cpu_map_setup ( unsigned int cpu )
{
struct cpu_cacheinfo * this_cpu_ci = get_cpu_cacheinfo ( cpu ) ;
struct cacheinfo * this_leaf , * sib_leaf ;
unsigned int index ;
2016-10-28 11:45:29 +03:00
int ret = 0 ;
2014-09-30 17:48:25 +04:00
2016-10-28 11:45:28 +03:00
if ( this_cpu_ci - > cpu_map_populated )
return 0 ;
2022-07-04 13:15:50 +03:00
/*
* skip setting up cache properties if LLC is valid , just need
* to update the shared cpu_map if the cache attributes were
* populated early before all the cpus are brought online
*/
if ( ! last_level_cache_is_valid ( cpu ) ) {
ret = cache_setup_properties ( cpu ) ;
if ( ret )
return ret ;
}
2014-09-30 17:48:25 +04:00
for ( index = 0 ; index < cache_leaves ( cpu ) ; index + + ) {
unsigned int i ;
2022-07-04 13:15:47 +03:00
this_leaf = per_cpu_cacheinfo_idx ( cpu , index ) ;
2014-09-30 17:48:25 +04:00
cpumask_set_cpu ( cpu , & this_leaf - > shared_cpu_map ) ;
for_each_online_cpu ( i ) {
struct cpu_cacheinfo * sib_cpu_ci = get_cpu_cacheinfo ( i ) ;
if ( i = = cpu | | ! sib_cpu_ci - > info_list )
continue ; /* skip if itself or no cacheinfo */
2022-07-04 13:15:52 +03:00
2022-07-04 13:15:47 +03:00
sib_leaf = per_cpu_cacheinfo_idx ( i , index ) ;
2014-09-30 17:48:25 +04:00
if ( cache_leaves_are_shared ( this_leaf , sib_leaf ) ) {
cpumask_set_cpu ( cpu , & sib_leaf - > shared_cpu_map ) ;
cpumask_set_cpu ( i , & this_leaf - > shared_cpu_map ) ;
}
}
2019-05-28 05:16:53 +03:00
/* record the maximum cache line size */
if ( this_leaf - > coherency_line_size > coherency_max_size )
coherency_max_size = this_leaf - > coherency_line_size ;
2014-09-30 17:48:25 +04:00
}
return 0 ;
}
static void cache_shared_cpu_map_remove ( unsigned int cpu )
{
struct cacheinfo * this_leaf , * sib_leaf ;
unsigned int sibling , index ;
for ( index = 0 ; index < cache_leaves ( cpu ) ; index + + ) {
2022-07-04 13:15:47 +03:00
this_leaf = per_cpu_cacheinfo_idx ( cpu , index ) ;
2014-09-30 17:48:25 +04:00
for_each_cpu ( sibling , & this_leaf - > shared_cpu_map ) {
2022-07-04 13:15:52 +03:00
struct cpu_cacheinfo * sib_cpu_ci =
get_cpu_cacheinfo ( sibling ) ;
2015-08-08 11:46:02 +03:00
2022-07-04 13:15:52 +03:00
if ( sibling = = cpu | | ! sib_cpu_ci - > info_list )
continue ; /* skip if itself or no cacheinfo */
2015-08-08 11:46:02 +03:00
2022-07-04 13:15:47 +03:00
sib_leaf = per_cpu_cacheinfo_idx ( sibling , index ) ;
2014-09-30 17:48:25 +04:00
cpumask_clear_cpu ( cpu , & sib_leaf - > shared_cpu_map ) ;
cpumask_clear_cpu ( sibling , & this_leaf - > shared_cpu_map ) ;
}
2018-05-12 02:58:02 +03:00
if ( of_have_populated_dt ( ) )
of_node_put ( this_leaf - > fw_token ) ;
2014-09-30 17:48:25 +04:00
}
}
static void free_cache_attributes ( unsigned int cpu )
{
2015-08-08 11:46:02 +03:00
if ( ! per_cpu_cacheinfo ( cpu ) )
return ;
2014-09-30 17:48:25 +04:00
cache_shared_cpu_map_remove ( cpu ) ;
kfree ( per_cpu_cacheinfo ( cpu ) ) ;
per_cpu_cacheinfo ( cpu ) = NULL ;
2021-07-14 04:32:55 +03:00
cache_leaves ( cpu ) = 0 ;
2014-09-30 17:48:25 +04:00
}
int __weak init_cache_level ( unsigned int cpu )
{
return - ENOENT ;
}
int __weak populate_cache_leaves ( unsigned int cpu )
{
return - ENOENT ;
}
2022-07-04 13:15:50 +03:00
int detect_cache_attributes ( unsigned int cpu )
2014-09-30 17:48:25 +04:00
{
int ret ;
2022-07-04 13:15:50 +03:00
/* Since early detection of the cacheinfo is allowed via this
* function and this also gets called as CPU hotplug callbacks via
* cacheinfo_cpu_online , the initialisation can be skipped and only
* CPU maps can be updated as the CPU online status would be update
* if called via cacheinfo_cpu_online path .
*/
if ( per_cpu_cacheinfo ( cpu ) )
goto update_cpu_map ;
2015-05-27 13:26:13 +03:00
if ( init_cache_level ( cpu ) | | ! cache_leaves ( cpu ) )
2014-09-30 17:48:25 +04:00
return - ENOENT ;
per_cpu_cacheinfo ( cpu ) = kcalloc ( cache_leaves ( cpu ) ,
2022-07-20 15:55:38 +03:00
sizeof ( struct cacheinfo ) , GFP_ATOMIC ) ;
2022-07-04 13:15:50 +03:00
if ( per_cpu_cacheinfo ( cpu ) = = NULL ) {
cache_leaves ( cpu ) = 0 ;
2014-09-30 17:48:25 +04:00
return - ENOMEM ;
2022-07-04 13:15:50 +03:00
}
2014-09-30 17:48:25 +04:00
2018-05-12 02:57:57 +03:00
/*
* populate_cache_leaves ( ) may completely setup the cache leaves and
* shared_cpu_map or it may leave it partially setup .
*/
2014-09-30 17:48:25 +04:00
ret = populate_cache_leaves ( cpu ) ;
if ( ret )
goto free_ci ;
2022-07-04 13:15:50 +03:00
update_cpu_map :
2014-09-30 17:48:25 +04:00
/*
2018-05-12 02:57:58 +03:00
* For systems using DT for cache hierarchy , fw_token
* and shared_cpu_map will be set up here only if they are
* not populated already
2014-09-30 17:48:25 +04:00
*/
ret = cache_shared_cpu_map_setup ( cpu ) ;
2015-03-17 20:28:46 +03:00
if ( ret ) {
2016-10-28 11:45:29 +03:00
pr_warn ( " Unable to detect cache hierarchy for CPU %d \n " , cpu ) ;
2014-09-30 17:48:25 +04:00
goto free_ci ;
2015-03-17 20:28:46 +03:00
}
2016-10-28 11:45:31 +03:00
2014-09-30 17:48:25 +04:00
return 0 ;
free_ci :
free_cache_attributes ( cpu ) ;
return ret ;
}
/* pointer to cpuX/cache device */
static DEFINE_PER_CPU ( struct device * , ci_cache_dev ) ;
# define per_cpu_cache_dev(cpu) (per_cpu(ci_cache_dev, cpu))
static cpumask_t cache_dev_map ;
/* pointer to array of devices for cpuX/cache/indexY */
static DEFINE_PER_CPU ( struct device * * , ci_index_dev ) ;
# define per_cpu_index_dev(cpu) (per_cpu(ci_index_dev, cpu))
# define per_cache_index_dev(cpu, idx) ((per_cpu_index_dev(cpu))[idx])
# define show_one(file_name, object) \
static ssize_t file_name # # _show ( struct device * dev , \
struct device_attribute * attr , char * buf ) \
{ \
struct cacheinfo * this_leaf = dev_get_drvdata ( dev ) ; \
2020-09-16 23:40:42 +03:00
return sysfs_emit ( buf , " %u \n " , this_leaf - > object ) ; \
2014-09-30 17:48:25 +04:00
}
2016-10-22 16:19:49 +03:00
show_one ( id , id ) ;
2014-09-30 17:48:25 +04:00
show_one ( level , level ) ;
show_one ( coherency_line_size , coherency_line_size ) ;
show_one ( number_of_sets , number_of_sets ) ;
show_one ( physical_line_partition , physical_line_partition ) ;
show_one ( ways_of_associativity , ways_of_associativity ) ;
static ssize_t size_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct cacheinfo * this_leaf = dev_get_drvdata ( dev ) ;
drivers core: Use sysfs_emit and sysfs_emit_at for show(device *...) functions
Convert the various sprintf fmaily calls in sysfs device show functions
to sysfs_emit and sysfs_emit_at for PAGE_SIZE buffer safety.
Done with:
$ spatch -sp-file sysfs_emit_dev.cocci --in-place --max-width=80 .
And cocci script:
$ cat sysfs_emit_dev.cocci
@@
identifier d_show;
identifier dev, attr, buf;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- sprintf(buf,
+ sysfs_emit(buf,
...);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- snprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- scnprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
expression chr;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- strcpy(buf, chr);
+ sysfs_emit(buf, chr);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
len =
- sprintf(buf,
+ sysfs_emit(buf,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
len =
- snprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
len =
- scnprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
- len += scnprintf(buf + len, PAGE_SIZE - len,
+ len += sysfs_emit_at(buf, len,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
expression chr;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
...
- strcpy(buf, chr);
- return strlen(buf);
+ return sysfs_emit(buf, chr);
}
Signed-off-by: Joe Perches <joe@perches.com>
Link: https://lore.kernel.org/r/3d033c33056d88bbe34d4ddb62afd05ee166ab9a.1600285923.git.joe@perches.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2020-09-16 23:40:39 +03:00
return sysfs_emit ( buf , " %uK \n " , this_leaf - > size > > 10 ) ;
2014-09-30 17:48:25 +04:00
}
2020-09-16 23:40:44 +03:00
static ssize_t shared_cpu_map_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
2014-09-30 17:48:25 +04:00
{
struct cacheinfo * this_leaf = dev_get_drvdata ( dev ) ;
const struct cpumask * mask = & this_leaf - > shared_cpu_map ;
2020-09-16 23:40:44 +03:00
return sysfs_emit ( buf , " %*pb \n " , nr_cpu_ids , mask ) ;
2014-09-30 17:48:25 +04:00
}
static ssize_t shared_cpu_list_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
2020-09-16 23:40:44 +03:00
struct cacheinfo * this_leaf = dev_get_drvdata ( dev ) ;
const struct cpumask * mask = & this_leaf - > shared_cpu_map ;
return sysfs_emit ( buf , " %*pbl \n " , nr_cpu_ids , mask ) ;
2014-09-30 17:48:25 +04:00
}
static ssize_t type_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct cacheinfo * this_leaf = dev_get_drvdata ( dev ) ;
2020-09-16 23:40:40 +03:00
const char * output ;
2014-09-30 17:48:25 +04:00
switch ( this_leaf - > type ) {
case CACHE_TYPE_DATA :
2020-09-16 23:40:40 +03:00
output = " Data " ;
break ;
2014-09-30 17:48:25 +04:00
case CACHE_TYPE_INST :
2020-09-16 23:40:40 +03:00
output = " Instruction " ;
break ;
2014-09-30 17:48:25 +04:00
case CACHE_TYPE_UNIFIED :
2020-09-16 23:40:40 +03:00
output = " Unified " ;
break ;
2014-09-30 17:48:25 +04:00
default :
return - EINVAL ;
}
2020-09-16 23:40:40 +03:00
return sysfs_emit ( buf , " %s \n " , output ) ;
2014-09-30 17:48:25 +04:00
}
static ssize_t allocation_policy_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct cacheinfo * this_leaf = dev_get_drvdata ( dev ) ;
unsigned int ci_attr = this_leaf - > attributes ;
2020-09-16 23:40:40 +03:00
const char * output ;
2014-09-30 17:48:25 +04:00
if ( ( ci_attr & CACHE_READ_ALLOCATE ) & & ( ci_attr & CACHE_WRITE_ALLOCATE ) )
2020-09-16 23:40:40 +03:00
output = " ReadWriteAllocate " ;
2014-09-30 17:48:25 +04:00
else if ( ci_attr & CACHE_READ_ALLOCATE )
2020-09-16 23:40:40 +03:00
output = " ReadAllocate " ;
2014-09-30 17:48:25 +04:00
else if ( ci_attr & CACHE_WRITE_ALLOCATE )
2020-09-16 23:40:40 +03:00
output = " WriteAllocate " ;
else
return 0 ;
return sysfs_emit ( buf , " %s \n " , output ) ;
2014-09-30 17:48:25 +04:00
}
static ssize_t write_policy_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct cacheinfo * this_leaf = dev_get_drvdata ( dev ) ;
unsigned int ci_attr = this_leaf - > attributes ;
int n = 0 ;
if ( ci_attr & CACHE_WRITE_THROUGH )
drivers core: Use sysfs_emit and sysfs_emit_at for show(device *...) functions
Convert the various sprintf fmaily calls in sysfs device show functions
to sysfs_emit and sysfs_emit_at for PAGE_SIZE buffer safety.
Done with:
$ spatch -sp-file sysfs_emit_dev.cocci --in-place --max-width=80 .
And cocci script:
$ cat sysfs_emit_dev.cocci
@@
identifier d_show;
identifier dev, attr, buf;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- sprintf(buf,
+ sysfs_emit(buf,
...);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- snprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- scnprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
expression chr;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- strcpy(buf, chr);
+ sysfs_emit(buf, chr);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
len =
- sprintf(buf,
+ sysfs_emit(buf,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
len =
- snprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
len =
- scnprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
- len += scnprintf(buf + len, PAGE_SIZE - len,
+ len += sysfs_emit_at(buf, len,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
expression chr;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
...
- strcpy(buf, chr);
- return strlen(buf);
+ return sysfs_emit(buf, chr);
}
Signed-off-by: Joe Perches <joe@perches.com>
Link: https://lore.kernel.org/r/3d033c33056d88bbe34d4ddb62afd05ee166ab9a.1600285923.git.joe@perches.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2020-09-16 23:40:39 +03:00
n = sysfs_emit ( buf , " WriteThrough \n " ) ;
2014-09-30 17:48:25 +04:00
else if ( ci_attr & CACHE_WRITE_BACK )
drivers core: Use sysfs_emit and sysfs_emit_at for show(device *...) functions
Convert the various sprintf fmaily calls in sysfs device show functions
to sysfs_emit and sysfs_emit_at for PAGE_SIZE buffer safety.
Done with:
$ spatch -sp-file sysfs_emit_dev.cocci --in-place --max-width=80 .
And cocci script:
$ cat sysfs_emit_dev.cocci
@@
identifier d_show;
identifier dev, attr, buf;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- sprintf(buf,
+ sysfs_emit(buf,
...);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- snprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- scnprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
expression chr;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
return
- strcpy(buf, chr);
+ sysfs_emit(buf, chr);
...>
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
len =
- sprintf(buf,
+ sysfs_emit(buf,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
len =
- snprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
len =
- scnprintf(buf, PAGE_SIZE,
+ sysfs_emit(buf,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
identifier len;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
<...
- len += scnprintf(buf + len, PAGE_SIZE - len,
+ len += sysfs_emit_at(buf, len,
...);
...>
return len;
}
@@
identifier d_show;
identifier dev, attr, buf;
expression chr;
@@
ssize_t d_show(struct device *dev, struct device_attribute *attr, char *buf)
{
...
- strcpy(buf, chr);
- return strlen(buf);
+ return sysfs_emit(buf, chr);
}
Signed-off-by: Joe Perches <joe@perches.com>
Link: https://lore.kernel.org/r/3d033c33056d88bbe34d4ddb62afd05ee166ab9a.1600285923.git.joe@perches.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2020-09-16 23:40:39 +03:00
n = sysfs_emit ( buf , " WriteBack \n " ) ;
2014-09-30 17:48:25 +04:00
return n ;
}
2016-10-22 16:19:49 +03:00
static DEVICE_ATTR_RO ( id ) ;
2014-09-30 17:48:25 +04:00
static DEVICE_ATTR_RO ( level ) ;
static DEVICE_ATTR_RO ( type ) ;
static DEVICE_ATTR_RO ( coherency_line_size ) ;
static DEVICE_ATTR_RO ( ways_of_associativity ) ;
static DEVICE_ATTR_RO ( number_of_sets ) ;
static DEVICE_ATTR_RO ( size ) ;
static DEVICE_ATTR_RO ( allocation_policy ) ;
static DEVICE_ATTR_RO ( write_policy ) ;
static DEVICE_ATTR_RO ( shared_cpu_map ) ;
static DEVICE_ATTR_RO ( shared_cpu_list ) ;
static DEVICE_ATTR_RO ( physical_line_partition ) ;
static struct attribute * cache_default_attrs [ ] = {
2016-10-22 16:19:49 +03:00
& dev_attr_id . attr ,
2014-09-30 17:48:25 +04:00
& dev_attr_type . attr ,
& dev_attr_level . attr ,
& dev_attr_shared_cpu_map . attr ,
& dev_attr_shared_cpu_list . attr ,
& dev_attr_coherency_line_size . attr ,
& dev_attr_ways_of_associativity . attr ,
& dev_attr_number_of_sets . attr ,
& dev_attr_size . attr ,
& dev_attr_allocation_policy . attr ,
& dev_attr_write_policy . attr ,
& dev_attr_physical_line_partition . attr ,
NULL
} ;
static umode_t
cache_default_attrs_is_visible ( struct kobject * kobj ,
struct attribute * attr , int unused )
{
struct device * dev = kobj_to_dev ( kobj ) ;
struct cacheinfo * this_leaf = dev_get_drvdata ( dev ) ;
const struct cpumask * mask = & this_leaf - > shared_cpu_map ;
umode_t mode = attr - > mode ;
2016-10-22 16:19:49 +03:00
if ( ( attr = = & dev_attr_id . attr ) & & ( this_leaf - > attributes & CACHE_ID ) )
return mode ;
2014-09-30 17:48:25 +04:00
if ( ( attr = = & dev_attr_type . attr ) & & this_leaf - > type )
return mode ;
if ( ( attr = = & dev_attr_level . attr ) & & this_leaf - > level )
return mode ;
if ( ( attr = = & dev_attr_shared_cpu_map . attr ) & & ! cpumask_empty ( mask ) )
return mode ;
if ( ( attr = = & dev_attr_shared_cpu_list . attr ) & & ! cpumask_empty ( mask ) )
return mode ;
if ( ( attr = = & dev_attr_coherency_line_size . attr ) & &
this_leaf - > coherency_line_size )
return mode ;
if ( ( attr = = & dev_attr_ways_of_associativity . attr ) & &
this_leaf - > size ) /* allow 0 = full associativity */
return mode ;
if ( ( attr = = & dev_attr_number_of_sets . attr ) & &
this_leaf - > number_of_sets )
return mode ;
if ( ( attr = = & dev_attr_size . attr ) & & this_leaf - > size )
return mode ;
if ( ( attr = = & dev_attr_write_policy . attr ) & &
( this_leaf - > attributes & CACHE_WRITE_POLICY_MASK ) )
return mode ;
if ( ( attr = = & dev_attr_allocation_policy . attr ) & &
( this_leaf - > attributes & CACHE_ALLOCATE_POLICY_MASK ) )
return mode ;
if ( ( attr = = & dev_attr_physical_line_partition . attr ) & &
this_leaf - > physical_line_partition )
return mode ;
return 0 ;
}
static const struct attribute_group cache_default_group = {
. attrs = cache_default_attrs ,
. is_visible = cache_default_attrs_is_visible ,
} ;
static const struct attribute_group * cache_default_groups [ ] = {
& cache_default_group ,
NULL ,
} ;
static const struct attribute_group * cache_private_groups [ ] = {
& cache_default_group ,
NULL , /* Place holder for private group */
NULL ,
} ;
const struct attribute_group *
__weak cache_get_priv_group ( struct cacheinfo * this_leaf )
{
return NULL ;
}
static const struct attribute_group * *
cache_get_attribute_groups ( struct cacheinfo * this_leaf )
{
const struct attribute_group * priv_group =
cache_get_priv_group ( this_leaf ) ;
if ( ! priv_group )
return cache_default_groups ;
if ( ! cache_private_groups [ 1 ] )
cache_private_groups [ 1 ] = priv_group ;
return cache_private_groups ;
}
/* Add/Remove cache interface for CPU device */
static void cpu_cache_sysfs_exit ( unsigned int cpu )
{
int i ;
struct device * ci_dev ;
if ( per_cpu_index_dev ( cpu ) ) {
for ( i = 0 ; i < cache_leaves ( cpu ) ; i + + ) {
ci_dev = per_cache_index_dev ( cpu , i ) ;
if ( ! ci_dev )
continue ;
device_unregister ( ci_dev ) ;
}
kfree ( per_cpu_index_dev ( cpu ) ) ;
per_cpu_index_dev ( cpu ) = NULL ;
}
device_unregister ( per_cpu_cache_dev ( cpu ) ) ;
per_cpu_cache_dev ( cpu ) = NULL ;
}
static int cpu_cache_sysfs_init ( unsigned int cpu )
{
struct device * dev = get_cpu_device ( cpu ) ;
if ( per_cpu_cacheinfo ( cpu ) = = NULL )
return - ENOENT ;
per_cpu_cache_dev ( cpu ) = cpu_device_create ( dev , NULL , NULL , " cache " ) ;
if ( IS_ERR ( per_cpu_cache_dev ( cpu ) ) )
return PTR_ERR ( per_cpu_cache_dev ( cpu ) ) ;
/* Allocate all required memory */
per_cpu_index_dev ( cpu ) = kcalloc ( cache_leaves ( cpu ) ,
sizeof ( struct device * ) , GFP_KERNEL ) ;
if ( unlikely ( per_cpu_index_dev ( cpu ) = = NULL ) )
goto err_out ;
return 0 ;
err_out :
cpu_cache_sysfs_exit ( cpu ) ;
return - ENOMEM ;
}
static int cache_add_dev ( unsigned int cpu )
{
unsigned int i ;
int rc ;
struct device * ci_dev , * parent ;
struct cacheinfo * this_leaf ;
const struct attribute_group * * cache_groups ;
rc = cpu_cache_sysfs_init ( cpu ) ;
if ( unlikely ( rc < 0 ) )
return rc ;
parent = per_cpu_cache_dev ( cpu ) ;
for ( i = 0 ; i < cache_leaves ( cpu ) ; i + + ) {
2022-07-04 13:15:47 +03:00
this_leaf = per_cpu_cacheinfo_idx ( cpu , i ) ;
2014-09-30 17:48:25 +04:00
if ( this_leaf - > disable_sysfs )
continue ;
2018-10-04 18:20:05 +03:00
if ( this_leaf - > type = = CACHE_TYPE_NOCACHE )
break ;
2014-09-30 17:48:25 +04:00
cache_groups = cache_get_attribute_groups ( this_leaf ) ;
ci_dev = cpu_device_create ( parent , this_leaf , cache_groups ,
" index%1u " , i ) ;
if ( IS_ERR ( ci_dev ) ) {
rc = PTR_ERR ( ci_dev ) ;
goto err ;
}
per_cache_index_dev ( cpu , i ) = ci_dev ;
}
cpumask_set_cpu ( cpu , & cache_dev_map ) ;
return 0 ;
err :
cpu_cache_sysfs_exit ( cpu ) ;
return rc ;
}
2016-11-03 17:50:08 +03:00
static int cacheinfo_cpu_online ( unsigned int cpu )
2014-09-30 17:48:25 +04:00
{
2016-11-03 17:50:08 +03:00
int rc = detect_cache_attributes ( cpu ) ;
2014-09-30 17:48:25 +04:00
2016-11-03 17:50:08 +03:00
if ( rc )
return rc ;
rc = cache_add_dev ( cpu ) ;
if ( rc )
free_cache_attributes ( cpu ) ;
return rc ;
2014-09-30 17:48:25 +04:00
}
2016-11-03 17:50:08 +03:00
static int cacheinfo_cpu_pre_down ( unsigned int cpu )
2014-09-30 17:48:25 +04:00
{
2016-11-03 17:50:08 +03:00
if ( cpumask_test_and_clear_cpu ( cpu , & cache_dev_map ) )
cpu_cache_sysfs_exit ( cpu ) ;
free_cache_attributes ( cpu ) ;
return 0 ;
2014-09-30 17:48:25 +04:00
}
static int __init cacheinfo_sysfs_init ( void )
{
2019-06-24 20:36:56 +03:00
return cpuhp_setup_state ( CPUHP_AP_BASE_CACHEINFO_ONLINE ,
" base/cacheinfo:online " ,
2016-11-03 17:50:08 +03:00
cacheinfo_cpu_online , cacheinfo_cpu_pre_down ) ;
2014-09-30 17:48:25 +04:00
}
device_initcall ( cacheinfo_sysfs_init ) ;