476 lines
12 KiB
C
476 lines
12 KiB
C
|
/*
|
||
|
*------------------------------------------------------------------
|
||
|
* Copyright (c) 2005-2016 Cisco and/or its affiliates.
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
* you may not use this file except in compliance with the License.
|
||
|
* You may obtain a copy of the License at:
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
*/
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
#include <unistd.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <sys/fcntl.h>
|
||
|
#include <sys/mman.h>
|
||
|
#include <arpa/inet.h>
|
||
|
#include <stdio.h>
|
||
|
#include <gtk/gtk.h>
|
||
|
#include "g2.h"
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
/*
|
||
|
* globals
|
||
|
*/
|
||
|
boolean g_little_endian;
|
||
|
event_t *g_events;
|
||
|
ulong g_nevents;
|
||
|
pid_sort_t *g_pids;
|
||
|
pid_sort_t *g_original_pids;
|
||
|
int g_npids;
|
||
|
pid_data_t *g_pid_data_list;
|
||
|
|
||
|
/*
|
||
|
* locals
|
||
|
*/
|
||
|
pid_data_t **s_pidhash;
|
||
|
|
||
|
/*
|
||
|
* config parameters
|
||
|
*/
|
||
|
|
||
|
double ticks_per_ns=1000.0;
|
||
|
boolean ticks_per_ns_set;
|
||
|
|
||
|
/****************************************************************************
|
||
|
* event_init
|
||
|
****************************************************************************/
|
||
|
|
||
|
void event_init(void)
|
||
|
{
|
||
|
ulong endian;
|
||
|
char *ep;
|
||
|
char *askstr;
|
||
|
int tmp;
|
||
|
|
||
|
ep = (char *)&endian;
|
||
|
endian = 0x12345678;
|
||
|
if (*ep != 0x12)
|
||
|
g_little_endian = TRUE;
|
||
|
else
|
||
|
g_little_endian = FALSE;
|
||
|
|
||
|
askstr = getprop("dont_ask_ticks_per_ns_initially");
|
||
|
|
||
|
if (askstr && (*askstr == 't' || *askstr == 'T')) {
|
||
|
tmp = atol(getprop_default("ticks_per_ns", 0));
|
||
|
if (tmp > 0) {
|
||
|
ticks_per_ns = tmp;
|
||
|
ticks_per_ns_set = TRUE;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* find_or_add_pid
|
||
|
****************************************************************************/
|
||
|
|
||
|
pid_data_t *find_or_add_pid (ulong pid)
|
||
|
{
|
||
|
pid_data_t *pp;
|
||
|
ulong bucket;
|
||
|
|
||
|
bucket = pid % PIDHASH_NBUCKETS;
|
||
|
|
||
|
pp = s_pidhash[bucket];
|
||
|
|
||
|
if (pp == 0) {
|
||
|
pp = g_malloc0(sizeof(pid_data_t));
|
||
|
pp->pid_value = pid;
|
||
|
s_pidhash[bucket] = pp;
|
||
|
g_npids++;
|
||
|
return(pp);
|
||
|
}
|
||
|
while (pp) {
|
||
|
if (pp->pid_value == pid)
|
||
|
return(pp);
|
||
|
pp = pp->next;
|
||
|
}
|
||
|
|
||
|
pp = g_malloc0(sizeof(pid_data_t));
|
||
|
pp->pid_value = pid;
|
||
|
pp->next = s_pidhash[bucket];
|
||
|
s_pidhash[bucket] = pp;
|
||
|
g_npids++;
|
||
|
return(pp);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* pid_cmp
|
||
|
****************************************************************************/
|
||
|
|
||
|
int pid_cmp(const void *a1, const void *a2)
|
||
|
{
|
||
|
pid_sort_t *p1 = (pid_sort_t *)a1;
|
||
|
pid_sort_t *p2 = (pid_sort_t *)a2;
|
||
|
|
||
|
if (p1->pid_value < p2->pid_value)
|
||
|
return(-1);
|
||
|
else if (p1->pid_value == p2->pid_value)
|
||
|
return(0);
|
||
|
else
|
||
|
return(1);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* make_sorted_pid_vector
|
||
|
****************************************************************************/
|
||
|
|
||
|
static void make_sorted_pid_vector(void)
|
||
|
{
|
||
|
pid_data_t *pp;
|
||
|
pid_data_t **p_previous;
|
||
|
pid_sort_t *psp;
|
||
|
int i;
|
||
|
|
||
|
psp = g_pids = g_malloc(sizeof(pid_sort_t)*g_npids);
|
||
|
|
||
|
for (i = 0; i < PIDHASH_NBUCKETS; i++) {
|
||
|
pp = s_pidhash[i];
|
||
|
while(pp) {
|
||
|
psp->pid = pp;
|
||
|
psp->pid_value = pp->pid_value;
|
||
|
psp++;
|
||
|
pp = pp->next;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
qsort(&g_pids[0], g_npids, sizeof(pid_sort_t), pid_cmp);
|
||
|
|
||
|
/* put the sort order into the pid objects */
|
||
|
psp = g_pids;
|
||
|
|
||
|
/*
|
||
|
* This is rather gross.
|
||
|
*
|
||
|
* We happen to know that whenever this function is called, the hash table
|
||
|
* structure itself is immediately torn down. So the "next" pointers in the
|
||
|
* pid_data_t elements are about to become useless.
|
||
|
*
|
||
|
* So we re-use them, to link all the pid_data_t elements together into a
|
||
|
* single unified linked list, with g_pid_data_list pointing to the head.
|
||
|
* This means we can walk all the pid_data_t objects if we really want to.
|
||
|
* Reading snapshots from disk is one example.
|
||
|
*
|
||
|
* Alternatively we could just leave the hash table in place; this is
|
||
|
* far nicer, but as it happens, trading O(n) lookups for O(1) lookups
|
||
|
* isn't actually a problem for the restricted post-tear-down usage. So for
|
||
|
* now we take the memory savings and swap our hash table for a list.
|
||
|
*/
|
||
|
p_previous = &g_pid_data_list;
|
||
|
for (i = 0; i < g_npids; i++) {
|
||
|
pp = psp->pid;
|
||
|
pp->pid_index = i;
|
||
|
*p_previous = pp;
|
||
|
p_previous = &pp->next;
|
||
|
psp++;
|
||
|
}
|
||
|
*p_previous = NULL;
|
||
|
|
||
|
/*
|
||
|
* Squirrel away original (sorted) vector, so we can
|
||
|
* toggle between "chase" mode, snapshots, and the original
|
||
|
* display method on short notice
|
||
|
*/
|
||
|
g_original_pids = g_malloc(sizeof(pid_sort_t)*g_npids);
|
||
|
memcpy (g_original_pids, g_pids, sizeof(pid_sort_t)*g_npids);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* read_events
|
||
|
****************************************************************************/
|
||
|
|
||
|
void read_events(char *filename)
|
||
|
{
|
||
|
ulong *ulp;
|
||
|
ulong size;
|
||
|
event_t *ep;
|
||
|
raw_event_t *rep;
|
||
|
ulonglong start_time=0ULL;
|
||
|
ulonglong low_time;
|
||
|
boolean once=TRUE;
|
||
|
int i;
|
||
|
char tmpbuf [128];
|
||
|
|
||
|
ulp = (ulong *)mapfile(filename, &size);
|
||
|
|
||
|
if (ulp == NULL) {
|
||
|
sprintf(tmpbuf, "Couldn't open %s\n", filename);
|
||
|
infobox("Read Event Log Failure", tmpbuf);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
g_nevents = ntohl(*ulp);
|
||
|
|
||
|
if (size != (g_nevents*sizeof(raw_event_t) + sizeof(g_nevents))) {
|
||
|
sprintf(tmpbuf, "%s was damaged, or isn't an event log.\n", filename);
|
||
|
infobox("Bad Input File", tmpbuf);
|
||
|
g_nevents = 0;
|
||
|
unmapfile((char *)ulp, size);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
rep = (raw_event_t *)(ulp+1);
|
||
|
|
||
|
if (g_events)
|
||
|
g_free(g_events);
|
||
|
|
||
|
g_events = (event_t *)g_malloc(g_nevents * sizeof(event_t));
|
||
|
ep = g_events;
|
||
|
|
||
|
while (g_npids > 0) {
|
||
|
g_free((g_pids + g_npids-1)->pid);
|
||
|
g_npids--;
|
||
|
}
|
||
|
if (g_pids) {
|
||
|
g_free(g_pids);
|
||
|
g_free(g_original_pids);
|
||
|
g_pids = 0;
|
||
|
g_original_pids = 0;
|
||
|
}
|
||
|
|
||
|
s_pidhash = (pid_data_t **)g_malloc0(
|
||
|
PIDHASH_NBUCKETS*sizeof(pid_data_t *));
|
||
|
|
||
|
/* $$$ add a SEGV handler... */
|
||
|
for (i = 0; i < g_nevents; i++) {
|
||
|
if (once) {
|
||
|
once = FALSE;
|
||
|
start_time = ((ulonglong)ntohl(rep->time[0]));
|
||
|
start_time <<= 32;
|
||
|
low_time = ntohl(rep->time[1]);
|
||
|
low_time &= 0xFFFFFFFF;
|
||
|
start_time |= low_time;
|
||
|
ep->time = 0LL;
|
||
|
} else {
|
||
|
ep->time = ((ulonglong)ntohl(rep->time[0]));
|
||
|
ep->time <<= 32;
|
||
|
low_time = ntohl(rep->time[1]);
|
||
|
low_time &= 0xFFFFFFFF;
|
||
|
ep->time |= low_time;
|
||
|
ep->time -= start_time;
|
||
|
ep->time /= ticks_per_ns;
|
||
|
}
|
||
|
ep->code = ntohl(rep->code);
|
||
|
ep->pid = find_or_add_pid(ntohl(rep->pid));
|
||
|
ep->datum = ntohl(rep->datum);
|
||
|
ep->flags = 0;
|
||
|
ep++;
|
||
|
rep++;
|
||
|
}
|
||
|
|
||
|
unmapfile((char *)ulp, size);
|
||
|
|
||
|
make_sorted_pid_vector();
|
||
|
g_free(s_pidhash);
|
||
|
s_pidhash = 0;
|
||
|
|
||
|
/* Give the view-1 world a chance to reset a few things... */
|
||
|
view1_read_events_callback();
|
||
|
}
|
||
|
|
||
|
static event_t *add_ep;
|
||
|
|
||
|
/****************************************************************************
|
||
|
* cpel_event_init
|
||
|
****************************************************************************/
|
||
|
void cpel_event_init (ulong nevents)
|
||
|
{
|
||
|
g_nevents = nevents;
|
||
|
if (g_events)
|
||
|
g_free(g_events);
|
||
|
add_ep = g_events = (event_t *)g_malloc(g_nevents * sizeof(event_t));
|
||
|
while (g_npids > 0) {
|
||
|
g_free((g_pids + g_npids-1)->pid);
|
||
|
g_npids--;
|
||
|
}
|
||
|
if (g_pids) {
|
||
|
g_free(g_pids);
|
||
|
g_free(g_original_pids);
|
||
|
g_pids = 0;
|
||
|
g_original_pids = 0;
|
||
|
}
|
||
|
s_pidhash = (pid_data_t **)g_malloc0(
|
||
|
PIDHASH_NBUCKETS*sizeof(pid_data_t *));
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* add_cpel_event
|
||
|
****************************************************************************/
|
||
|
|
||
|
void add_cpel_event(ulonglong delta, ulong track, ulong event, ulong datum)
|
||
|
{
|
||
|
event_t *ep;
|
||
|
|
||
|
ep = add_ep++;
|
||
|
ep->time = delta;
|
||
|
ep->pid = find_or_add_pid(track);
|
||
|
ep->code = event;
|
||
|
ep->datum = datum;
|
||
|
ep->flags = 0;
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* add_clib_event
|
||
|
****************************************************************************/
|
||
|
|
||
|
void add_clib_event(double delta, unsigned short track,
|
||
|
unsigned short event, unsigned int index)
|
||
|
{
|
||
|
event_t *ep;
|
||
|
|
||
|
ep = add_ep++;
|
||
|
ep->time = (ulonglong) (delta * 1e9); /* time in intger nanoseconds */
|
||
|
ep->pid = find_or_add_pid(track);
|
||
|
ep->code = event;
|
||
|
ep->datum = index;
|
||
|
ep->flags = EVENT_FLAG_CLIB;
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* cpel_event_finalize
|
||
|
****************************************************************************/
|
||
|
|
||
|
void cpel_event_finalize(void)
|
||
|
{
|
||
|
make_sorted_pid_vector();
|
||
|
g_free(s_pidhash);
|
||
|
s_pidhash = 0;
|
||
|
|
||
|
/* Give the view-1 world a chance to reset a few things... */
|
||
|
view1_read_events_callback();
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* mapfile
|
||
|
****************************************************************************/
|
||
|
|
||
|
char *mapfile (char *file, ulong *sizep)
|
||
|
{
|
||
|
struct stat statb;
|
||
|
char *rv;
|
||
|
int maphfile;
|
||
|
size_t mapfsize;
|
||
|
|
||
|
maphfile = open (file, O_RDONLY);
|
||
|
|
||
|
if (maphfile < 0)
|
||
|
return (NULL);
|
||
|
|
||
|
if (fstat (maphfile, &statb) < 0) {
|
||
|
return (NULL);
|
||
|
}
|
||
|
|
||
|
/* Don't try to mmap directories, FIFOs, semaphores, etc. */
|
||
|
if (! (statb.st_mode & S_IFREG)) {
|
||
|
return (NULL);
|
||
|
}
|
||
|
|
||
|
mapfsize = statb.st_size;
|
||
|
|
||
|
if (mapfsize < 3) {
|
||
|
close (maphfile);
|
||
|
return (NULL);
|
||
|
}
|
||
|
|
||
|
rv = mmap (0, mapfsize, PROT_READ, MAP_SHARED, maphfile, 0);
|
||
|
|
||
|
if (rv == 0) {
|
||
|
g_error ("%s mapping problem, I quit...\n", file);
|
||
|
}
|
||
|
|
||
|
close (maphfile);
|
||
|
|
||
|
if (madvise (rv, mapfsize, MADV_SEQUENTIAL) < 0) {
|
||
|
return (rv);
|
||
|
}
|
||
|
|
||
|
if (sizep) {
|
||
|
*sizep = mapfsize;
|
||
|
}
|
||
|
return (rv);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* unmapfile
|
||
|
****************************************************************************/
|
||
|
|
||
|
boolean unmapfile (char *addr, ulong size)
|
||
|
{
|
||
|
if (munmap (addr, size) < 0) {
|
||
|
g_warning("Unmap error, addr 0x%lx size 0x%x\n",
|
||
|
(unsigned long) addr, (unsigned int)size);
|
||
|
return(FALSE);
|
||
|
}
|
||
|
return(TRUE);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* find_event_index
|
||
|
* Binary search for first event whose time is >= t
|
||
|
****************************************************************************/
|
||
|
|
||
|
int find_event_index (ulonglong t)
|
||
|
{
|
||
|
int index, bottom, top;
|
||
|
event_t *ep;
|
||
|
|
||
|
bottom = g_nevents-1;
|
||
|
top = 0;
|
||
|
|
||
|
while (1) {
|
||
|
index = (bottom + top) / 2;
|
||
|
|
||
|
ep = (g_events + index);
|
||
|
|
||
|
if (ep->time == t)
|
||
|
return(index);
|
||
|
|
||
|
if (top >= bottom) {
|
||
|
while (index > 0 && ep->time > t) {
|
||
|
ep--;
|
||
|
index--;
|
||
|
}
|
||
|
while (index < g_nevents && ep->time < t) {
|
||
|
ep++;
|
||
|
index++;
|
||
|
}
|
||
|
return(index);
|
||
|
}
|
||
|
|
||
|
if (ep->time < t)
|
||
|
top = index + 1;
|
||
|
else
|
||
|
bottom = index - 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* events_about
|
||
|
****************************************************************************/
|
||
|
|
||
|
void events_about (char *tmpbuf)
|
||
|
{
|
||
|
sprintf(tmpbuf+strlen(tmpbuf), "%d total events, %.3f ticks per us\n",
|
||
|
(int)g_nevents, ticks_per_ns);
|
||
|
}
|