diff options
| author | Thomas Richter <tmricht@linux.vnet.ibm.com> | 2018-02-13 10:14:16 -0500 |
|---|---|---|
| committer | Arnaldo Carvalho de Melo <acme@redhat.com> | 2018-02-16 13:15:23 -0500 |
| commit | eca0fa28cd0df7369701dbee0e30ddce19c039b8 (patch) | |
| tree | ec81bfc68943d0160a585256700092e70770ccb2 /tools | |
| parent | 4281da235e3de91bb8deae44bc6506336ceaa88a (diff) | |
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>
Diffstat (limited to 'tools')
| -rw-r--r-- | tools/perf/arch/s390/util/header.c | 130 |
1 files changed, 125 insertions, 5 deletions
diff --git a/tools/perf/arch/s390/util/header.c b/tools/perf/arch/s390/util/header.c index 9fa6c3e5782c..a78064c25ced 100644 --- a/tools/perf/arch/s390/util/header.c +++ b/tools/perf/arch/s390/util/header.c | |||
| @@ -1,8 +1,9 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Implementation of get_cpuid(). | 2 | * Implementation of get_cpuid(). |
| 3 | * | 3 | * |
| 4 | * Copyright 2014 IBM Corp. | 4 | * Copyright IBM Corp. 2014, 2018 |
| 5 | * Author(s): Alexander Yarygin <yarygin@linux.vnet.ibm.com> | 5 | * Author(s): Alexander Yarygin <yarygin@linux.vnet.ibm.com> |
| 6 | * Thomas Richter <tmricht@linux.vnet.ibm.com> | ||
| 6 | * | 7 | * |
| 7 | * This program is free software; you can redistribute it and/or modify | 8 | * This program is free software; you can redistribute it and/or modify |
| 8 | * it under the terms of the GNU General Public License (version 2 only) | 9 | * it under the terms of the GNU General Public License (version 2 only) |
| @@ -13,16 +14,135 @@ | |||
| 13 | #include <unistd.h> | 14 | #include <unistd.h> |
| 14 | #include <stdio.h> | 15 | #include <stdio.h> |
| 15 | #include <string.h> | 16 | #include <string.h> |
| 17 | #include <ctype.h> | ||
| 16 | 18 | ||
| 17 | #include "../../util/header.h" | 19 | #include "../../util/header.h" |
| 20 | #include "../../util/util.h" | ||
| 21 | |||
| 22 | #define SYSINFO_MANU "Manufacturer:" | ||
| 23 | #define SYSINFO_TYPE "Type:" | ||
| 24 | #define SYSINFO_MODEL "Model:" | ||
| 25 | #define SRVLVL_CPUMF "CPU-MF:" | ||
| 26 | #define SRVLVL_VERSION "version=" | ||
| 27 | #define SRVLVL_AUTHORIZATION "authorization=" | ||
| 28 | #define SYSINFO "/proc/sysinfo" | ||
| 29 | #define SRVLVL "/proc/service_levels" | ||
| 18 | 30 | ||
| 19 | int get_cpuid(char *buffer, size_t sz) | 31 | int get_cpuid(char *buffer, size_t sz) |
| 20 | { | 32 | { |
| 21 | const char *cpuid = "IBM/S390"; | 33 | char *cp, *line = NULL, *line2; |
| 34 | char type[8], model[33], version[8], manufacturer[32], authorization[8]; | ||
| 35 | int tpsize = 0, mdsize = 0, vssize = 0, mfsize = 0, atsize = 0; | ||
| 36 | int read; | ||
| 37 | unsigned long line_sz; | ||
| 38 | size_t nbytes; | ||
| 39 | FILE *sysinfo; | ||
| 40 | |||
| 41 | /* | ||
| 42 | * Scan /proc/sysinfo line by line and read out values for | ||
| 43 | * Manufacturer:, Type: and Model:, for example: | ||
| 44 | * Manufacturer: IBM | ||
| 45 | * Type: 2964 | ||
| 46 | * Model: 702 N96 | ||
| 47 | * The first word is the Model Capacity and the second word is | ||
| 48 | * Model (can be omitted). Both words have a maximum size of 16 | ||
| 49 | * bytes. | ||
| 50 | */ | ||
| 51 | memset(manufacturer, 0, sizeof(manufacturer)); | ||
| 52 | memset(type, 0, sizeof(type)); | ||
| 53 | memset(model, 0, sizeof(model)); | ||
| 54 | memset(version, 0, sizeof(version)); | ||
| 55 | memset(authorization, 0, sizeof(authorization)); | ||
| 56 | |||
| 57 | sysinfo = fopen(SYSINFO, "r"); | ||
| 58 | if (sysinfo == NULL) | ||
| 59 | return -1; | ||
| 60 | |||
| 61 | while ((read = getline(&line, &line_sz, sysinfo)) != -1) { | ||
| 62 | if (!strncmp(line, SYSINFO_MANU, strlen(SYSINFO_MANU))) { | ||
| 63 | line2 = line + strlen(SYSINFO_MANU); | ||
| 64 | |||
| 65 | while ((cp = strtok_r(line2, "\n ", &line2))) { | ||
| 66 | mfsize += scnprintf(manufacturer + mfsize, | ||
| 67 | sizeof(manufacturer) - mfsize, "%s", cp); | ||
| 68 | } | ||
| 69 | } | ||
| 70 | |||
| 71 | if (!strncmp(line, SYSINFO_TYPE, strlen(SYSINFO_TYPE))) { | ||
| 72 | line2 = line + strlen(SYSINFO_TYPE); | ||
| 73 | |||
| 74 | while ((cp = strtok_r(line2, "\n ", &line2))) { | ||
| 75 | tpsize += scnprintf(type + tpsize, | ||
| 76 | sizeof(type) - tpsize, "%s", cp); | ||
| 77 | } | ||
| 78 | } | ||
| 79 | |||
| 80 | if (!strncmp(line, SYSINFO_MODEL, strlen(SYSINFO_MODEL))) { | ||
| 81 | line2 = line + strlen(SYSINFO_MODEL); | ||
| 82 | |||
| 83 | while ((cp = strtok_r(line2, "\n ", &line2))) { | ||
| 84 | mdsize += scnprintf(model + mdsize, sizeof(type) - mdsize, | ||
| 85 | "%s%s", model[0] ? "," : "", cp); | ||
| 86 | } | ||
| 87 | break; | ||
| 88 | } | ||
| 89 | } | ||
| 90 | fclose(sysinfo); | ||
| 22 | 91 | ||
| 23 | if (strlen(cpuid) + 1 > sz) | 92 | /* Missing manufacturer, type or model information should not happen */ |
| 93 | if (!manufacturer[0] || !type[0] || !model[0]) | ||
| 24 | return -1; | 94 | return -1; |
| 25 | 95 | ||
| 26 | strcpy(buffer, cpuid); | 96 | /* |
| 27 | return 0; | 97 | * Scan /proc/service_levels and return the CPU-MF counter facility |
| 98 | * version number and authorization level. | ||
| 99 | * Optional, does not exist on z/VM guests. | ||
| 100 | */ | ||
| 101 | sysinfo = fopen(SRVLVL, "r"); | ||
| 102 | if (sysinfo == NULL) | ||
| 103 | goto skip_sysinfo; | ||
| 104 | while ((read = getline(&line, &line_sz, sysinfo)) != -1) { | ||
| 105 | if (strncmp(line, SRVLVL_CPUMF, strlen(SRVLVL_CPUMF))) | ||
| 106 | continue; | ||
| 107 | |||
| 108 | line2 = line + strlen(SRVLVL_CPUMF); | ||
| 109 | while ((cp = strtok_r(line2, "\n ", &line2))) { | ||
| 110 | if (!strncmp(cp, SRVLVL_VERSION, | ||
| 111 | strlen(SRVLVL_VERSION))) { | ||
| 112 | char *sep = strchr(cp, '='); | ||
| 113 | |||
| 114 | vssize += scnprintf(version + vssize, | ||
| 115 | sizeof(version) - vssize, "%s", sep + 1); | ||
| 116 | } | ||
| 117 | if (!strncmp(cp, SRVLVL_AUTHORIZATION, | ||
| 118 | strlen(SRVLVL_AUTHORIZATION))) { | ||
| 119 | char *sep = strchr(cp, '='); | ||
| 120 | |||
| 121 | atsize += scnprintf(authorization + atsize, | ||
| 122 | sizeof(authorization) - atsize, "%s", sep + 1); | ||
| 123 | } | ||
| 124 | } | ||
| 125 | } | ||
| 126 | fclose(sysinfo); | ||
| 127 | |||
| 128 | skip_sysinfo: | ||
| 129 | free(line); | ||
| 130 | |||
| 131 | if (version[0] && authorization[0] ) | ||
| 132 | nbytes = snprintf(buffer, sz, "%s,%s,%s,%s,%s", | ||
| 133 | manufacturer, type, model, version, | ||
| 134 | authorization); | ||
| 135 | else | ||
| 136 | nbytes = snprintf(buffer, sz, "%s,%s,%s", manufacturer, type, | ||
| 137 | model); | ||
| 138 | return (nbytes >= sz) ? -1 : 0; | ||
| 139 | } | ||
| 140 | |||
| 141 | char *get_cpuid_str(struct perf_pmu *pmu __maybe_unused) | ||
| 142 | { | ||
| 143 | char *buf = malloc(128); | ||
| 144 | |||
| 145 | if (buf && get_cpuid(buf, 128) < 0) | ||
| 146 | zfree(&buf); | ||
| 147 | return buf; | ||
| 28 | } | 148 | } |
