diff options
Diffstat (limited to 'qa/src/churnctx.c')
-rw-r--r-- | qa/src/churnctx.c | 429 |
1 files changed, 429 insertions, 0 deletions
diff --git a/qa/src/churnctx.c b/qa/src/churnctx.c new file mode 100644 index 0000000..bfedad7 --- /dev/null +++ b/qa/src/churnctx.c @@ -0,0 +1,429 @@ +/* + * Copyright (c) 1995-2001 Silicon Graphics, Inc. All Rights Reserved. + * Copyright (c) 2014 Ken McDonell. All Rights Reserved. + * + * Churn a context ... looking for memory leaks a la + * http://oss.sgi.com/bugzilla/show_bug.cgi?id=1057 + */ + +#include <pcp/pmapi.h> +#include <pcp/impl.h> + +#define BUILD_STANDALONE 1 + +int +main(int argc, char **argv) +{ + int c; + int sts; + int ctx; + int dupctx = 0; + int nmetric; + int iter; + int errflag = 0; + int type = 0; + char *host = NULL; /* pander to gcc */ + int mode = PM_MODE_INTERP; /* mode for archives */ + char *configfile = NULL; + char *start = NULL; + char *finish = NULL; + char *align = NULL; + char *offset = NULL; + char *logfile = NULL; + pmLogLabel label; /* get hostname for archives */ + int zflag = 0; /* for -z */ + char *tz = NULL; /* for -Z timezone */ + int tzh; /* initial timezone handle */ + char local[MAXHOSTNAMELEN]; + char *pmnsfile = PM_NS_DEFAULT; + int samples = 1; + char *endnum; + char **name = NULL; + pmID *pmid = NULL; + pmResult *rp; + char *highwater = NULL; + struct timeval delta = { 15, 0 }; + struct timeval startTime; + struct timeval endTime; + struct timeval appStart; + struct timeval appEnd; + struct timeval appOffset; + + /* trim cmd name of leading directory components */ + __pmSetProgname(argv[0]); + + setvbuf(stdout, NULL, _IONBF, 0); + setvbuf(stderr, NULL, _IONBF, 0); + + while ((c = getopt(argc, argv, "a:A:c:D:dh:l:Ln:O:s:S:t:T:U:zZ:?")) != EOF) { + switch (c) { + + case 'a': /* archive name */ + if (type != 0) { +#ifdef BUILD_STANDALONE + fprintf(stderr, "%s: at most one of -a, -h and -L allowed\n", pmProgname); +#else + fprintf(stderr, "%s: at most one of -a and -h allowed\n", pmProgname); +#endif + errflag++; + } + type = PM_CONTEXT_ARCHIVE; + host = optarg; + break; + + case 'A': /* time alignment */ + align = optarg; + break; + + case 'c': /* configfile */ + if (configfile != NULL) { + fprintf(stderr, "%s: at most one -c option allowed\n", pmProgname); + errflag++; + } + configfile = optarg; + break; + + case 'D': /* debug flag */ + sts = __pmParseDebug(optarg); + if (sts < 0) { + fprintf(stderr, "%s: unrecognized debug flag specification (%s)\n", + pmProgname, optarg); + errflag++; + } + else + pmDebug |= sts; + break; + + case 'd': /* dup context rather than new context */ + dupctx = 1; + break; + + case 'h': /* contact PMCD on this hostname */ + if (type != 0) { +#ifdef BUILD_STANDALONE + fprintf(stderr, "%s: at most one of -a, -h and -L allowed\n", pmProgname); +#else + fprintf(stderr, "%s: at most one of -a and -h allowed\n", pmProgname); +#endif + errflag++; + } + host = optarg; + type = PM_CONTEXT_HOST; + break; + +#ifdef BUILD_STANDALONE + case 'L': /* LOCAL, no PMCD */ + if (type != 0) { + fprintf(stderr, "%s: at most one of -a, -h, -L and -U allowed\n", pmProgname); + errflag++; + } + host = NULL; + type = PM_CONTEXT_LOCAL; + putenv("PMDA_LOCAL_PROC="); /* if proc PMDA needed */ + putenv("PMDA_LOCAL_SAMPLE="); /* if sampledso PMDA needed */ + break; +#endif + + case 'l': /* logfile */ + logfile = optarg; + break; + + case 'n': /* alternative name space file */ + pmnsfile = optarg; + break; + + case 'O': /* sample offset time */ + offset = optarg; + break; + + case 's': /* sample count */ + samples = (int)strtol(optarg, &endnum, 10); + if (*endnum != '\0' || samples < 0) { + fprintf(stderr, "%s: -s requires numeric argument\n", pmProgname); + errflag++; + } + break; + + case 'S': /* start time */ + start = optarg; + break; + + case 't': /* change update interval */ + if (pmParseInterval(optarg, &delta, &endnum) < 0) { + fprintf(stderr, "%s: illegal -t argument\n", pmProgname); + fputs(endnum, stderr); + free(endnum); + errflag++; + } + break; + + case 'T': /* terminate time */ + finish = optarg; + break; + + case 'U': /* uninterpolated archive log */ + if (type != 0) { +#ifdef BUILD_STANDALONE + fprintf(stderr, "%s: at most one of -a, -h, -L and -U allowed\n", pmProgname); +#else + fprintf(stderr, "%s: at most one of -a, -h and -U allowed\n", pmProgname); +#endif + errflag++; + } + type = PM_CONTEXT_ARCHIVE; + mode = PM_MODE_FORW; + host = optarg; + break; + + case 'z': /* timezone from host */ + if (tz != NULL) { + fprintf(stderr, "%s: at most one of -Z and/or -z allowed\n", pmProgname); + errflag++; + } + zflag++; + break; + + case 'Z': /* $TZ timezone */ + if (zflag) { + fprintf(stderr, "%s: at most one of -Z and/or -z allowed\n", pmProgname); + errflag++; + } + tz = optarg; + break; + + case '?': + default: + errflag++; + break; + } + } + + if (zflag && type == 0) { + fprintf(stderr, "%s: -z requires an explicit -a, -h or -U option\n", pmProgname); + errflag++; + } + + if (errflag) { + fprintf(stderr, +"Usage: %s [options] [metrics ...]\n\ +\n\ +Options:\n\ + -a archive metrics source is a PCP log archive\n\ + -A align align sample times on natural boundaries\n\ + -c configfile file to load configuration from\n\ + -d use pmDupContext [default: pmNewContext]\n\ + -h host metrics source is PMCD on host\n\ + -l logfile redirect diagnostics and trace output\n" +#ifdef BUILD_STANDALONE +" -L use local context instead of PMCD\n" +#endif +" -n pmnsfile use an alternative PMNS\n\ + -O offset initial offset into the time window\n\ + -s samples terminate after this many iterations [default 1]\n\ + -S starttime start of the time window\n\ + -t interval sample interval [default 15.0 seconds]\n\ + -T endtime end of the time window\n\ + -z set reporting timezone to local time of metrics source\n\ + -Z timezone set reporting timezone\n", + pmProgname); + exit(1); + } + + if (logfile != NULL) { + __pmOpenLog(pmProgname, logfile, stderr, &sts); + if (sts < 0) { + fprintf(stderr, "%s: Could not open logfile \"%s\"\n", pmProgname, logfile); + } + } + + if (pmnsfile != PM_NS_DEFAULT && (sts = pmLoadNameSpace(pmnsfile)) < 0) { + printf("%s: Cannot load namespace from \"%s\": %s\n", pmProgname, + pmnsfile, pmErrStr(sts)); + exit(1); + } + + if (type == 0) { + type = PM_CONTEXT_HOST; + (void)gethostname(local, MAXHOSTNAMELEN); + local[MAXHOSTNAMELEN-1] = '\0'; + host = local; + } + if ((ctx = pmNewContext(type, host)) < 0) { + if (type == PM_CONTEXT_HOST) + fprintf(stderr, "%s: Cannot connect to PMCD on host \"%s\": %s\n", + pmProgname, host, pmErrStr(ctx)); + else + fprintf(stderr, "%s: Cannot open archive \"%s\": %s\n", + pmProgname, host, pmErrStr(ctx)); + exit(1); + } + + if (type == PM_CONTEXT_ARCHIVE) { + if ((sts = pmGetArchiveLabel(&label)) < 0) { + fprintf(stderr, "%s: Cannot get archive label record: %s\n", + pmProgname, pmErrStr(sts)); + exit(1); + } + startTime = label.ll_start; + if ((sts = pmGetArchiveEnd(&endTime)) < 0) { + endTime.tv_sec = INT_MAX; + endTime.tv_usec = 0; + fflush(stdout); + fprintf(stderr, "%s: Cannot locate end of archive: %s\n", + pmProgname, pmErrStr(sts)); + fprintf(stderr, "\nWARNING: This archive is sufficiently damaged that it may not be possible to\n"); + fprintf(stderr, " produce complete information. Continuing and hoping for the best.\n\n"); + fflush(stderr); + } + } + else { + gettimeofday(&startTime, NULL); + endTime.tv_sec = INT_MAX; + } + + if (zflag) { + if ((tzh = pmNewContextZone()) < 0) { + fprintf(stderr, "%s: Cannot set context timezone: %s\n", + pmProgname, pmErrStr(tzh)); + exit(1); + } + if (type == PM_CONTEXT_ARCHIVE) + printf("Note: timezone set to local timezone of host \"%s\" from archive\n\n", + label.ll_hostname); + else + printf("Note: timezone set to local timezone of host \"%s\"\n\n", host); + } + else if (tz != NULL) { + if ((tzh = pmNewZone(tz)) < 0) { + fprintf(stderr, "%s: Cannot set timezone to \"%s\": %s\n", + pmProgname, tz, pmErrStr(tzh)); + exit(1); + } + printf("Note: timezone set to \"TZ=%s\"\n\n", tz); + } + + /* non-flag args are argv[optind] ... argv[argc-1] */ + nmetric = 0; + while (optind < argc) { + nmetric++; + if ((name = (char **)realloc(name, nmetric*sizeof(name[0]))) == NULL) { + fprintf(stderr, "name[%d] malloc failed @ %s\n", nmetric-1, argv[optind]); + exit(1); + } + name[nmetric-1] = argv[optind]; + if ((pmid = (pmID *)realloc(pmid, nmetric*sizeof(pmid[0]))) == NULL) { + fprintf(stderr, "pmid[%d] malloc failed @ %s\n", nmetric-1, argv[optind]); + exit(1); + } + if ((sts = pmLookupName(1, &name[nmetric-1], &pmid[nmetric-1])) < 0) { + fprintf(stderr, "Warning: pmLookupName(\"%s\",...) failed: %s\n", name[nmetric-1], pmErrStr(sts)); + } + optind++; + } + + if (align != NULL && type != PM_CONTEXT_ARCHIVE) { + fprintf(stderr, "%s: -A option only supported for PCP archives, alignment request ignored\n", + pmProgname); + align = NULL; + } + + sts = pmParseTimeWindow(start, finish, align, offset, &startTime, + &endTime, &appStart, &appEnd, &appOffset, + &endnum); + if (sts < 0) { + fprintf(stderr, "%s: illegal time window specification\n%s", pmProgname, endnum); + exit(1); + } + + iter = 1; + while (samples == -1 || samples-- > 0) { + int new_ctx; + char *check; + + if (dupctx) { + if ((new_ctx = pmDupContext()) < 0) { + fprintf(stderr, "%s: pmDupContext failed: %s\n", pmProgname, pmErrStr(new_ctx)); + exit(1); + } + } + else { + if ((new_ctx = pmNewContext(type, host)) < 0) { + fprintf(stderr, "%s: pmNewContext failed: %s\n", pmProgname, pmErrStr(new_ctx)); + exit(1); + } + + } + + if ((sts = pmDestroyContext(ctx)) < 0) { + fprintf(stderr, "%s: pmDestroyContex failed: %s\n", pmProgname, pmErrStr(sts)); + exit(1); + } + + if ((sts = pmUseContext(new_ctx)) < 0) { + fprintf(stderr, "%s: pmUseContext(%d) failed: %s\n", pmProgname, new_ctx, pmErrStr(sts)); + exit(1); + } + ctx = new_ctx; + /* dump out PDU buffer pool state */ + __pmFindPDUBuf(-1); + /* check for outrageous memory leaks */ + check = (char *)sbrk(0); + if (highwater != NULL) { + if (check - highwater > 4096) { + printf("Memory growth (iteration %d): %ld\n", iter, (long)(check - highwater)); + highwater = check; + } + } + else + highwater = check; + + if (type == PM_CONTEXT_ARCHIVE) { + if (mode == PM_MODE_INTERP) { + int delta_msec; + delta_msec = delta.tv_sec*1000 + delta.tv_usec/1000; + if ((sts = pmSetMode(mode, &appStart, delta_msec)) < 0) { + fprintf(stderr, "%s: pmSetMode: %s\n", pmProgname, pmErrStr(sts)); + exit(1); + } + } + appStart.tv_sec += delta.tv_sec; + appStart.tv_usec += delta.tv_usec; + if (appStart.tv_usec > 1000000) { + appStart.tv_usec -= 1000000; + appStart.tv_sec++; + } + } + + if (nmetric > 0) { + if ((sts = pmFetch(nmetric, pmid, &rp)) < 0) { + fprintf(stderr, "%s: pmFetch failed: %s\n", pmProgname, pmErrStr(sts)); + exit(1); + } + else { + int i; + char now[26]; + time_t stamp; + stamp = rp->timestamp.tv_sec; + printf("%s", pmCtime(&stamp, now)); + for (i = 0; i < nmetric; i++) { + printf("%s: %d values\n", pmIDStr(rp->vset[i]->pmid), rp->vset[i]->numval); + } + pmFreeResult(rp); + } + + if (type != PM_CONTEXT_ARCHIVE) { + __pmtimevalSleep(delta); + } + } + else + printf("Nothing to be fetched\n"); + iter++; + } + + if ((sts = pmDestroyContext(ctx)) < 0) { + fprintf(stderr, "%s: final pmDestroyContex failed: %s\n", pmProgname, pmErrStr(sts)); + exit(1); + } + + return 0; +} |