summaryrefslogtreecommitdiffstats
path: root/mdk-stage1/dietlibc/profiling/profil.c
blob: fa9a0ef2b507fcb1c4ece3c4c32f46c51e6cc541 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
/******************************************************
  Copyright (C) 2001, 2002 Thomas M. Ogrisegg

   This is free software. You can redistribute and modify
   it under the terms of the GNU General Public License.

   This file is part of the profiling support for dietlibc

   profil (3) generic implementation 

 *************************************************************/

#include <asm/sigcontext.h>
#include <sys/types.h>
#include <sys/time.h>
#include <unistd.h>
#include <signal.h>

#define SHORT_SIZE sizeof (short)
#define MAX_SHORT 65536

#ifdef DEBUG
# include <stdio.h>
# define debug printf
#else
# define debug
#endif

#ifndef u_short
# define u_short unsigned short
#endif

#ifndef u_int
# define u_int unsigned int
#endif

static unsigned short *buffer = NULL;
static size_t maxhits  = 0;
static unsigned long low_pc = 0;
static unsigned long pscale = 0;

/* profiler - helper function for profil(3) */
static void
profiler (int signal, struct sigcontext ctx)
{
	size_t s = PC(ctx)-low_pc;
	(void)signal;
	if ((PC(ctx)) < low_pc) return;
	s >>= 1;
	if (s < maxhits)
		++buffer[s];
}

/* profil(3) - start or stop the profiling timer */
int
profil (u_short *buf, size_t bufsiz, size_t offset, u_int scale)
{
	struct itimerval itv = { { 0, 1 }, { 0, 1 } };
	struct sigaction sa;
	if (!buf) {
		sigaction (SIGPROF, NULL, NULL);
		setitimer (ITIMER_PROF, NULL, NULL);
		return (0);
	}
	sa.sa_handler = (sighandler_t)&profiler;
	sa.sa_flags   = SA_RESTART;
	sigfillset (&sa.sa_mask);
	sigaction (SIGPROF, &sa, NULL);
	pscale = scale;
	buffer = buf;
	low_pc = offset;
	maxhits = bufsiz/SHORT_SIZE;

	return (setitimer (ITIMER_PROF, &itv, &itv));
}