perf record: Provide detailed information on s390 CPU
When perf record ... is setup to record data, the s390 cpu information was a fixed string "IBM/S390". Replace this string with one containing more information about the machine. The information included in the cpuid is a comma separated list: manufacturer,type,model-capacity,model[,version,authorization] with - manufacturer: up to 16 byte name of the manufacturer (IBM). - type: a four digit number refering to the machine generation. - model-capacitiy: up to 16 characters describing number of cpus etc. - model: up to 16 characters describing model. - version: the CPU-MF counter facility version number, available on LPARs only, omitted on z/VM guests. - authorization: the CPU-MF counter facility authorization level, available on LPARs only, omitted on z/VM guests. Before: [root@s8360047 perf]# ./perf record -- sleep 1 [ perf record: Woken up 1 times to write data ] [ perf record: Captured and wrote 0.001 MB perf.data (4 samples) ] [root@s8360047 perf]# ./perf report --header | fgrep cpuid # cpuid : IBM/S390 [root@s8360047 perf]# After: [root@s35lp76 perf]# ./perf report --header|fgrep cpuid # cpuid : IBM,3906,704,M03,3.5,002f [root@s35lp76 perf]# Signed-off-by: Thomas Richter <tmricht@linux.vnet.ibm.com> Reviewed-by: Hendrik Brueckner <brueckner@linux.vnet.ibm.com> Cc: Heiko Carstens <heiko.carstens@de.ibm.com> Cc: Martin Schwidefsky <schwidefsky@de.ibm.com> Link: http://lkml.kernel.org/r/20180213151419.80737-1-tmricht@linux.vnet.ibm.com [ Use scnprintf instead of strncat to fix build errors on gcc GNU C99 5.4.0 20160609 -march=zEC12 -m64 -mzarch -ggdb3 -O6 -std=gnu99 -fPIC -fno-omit-frame-pointer -funwind-tables -fstack-protector-all ] Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
This commit is contained in:
parent
4281da235e
commit
eca0fa28cd
|
@ -1,8 +1,9 @@
|
|||
/*
|
||||
* Implementation of get_cpuid().
|
||||
*
|
||||
* Copyright 2014 IBM Corp.
|
||||
* Copyright IBM Corp. 2014, 2018
|
||||
* Author(s): Alexander Yarygin <yarygin@linux.vnet.ibm.com>
|
||||
* Thomas Richter <tmricht@linux.vnet.ibm.com>
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License (version 2 only)
|
||||
|
@ -13,16 +14,135 @@
|
|||
#include <unistd.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <ctype.h>
|
||||
|
||||
#include "../../util/header.h"
|
||||
#include "../../util/util.h"
|
||||
|
||||
#define SYSINFO_MANU "Manufacturer:"
|
||||
#define SYSINFO_TYPE "Type:"
|
||||
#define SYSINFO_MODEL "Model:"
|
||||
#define SRVLVL_CPUMF "CPU-MF:"
|
||||
#define SRVLVL_VERSION "version="
|
||||
#define SRVLVL_AUTHORIZATION "authorization="
|
||||
#define SYSINFO "/proc/sysinfo"
|
||||
#define SRVLVL "/proc/service_levels"
|
||||
|
||||
int get_cpuid(char *buffer, size_t sz)
|
||||
{
|
||||
const char *cpuid = "IBM/S390";
|
||||
char *cp, *line = NULL, *line2;
|
||||
char type[8], model[33], version[8], manufacturer[32], authorization[8];
|
||||
int tpsize = 0, mdsize = 0, vssize = 0, mfsize = 0, atsize = 0;
|
||||
int read;
|
||||
unsigned long line_sz;
|
||||
size_t nbytes;
|
||||
FILE *sysinfo;
|
||||
|
||||
if (strlen(cpuid) + 1 > sz)
|
||||
/*
|
||||
* Scan /proc/sysinfo line by line and read out values for
|
||||
* Manufacturer:, Type: and Model:, for example:
|
||||
* Manufacturer: IBM
|
||||
* Type: 2964
|
||||
* Model: 702 N96
|
||||
* The first word is the Model Capacity and the second word is
|
||||
* Model (can be omitted). Both words have a maximum size of 16
|
||||
* bytes.
|
||||
*/
|
||||
memset(manufacturer, 0, sizeof(manufacturer));
|
||||
memset(type, 0, sizeof(type));
|
||||
memset(model, 0, sizeof(model));
|
||||
memset(version, 0, sizeof(version));
|
||||
memset(authorization, 0, sizeof(authorization));
|
||||
|
||||
sysinfo = fopen(SYSINFO, "r");
|
||||
if (sysinfo == NULL)
|
||||
return -1;
|
||||
|
||||
strcpy(buffer, cpuid);
|
||||
return 0;
|
||||
while ((read = getline(&line, &line_sz, sysinfo)) != -1) {
|
||||
if (!strncmp(line, SYSINFO_MANU, strlen(SYSINFO_MANU))) {
|
||||
line2 = line + strlen(SYSINFO_MANU);
|
||||
|
||||
while ((cp = strtok_r(line2, "\n ", &line2))) {
|
||||
mfsize += scnprintf(manufacturer + mfsize,
|
||||
sizeof(manufacturer) - mfsize, "%s", cp);
|
||||
}
|
||||
}
|
||||
|
||||
if (!strncmp(line, SYSINFO_TYPE, strlen(SYSINFO_TYPE))) {
|
||||
line2 = line + strlen(SYSINFO_TYPE);
|
||||
|
||||
while ((cp = strtok_r(line2, "\n ", &line2))) {
|
||||
tpsize += scnprintf(type + tpsize,
|
||||
sizeof(type) - tpsize, "%s", cp);
|
||||
}
|
||||
}
|
||||
|
||||
if (!strncmp(line, SYSINFO_MODEL, strlen(SYSINFO_MODEL))) {
|
||||
line2 = line + strlen(SYSINFO_MODEL);
|
||||
|
||||
while ((cp = strtok_r(line2, "\n ", &line2))) {
|
||||
mdsize += scnprintf(model + mdsize, sizeof(type) - mdsize,
|
||||
"%s%s", model[0] ? "," : "", cp);
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
fclose(sysinfo);
|
||||
|
||||
/* Missing manufacturer, type or model information should not happen */
|
||||
if (!manufacturer[0] || !type[0] || !model[0])
|
||||
return -1;
|
||||
|
||||
/*
|
||||
* Scan /proc/service_levels and return the CPU-MF counter facility
|
||||
* version number and authorization level.
|
||||
* Optional, does not exist on z/VM guests.
|
||||
*/
|
||||
sysinfo = fopen(SRVLVL, "r");
|
||||
if (sysinfo == NULL)
|
||||
goto skip_sysinfo;
|
||||
while ((read = getline(&line, &line_sz, sysinfo)) != -1) {
|
||||
if (strncmp(line, SRVLVL_CPUMF, strlen(SRVLVL_CPUMF)))
|
||||
continue;
|
||||
|
||||
line2 = line + strlen(SRVLVL_CPUMF);
|
||||
while ((cp = strtok_r(line2, "\n ", &line2))) {
|
||||
if (!strncmp(cp, SRVLVL_VERSION,
|
||||
strlen(SRVLVL_VERSION))) {
|
||||
char *sep = strchr(cp, '=');
|
||||
|
||||
vssize += scnprintf(version + vssize,
|
||||
sizeof(version) - vssize, "%s", sep + 1);
|
||||
}
|
||||
if (!strncmp(cp, SRVLVL_AUTHORIZATION,
|
||||
strlen(SRVLVL_AUTHORIZATION))) {
|
||||
char *sep = strchr(cp, '=');
|
||||
|
||||
atsize += scnprintf(authorization + atsize,
|
||||
sizeof(authorization) - atsize, "%s", sep + 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
fclose(sysinfo);
|
||||
|
||||
skip_sysinfo:
|
||||
free(line);
|
||||
|
||||
if (version[0] && authorization[0] )
|
||||
nbytes = snprintf(buffer, sz, "%s,%s,%s,%s,%s",
|
||||
manufacturer, type, model, version,
|
||||
authorization);
|
||||
else
|
||||
nbytes = snprintf(buffer, sz, "%s,%s,%s", manufacturer, type,
|
||||
model);
|
||||
return (nbytes >= sz) ? -1 : 0;
|
||||
}
|
||||
|
||||
char *get_cpuid_str(struct perf_pmu *pmu __maybe_unused)
|
||||
{
|
||||
char *buf = malloc(128);
|
||||
|
||||
if (buf && get_cpuid(buf, 128) < 0)
|
||||
zfree(&buf);
|
||||
return buf;
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue