PAPI 7.1.0.0
Loading...
Searching...
No Matches
pfmlib_os_linux_v2.c
Go to the documentation of this file.
1/*
2 * pfmlib_os_linux_v2.c: Perfmon2 syscall API
3 *
4 * Copyright (c) 2003-2006 Hewlett-Packard Development Company, L.P.
5 * Contributed by Stephane Eranian <eranian@hpl.hp.com>
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a copy
8 * of this software and associated documentation files (the "Software"), to deal
9 * in the Software without restriction, including without limitation the rights
10 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
11 * of the Software, and to permit persons to whom the Software is furnished to do so,
12 * subject to the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be included in all
15 * copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
18 * INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
19 * PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
20 * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
21 * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE
22 * OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23 */
24#include <sys/types.h>
25#include <stdint.h>
26#include <stdio.h>
27#include <stdarg.h>
28#include <string.h>
29#include <stdlib.h>
30#include <unistd.h>
31#include <errno.h>
32#include <syscall.h>
33#include <perfmon/perfmon.h>
34
35#include <perfmon/pfmlib.h>
36#include "pfmlib_priv.h"
37
38/*
39 * v2.x interface
40 */
41#define PFM_pfm_create_context (_pfmlib_get_sys_base()+0)
42#define PFM_pfm_write_pmcs (_pfmlib_get_sys_base()+1)
43#define PFM_pfm_write_pmds (_pfmlib_get_sys_base()+2)
44#define PFM_pfm_read_pmds (_pfmlib_get_sys_base()+3)
45#define PFM_pfm_load_context (_pfmlib_get_sys_base()+4)
46#define PFM_pfm_start (_pfmlib_get_sys_base()+5)
47#define PFM_pfm_stop (_pfmlib_get_sys_base()+6)
48#define PFM_pfm_restart (_pfmlib_get_sys_base()+7)
49#define PFM_pfm_create_evtsets (_pfmlib_get_sys_base()+8)
50#define PFM_pfm_getinfo_evtsets (_pfmlib_get_sys_base()+9)
51#define PFM_pfm_delete_evtsets (_pfmlib_get_sys_base()+10)
52#define PFM_pfm_unload_context (_pfmlib_get_sys_base()+11)
53
54/*
55 * argument to v2.2 pfm_create_context()
56 * ALWAYS use pfarg_ctx_t in programs, libpfm
57 * does convert ths structure on the fly if v2.2. is detected
58 */
59typedef struct {
60 unsigned char ctx_smpl_buf_id[16]; /* which buffer format to use */
61 uint32_t ctx_flags; /* noblock/block/syswide */
62 int32_t ctx_fd; /* ret arg: fd for context */
63 uint64_t ctx_smpl_buf_size; /* ret arg: actual buffer sz */
64 uint64_t ctx_reserved3[12]; /* for future use */
66
67/*
68 * perfmon2 compatibility layer with perfmon3
69 */
70#ifndef PFMLIB_OLD_PFMV2
71static int
72pfm_create_context_2v3(pfarg_ctx_t *ctx, char *name, void *smpl_arg, size_t smpl_size)
73{
74 pfarg_sinfo_t cinfo;
75 uint32_t fl;
76
77 /*
78 * simulate kernel returning error on NULL ctx
79 */
80 if (!ctx) {
81 errno = EINVAL;
82 return -1;
83 }
84
85 /*
86 * if sampling format is used, then force SMPL_FMT
87 * and PFM_FL_SINFO because it comes first
88 */
89 fl = ctx->ctx_flags;
90 if (name || smpl_arg || smpl_size)
91 fl |= PFM_FL_SMPL_FMT;
92
93 return pfm_create(fl, &cinfo, name, smpl_arg, smpl_size);
94}
95
96static int
98{
99 pfarg_pmr_t *pmrs;
100 int errno_save;
101 int i, ret;
102 size_t sz;
103
104 sz = count * sizeof(pfarg_pmr_t);
105
106 if (!pmcs)
107 return pfm_write(fd, 0, PFM_RW_PMC, NULL, sz);
108
109 pmrs = calloc(count, sizeof(*pmrs));
110 if (!pmrs) {
111 errno = ENOMEM;
112 return -1;
113 }
114
115 for (i=0 ; i < count; i++) {
116 pmrs[i].reg_num = pmcs[i].reg_num;
117 pmrs[i].reg_set = pmcs[i].reg_set;
118 pmrs[i].reg_flags = pmcs[i].reg_flags;
119 pmrs[i].reg_value = pmcs[i].reg_value;
120 }
121
122 ret = pfm_write(fd, 0, PFM_RW_PMC, pmrs, sz);
123 errno_save = errno;
124 free(pmrs);
125 errno = errno_save;
126 return ret;
127}
128
129static int
131{
132 pfarg_pmd_attr_t *pmas;
133 size_t sz;
134 int errno_save;
135 int i, ret;
136
137 sz = count * sizeof(*pmas);
138
139 if (!pmds)
140 return pfm_write(fd, 0, PFM_RW_PMD, NULL, sz);
141
142 pmas = calloc(count, sizeof(*pmas));
143 if (!pmas) {
144 errno = ENOMEM;
145 return -1;
146 }
147
148 for (i=0 ; i < count; i++) {
149 pmas[i].reg_num = pmds[i].reg_num;
150 pmas[i].reg_set = pmds[i].reg_set;
151 pmas[i].reg_flags = pmds[i].reg_flags;
152 pmas[i].reg_value = pmds[i].reg_value;
153
154 pmas[i].reg_long_reset = pmds[i].reg_long_reset;
155 pmas[i].reg_short_reset = pmds[i].reg_short_reset;
156 /* skip last_value not used on write */
157
159
160 memcpy(pmas[i].reg_smpl_pmds, pmds[i].reg_smpl_pmds, sizeof(pmds[i].reg_smpl_pmds));
161 memcpy(pmas[i].reg_reset_pmds, pmds[i].reg_reset_pmds, sizeof(pmds[i].reg_reset_pmds));
162
163 pmas[i].reg_smpl_eventid = pmds[i].reg_smpl_eventid;
164 pmas[i].reg_random_mask = pmds[i].reg_random_mask;
165 }
166
167 ret = pfm_write(fd, 0, PFM_RW_PMD_ATTR, pmas, sz);
168
169 errno_save = errno;
170 free(pmas);
171 errno = errno_save;
172
173 return ret;
174}
175
176static int
178{
179 pfarg_pmd_attr_t *pmas;
180 int errno_save;
181 int i, ret;
182 size_t sz;
183
184 sz = count * sizeof(*pmas);
185
186 if (!pmds)
187 return pfm_write(fd, 0, PFM_RW_PMD, NULL, sz);
188
189 pmas = calloc(count, sizeof(*pmas));
190 if (!pmas) {
191 errno = ENOMEM;
192 return -1;
193 }
194
195 for (i=0 ; i < count; i++) {
196 pmas[i].reg_num = pmds[i].reg_num;
197 pmas[i].reg_set = pmds[i].reg_set;
198 pmas[i].reg_flags = pmds[i].reg_flags;
199 pmas[i].reg_value = pmds[i].reg_value;
200 }
201
202 ret = pfm_read(fd, 0, PFM_RW_PMD_ATTR, pmas, sz);
203
204 errno_save = errno;
205
206 for (i=0 ; i < count; i++) {
207 pmds[i].reg_value = pmas[i].reg_value;
208
209 pmds[i].reg_long_reset = pmas[i].reg_long_reset;
210 pmds[i].reg_short_reset = pmas[i].reg_short_reset;
211 pmds[i].reg_last_reset_val = pmas[i].reg_last_value;
212
214 /* skip reg_smpl_pmds */
215 /* skip reg_reset_pmds */
216 /* skip reg_smpl_eventid */
217 /* skip reg_random_mask */
218 }
219 free(pmas);
220 errno = errno_save;
221 return ret;
222}
223
224static int
226{
227 if (!load) {
228 errno = EINVAL;
229 return -1;
230 }
231 return pfm_attach(fd, 0, load->load_pid);
232}
233
234static int
236{
237 if (start) {
238 __pfm_vbprintf("pfarg_start_t not supported in v3.x\n");
239 errno = EINVAL;
240 return -1;
241 }
242 return pfm_set_state(fd, 0, PFM_ST_START);
243}
244static int
246{
247 return pfm_set_state(fd, 0, PFM_ST_STOP);
248}
249
250static int
252{
253 return pfm_set_state(fd, 0, PFM_ST_RESTART);
254}
255
256static int
258{
259 /* set_desc an setdesc are identical so we can cast */
260 return pfm_create_sets(fd, 0, (pfarg_set_desc_t *)setd, count * sizeof(pfarg_setdesc_t));
261}
262
263static int
265{
266 __pfm_vbprintf("pfm_delete_evtsets not supported in v3.x\n");
267 errno = EINVAL;
268 return -1;
269}
270
271static int
273{
274 pfarg_sinfo_t cinfo;
275 pfarg_set_info_t *sif;
276 int fdx, i, ret, errno_save;
277
278 if (!info) {
279 errno = EFAULT;
280 return -1;
281 }
282 /*
283 * initialize bitmask to all available and defer checking
284 * until kernel. That means libpfm must be misled but we
285 * have no other way of fixing this
286 */
287 memset(&cinfo, -1, sizeof(cinfo));
288
289 /*
290 * XXX: relies on the fact that cinfo is independent
291 * of the session type (which is wrong in the future)
292 */
293 fdx = pfm_create(0, &cinfo);
294 if (fdx > -1)
295 close(fdx);
296
297 sif = calloc(count, sizeof(*sif));
298 if (!sif) {
299 errno = ENOMEM;
300 return -1;
301 }
302
303 for (i=0 ; i < count; i++)
304 sif[i].set_id = info[i].set_id;
305
306 ret = pfm_getinfo_sets(fd, 0, sif, count * sizeof(pfarg_set_info_t));
307 errno_save = errno;
308 if (ret)
309 goto skip;
310
311 for (i=0 ; i < count; i++) {
312 info[i].set_flags = 0;
313
314 memcpy(info[i].set_ovfl_pmds,
315 sif[i].set_ovfl_pmds,
316 sizeof(info[i].set_ovfl_pmds));
317
318 info[i].set_runs = sif[i].set_runs;
319 info[i].set_timeout = sif[i].set_timeout;
320 info[i].set_act_duration = sif[i].set_duration;
321
322 memcpy(info[i].set_avail_pmcs,
323 cinfo.sif_avail_pmcs,
324 sizeof(info[i].set_avail_pmcs));
325
326 memcpy(info[i].set_avail_pmds,
327 cinfo.sif_avail_pmds,
328 sizeof(info[i].set_avail_pmds));
329 }
330skip:
331 free(sif);
332 errno = errno_save;
333 return ret;
334}
335
336static int
338{
339 return pfm_attach(fd, 0, PFM_NO_TARGET);
340}
341
342#else /* PFMLIB_OLD_PFMV2 */
343
344static int
345pfm_create_context_2v3(pfarg_ctx_t *ctx, char *name, void *smpl_arg, size_t smpl_size)
346{
347 return -1;
348}
349
350static int
351pfm_write_pmcs_2v3(int fd, pfarg_pmc_t *pmcs, int count)
352{
353 return -1;
354}
355
356static int
357pfm_write_pmds_2v3(int fd, pfarg_pmd_t *pmds, int count)
358{
359 return -1;
360}
361
362static int
363pfm_read_pmds_2v3(int fd, pfarg_pmd_t *pmds, int count)
364{
365 return -1;
366}
367
368static int
370{
371 return -1;
372}
373
374static int
376{
377 return -1;
378}
379static int
380pfm_stop_2v3(int fd)
381{
382 return -1;
383}
384
385static int
386pfm_restart_2v3(int fd)
387{
388 return -1;
389}
390
391static int
393{
394 return -1;
395}
396
397static int
399{
400 return -1;
401}
402
403static int
405{
406 return -1;
407}
408
409static int
411{
412 return -1;
413}
414#endif /* PFMLIB_OLD_PFMV2 */
415
416int
418{
419 if (_pfmlib_major_version < 3)
420 return (int)syscall(PFM_pfm_load_context, fd, load);
421 return pfm_load_context_2v3(fd, load);
422}
423
424int
426{
427 if (_pfmlib_major_version < 3)
428 return (int)syscall(PFM_pfm_start, fd, start);
429 return pfm_start_2v3(fd, start);
430}
431
432int
433pfm_stop(int fd)
434{
435 if (_pfmlib_major_version < 3)
436 return (int)syscall(PFM_pfm_stop, fd);
437 return pfm_stop_2v3(fd);
438}
439
440int
442{
443 if (_pfmlib_major_version < 3)
444 return (int)syscall(PFM_pfm_restart, fd);
445 return pfm_restart_2v3(fd);
446}
447
448int
450{
451 if (_pfmlib_major_version < 3)
452 return (int)syscall(PFM_pfm_create_evtsets, fd, setd, count);
453 return pfm_create_evtsets_2v3(fd, setd, count);
454}
455
456int
458{
459 if (_pfmlib_major_version < 3)
460 return (int)syscall(PFM_pfm_delete_evtsets, fd, setd, count);
461 return pfm_delete_evtsets_2v3(fd, setd, count);
462}
463
464int
466{
467 if (_pfmlib_major_version < 3)
468 return (int)syscall(PFM_pfm_getinfo_evtsets, fd, info, count);
469
470 return pfm_getinfo_evtsets_2v3(fd, info, count);
471}
472
473int
475{
476 if (_pfmlib_major_version < 3)
477 return (int)syscall(PFM_pfm_unload_context, fd);
478
479 return pfm_unload_context_2v3(fd);
480}
481
482int
483pfm_create_context(pfarg_ctx_t *ctx, char *name, void *smpl_arg, size_t smpl_size)
484{
485 if (_pfmlib_major_version < 3) {
486 /*
487 * In perfmon v2.2, the pfm_create_context() call had a
488 * different return value. It used to return errno, in v2.3
489 * it returns the file descriptor.
490 */
491 if (_pfmlib_minor_version < 3) {
492 int r;
493 pfarg_ctx22_t ctx22;
494
495 /* transfer the v2.3 contents to v2.2 for sys call */
496 memset (&ctx22, 0, sizeof(ctx22));
497 if (name != NULL) {
498 memcpy (ctx22.ctx_smpl_buf_id, name, 16);
499 }
500 ctx22.ctx_flags = ctx->ctx_flags;
501 /* ctx22.ctx_fd returned */
502 /* ctx22.ctx_smpl_buf_size returned */
503 memcpy (ctx22.ctx_reserved3, &ctx->ctx_reserved1, 64);
504
505 r = syscall (PFM_pfm_create_context, &ctx22, smpl_arg, smpl_size);
506
507 /* transfer the v2.2 contents back to v2.3 */
508 ctx->ctx_flags = ctx22.ctx_flags;
509 memcpy (&ctx->ctx_reserved1, ctx22.ctx_reserved3, 64);
510
511 return (r < 0 ? r : ctx22.ctx_fd);
512 } else {
513 return (int)syscall(PFM_pfm_create_context, ctx, name, smpl_arg, smpl_size);
514 }
515 }
516 return pfm_create_context_2v3(ctx, name, smpl_arg, smpl_size);
517}
518
519int
521{
522 if (_pfmlib_major_version < 3)
523 return (int)syscall(PFM_pfm_write_pmcs, fd, pmcs, count);
524 return pfm_write_pmcs_2v3(fd, pmcs, count);
525}
526
527int
529{
530 if (_pfmlib_major_version < 3)
531 return (int)syscall(PFM_pfm_write_pmds, fd, pmds, count);
532 return pfm_write_pmds_2v3(fd, pmds, count);
533}
534
535int
536pfm_read_pmds(int fd, pfarg_pmd_t *pmds, int count)
537{
538 if (_pfmlib_major_version < 3)
539 return (int)syscall(PFM_pfm_read_pmds, fd, pmds, count);
540 return pfm_read_pmds_2v3(fd, pmds, count);
541}
542
543#ifdef __ia64__
544#define __PFMLIB_OS_COMPILE
545#include <perfmon/pfmlib.h>
546
547/*
548 * this is the old perfmon2 interface, maintained for backward
549 * compatibility reasons with older applications. This is for IA-64 ONLY.
550 */
551int
552perfmonctl(int fd, int cmd, void *arg, int narg)
553{
554 return syscall(__NR_perfmonctl, fd, cmd, arg, narg);
555}
556#endif /* __ia64__ */
int i
int close(int fd)
Definition: appio.c:179
int errno
static long count
static struct timeval start
os_err_t pfm_read(int fd, int flags, int type, void *reg, size_t n)
os_err_t pfm_set_state(int fd, int flags, int state)
os_err_t pfm_create(int flags, pfarg_sinfo_t *sif,...)
os_err_t pfm_write(int fd, int flags, int type, void *reg, size_t n)
os_err_t pfm_getinfo_sets(int fd, int flags, pfarg_set_info_t *s, size_t sz)
os_err_t pfm_attach(int fd, int flags, int target)
os_err_t pfm_create_sets(int fd, int flags, pfarg_set_desc_t *s, size_t sz)
static int skip
int perfmonctl(int fd, int cmd, void *arg, int narg)
int _pfmlib_major_version
int _pfmlib_minor_version
int pfm_load_context(int fd, pfarg_load_t *load)
static int pfm_load_context_2v3(int fd, pfarg_load_t *load)
#define PFM_pfm_start
int pfm_start(int fd, pfarg_start_t *start)
#define PFM_pfm_create_context
int pfm_create_evtsets(int fd, pfarg_setdesc_t *setd, int count)
#define PFM_pfm_read_pmds
static int pfm_getinfo_evtsets_2v3(int fd, pfarg_setinfo_t *info, int count)
static int pfm_read_pmds_2v3(int fd, pfarg_pmd_t *pmds, int count)
static int pfm_restart_2v3(int fd)
int pfm_read_pmds(int fd, pfarg_pmd_t *pmds, int count)
#define PFM_pfm_restart
int pfm_stop(int fd)
#define PFM_pfm_load_context
static int pfm_stop_2v3(int fd)
int pfm_getinfo_evtsets(int fd, pfarg_setinfo_t *info, int count)
static int pfm_create_evtsets_2v3(int fd, pfarg_setdesc_t *setd, int count)
#define PFM_pfm_unload_context
int pfm_delete_evtsets(int fd, pfarg_setdesc_t *setd, int count)
static int pfm_write_pmcs_2v3(int fd, pfarg_pmc_t *pmcs, int count)
static int pfm_unload_context_2v3(int fd)
int pfm_write_pmds(int fd, pfarg_pmd_t *pmds, int count)
int pfm_unload_context(int fd)
static int pfm_delete_evtsets_2v3(int fd, pfarg_setdesc_t *setd, int count)
#define PFM_pfm_delete_evtsets
int pfm_write_pmcs(int fd, pfarg_pmc_t *pmcs, int count)
#define PFM_pfm_write_pmcs
#define PFM_pfm_getinfo_evtsets
#define PFM_pfm_write_pmds
static int pfm_write_pmds_2v3(int fd, pfarg_pmd_t *pmds, int count)
static int pfm_start_2v3(int fd, pfarg_start_t *start)
#define PFM_pfm_create_evtsets
int pfm_create_context(pfarg_ctx_t *ctx, char *name, void *smpl_arg, size_t smpl_size)
#define PFM_pfm_stop
int pfm_restart(int fd)
static int pfm_create_context_2v3(pfarg_ctx_t *ctx, char *name, void *smpl_arg, size_t smpl_size)
void __pfm_vbprintf(const char *fmt,...)
Definition: pfmlib_priv.c:52
const char * name
Definition: rocs.c:225
uint64_t ctx_reserved3[12]
uint64_t ctx_smpl_buf_size
unsigned char ctx_smpl_buf_id[16]
uint32_t ctx_flags
Definition: perfmon_v2.h:19
uint32_t ctx_reserved1
Definition: perfmon_v2.h:20
uint32_t load_pid
Definition: perfmon_v2.h:69
uint32_t reg_flags
Definition: perfmon_v2.h:30
uint16_t reg_set
Definition: perfmon_v2.h:29
uint16_t reg_num
Definition: perfmon_v2.h:28
uint64_t reg_value
Definition: perfmon_v2.h:31
uint64_t reg_value
Definition: perfmon_v2.h:42
uint64_t reg_short_reset
Definition: perfmon_v2.h:44
uint64_t reg_ovfl_switch_cnt
Definition: perfmon_v2.h:46
uint32_t reg_flags
Definition: perfmon_v2.h:41
uint64_t reg_random_mask
Definition: perfmon_v2.h:50
uint64_t reg_last_reset_val
Definition: perfmon_v2.h:45
uint16_t reg_set
Definition: perfmon_v2.h:40
uint64_t reg_long_reset
Definition: perfmon_v2.h:43
uint16_t reg_num
Definition: perfmon_v2.h:39
uint64_t reg_smpl_eventid
Definition: perfmon_v2.h:49
uint64_t set_avail_pmcs[PFM_PMC_BV]
Definition: perfmon_v2.h:92
uint64_t set_avail_pmds[PFM_PMD_BV]
Definition: perfmon_v2.h:93
uint64_t set_act_duration
Definition: perfmon_v2.h:91
uint32_t set_flags
Definition: perfmon_v2.h:87
uint64_t set_runs
Definition: perfmon_v2.h:89
uint64_t set_timeout
Definition: perfmon_v2.h:90
uint64_t sif_avail_pmds[PFM_PMD_BV]
uint64_t sif_avail_pmcs[PFM_PMC_BV]