Files
mckernel/executer/kernel/mcctrl/sysfs_files.c
Balazs Gerofi 5f5b9f94d1 Revert "get_one_cpu_topology: Renumber core_id (physical core id)"
This reverts commit 0a4e6b49b4.

Change-Id: Icd9f2cda63d0daf661a40b146c72608b82cf2061
2020-06-16 13:26:55 +09:00

1150 lines
28 KiB
C

// sysfs_files.c COPYRIGHT FUJITSU LIMITED 2016-2018
/**
* \file sysfs_files.c
* License details are found in the file LICENSE.
* \brief
* implement McKernel's sysfs files, IHK-Master side
* \author Gou Nakamura <go.nakamura.yw@hitachi-solutions.com> \par
* Copyright (C) 2016 RIKEN AICS
*/
/*
* HISTORY:
*/
#include <linux/kernel.h>
#include <linux/slab.h>
#include <linux/uaccess.h>
#include <linux/version.h>
#include "config.h"
#include "mcctrl.h"
#include "sysfs_msg.h"
#define dprintk(...) do { if (0) printk(__VA_ARGS__); } while (0)
#define wprintk(...) do { if (1) printk(KERN_WARNING __VA_ARGS__); } while (0)
#define eprintk(...) do { if (1) printk(KERN_ERR __VA_ARGS__); } while (0)
static ssize_t
show_int(struct sysfsm_ops *ops, void *instance, void *buf, size_t size)
{
int *p = instance;
return snprintf(buf, size, "%d\n", *p);
} /* show_int() */
struct sysfsm_ops show_int_ops = {
.show = &show_int,
};
void setup_local_snooping_samples(ihk_os_t os)
{
static long lvalue = 0xf123456789abcde0;
static char *svalue = "string(local)";
int error;
struct sysfsm_bitmap_param param;
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_d32, &lvalue, 0444, "/sys/test/local/d32");
if (error) {
panic("setup_local_snooping_samples: d32");
}
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_d64, &lvalue, 0444, "/sys/test/local/d64");
if (error) {
panic("setup_local_snooping_samples: d64");
}
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_u32, &lvalue, 0444, "/sys/test/local/u32");
if (error) {
panic("setup_local_snooping_samples: u32");
}
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_u64, &lvalue, 0444, "/sys/test/local/u64");
if (error) {
panic("setup_local_snooping_samples: u64");
}
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_s, svalue, 0444, "/sys/test/local/s");
if (error) {
panic("setup_local_snooping_samples: s");
}
param.nbits = 40;
param.ptr = &lvalue;
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_pbl, &param, 0444, "/sys/test/local/pbl");
if (error) {
panic("setup_local_snooping_samples: pbl");
}
param.nbits = 40;
param.ptr = &lvalue;
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_pb, &param, 0444, "/sys/test/local/pb");
if (error) {
panic("setup_local_snooping_samples: pb");
}
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_u32K, &lvalue, 0444, "/sys/test/local/u32K");
if (error) {
panic("setup_local_snooping_samples: u32K");
}
return;
}
void setup_local_snooping_files(ihk_os_t os)
{
struct mcctrl_usrdata *udp = ihk_host_os_get_usrdata(os);
struct sysfsm_bitmap_param param;
static unsigned long cpu_offline = 0x0;
int i;
int error;
if (!udp) {
panic("%s: error: mcctrl_usrdata not found\n",
__func__);
}
memset(udp->cpu_online, 0, sizeof(udp->cpu_online));
for (i = 0; i < udp->cpu_info->n_cpus; i++) {
set_bit(i, udp->cpu_online);
}
param.nbits = CPU_LONGS * BITS_PER_LONG;
param.ptr = &udp->cpu_online;
dprintk("mcctrl:setup_local_snooping_files: CPU_LONGS=%d, BITS_PER_LONG=%d\n",
CPU_LONGS, BITS_PER_LONG);
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"/sys/devices/system/cpu/online");
if (error) {
panic("setup_local_snooping_files: devices/system/cpu/online");
}
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"/sys/devices/system/cpu/possible");
if (error) {
panic("setup_local_snooping_files: devices/system/cpu/possible");
}
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"/sys/devices/system/cpu/present");
if (error) {
panic("setup_local_snooping_files: devices/system/cpu/present");
}
param.nbits = BITS_PER_LONG;
param.ptr = &cpu_offline;
error = sysfsm_createf(os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"/sys/devices/system/cpu/offline");
if (error) {
panic("setup_local_snooping_files: devices/system/cpu/offline");
}
return;
}
static void free_node_topology(struct mcctrl_usrdata *udp)
{
struct node_topology *node;
struct node_topology *next;
list_for_each_entry_safe(node, next, &udp->node_topology_list, chain) {
list_del(&node->chain);
kfree(node);
}
return;
} /* free_node_topology() */
static void free_cpu_topology_one(struct mcctrl_usrdata *udp,
struct mcctrl_cpu_topology *cpu)
{
struct cache_topology *cache;
struct cache_topology *next;
list_for_each_entry_safe(cache, next, &cpu->cache_list, chain) {
list_del(&cache->chain);
kfree(cache);
}
kfree(cpu);
return;
} /* free_cpu_topology_one() */
static void free_cpu_topology(struct mcctrl_usrdata *udp)
{
struct mcctrl_cpu_topology *cpu;
struct mcctrl_cpu_topology *next;
list_for_each_entry_safe(cpu, next, &udp->cpu_topology_list, chain) {
list_del(&cpu->chain);
free_cpu_topology_one(udp, cpu);
}
return;
} /* free_cpu_topology() */
void free_topology_info(ihk_os_t os)
{
struct mcctrl_usrdata *udp = ihk_host_os_get_usrdata(os);
if (!udp) {
pr_warn("%s: warning: mcctrl_usrdata not found\n", __func__);
return;
}
free_node_topology(udp);
free_cpu_topology(udp);
return;
} /* free_topology_info() */
/*
* CPU and NUMA node mapping conversion functions.
*/
int mckernel_cpu_2_linux_cpu(struct mcctrl_usrdata *udp, int cpu_id)
{
return (cpu_id < udp->cpu_info->n_cpus) ?
udp->cpu_info->mapping[cpu_id] : -1;
}
int mckernel_cpu_2_hw_id(struct mcctrl_usrdata *udp, int cpu_id)
{
return (cpu_id < udp->cpu_info->n_cpus) ?
udp->cpu_info->hw_ids[cpu_id] : -1;
}
int linux_cpu_2_mckernel_cpu(struct mcctrl_usrdata *udp, int cpu_id)
{
int i;
for (i = 0; i < udp->cpu_info->n_cpus; ++i) {
if (udp->cpu_info->mapping[i] == cpu_id)
return i;
}
return -1;
}
#if 0
int hw_id_2_mckernel_cpu(struct mcctrl_usrdata *udp, int hw_id)
{
int i;
for (i = 0; i < udp->cpu_info->n_cpus; ++i) {
if (udp->cpu_info->hw_ids[i] == hw_id) {
return i;
}
}
return -1;
}
int hw_id_2_linux_cpu(struct mcctrl_usrdata *udp, int hw_id)
{
int i;
for (i = 0; i < udp->cpu_info->n_cpus; ++i) {
if (udp->cpu_info->hw_ids[i] == hw_id) {
return mckernel_cpu_2_linux_cpu(udp, i);
}
}
return -1;
}
int linux_cpu_2_hw_id(struct mcctrl_usrdata *udp, int cpu)
{
int mckernel_cpu = linux_cpu_2_mckernel_cpu(udp, cpu);
return (mckernel_cpu >= 0 && mckernel_cpu < udp->cpu_info->n_cpus) ?
udp->cpu_info->hw_ids[mckernel_cpu] : -1;
}
#endif
int mckernel_numa_2_linux_numa(struct mcctrl_usrdata *udp, int numa_id)
{
return (numa_id < udp->mem_info->n_numa_nodes) ?
udp->mem_info->numa_mapping[numa_id] : -1;
}
int linux_numa_2_mckernel_numa(struct mcctrl_usrdata *udp, int numa_id)
{
int i;
for (i = 0; i < udp->mem_info->n_numa_nodes; ++i) {
if (udp->mem_info->numa_mapping[i] == numa_id)
return i;
}
return -1;
}
static int translate_cpumap(struct mcctrl_usrdata *udp,
cpumask_t *linmap, cpumask_t *mckmap)
{
int error;
int lincpu;
int mckcpu;
dprintk("translate_cpumap(%p,%p,%p)\n", udp, linmap, mckmap);
cpumask_clear(mckmap);
for_each_cpu(lincpu, linmap) {
mckcpu = linux_cpu_2_mckernel_cpu(udp, lincpu);
if (mckcpu >= 0) {
cpumask_set_cpu(mckcpu, mckmap);
}
}
error = 0;
dprintk("translate_cpumap(%p,%p,%p): %d\n", udp, linmap, mckmap, error);
return error;
} /* translate_cpumap() */
static struct cache_topology *get_cache_topology(struct mcctrl_usrdata *udp,
struct mcctrl_cpu_topology *cpu_topo, struct ihk_cache_topology *saved)
{
int error;
struct cache_topology *topo = NULL;
dprintk("get_cache_topology(%p,%p)\n", cpu_topo, saved);
topo = kmalloc(sizeof(*topo), GFP_KERNEL);
if (!topo) {
error = -ENOMEM;
eprintk("mcctrl:get_cache_topology:"
"kmalloc failed. %d\n", error);
goto out;
}
topo->saved = saved;
error = translate_cpumap(udp, &topo->saved->shared_cpu_map,
&topo->shared_cpu_map);
if (error) {
eprintk("mcctrl:get_cache_topology:"
"translate_cpumap failed. %d\n", error);
goto out;
}
error = 0;
out:
if (error && !IS_ERR_OR_NULL(topo)) {
kfree(topo);
}
dprintk("get_cache_topology(%p,%p): %d %p\n",
cpu_topo, saved, error, topo);
return (error)? ERR_PTR(error): topo;
} /* get_cache_topology() */
static struct mcctrl_cpu_topology *get_one_cpu_topology(struct mcctrl_usrdata *udp,
int index)
{
int error;
ihk_device_t dev = ihk_os_to_dev(udp->os);
struct mcctrl_cpu_topology *topology = NULL;
struct cache_topology *cache;
struct ihk_cache_topology *saved_cache;
dprintk("get_one_cpu_topology(%p,%d)\n", udp, index);
topology = kmalloc(sizeof(*topology), GFP_KERNEL);
if (!topology) {
error = -ENOMEM;
eprintk("mcctrl:get_one_cpu_topology:"
"kmalloc failed. %d\n", error);
goto out;
}
INIT_LIST_HEAD(&topology->cache_list);
topology->mckernel_cpu_id = index;
topology->saved = ihk_device_get_cpu_topology(dev,
mckernel_cpu_2_hw_id(udp, index));
if (!topology->saved) {
error = -ENOENT;
eprintk("mcctrl:get_one_cpu_topology:"
"ihk_device_get_cpu_topology failed. %d\n",
error);
goto out;
}
error = translate_cpumap(udp,
&topology->saved->core_siblings,
&topology->core_siblings);
if (error) {
eprintk("mcctrl:get_one_cpu_topology:"
"translate_cpumap(core_siblings) failed."
" %d\n", error);
goto out;
}
error = translate_cpumap(udp,
&topology->saved->thread_siblings,
&topology->thread_siblings);
if (error) {
eprintk("mcctrl:get_one_cpu_topology:"
"translate_cpumap(thread_siblings) failed."
" %d\n", error);
goto out;
}
list_for_each_entry(saved_cache,
&topology->saved->cache_topology_list, chain) {
cache = get_cache_topology(udp, topology, saved_cache);
if (IS_ERR(cache)) {
error = PTR_ERR(cache);
eprintk("mcctrl:get_one_cpu_topology:"
"get_cache_topology failed. %d\n",
error);
goto out;
} else if (!cache) {
error = -ENOENT;
goto out;
}
list_add(&cache->chain, &topology->cache_list);
}
error = 0;
out:
if (error && !IS_ERR_OR_NULL(topology)) {
free_cpu_topology_one(udp, topology);
}
dprintk("get_one_cpu_topology(%p,%d): %d %p\n",
udp, index, error, topology);
return (error)? ERR_PTR(error): topology;
} /* get_one_cpu_topology() */
static int get_cpu_topology(struct mcctrl_usrdata *udp)
{
int error;
int index;
struct mcctrl_cpu_topology *topology;
dprintk("get_cpu_topology(%p)\n", udp);
for (index = 0; index < udp->cpu_info->n_cpus; ++index) {
topology = get_one_cpu_topology(udp, index);
if (IS_ERR(topology)) {
error = PTR_ERR(topology);
eprintk("mcctrl:get_cpu_topology: "
"get_one_cpu_topology failed. %d\n",
error);
goto out;
}
list_add(&topology->chain, &udp->cpu_topology_list);
}
error = 0;
out:
dprintk("get_cpu_topology(%p): %d\n", udp, error);
return error;
} /* get_cpu_topology() */
static void setup_cpu_sysfs_cache_files(struct mcctrl_usrdata *udp,
struct mcctrl_cpu_topology *cpu, struct cache_topology *cache)
{
char *prefix = "/sys/devices/system/cpu";
int cpu_number = cpu->mckernel_cpu_id;
int index = cache->saved->index;
struct sysfsm_bitmap_param param;
dprintk("setup_cpu_sysfs_cache_files(%p,%p,%p)\n", udp, cpu, cache);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_d64,
&cache->saved->level, 0444,
"%s/cpu%d/cache/index%d/level",
prefix, cpu_number, index);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_s,
cache->saved->type, 0444,
"%s/cpu%d/cache/index%d/type",
prefix, cpu_number, index);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_s,
cache->saved->size_str, 0444,
"%s/cpu%d/cache/index%d/size",
prefix, cpu_number, index);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_d64,
&cache->saved->coherency_line_size, 0444,
"%s/cpu%d/cache/index%d/coherency_line_size",
prefix, cpu_number, index);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_d64,
&cache->saved->number_of_sets, 0444,
"%s/cpu%d/cache/index%d/number_of_sets",
prefix, cpu_number, index);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_d64,
&cache->saved->physical_line_partition, 0444,
"%s/cpu%d/cache/index%d/physical_line_partition",
prefix, cpu_number, index);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_d64,
&cache->saved->ways_of_associativity, 0444,
"%s/cpu%d/cache/index%d/ways_of_associativity",
prefix, cpu_number, index);
param.nbits = nr_cpu_ids;
param.ptr = &cache->shared_cpu_map;
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pb, &param, 0444,
"%s/cpu%d/cache/index%d/shared_cpu_map",
prefix, cpu_number, index);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"%s/cpu%d/cache/index%d/shared_cpu_list",
prefix, cpu_number, index);
dprintk("setup_cpu_sysfs_cache_files(%p,%p,%p):\n", udp, cpu, cache);
return;
} /* setup_cpu_sysfs_cache_files() */
static void setup_cpu_sysfs_files(struct mcctrl_usrdata *udp,
struct mcctrl_cpu_topology *cpu)
{
char *prefix = "/sys/devices/system/cpu";
int cpu_number = cpu->mckernel_cpu_id;
struct sysfsm_bitmap_param param;
struct cache_topology *cache;
dprintk("setup_cpu_sysfs_files(%p,%p)\n", udp, cpu);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_d32,
&cpu->saved->physical_package_id, 0444,
"%s/cpu%d/topology/physical_package_id",
prefix, cpu_number);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_d32,
&cpu->saved->core_id, 0444,
"%s/cpu%d/topology/core_id",
prefix, cpu_number);
param.nbits = nr_cpu_ids;
param.ptr = &cpu->core_siblings;
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pb, &param, 0444,
"%s/cpu%d/topology/core_siblings",
prefix, cpu_number);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"%s/cpu%d/topology/core_siblings_list",
prefix, cpu_number);
param.nbits = nr_cpu_ids;
param.ptr = &cpu->thread_siblings;
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pb, &param, 0444,
"%s/cpu%d/topology/thread_siblings",
prefix, cpu_number);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"%s/cpu%d/topology/thread_siblings_list",
prefix, cpu_number);
list_for_each_entry(cache, &cpu->cache_list, chain) {
setup_cpu_sysfs_cache_files(udp, cpu, cache);
}
dprintk("setup_cpu_sysfs_files(%p,%p):\n", udp, cpu);
return;
} /* setup_cpu_sysfs_files() */
static void setup_cpus_sysfs_files(struct mcctrl_usrdata *udp)
{
int error;
struct mcctrl_cpu_topology *cpu;
error = get_cpu_topology(udp);
if (error) {
eprintk("mcctrl:setup_cpus_sysfs_files:"
"get_cpu_topology failed. %d\n", error);
goto out;
}
list_for_each_entry(cpu, &udp->cpu_topology_list, chain) {
setup_cpu_sysfs_files(udp, cpu);
}
error = 0;
out:
dprintk("setup_cpu_file(%p):\n", udp);
return;
} /* setup_cpus_sysfs_files() */
static struct node_topology *get_one_node_topology(struct mcctrl_usrdata *udp,
struct ihk_node_topology *saved)
{
int error;
struct node_topology *node = NULL;
node = kmalloc(sizeof(*node), GFP_KERNEL);
if (!node) {
error = -ENOMEM;
eprintk("mcctrl:get_one_node_topology:"
"kmalloc failed. %d\n", error);
goto out;
}
node->saved = saved;
error = translate_cpumap(udp, &node->saved->cpumap, &node->cpumap);
if (error) {
eprintk("mcctrl:get_one_node_topology:"
"translate_cpumap failed. %d\n", error);
goto out;
}
error = 0;
out:
if (error && !IS_ERR_OR_NULL(node)) {
kfree(node);
}
return (error)? ERR_PTR(error): node;
} /* get_one_node_topology() */
static int get_node_topology(struct mcctrl_usrdata *udp)
{
int error;
ihk_device_t dev = ihk_os_to_dev(udp->os);
int node;
struct ihk_node_topology *saved;
struct node_topology *topology;
dprintk("get_node_topology(%p)\n", udp);
for (node = 0; node < udp->mem_info->n_numa_nodes; ++node) {
saved = ihk_device_get_node_topology(dev,
mckernel_numa_2_linux_numa(udp, node));
if (IS_ERR(saved)) {
break;
}
if (!saved) {
continue;
}
topology = get_one_node_topology(udp, saved);
if (IS_ERR(topology)) {
error = PTR_ERR(topology);
eprintk("mcctrl:get_node_topology:"
"get_one_node_topology failed. %d\n",
error);
goto out;
}
topology->mckernel_numa_id = node;
list_add(&topology->chain, &udp->node_topology_list);
}
error = 0;
out:
dprintk("get_node_topology(%p): %d\n", udp, error);
return error;
} /* get_node_topology() */
static int setup_node_files(struct mcctrl_usrdata *udp)
{
int error;
int node;
struct node_topology *p;
struct sysfsm_bitmap_param param;
dprintk("setup_node_files(%p)\n", udp);
error = get_node_topology(udp);
if (error) {
eprintk("mcctrl:setup_node_files:"
"get_node_topology failed. %d\n", error);
goto out;
}
memset(&udp->numa_online, 0, sizeof(udp->numa_online));
for (node = 0; node < udp->mem_info->n_numa_nodes; ++node) {
node_set(node, udp->numa_online);
}
param.nbits = MAX_NUMNODES;
param.ptr = &udp->numa_online;
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"/sys/devices/system/node/online");
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"/sys/devices/system/node/possible");
list_for_each_entry(p, &udp->node_topology_list, chain) {
struct sysfs_handle node_handle, cpu_handle;
int cpu;
size_t offset = 0;
param.nbits = nr_cpu_ids;
param.ptr = &p->cpumap;
for (node = 0; node < udp->mem_info->n_numa_nodes; ++node) {
if (node > 0) {
offset += snprintf(&p->mckernel_numa_distance_s[offset],
NODE_DISTANCE_S_SIZE - offset, "%s", " ");
}
offset += snprintf(&p->mckernel_numa_distance_s[offset],
NODE_DISTANCE_S_SIZE - offset, "%d",
node_distance(
mckernel_numa_2_linux_numa(udp, p->mckernel_numa_id),
mckernel_numa_2_linux_numa(udp, node)
));
}
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_s,
p->mckernel_numa_distance_s, 0444,
"/sys/devices/system/node/node%d/distance",
p->mckernel_numa_id);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pb, &param, 0444,
"/sys/devices/system/node/node%d/cpumap",
p->mckernel_numa_id);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"/sys/devices/system/node/node%d/cpulist",
p->mckernel_numa_id);
error = sysfsm_lookupf(udp->os, &node_handle,
"/sys/devices/system/node/node%d",
p->mckernel_numa_id);
if (error) {
panic("sysfsm_lookupf(node)");
}
error = sysfsm_symlinkf(udp->os, node_handle,
"/sys/bus/node/devices/node%d",
p->mckernel_numa_id);
if (error) {
panic("sysfsm_symlinkf(bus node)");
}
/* Add CPU symlinks for this node */
for (cpu = 0; cpu < udp->cpu_info->n_cpus; ++cpu) {
if (linux_numa_2_mckernel_numa(udp,
cpu_to_node(mckernel_cpu_2_linux_cpu(udp, cpu)))
!= p->mckernel_numa_id) {
continue;
}
error = sysfsm_symlinkf(udp->os, node_handle,
"/sys/devices/system/cpu/cpu%d/node%d",
cpu, p->mckernel_numa_id);
if (error) {
panic("sysfsm_symlinkf(node in CPU)");
}
error = sysfsm_lookupf(udp->os, &cpu_handle,
"/sys/devices/system/cpu/cpu%d", cpu);
if (error) {
panic("sysfsm_lookupf(CPU in node)");
}
error = sysfsm_symlinkf(udp->os, cpu_handle,
"/sys/devices/system/node/node%d/cpu%d",
p->mckernel_numa_id, cpu);
if (error) {
panic("sysfsm_symlinkf(CPU in node)");
}
}
}
error = 0;
out:
dprintk("setup_node_files(%p): %d\n", udp, error);
return error;
} /* setup_node_files() */
#ifdef SETUP_PCI_FILES
static int read_file(void *buf, size_t size, char *fmt, va_list ap)
{
int error;
int er;
char *filename = NULL;
int n;
struct file *fp = NULL;
loff_t off;
ssize_t ss;
dprintk("read_file(%p,%ld,%s,%p)\n", buf, size, fmt, ap);
filename = kmalloc(PATH_MAX, GFP_KERNEL);
if (!filename) {
error = -ENOMEM;
eprintk("mcctrl:read_file:kmalloc failed. %d\n", error);
goto out;
}
n = vsnprintf(filename, PATH_MAX, fmt, ap);
if (n >= PATH_MAX) {
error = -ENAMETOOLONG;
eprintk("mcctrl:read_file:vsnprintf failed. %d\n", error);
goto out;
}
fp = filp_open(filename, O_RDONLY, 0);
if (IS_ERR(fp)) {
error = PTR_ERR(fp);
eprintk("mcctrl:read_file:filp_open failed. %d\n", error);
goto out;
}
off = 0;
#if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 14, 0)
ss = kernel_read(fp, buf, size, &off);
#else
ss = kernel_read(fp, off, buf, size);
#endif
if (ss < 0) {
error = ss;
eprintk("mcctrl:read_file:kernel_read failed. %d\n", error);
goto out;
}
if (ss >= size) {
error = -ENOSPC;
eprintk("mcctrl:read_file:buffer overflow. %d\n", error);
goto out;
}
*(char *)(buf + ss) = '\0';
error = 0;
out:
if (!IS_ERR_OR_NULL(fp)) {
er = filp_close(fp, NULL);
if (er) {
eprintk("mcctrl:read_file:"
"filp_close failed. %d\n", error);
}
}
kfree(filename);
dprintk("read_file(%p,%ld,%s,%p): %d\n", buf, size, fmt, ap, error);
return error;
} /* read_file() */
static int read_long(long *valuep, char *fmt, ...)
{
int error;
char *buf = NULL;
va_list ap;
int n;
dprintk("read_long(%p,%s)\n", valuep, fmt);
buf = (void *)__get_free_pages(GFP_KERNEL, 0);
if (!buf) {
error = -ENOMEM;
eprintk("mcctrl:read_long:"
"__get_free_pages failed. %d\n", error);
goto out;
}
va_start(ap, fmt);
error = read_file(buf, PAGE_SIZE, fmt, ap);
va_end(ap);
if (error) {
eprintk("mcctrl:read_long:read_file failed. %d\n", error);
goto out;
}
n = sscanf(buf, "%ld", valuep);
if (n != 1) {
error = -EIO;
eprintk("mcctrl:read_long:sscanf failed. %d\n", error);
goto out;
}
error = 0;
out:
free_pages((long)buf, 0);
dprintk("read_long(%p,%s): %d\n", valuep, fmt, error);
return error;
} /* read_long() */
static int read_link(char *buf, size_t bufsize, char *fmt, ...)
{
int error;
char *filename = NULL;
va_list ap;
int n;
mm_segment_t old_fs;
ssize_t ss;
dprintk("read_link(%p,%#lx,%s)\n", buf, bufsize, fmt);
filename = kmalloc(PATH_MAX, GFP_KERNEL);
if (!filename) {
error = -ENOMEM;
eprintk("mcctrl:read_link:kmalloc failed. %d\n", error);
goto out;
}
va_start(ap, fmt);
n = vsnprintf(filename, PATH_MAX, fmt, ap);
va_end(ap);
if (n >= PATH_MAX) {
error = -ENAMETOOLONG;
eprintk("mcctrl:read_link:snprintf failed. %d\n", error);
goto out;
}
old_fs = get_fs();
set_fs(KERNEL_DS);
ss = mcctrl_sys_readlinkat(AT_FDCWD, filename, buf, bufsize);
set_fs(old_fs);
if (ss < 0) {
error = ss;
eprintk("mcctrl:read_link:sys_readlink failed. %d\n", error);
goto out;
}
if (ss >= bufsize) {
error = -ENOSPC;
eprintk("mcctrl:read_link:linkname too long. %d\n", error);
goto out;
}
buf[ss] = '\0';
error = 0;
out:
kfree(filename);
dprintk("read_link(%p,%#lx,%s): %d\n", buf, bufsize, fmt, error);
return error;
} /* read_link() */
static int setup_one_pci(struct mcctrl_usrdata *udp, const char *name)
{
int error;
char *buf = NULL;
long node;
struct sysfsm_bitmap_param param;
dprintk("setup_one_pci(%p,%s)\n", udp, name);
buf = (void *)__get_free_pages(GFP_KERNEL, 0);
if (!buf) {
error = -ENOMEM;
eprintk("mcctrl:setup_one_pci:"
"__get_free_pages failed. %d\n", error);
goto out;
}
error = read_long(&node, "/sys/bus/pci/devices/%s/numa_node", name);
if (error) {
eprintk("mcctrl:setup_one_pci:read_long failed. %d\n", error);
goto out;
}
error = read_link(buf, PAGE_SIZE, "/sys/bus/pci/devices/%s", name);
if (error) {
eprintk("mcctrl:setup_one_pci:read_link failed. %d\n", error);
goto out;
}
if (strncmp(buf, "../../../devices/", 17)) {
error = -ENOENT;
eprintk("mcctrl:setup_one_pci:"
"realpath is not /sys/devices. %d\n", error);
goto out;
}
param.ptr = &udp->cpu_online;
param.nbits = nr_cpu_ids;
if (node >= 0) {
struct node_topology *node_topo;
list_for_each_entry(node_topo,
&udp->node_topology_list, chain) {
if (node_topo->saved->node_number == node) {
param.ptr = &node_topo->cpumap;
param.nbits = nr_cpu_ids;
break;
}
}
}
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pb, &param, 0444,
"/sys/%s/local_cpus", buf+9);
sysfsm_createf(udp->os, SYSFS_SNOOPING_OPS_pbl, &param, 0444,
"/sys/%s/local_cpulist", buf+9);
error = 0;
out:
free_pages((long)buf, 0);
dprintk("setup_one_pci(%p,%s): %d\n", udp, name, error);
return error;
} /* setup_one_pci() */
LIST_HEAD(pci_file_name_list);
struct pci_file_name {
char *name;
struct list_head chain;
};
#if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 11, 0) || \
(defined(RHEL_RELEASE_CODE) && RHEL_RELEASE_CODE >= RHEL_RELEASE_VERSION(7, 5))
struct mcctrl_filler_args {
struct dir_context ctx;
void *buf;
};
#if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 19, 0)
static int pci_file_name_gen(struct dir_context *ctx, const char *name,
int namlen, loff_t offset, u64 ino, unsigned int d_type)
#else
static int pci_file_name_gen(void *ctx, const char *name,
int namlen, loff_t offset, u64 ino, unsigned int d_type)
#endif
{
struct mcctrl_filler_args *args
= container_of(ctx, struct mcctrl_filler_args, ctx);
void *buf = args->buf;
#else
static int pci_file_name_gen(void *buf, const char *name, int namlen,
loff_t offset, u64 ino, unsigned d_type)
{
#endif
struct pci_file_name *p;
int error = -1;
dprintk("pci_file_name_gen(%p,%s,%d,%#lx,%#lx,%d)\n",
buf, name, namlen, (long)offset, (long)ino, d_type);
/* check namlen, name exmple, "0000:00:00.0" 12 chars */
/* otherstring, return function */
if (namlen != 12) {
error = 0;
goto out;
}
p = kmalloc(sizeof(*p), GFP_KERNEL);
if (!p) {
error = -ENOMEM;
eprintk("mcctrl:pci_file_name_gen:kmalloc failed. %d\n", error);
goto out;
}
p->name = kmalloc(sizeof(namlen + 1), GFP_KERNEL);
if (!p->name) {
error = -ENOMEM;
eprintk("mcctrl:pci_file_name_gen:kmalloc failed. %d\n", error);
kfree(p);
goto out;
}
memset(p->name, '\0', namlen + 1);
memcpy(p->name, name, namlen);
list_add(&p->chain, &pci_file_name_list);
error = 0;
out:
dprintk("pci_file_name_gen(%p,%s,%d,%#lx,%#lx,%d): %d\n",
buf, name, namlen, (long)offset, (long)ino, d_type, error);
return error;
}
static inline int mcctrl_vfs_readdir(struct file *file, filldir_t filler,
void *buf)
{
#if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 11, 0) || \
(defined(RHEL_RELEASE_CODE) && RHEL_RELEASE_CODE >= RHEL_RELEASE_VERSION(7, 5))
struct mcctrl_filler_args args = {
.ctx.actor = filler,
.buf = buf,
};
return iterate_dir(file, &args.ctx);
#else
return vfs_readdir(file, filler, buf);
#endif
} /* mcctrl_vfs_readdir() */
static int setup_pci_files(struct mcctrl_usrdata *udp)
{
int error;
int er;
struct file *fp = NULL;
int ret = 0;
struct pci_file_name *cur;
struct pci_file_name *next;
dprintk("setup_pci_files(%p)\n", udp);
fp = filp_open("/sys/bus/pci/devices", O_DIRECTORY, 0);
if (IS_ERR(fp)) {
error = PTR_ERR(fp);
eprintk("mcctrl:setup_pci_files:filp_open failed. %d\n", error);
goto out;
}
error = mcctrl_vfs_readdir(fp, &pci_file_name_gen, udp);
if (error) {
eprintk("mcctrl:setup_pci_files:"
"mcctrl_vfs_readdir failed. %d\n", error);
goto out;
}
list_for_each_entry_safe(cur, next, &pci_file_name_list, chain) {
if (!ret) {
ret = setup_one_pci(udp, cur->name);
}
list_del(&cur->chain);
kfree(cur->name);
kfree(cur);
}
error = 0;
out:
if (!IS_ERR_OR_NULL(fp)) {
er = filp_close(fp, NULL);
if (er) {
eprintk("mcctrl:setup_pci_files:"
"filp_close failed. %d\n", er);
}
}
dprintk("setup_pci_files(%p): %d\n", udp, error);
return error;
} /* setup_pci_files() */
#endif // SETUP_PCI_FILES
void setup_sysfs_files(ihk_os_t os)
{
static int a_value = 35;
int error;
struct sysfs_handle handle;
struct mcctrl_usrdata *udp = ihk_host_os_get_usrdata(os);
if (!udp) {
panic("%s: error: mcctrl_usrdata not found\n",
__func__);
}
error = sysfsm_mkdirf(os, NULL, "/sys/test/x.dir");
if (error) {
panic("sysfsm_mkdir(x.dir)");
}
error = sysfsm_createf(os, &show_int_ops, &a_value, 0444,
"/sys/test/a.dir/a_value");
if (error) {
panic("sysfsm_createf");
}
error = sysfsm_lookupf(os, &handle, "/sys/test/%s", "a.dir");
if (error) {
panic("sysfsm_lookupf(a.dir)");
}
error = sysfsm_symlinkf(os, handle, "/sys/test/%c.dir", 'L');
if (error) {
panic("sysfsm_symlinkf");
}
error = sysfsm_unlinkf(os, 0, "/sys/test/%s.dir", "x");
if (error) {
panic("sysfsm_unlinkf");
}
//setup_local_snooping_samples(os);
setup_local_snooping_files(os);
setup_cpus_sysfs_files(udp);
setup_node_files(udp);
#ifdef SETUP_PCI_FILES
setup_pci_files(udp);
#endif
/* Indicate sysfs files setup completion for boot script */
error = sysfsm_mkdirf(os, NULL, "/sys/setup_complete");
if (error) {
panic("sysfsm_mkdir(complete)");
}
return;
} /* setup_files() */
/**** End of File ****/