mirror of
git://sourceware.org/git/glibc.git
synced 2024-11-27 03:41:23 +08:00
11336c165c
Wed Oct 9 00:30:33 1996 Ulrich Drepper <drepper@cygnus.com> * inet/getnetgrent_r.c: Correct netgroup implementation. A specification of a netgroup can also name another netgroup. * nss/nss_files/files-netgrp.c: Likewise. * inet/netgroup.h: Add fields to hold additional information. Tue Oct 8 21:51:14 1996 Arnold D. Robbins <arnold@skeeve.atl.ga.us> * posix/getopt.c: Implement POSIX -W option handling. When the option string contains "W;" -W foo is handled like --foo. Tue Oct 8 12:27:26 1996 Ulrich Drepper <drepper@cygnus.com> * sysdeps/unix/sysv/linux/i386/clone.S: Define _ERRNO_H before inclusing <errnos.h> so we really get error symbols defined. * sysdeps/unix/sysv/linux/m68k/clone.S: Likewise. * sysdeps/unix/sysv/linux/alpha/clone.S: Likewise. * sunrpc/auth_unix.c (authunix_create_default): Don't use fixed size array for getgroups call. Instead get maximal number via sysconf. But discard list to NGRPS elements before calling XDR function since Sun's code cannot handle longer lists. Based on a patch by Thorsten Kukuk <kukuk@weber.uni-paderborn.de>. * sysdeps/stub/e_j0l.c: Define y0l as well. * sysdeps/stub/e_j1l.c: Define y1l as well. * sysdeps/stub/e_jnl.c: Define ynl as well. * posix/unistd.h: Correct prototype for execlp. * sysdeps/unix/sysv/linux/posix_opt.h: Define _POSIX_NO_TRUNC. Mon Oct 7 22:18:03 1996 Andreas Schwab <schwab@issan.informatik.uni-dortmund.de> * Makeconfig (sysdep-library-path): New variable. (built-program-cmd): Use it here to properly build a colon separated library path. Mon Oct 7 22:11:55 1996 Andreas Schwab <schwab@issan.informatik.uni-dortmund.de> * crypt/md5-crypt.c (md5_crypt_r): Add missing parens around & within comparison. Fix comments. (md5_crypt): Fix comment. Tue Oct 8 05:10:49 1996 Ulrich Drepper <drepper@cygnus.com> * sysdeps/unix/sysv/linux/errnos.h: Only include <linux/errnos.h> when _ERRNO_H is defined. [!_ERRNO_H && __need_Emath]: Define value for EDOM and ERANGE. Should there ever be a Linux port where the numeric values are not 33 and 34 this file must change. Mon Oct 7 13:54:04 1996 Ulrich Drepper <drepper@cygnus.com> * libio/iofgets.c (_IO_fgets): Use _IO_flockfile instead of __flockfile. Mon Oct 7 11:01:45 1996 Andreas Jaeger <aj@arthur.pfalz.de> * string/tst-strlen.c (main): Provide prototype. * malloc/mallocbug.c: Likewise. * io/test-utime.c: Likewise. * sysdeps/generic/crypt-entry.c: Correct typo. Mon Oct 7 13:42:20 1996 Ulrich Drepper <drepper@cygnus.com> * elf/dl-open.c (_dl_open): Check against _dl_sysdep_start to test for libc.a. Checking _DYNAMIC is of no worth here. Suggested by Andreas Schwab <schwab@issan.informatik.uni-dortmund.de>. * nss/nss_files/files-parse.c: Define LOOKUP_NAME even if EXTERN_PARSER is defined. Sat Oct 5 17:05:36 1996 Andreas Jaeger <aj@arthur.pfalz.de> * Rules (dep-dummy-lib): Correct prototype for __dummy__. * crypt/md5-crypt.c: Fix typos in comments. * gmon/gmon.c: Provide prototypes. * db/makedb.c: Likewise. * locale/programs/xmalloc.c: Likewise. * stdio-common/bug11.c (main): Likewise. * stdio-common/bug7.c (main): Likewise. * stdio-common/bug8.c (main): Likewise. * stdio-common/bug9.c (main): Likewise. * stdio-common/scanf1.c (main): Likewise. * stdio-common/scanf2.c (main): Likewise. * stdio-common/scanf5.c (main): Likewise. * stdio-common/scanf6.c (main): Likewise. * stdio-common/scanf7.c (main): Likewise. * stdio-common/scanf8.c (main): Likewise. * stdio-common/temptest.c (main): Likewise. * stdio-common/test-fwrite.c (main): Likewise. * stdio-common/tst-printf.c: Likewise. * stdio-common/tstdiomisc.c: Likewise. * stdio-common/tstgetln.c (main): Likewise. * stdlib/testmb.c (main): Likewise. * stdio-common/scanf4.c (main): Correct prototype, remove unused variable n. * stdio-common/scanf3.c (main): Correct prototype, change declaration of s to reduce warnings. * stdio-common/bug10.c (main): Likewise. * stdio-common/tfformat.c: Provide prototype, remove unused reference to dump_stats, add braces in sprint_doubles. * stdio-common/tiformat.c: Likewise. * stdio-common/test_rdwr.c (main): Parameter `where' is long in printf call. Mon Oct 7 14:04:26 1996 NIIBE Yutaka <gniibe@mri.co.jp> * sysdeps/unix/sysv/linux/sys/soundcard.h: Fix typo. * shadow/putspent.c: Don't write colon after flag field. * sysdeps/unix/sysv/linux/net/if_ppp.h: New file. Wrapper around kernel header. * sysdeps/unix/sysv/linux/net/ppp-comp.h: Likewise. * sysdeps/unix/sysv/linux/net/ppp_defs.h: Likewise. * sysdeps/unix/sysv/linux/Dist: Mention new files. Mon Oct 7 00:58:19 1996 Ulrich Drepper <drepper@cygnus.com> * sysdeps/i386/i586/strchr.S: Correct handling of first bytes to get pointer aligned. Reported by Matthias Urlichs <smurf@noris.de>. * sysdeps/posix/euidaccess.c [_LIBC]: Avoid calling get?id functions by using __libc_enable_secure variable. * sysdeps/libm-i387/s_copysignl.S: Correct loading of return value.
277 lines
7.4 KiB
C
277 lines
7.4 KiB
C
/*-
|
|
* Copyright (c) 1983, 1992, 1993
|
|
* The Regents of the University of California. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by the University of
|
|
* California, Berkeley and its contributors.
|
|
* 4. Neither the name of the University nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
#include <sys/param.h>
|
|
#include <sys/time.h>
|
|
#include <sys/gmon.h>
|
|
#include <sys/gmon_out.h>
|
|
|
|
#include <stdio.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
|
|
extern int __profile_frequency __P ((void));
|
|
|
|
struct __bb *__bb_head; /* Head of basic-block list or NULL. */
|
|
|
|
struct gmonparam _gmonparam = { GMON_PROF_OFF };
|
|
|
|
/*
|
|
* See profil(2) where this is described:
|
|
*/
|
|
static int s_scale;
|
|
#define SCALE_1_TO_1 0x10000L
|
|
|
|
#define ERR(s) write(2, s, sizeof(s) - 1)
|
|
|
|
void moncontrol __P ((int mode));
|
|
static void write_hist __P ((int fd));
|
|
static void write_call_graph __P ((int fd));
|
|
static void write_bb_counts __P ((int fd));
|
|
|
|
/*
|
|
* Control profiling
|
|
* profiling is what mcount checks to see if
|
|
* all the data structures are ready.
|
|
*/
|
|
void
|
|
moncontrol (mode)
|
|
int mode;
|
|
{
|
|
struct gmonparam *p = &_gmonparam;
|
|
|
|
if (mode)
|
|
{
|
|
/* start */
|
|
profil((void *) p->kcount, p->kcountsize, p->lowpc, s_scale);
|
|
p->state = GMON_PROF_ON;
|
|
}
|
|
else
|
|
{
|
|
/* stop */
|
|
profil((void *) 0, 0, 0, 0);
|
|
p->state = GMON_PROF_OFF;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
monstartup (lowpc, highpc)
|
|
u_long lowpc;
|
|
u_long highpc;
|
|
{
|
|
register int o;
|
|
char *cp;
|
|
struct gmonparam *p = &_gmonparam;
|
|
|
|
/*
|
|
* round lowpc and highpc to multiples of the density we're using
|
|
* so the rest of the scaling (here and in gprof) stays in ints.
|
|
*/
|
|
p->lowpc = ROUNDDOWN(lowpc, HISTFRACTION * sizeof(HISTCOUNTER));
|
|
p->highpc = ROUNDUP(highpc, HISTFRACTION * sizeof(HISTCOUNTER));
|
|
p->textsize = p->highpc - p->lowpc;
|
|
p->kcountsize = p->textsize / HISTFRACTION;
|
|
p->hashfraction = HASHFRACTION;
|
|
p->log_hashfraction = -1;
|
|
if ((HASHFRACTION & (HASHFRACTION - 1)) == 0) {
|
|
/* if HASHFRACTION is a power of two, mcount can use shifting
|
|
instead of integer division. Precompute shift amount. */
|
|
p->log_hashfraction = ffs(p->hashfraction * sizeof(*p->froms)) - 1;
|
|
}
|
|
p->fromssize = p->textsize / HASHFRACTION;
|
|
p->tolimit = p->textsize * ARCDENSITY / 100;
|
|
if (p->tolimit < MINARCS)
|
|
p->tolimit = MINARCS;
|
|
else if (p->tolimit > MAXARCS)
|
|
p->tolimit = MAXARCS;
|
|
p->tossize = p->tolimit * sizeof(struct tostruct);
|
|
|
|
cp = malloc (p->kcountsize + p->fromssize + p->tossize);
|
|
if (! cp)
|
|
{
|
|
ERR("monstartup: out of memory\n");
|
|
return;
|
|
}
|
|
bzero(cp, p->kcountsize + p->fromssize + p->tossize);
|
|
p->tos = (struct tostruct *)cp;
|
|
cp += p->tossize;
|
|
p->kcount = (u_short *)cp;
|
|
cp += p->kcountsize;
|
|
p->froms = (u_short *)cp;
|
|
|
|
p->tos[0].link = 0;
|
|
|
|
o = p->highpc - p->lowpc;
|
|
if (p->kcountsize < (u_long) o)
|
|
{
|
|
#ifndef hp300
|
|
s_scale = ((float)p->kcountsize / o ) * SCALE_1_TO_1;
|
|
#else
|
|
/* avoid floating point operations */
|
|
int quot = o / p->kcountsize;
|
|
|
|
if (quot >= 0x10000)
|
|
s_scale = 1;
|
|
else if (quot >= 0x100)
|
|
s_scale = 0x10000 / quot;
|
|
else if (o >= 0x800000)
|
|
s_scale = 0x1000000 / (o / (p->kcountsize >> 8));
|
|
else
|
|
s_scale = 0x1000000 / ((o << 8) / p->kcountsize);
|
|
#endif
|
|
} else
|
|
s_scale = SCALE_1_TO_1;
|
|
|
|
moncontrol(1);
|
|
}
|
|
|
|
|
|
static void
|
|
write_hist (fd)
|
|
int fd;
|
|
{
|
|
const u_char tag = GMON_TAG_TIME_HIST;
|
|
struct gmon_hist_hdr thdr;
|
|
|
|
if (_gmonparam.kcountsize > 0)
|
|
{
|
|
thdr.low_pc = _gmonparam.lowpc;
|
|
thdr.high_pc = _gmonparam.highpc;
|
|
thdr.hist_size = _gmonparam.kcountsize / sizeof(HISTCOUNTER);
|
|
thdr.prof_rate = __profile_frequency();
|
|
strncpy(thdr.dimen, "seconds", sizeof(thdr.dimen));
|
|
thdr.dimen_abbrev = 's';
|
|
|
|
write(fd, &tag, sizeof(tag));
|
|
write(fd, &thdr, sizeof(thdr));
|
|
write(fd, _gmonparam.kcount, _gmonparam.kcountsize);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
write_call_graph (fd)
|
|
int fd;
|
|
{
|
|
const u_char tag = GMON_TAG_CG_ARC;
|
|
struct gmon_cg_arc_record raw_arc;
|
|
int from_index, to_index, from_len;
|
|
u_long frompc;
|
|
|
|
from_len = _gmonparam.fromssize / sizeof(*_gmonparam.froms);
|
|
for (from_index = 0; from_index < from_len; ++from_index)
|
|
{
|
|
if (_gmonparam.froms[from_index] == 0)
|
|
continue;
|
|
|
|
frompc = _gmonparam.lowpc;
|
|
frompc += (from_index * _gmonparam.hashfraction
|
|
* sizeof(*_gmonparam.froms));
|
|
for (to_index = _gmonparam.froms[from_index];
|
|
to_index != 0;
|
|
to_index = _gmonparam.tos[to_index].link)
|
|
{
|
|
raw_arc.from_pc = frompc;
|
|
raw_arc.self_pc = _gmonparam.tos[to_index].selfpc;
|
|
raw_arc.count = _gmonparam.tos[to_index].count;
|
|
|
|
write(fd, &tag, sizeof(tag));
|
|
write(fd, &raw_arc, sizeof(raw_arc));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
write_bb_counts (fd)
|
|
int fd;
|
|
{
|
|
struct __bb *grp;
|
|
const u_char tag = GMON_TAG_BB_COUNT;
|
|
int ncounts;
|
|
int i;
|
|
|
|
/* Write each group of basic-block info (all basic-blocks in a
|
|
compilation unit form a single group). */
|
|
|
|
for (grp = __bb_head; grp; grp = grp->next)
|
|
{
|
|
ncounts = grp->ncounts;
|
|
write(fd, &tag, sizeof(tag));
|
|
write(fd, &ncounts, sizeof(ncounts));
|
|
for (i = 0; i < ncounts; ++i)
|
|
{
|
|
write(fd, &grp->addresses[i], sizeof(grp->addresses[0]));
|
|
write(fd, &grp->counts[i], sizeof(grp->counts[0]));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
_mcleanup ()
|
|
{
|
|
struct gmon_hdr ghdr;
|
|
int fd;
|
|
|
|
moncontrol(0);
|
|
fd = open("gmon.out", O_CREAT|O_TRUNC|O_WRONLY, 0666);
|
|
if (fd < 0)
|
|
{
|
|
perror("_mcleanup: gmon.out");
|
|
return;
|
|
}
|
|
|
|
/* write gmon.out header: */
|
|
memset(&ghdr, 0, sizeof(ghdr));
|
|
memcpy(&ghdr.cookie[0], GMON_MAGIC, sizeof(ghdr.cookie));
|
|
ghdr.version = GMON_VERSION;
|
|
write(fd, &ghdr, sizeof(ghdr));
|
|
|
|
/* write PC histogram: */
|
|
write_hist(fd);
|
|
|
|
/* write call-graph: */
|
|
write_call_graph(fd);
|
|
|
|
/* write basic-block execution counts: */
|
|
write_bb_counts(fd);
|
|
|
|
close(fd);
|
|
}
|