1
2 /*--------------------------------------------------------------------*/
3 /*--- A program that merges multiple cachegrind output files. ---*/
4 /*--- cg_merge.c ---*/
5 /*--------------------------------------------------------------------*/
6
7 /*
8 This file is part of Cachegrind, a Valgrind tool for cache
9 profiling programs.
10
11 Copyright (C) 2002-2017 Nicholas Nethercote
12 njn@valgrind.org
13
14 AVL tree code derived from
15 ANSI C Library for maintenance of AVL Balanced Trees
16 (C) 2000 Daniel Nagy, Budapest University of Technology and Economics
17 Released under GNU General Public License (GPL) version 2
18
19 This program is free software; you can redistribute it and/or
20 modify it under the terms of the GNU General Public License as
21 published by the Free Software Foundation; either version 2 of the
22 License, or (at your option) any later version.
23
24 This program is distributed in the hope that it will be useful, but
25 WITHOUT ANY WARRANTY; without even the implied warranty of
26 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
27 General Public License for more details.
28
29 You should have received a copy of the GNU General Public License
30 along with this program; if not, write to the Free Software
31 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
32 02111-1307, USA.
33
34 The GNU General Public License is contained in the file COPYING.
35 */
36
37 #include <stdio.h>
38 #include <stdlib.h>
39 #include <assert.h>
40 #include <string.h>
41 #include <ctype.h>
42
43 typedef signed long Word;
44 typedef unsigned long UWord;
45 typedef unsigned char Bool;
46 #define True ((Bool)1)
47 #define False ((Bool)0)
48 typedef signed int Int;
49 typedef unsigned int UInt;
50 typedef unsigned long long int ULong;
51 typedef signed char Char;
52 typedef size_t SizeT;
53
54
55 //------------------------------------------------------------------//
56 //--- WordFM ---//
57 //--- Public interface ---//
58 //------------------------------------------------------------------//
59
60 typedef struct _WordFM WordFM; /* opaque */
61
62 /* Initialise a WordFM */
63 void initFM ( WordFM* t,
64 void* (*alloc_nofail)( SizeT ),
65 void (*dealloc)(void*),
66 Word (*kCmp)(Word,Word) );
67
68 /* Allocate and initialise a WordFM */
69 WordFM* newFM( void* (*alloc_nofail)( SizeT ),
70 void (*dealloc)(void*),
71 Word (*kCmp)(Word,Word) );
72
73 /* Free up the FM. If kFin is non-NULL, it is applied to keys
74 before the FM is deleted; ditto with vFin for vals. */
75 void deleteFM ( WordFM*, void(*kFin)(Word), void(*vFin)(Word) );
76
77 /* Add (k,v) to fm. If a binding for k already exists, it is updated
78 to map to this new v. In that case we should really return the
79 previous v so that caller can finalise it. Oh well. */
80 void addToFM ( WordFM* fm, Word k, Word v );
81
82 // Delete key from fm, returning associated val if found
83 Bool delFromFM ( WordFM* fm, /*OUT*/Word* oldV, Word key );
84
85 // Look up in fm, assigning found val at spec'd address
86 Bool lookupFM ( WordFM* fm, /*OUT*/Word* valP, Word key );
87
88 Word sizeFM ( WordFM* fm );
89
90 // set up FM for iteration
91 void initIterFM ( WordFM* fm );
92
93 // get next key/val pair. Will assert if fm has been modified
94 // or looked up in since initIterFM was called.
95 Bool nextIterFM ( WordFM* fm, /*OUT*/Word* pKey, /*OUT*/Word* pVal );
96
97 // clear the I'm iterating flag
98 void doneIterFM ( WordFM* fm );
99
100 // Deep copy a FM. If dopyK is NULL, keys are copied verbatim.
101 // If non-null, dopyK is applied to each key to generate the
102 // version in the new copy. In that case, if the argument to dopyK
103 // is non-NULL but the result is NULL, it is assumed that dopyK
104 // could not allocate memory, in which case the copy is abandoned
105 // and NULL is returned. Ditto with dopyV for values.
106 WordFM* dopyFM ( WordFM* fm, Word(*dopyK)(Word), Word(*dopyV)(Word) );
107
108 //------------------------------------------------------------------//
109 //--- end WordFM ---//
110 //--- Public interface ---//
111 //------------------------------------------------------------------//
112
113
114 static const char* argv0 = "cg_merge";
115
116 /* Keep track of source filename/line no so as to be able to
117 print decent error messages. */
118 typedef
119 struct {
120 FILE* fp;
121 UInt lno;
122 char* filename;
123 }
124 SOURCE;
125
printSrcLoc(SOURCE * s)126 static void printSrcLoc ( SOURCE* s )
127 {
128 fprintf(stderr, "%s: near %s line %u\n", argv0, s->filename, s->lno-1);
129 }
130
131 __attribute__((noreturn))
mallocFail(SOURCE * s,const char * who)132 static void mallocFail ( SOURCE* s, const char* who )
133 {
134 fprintf(stderr, "%s: out of memory in %s\n", argv0, who );
135 printSrcLoc( s );
136 exit(2);
137 }
138
139 __attribute__((noreturn))
parseError(SOURCE * s,const char * msg)140 static void parseError ( SOURCE* s, const char* msg )
141 {
142 fprintf(stderr, "%s: parse error: %s\n", argv0, msg );
143 printSrcLoc( s );
144 exit(1);
145 }
146
147 __attribute__((noreturn))
barf(SOURCE * s,const char * msg)148 static void barf ( SOURCE* s, const char* msg )
149 {
150 fprintf(stderr, "%s: %s\n", argv0, msg );
151 printSrcLoc( s );
152 exit(1);
153 }
154
155 // Read a line. Return the line read, or NULL if at EOF.
156 // The line is allocated dynamically but will be overwritten with
157 // every invocation. Caller must not free it.
readline(SOURCE * s)158 static const char *readline ( SOURCE* s )
159 {
160 static char *line = NULL;
161 static size_t linesiz = 0;
162
163 int ch, i = 0;
164
165 while (1) {
166 ch = getc(s->fp);
167 if (ch != EOF) {
168 if (i + 1 >= linesiz) {
169 linesiz += 500;
170 line = realloc(line, linesiz * sizeof *line);
171 if (line == NULL)
172 mallocFail(s, "readline:");
173 }
174 line[i++] = ch;
175 line[i] = 0;
176 if (ch == '\n') {
177 line[i-1] = 0;
178 s->lno++;
179 break;
180 }
181 } else {
182 if (ferror(s->fp)) {
183 perror(argv0);
184 barf(s, "I/O error while reading input file");
185 } else {
186 // hit EOF
187 break;
188 }
189 }
190 }
191 return i == 0 ? NULL : line;
192 }
193
streqn(const char * s1,const char * s2,size_t n)194 static Bool streqn ( const char* s1, const char* s2, size_t n )
195 {
196 return 0 == strncmp(s1, s2, n);
197 }
198
streq(const char * s1,const char * s2)199 static Bool streq ( const char* s1, const char* s2 )
200 {
201 return 0 == strcmp(s1, s2 );
202 }
203
204
205 ////////////////////////////////////////////////////////////////
206
207 typedef
208 struct {
209 char* fi_name;
210 char* fn_name;
211 }
212 FileFn;
213
214 typedef
215 struct {
216 Int n_counts;
217 ULong* counts;
218 }
219 Counts;
220
221 typedef
222 struct {
223 // null-terminated vector of desc_lines
224 char** desc_lines;
225
226 // Cmd line
227 char* cmd_line;
228
229 // Events line
230 char* events_line;
231 Int n_events;
232
233 // Summary line (copied from input)
234 char* summary_line;
235
236 /* Outermost map is
237 WordFM FileFn* innerMap
238 where innerMap is WordFM line-number=UWord Counts */
239 WordFM* outerMap;
240
241 // Summary counts (computed whilst parsing)
242 // should match .summary_line
243 Counts* summary;
244 }
245 CacheProfFile;
246
new_FileFn(char * file_name,char * fn_name)247 static FileFn* new_FileFn ( char* file_name, char* fn_name )
248 {
249 FileFn* ffn = malloc(sizeof(FileFn));
250 if (ffn == NULL)
251 return NULL;
252 ffn->fi_name = file_name;
253 ffn->fn_name = fn_name;
254 return ffn;
255 }
256
ddel_FileFn(FileFn * ffn)257 static void ddel_FileFn ( FileFn* ffn )
258 {
259 if (ffn->fi_name)
260 free(ffn->fi_name);
261 if (ffn->fn_name)
262 free(ffn->fn_name);
263 memset(ffn, 0, sizeof(FileFn));
264 free(ffn);
265 }
266
dopy_FileFn(FileFn * ff)267 static FileFn* dopy_FileFn ( FileFn* ff )
268 {
269 char *fi2, *fn2;
270 fi2 = strdup(ff->fi_name);
271 if (fi2 == NULL) return NULL;
272 fn2 = strdup(ff->fn_name);
273 if (fn2 == NULL) {
274 free(fi2);
275 return NULL;
276 }
277 return new_FileFn( fi2, fn2 );
278 }
279
new_Counts(Int n_counts,ULong * counts)280 static Counts* new_Counts ( Int n_counts, /*COPIED*/ULong* counts )
281 {
282 Int i;
283 Counts* cts = malloc(sizeof(Counts));
284 if (cts == NULL)
285 return NULL;
286
287 assert(n_counts >= 0);
288 cts->counts = malloc(n_counts * sizeof(ULong));
289 if (cts->counts == NULL) {
290 free(cts);
291 return NULL;
292 }
293
294 cts->n_counts = n_counts;
295 for (i = 0; i < n_counts; i++)
296 cts->counts[i] = counts[i];
297
298 return cts;
299 }
300
new_Counts_Zeroed(Int n_counts)301 static Counts* new_Counts_Zeroed ( Int n_counts )
302 {
303 Int i;
304 Counts* cts = malloc(sizeof(Counts));
305 if (cts == NULL)
306 return NULL;
307
308 assert(n_counts >= 0);
309 cts->counts = malloc(n_counts * sizeof(ULong));
310 if (cts->counts == NULL) {
311 free(cts);
312 return NULL;
313 }
314
315 cts->n_counts = n_counts;
316 for (i = 0; i < n_counts; i++)
317 cts->counts[i] = 0;
318
319 return cts;
320 }
321
sdel_Counts(Counts * cts)322 static void sdel_Counts ( Counts* cts )
323 {
324 memset(cts, 0, sizeof(Counts));
325 free(cts);
326 }
327
ddel_Counts(Counts * cts)328 static void ddel_Counts ( Counts* cts )
329 {
330 if (cts->counts)
331 free(cts->counts);
332 memset(cts, 0, sizeof(Counts));
333 free(cts);
334 }
335
dopy_Counts(Counts * cts)336 static Counts* dopy_Counts ( Counts* cts )
337 {
338 return new_Counts( cts->n_counts, cts->counts );
339 }
340
341 static
new_CacheProfFile(char ** desc_lines,char * cmd_line,char * events_line,Int n_events,char * summary_line,WordFM * outerMap,Counts * summary)342 CacheProfFile* new_CacheProfFile ( char** desc_lines,
343 char* cmd_line,
344 char* events_line,
345 Int n_events,
346 char* summary_line,
347 WordFM* outerMap,
348 Counts* summary )
349 {
350 CacheProfFile* cpf = malloc(sizeof(CacheProfFile));
351 if (cpf == NULL)
352 return NULL;
353 cpf->desc_lines = desc_lines;
354 cpf->cmd_line = cmd_line;
355 cpf->events_line = events_line;
356 cpf->n_events = n_events;
357 cpf->summary_line = summary_line;
358 cpf->outerMap = outerMap;
359 cpf->summary = summary;
360 return cpf;
361 }
362
dopy_InnerMap(WordFM * innerMap)363 static WordFM* dopy_InnerMap ( WordFM* innerMap )
364 {
365 return dopyFM ( innerMap, NULL,
366 (Word(*)(Word))dopy_Counts );
367 }
368
ddel_InnerMap(WordFM * innerMap)369 static void ddel_InnerMap ( WordFM* innerMap )
370 {
371 deleteFM( innerMap, NULL, (void(*)(Word))ddel_Counts );
372 }
373
ddel_CacheProfFile(CacheProfFile * cpf)374 static void ddel_CacheProfFile ( CacheProfFile* cpf )
375 {
376 char** p;
377 if (cpf->desc_lines) {
378 for (p = cpf->desc_lines; *p; p++)
379 free(*p);
380 free(cpf->desc_lines);
381 }
382 if (cpf->cmd_line)
383 free(cpf->cmd_line);
384 if (cpf->events_line)
385 free(cpf->events_line);
386 if (cpf->summary_line)
387 free(cpf->summary_line);
388 if (cpf->outerMap)
389 deleteFM( cpf->outerMap, (void(*)(Word))ddel_FileFn,
390 (void(*)(Word))ddel_InnerMap );
391 if (cpf->summary)
392 ddel_Counts(cpf->summary);
393
394 memset(cpf, 0, sizeof(CacheProfFile));
395 free(cpf);
396 }
397
showCounts(FILE * f,Counts * c)398 static void showCounts ( FILE* f, Counts* c )
399 {
400 Int i;
401 for (i = 0; i < c->n_counts; i++) {
402 fprintf(f, "%lld ", c->counts[i]);
403 }
404 }
405
show_CacheProfFile(FILE * f,CacheProfFile * cpf)406 static void show_CacheProfFile ( FILE* f, CacheProfFile* cpf )
407 {
408 Int i;
409 char** d;
410 FileFn* topKey;
411 WordFM* topVal;
412 UWord subKey;
413 Counts* subVal;
414
415 for (d = cpf->desc_lines; *d; d++)
416 fprintf(f, "%s\n", *d);
417 fprintf(f, "%s\n", cpf->cmd_line);
418 fprintf(f, "%s\n", cpf->events_line);
419
420 initIterFM( cpf->outerMap );
421 while (nextIterFM( cpf->outerMap, (Word*)(&topKey), (Word*)(&topVal) )) {
422 fprintf(f, "fl=%s\nfn=%s\n",
423 topKey->fi_name, topKey->fn_name );
424 initIterFM( topVal );
425 while (nextIterFM( topVal, (Word*)(&subKey), (Word*)(&subVal) )) {
426 fprintf(f, "%ld ", subKey );
427 showCounts( f, subVal );
428 fprintf(f, "\n");
429 }
430 doneIterFM( topVal );
431 }
432 doneIterFM( cpf->outerMap );
433
434 //fprintf(f, "%s\n", cpf->summary_line);
435 fprintf(f, "summary:");
436 for (i = 0; i < cpf->summary->n_counts; i++)
437 fprintf(f, " %lld", cpf->summary->counts[i]);
438 fprintf(f, "\n");
439 }
440
441 ////////////////////////////////////////////////////////////////
442
cmp_FileFn(Word s1,Word s2)443 static Word cmp_FileFn ( Word s1, Word s2 )
444 {
445 FileFn* ff1 = (FileFn*)s1;
446 FileFn* ff2 = (FileFn*)s2;
447 Word r = strcmp(ff1->fi_name, ff2->fi_name);
448 if (r == 0)
449 r = strcmp(ff1->fn_name, ff2->fn_name);
450 return r;
451 }
452
cmp_unboxed_UWord(Word s1,Word s2)453 static Word cmp_unboxed_UWord ( Word s1, Word s2 )
454 {
455 UWord u1 = (UWord)s1;
456 UWord u2 = (UWord)s2;
457 if (u1 < u2) return -1;
458 if (u1 > u2) return 1;
459 return 0;
460 }
461
462 ////////////////////////////////////////////////////////////////
463
parse_ULong(ULong * res,const char ** pptr)464 static Bool parse_ULong ( /*OUT*/ULong* res, /*INOUT*/const char** pptr)
465 {
466 ULong u64;
467 const char* ptr = *pptr;
468 while (isspace(*ptr)) ptr++;
469 if (!isdigit(*ptr)) {
470 *pptr = ptr;
471 return False; /* end of string, or junk */
472 }
473 u64 = 0;
474 while (isdigit(*ptr)) {
475 u64 = (u64 * 10) + (ULong)(*ptr - '0');
476 ptr++;
477 }
478 *res = u64;
479 *pptr = ptr;
480 return True;
481 }
482
483 // str is a line of integers, starting with a line number. Parse it,
484 // returning the first number in *lnno and the rest in a newly
485 // allocated Counts struct. If lnno is non-NULL, treat the first
486 // number as a line number and assign it to *lnno instead of
487 // incorporating it in the counts array.
488 static
splitUpCountsLine(SOURCE * s,UWord * lnno,const char * str)489 Counts* splitUpCountsLine ( SOURCE* s, /*OUT*/UWord* lnno, const char* str )
490 {
491 Bool ok;
492 Counts* counts;
493 ULong *tmpC = NULL;
494 UInt n_tmpC = 0, tmpCsize = 0;
495 while (1) {
496 if (n_tmpC >= tmpCsize) {
497 tmpCsize += 50;
498 tmpC = realloc(tmpC, tmpCsize * sizeof *tmpC);
499 if (tmpC == NULL)
500 mallocFail(s, "splitUpCountsLine:");
501 }
502 ok = parse_ULong( &tmpC[n_tmpC], &str );
503 if (!ok)
504 break;
505 n_tmpC++;
506 }
507 if (*str != 0)
508 parseError(s, "garbage in counts line");
509 if (lnno ? (n_tmpC < 2) : (n_tmpC < 1))
510 parseError(s, "too few counts in count line");
511
512 if (lnno) {
513 *lnno = (UWord)tmpC[0];
514 counts = new_Counts( n_tmpC-1, /*COPIED*/&tmpC[1] );
515 } else {
516 counts = new_Counts( n_tmpC, /*COPIED*/&tmpC[0] );
517 }
518 free(tmpC);
519
520 return counts;
521 }
522
addCounts(SOURCE * s,Counts * counts1,Counts * counts2)523 static void addCounts ( SOURCE* s, /*OUT*/Counts* counts1, Counts* counts2 )
524 {
525 Int i;
526 if (counts1->n_counts != counts2->n_counts)
527 parseError(s, "addCounts: inconsistent number of counts");
528 for (i = 0; i < counts1->n_counts; i++)
529 counts1->counts[i] += counts2->counts[i];
530 }
531
addCountsToMap(SOURCE * s,WordFM * counts_map,UWord lnno,Counts * newCounts)532 static Bool addCountsToMap ( SOURCE* s,
533 WordFM* counts_map,
534 UWord lnno, Counts* newCounts )
535 {
536 Counts* oldCounts;
537 // look up lnno in the map. If none present, add a binding
538 // lnno->counts. If present, add counts to the existing entry.
539 if (lookupFM( counts_map, (Word*)(&oldCounts), (Word)lnno )) {
540 // merge with existing binding
541 addCounts( s, oldCounts, newCounts );
542 return True;
543 } else {
544 // create new binding
545 addToFM( counts_map, (Word)lnno, (Word)newCounts );
546 return False;
547 }
548 }
549
550 static
handle_counts(SOURCE * s,CacheProfFile * cpf,const char * fi,const char * fn,const char * newCountsStr)551 void handle_counts ( SOURCE* s,
552 CacheProfFile* cpf,
553 const char* fi, const char* fn, const char* newCountsStr )
554 {
555 WordFM* countsMap;
556 Bool freeNewCounts;
557 UWord lnno;
558 Counts* newCounts;
559 FileFn* topKey;
560
561 if (0) printf("%s %s %s\n", fi, fn, newCountsStr );
562
563 // parse the numbers
564 newCounts = splitUpCountsLine( s, &lnno, newCountsStr );
565
566 // Did we get the right number?
567 if (newCounts->n_counts != cpf->n_events)
568 goto oom;
569
570 // allocate the key
571 topKey = malloc(sizeof(FileFn));
572 if (topKey) {
573 topKey->fi_name = strdup(fi);
574 topKey->fn_name = strdup(fn);
575 }
576 if (! (topKey && topKey->fi_name && topKey->fn_name))
577 mallocFail(s, "handle_counts:");
578
579 // search for it
580 if (lookupFM( cpf->outerMap, (Word*)(&countsMap), (Word)topKey )) {
581 // found it. Merge in new counts
582 freeNewCounts = addCountsToMap( s, countsMap, lnno, newCounts );
583 ddel_FileFn(topKey);
584 } else {
585 // not found in the top map. Create new entry
586 countsMap = newFM( malloc, free, cmp_unboxed_UWord );
587 if (!countsMap)
588 goto oom;
589 addToFM( cpf->outerMap, (Word)topKey, (Word)countsMap );
590 freeNewCounts = addCountsToMap( s, countsMap, lnno, newCounts );
591 }
592
593 // also add to running summary total
594 addCounts( s, cpf->summary, newCounts );
595
596 // if safe to do so, free up the count vector
597 if (freeNewCounts)
598 ddel_Counts(newCounts);
599
600 return;
601
602 oom:
603 parseError(s, "# counts doesn't match # events");
604 }
605
606
607 /* Parse a complete file from the stream in 's'. If a parse error
608 happens, do not return; instead exit via parseError(). If an
609 out-of-memory condition happens, do not return; instead exit via
610 mallocError().
611 */
parse_CacheProfFile(SOURCE * s)612 static CacheProfFile* parse_CacheProfFile ( SOURCE* s )
613 {
614 Int i;
615 char** tmp_desclines = NULL;
616 unsigned tmp_desclines_size = 0;
617 char* p;
618 int n_tmp_desclines = 0;
619 CacheProfFile* cpf;
620 Counts* summaryRead;
621 char* curr_fn = strdup("???");
622 char* curr_fl = strdup("???");
623 const char* line;
624
625 cpf = new_CacheProfFile( NULL, NULL, NULL, 0, NULL, NULL, NULL );
626 if (cpf == NULL)
627 mallocFail(s, "parse_CacheProfFile(1)");
628
629 // Parse "desc:" lines
630 while (1) {
631 line = readline(s);
632 if (!line)
633 break;
634 if (!streqn(line, "desc: ", 6))
635 break;
636 if (n_tmp_desclines >= tmp_desclines_size) {
637 tmp_desclines_size += 100;
638 tmp_desclines = realloc(tmp_desclines,
639 tmp_desclines_size * sizeof *tmp_desclines);
640 if (tmp_desclines == NULL)
641 mallocFail(s, "parse_CacheProfFile(1)");
642 }
643 tmp_desclines[n_tmp_desclines++] = strdup(line);
644 }
645
646 if (n_tmp_desclines == 0)
647 parseError(s, "parse_CacheProfFile: no DESC lines present");
648
649 cpf->desc_lines = malloc( (1+n_tmp_desclines) * sizeof(char*) );
650 if (cpf->desc_lines == NULL)
651 mallocFail(s, "parse_CacheProfFile(2)");
652
653 cpf->desc_lines[n_tmp_desclines] = NULL;
654 for (i = 0; i < n_tmp_desclines; i++)
655 cpf->desc_lines[i] = tmp_desclines[i];
656
657 // Parse "cmd:" line
658 if (!streqn(line, "cmd: ", 5))
659 parseError(s, "parse_CacheProfFile: no CMD line present");
660
661 cpf->cmd_line = strdup(line);
662 if (cpf->cmd_line == NULL)
663 mallocFail(s, "parse_CacheProfFile(3)");
664
665 // Parse "events:" line and figure out how many events there are
666 line = readline(s);
667 if (!line)
668 parseError(s, "parse_CacheProfFile: eof before EVENTS line");
669 if (!streqn(line, "events: ", 8))
670 parseError(s, "parse_CacheProfFile: no EVENTS line present");
671
672 // figure out how many events there are by counting the number
673 // of space-alphanum transitions in the events_line
674 cpf->events_line = strdup(line);
675 if (cpf->events_line == NULL)
676 mallocFail(s, "parse_CacheProfFile(3)");
677
678 cpf->n_events = 0;
679 assert(cpf->events_line[6] == ':');
680 for (p = &cpf->events_line[6]; *p; p++) {
681 if (p[0] == ' ' && isalpha(p[1]))
682 cpf->n_events++;
683 }
684
685 // create the running cross-check summary
686 cpf->summary = new_Counts_Zeroed( cpf->n_events );
687 if (cpf->summary == NULL)
688 mallocFail(s, "parse_CacheProfFile(4)");
689
690 // create the outer map (file+fn name --> inner map)
691 cpf->outerMap = newFM ( malloc, free, cmp_FileFn );
692 if (cpf->outerMap == NULL)
693 mallocFail(s, "parse_CacheProfFile(5)");
694
695 // process count lines
696 while (1) {
697 line = readline(s);
698 if (!line)
699 parseError(s, "parse_CacheProfFile: eof before SUMMARY line");
700
701 if (isdigit(line[0])) {
702 handle_counts(s, cpf, curr_fl, curr_fn, line);
703 continue;
704 }
705 else
706 if (streqn(line, "fn=", 3)) {
707 free(curr_fn);
708 curr_fn = strdup(line+3);
709 continue;
710 }
711 else
712 if (streqn(line, "fl=", 3)) {
713 free(curr_fl);
714 curr_fl = strdup(line+3);
715 continue;
716 }
717 else
718 if (streqn(line, "summary: ", 9)) {
719 break;
720 }
721 else
722 parseError(s, "parse_CacheProfFile: unexpected line in main data");
723 }
724
725 // finally, the "summary:" line
726 if (!streqn(line, "summary: ", 9))
727 parseError(s, "parse_CacheProfFile: missing SUMMARY line");
728
729 cpf->summary_line = strdup(line);
730 if (cpf->summary_line == NULL)
731 mallocFail(s, "parse_CacheProfFile(6)");
732
733 // there should be nothing more
734 line = readline(s);
735 if (line)
736 parseError(s, "parse_CacheProfFile: "
737 "extraneous content after SUMMARY line");
738
739 // check the summary counts are as expected
740 summaryRead = splitUpCountsLine( s, NULL, &cpf->summary_line[8] );
741 if (summaryRead == NULL)
742 mallocFail(s, "parse_CacheProfFile(7)");
743 if (summaryRead->n_counts != cpf->n_events)
744 parseError(s, "parse_CacheProfFile: wrong # counts in SUMMARY line");
745 for (i = 0; i < summaryRead->n_counts; i++) {
746 if (summaryRead->counts[i] != cpf->summary->counts[i]) {
747 parseError(s, "parse_CacheProfFile: "
748 "computed vs stated SUMMARY counts mismatch");
749 }
750 }
751 free(summaryRead->counts);
752 sdel_Counts(summaryRead);
753
754 // since the summary counts are OK, free up the summary_line text
755 // which contains the same info.
756 free(cpf->summary_line);
757 cpf->summary_line = NULL;
758
759 free(tmp_desclines);
760 free(curr_fn);
761 free(curr_fl);
762
763 // All looks OK
764 return cpf;
765 }
766
767
merge_CacheProfInfo(SOURCE * s,CacheProfFile * dst,CacheProfFile * src)768 static void merge_CacheProfInfo ( SOURCE* s,
769 /*MOD*/CacheProfFile* dst,
770 CacheProfFile* src )
771 {
772 /* For each (filefn, innerMap) in src
773 if filefn not in dst
774 add binding dopy(filefn)->dopy(innerMap) in src
775 else
776 // merge src->innerMap with dst->innerMap
777 for each (lineno, counts) in src->innerMap
778 if lineno not in dst->innerMap
779 add binding lineno->dopy(counts) to dst->innerMap
780 else
781 add counts into dst->innerMap[lineno]
782 */
783 /* Outer iterator: FileFn* -> WordFM* (inner iterator)
784 Inner iterator: UWord -> Counts*
785 */
786 FileFn* soKey;
787 WordFM* soVal;
788 WordFM* doVal;
789 UWord siKey;
790 Counts* siVal;
791 Counts* diVal;
792
793 /* First check mundane things: that the events: lines are
794 identical. */
795 if (!streq( dst->events_line, src->events_line ))
796 barf(s, "\"events:\" line of most recent file does "
797 "not match those previously processed");
798
799 initIterFM( src->outerMap );
800
801 // for (filefn, innerMap) in src
802 while (nextIterFM( src->outerMap, (Word*)&soKey, (Word*)&soVal )) {
803
804 // is filefn in dst?
805 if (! lookupFM( dst->outerMap, (Word*)&doVal, (Word)soKey )) {
806
807 // no .. add dopy(filefn) -> dopy(innerMap) to src
808 FileFn* c_soKey = dopy_FileFn(soKey);
809 WordFM* c_soVal = dopy_InnerMap(soVal);
810 if ((!c_soKey) || (!c_soVal)) goto oom;
811 addToFM( dst->outerMap, (Word)c_soKey, (Word)c_soVal );
812
813 } else {
814
815 // yes .. merge the two innermaps
816 initIterFM( soVal );
817
818 // for (lno, counts) in soVal (source inner map)
819 while (nextIterFM( soVal, (Word*)&siKey, (Word*)&siVal )) {
820
821 // is lno in the corresponding dst inner map?
822 if (! lookupFM( doVal, (Word*)&diVal, siKey )) {
823
824 // no .. add lineno->dopy(counts) to dst inner map
825 Counts* c_siVal = dopy_Counts( siVal );
826 if (!c_siVal) goto oom;
827 addToFM( doVal, siKey, (Word)c_siVal );
828
829 } else {
830
831 // yes .. merge counts into dst inner map val
832 addCounts( s, diVal, siVal );
833
834 }
835 }
836
837 }
838
839 }
840
841 // add the summaries too
842 addCounts(s, dst->summary, src->summary );
843
844 return;
845
846 oom:
847 mallocFail(s, "merge_CacheProfInfo");
848 }
849
usage(void)850 static void usage ( void )
851 {
852 fprintf(stderr, "%s: Merges multiple cachegrind output files into one\n",
853 argv0);
854 fprintf(stderr, "%s: usage: %s [-o outfile] [files-to-merge]\n",
855 argv0, argv0);
856 exit(1);
857 }
858
main(int argc,char ** argv)859 int main ( int argc, char** argv )
860 {
861 Int i;
862 SOURCE src;
863 CacheProfFile *cpf, *cpfTmp;
864
865 FILE* outfile = NULL;
866 char* outfilename = NULL;
867 Int outfileix = 0;
868
869 if (argv[0])
870 argv0 = argv[0];
871
872 if (argc < 2)
873 usage();
874
875 for (i = 1; i < argc; i++) {
876 if (streq(argv[i], "-h") || streq(argv[i], "--help"))
877 usage();
878 }
879
880 /* Scan args, looking for '-o outfilename'. */
881 for (i = 1; i < argc; i++) {
882 if (streq(argv[i], "-o")) {
883 if (i+1 < argc) {
884 outfilename = argv[i+1];
885 outfileix = i;
886 break;
887 } else {
888 usage();
889 }
890 }
891 }
892
893 cpf = NULL;
894
895 for (i = 1; i < argc; i++) {
896
897 if (i == outfileix) {
898 /* Skip '-o' and whatever follows it */
899 i += 1;
900 continue;
901 }
902
903 fprintf(stderr, "%s: parsing %s\n", argv0, argv[i]);
904 src.lno = 1;
905 src.filename = argv[i];
906 src.fp = fopen(src.filename, "r");
907 if (!src.fp) {
908 perror(argv0);
909 barf(&src, "Cannot open input file");
910 }
911 assert(src.fp);
912 cpfTmp = parse_CacheProfFile( &src );
913 fclose(src.fp);
914
915 /* If this isn't the first file, merge */
916 if (cpf == NULL) {
917 /* this is the first file */
918 cpf = cpfTmp;
919 } else {
920 /* not the first file; merge */
921 fprintf(stderr, "%s: merging %s\n", argv0, argv[i]);
922 merge_CacheProfInfo( &src, cpf, cpfTmp );
923 ddel_CacheProfFile( cpfTmp );
924 }
925
926 }
927
928 /* Now create the output file. */
929
930 if (cpf) {
931
932 fprintf(stderr, "%s: writing %s\n",
933 argv0, outfilename ? outfilename : "(stdout)" );
934
935 /* Write the output. */
936 if (outfilename) {
937 outfile = fopen(outfilename, "w");
938 if (!outfile) {
939 fprintf(stderr, "%s: can't create output file %s\n",
940 argv0, outfilename);
941 perror(argv0);
942 exit(1);
943 }
944 } else {
945 outfile = stdout;
946 }
947
948 show_CacheProfFile( outfile, cpf );
949 if (ferror(outfile)) {
950 fprintf(stderr, "%s: error writing output file %s\n",
951 argv0, outfilename ? outfilename : "(stdout)" );
952 perror(argv0);
953 if (outfile != stdout)
954 fclose(outfile);
955 exit(1);
956 }
957
958 fflush(outfile);
959 if (outfile != stdout)
960 fclose( outfile );
961
962 ddel_CacheProfFile( cpf );
963 }
964
965 return 0;
966 }
967
968
969 //------------------------------------------------------------------//
970 //--- WordFM ---//
971 //--- Implementation ---//
972 //------------------------------------------------------------------//
973
974 /* ------------ Implementation ------------ */
975
976 /* One element of the AVL tree */
977 typedef
978 struct _AvlNode {
979 Word key;
980 Word val;
981 struct _AvlNode* left;
982 struct _AvlNode* right;
983 Char balance;
984 }
985 AvlNode;
986
987 typedef
988 struct {
989 Word w;
990 Bool b;
991 }
992 MaybeWord;
993
994 #define WFM_STKMAX 32 // At most 2**32 entries can be iterated over
995
996 struct _WordFM {
997 AvlNode* root;
998 void* (*alloc_nofail)( SizeT );
999 void (*dealloc)(void*);
1000 Word (*kCmp)(Word,Word);
1001 AvlNode* nodeStack[WFM_STKMAX]; // Iterator node stack
1002 Int numStack[WFM_STKMAX]; // Iterator num stack
1003 Int stackTop; // Iterator stack pointer, one past end
1004 };
1005
1006 /* forward */
1007 static Bool avl_removeroot_wrk(AvlNode** t, Word(*kCmp)(Word,Word));
1008
1009 /* Swing to the left. Warning: no balance maintenance. */
avl_swl(AvlNode ** root)1010 static void avl_swl ( AvlNode** root )
1011 {
1012 AvlNode* a = *root;
1013 AvlNode* b = a->right;
1014 *root = b;
1015 a->right = b->left;
1016 b->left = a;
1017 }
1018
1019 /* Swing to the right. Warning: no balance maintenance. */
avl_swr(AvlNode ** root)1020 static void avl_swr ( AvlNode** root )
1021 {
1022 AvlNode* a = *root;
1023 AvlNode* b = a->left;
1024 *root = b;
1025 a->left = b->right;
1026 b->right = a;
1027 }
1028
1029 /* Balance maintenance after especially nasty swings. */
avl_nasty(AvlNode * root)1030 static void avl_nasty ( AvlNode* root )
1031 {
1032 switch (root->balance) {
1033 case -1:
1034 root->left->balance = 0;
1035 root->right->balance = 1;
1036 break;
1037 case 1:
1038 root->left->balance = -1;
1039 root->right->balance = 0;
1040 break;
1041 case 0:
1042 root->left->balance = 0;
1043 root->right->balance = 0;
1044 break;
1045 default:
1046 assert(0);
1047 }
1048 root->balance=0;
1049 }
1050
1051 /* Find size of a non-NULL tree. */
size_avl_nonNull(AvlNode * nd)1052 static Word size_avl_nonNull ( AvlNode* nd )
1053 {
1054 return 1 + (nd->left ? size_avl_nonNull(nd->left) : 0)
1055 + (nd->right ? size_avl_nonNull(nd->right) : 0);
1056 }
1057
1058 /* Insert element a into the AVL tree t. Returns True if the depth of
1059 the tree has grown. If element with that key is already present,
1060 just copy a->val to existing node, first returning old ->val field
1061 of existing node in *oldV, so that the caller can finalize it
1062 however it wants.
1063 */
1064 static
avl_insert_wrk(AvlNode ** rootp,MaybeWord * oldV,AvlNode * a,Word (* kCmp)(Word,Word))1065 Bool avl_insert_wrk ( AvlNode** rootp,
1066 /*OUT*/MaybeWord* oldV,
1067 AvlNode* a,
1068 Word (*kCmp)(Word,Word) )
1069 {
1070 Word cmpres;
1071
1072 /* initialize */
1073 a->left = 0;
1074 a->right = 0;
1075 a->balance = 0;
1076 oldV->b = False;
1077
1078 /* insert into an empty tree? */
1079 if (!(*rootp)) {
1080 (*rootp) = a;
1081 return True;
1082 }
1083
1084 cmpres = kCmp( (*rootp)->key, a->key );
1085
1086 if (cmpres > 0) {
1087 /* insert into the left subtree */
1088 if ((*rootp)->left) {
1089 AvlNode* left_subtree = (*rootp)->left;
1090 if (avl_insert_wrk(&left_subtree, oldV, a, kCmp)) {
1091 switch ((*rootp)->balance--) {
1092 case 1: return False;
1093 case 0: return True;
1094 case -1: break;
1095 default: assert(0);
1096 }
1097 if ((*rootp)->left->balance < 0) {
1098 avl_swr( rootp );
1099 (*rootp)->balance = 0;
1100 (*rootp)->right->balance = 0;
1101 } else {
1102 avl_swl( &((*rootp)->left) );
1103 avl_swr( rootp );
1104 avl_nasty( *rootp );
1105 }
1106 } else {
1107 (*rootp)->left = left_subtree;
1108 }
1109 return False;
1110 } else {
1111 (*rootp)->left = a;
1112 if ((*rootp)->balance--)
1113 return False;
1114 return True;
1115 }
1116 assert(0);/*NOTREACHED*/
1117 }
1118 else
1119 if (cmpres < 0) {
1120 /* insert into the right subtree */
1121 if ((*rootp)->right) {
1122 AvlNode* right_subtree = (*rootp)->right;
1123 if (avl_insert_wrk(&right_subtree, oldV, a, kCmp)) {
1124 switch((*rootp)->balance++){
1125 case -1: return False;
1126 case 0: return True;
1127 case 1: break;
1128 default: assert(0);
1129 }
1130 if ((*rootp)->right->balance > 0) {
1131 avl_swl( rootp );
1132 (*rootp)->balance = 0;
1133 (*rootp)->left->balance = 0;
1134 } else {
1135 avl_swr( &((*rootp)->right) );
1136 avl_swl( rootp );
1137 avl_nasty( *rootp );
1138 }
1139 } else {
1140 (*rootp)->right = right_subtree;
1141 }
1142 return False;
1143 } else {
1144 (*rootp)->right = a;
1145 if ((*rootp)->balance++)
1146 return False;
1147 return True;
1148 }
1149 assert(0);/*NOTREACHED*/
1150 }
1151 else {
1152 /* cmpres == 0, a duplicate - replace the val, but don't
1153 incorporate the node in the tree */
1154 oldV->b = True;
1155 oldV->w = (*rootp)->val;
1156 (*rootp)->val = a->val;
1157 return False;
1158 }
1159 }
1160
1161 /* Remove an element a from the AVL tree t. a must be part of
1162 the tree. Returns True if the depth of the tree has shrunk.
1163 */
1164 static
avl_remove_wrk(AvlNode ** rootp,AvlNode * a,Word (* kCmp)(Word,Word))1165 Bool avl_remove_wrk ( AvlNode** rootp,
1166 AvlNode* a,
1167 Word(*kCmp)(Word,Word) )
1168 {
1169 Bool ch;
1170 Word cmpres = kCmp( (*rootp)->key, a->key );
1171
1172 if (cmpres > 0){
1173 /* remove from the left subtree */
1174 AvlNode* left_subtree = (*rootp)->left;
1175 assert(left_subtree);
1176 ch = avl_remove_wrk(&left_subtree, a, kCmp);
1177 (*rootp)->left=left_subtree;
1178 if (ch) {
1179 switch ((*rootp)->balance++) {
1180 case -1: return True;
1181 case 0: return False;
1182 case 1: break;
1183 default: assert(0);
1184 }
1185 switch ((*rootp)->right->balance) {
1186 case 0:
1187 avl_swl( rootp );
1188 (*rootp)->balance = -1;
1189 (*rootp)->left->balance = 1;
1190 return False;
1191 case 1:
1192 avl_swl( rootp );
1193 (*rootp)->balance = 0;
1194 (*rootp)->left->balance = 0;
1195 return -1;
1196 case -1:
1197 break;
1198 default:
1199 assert(0);
1200 }
1201 avl_swr( &((*rootp)->right) );
1202 avl_swl( rootp );
1203 avl_nasty( *rootp );
1204 return True;
1205 }
1206 }
1207 else
1208 if (cmpres < 0) {
1209 /* remove from the right subtree */
1210 AvlNode* right_subtree = (*rootp)->right;
1211 assert(right_subtree);
1212 ch = avl_remove_wrk(&right_subtree, a, kCmp);
1213 (*rootp)->right = right_subtree;
1214 if (ch) {
1215 switch ((*rootp)->balance--) {
1216 case 1: return True;
1217 case 0: return False;
1218 case -1: break;
1219 default: assert(0);
1220 }
1221 switch ((*rootp)->left->balance) {
1222 case 0:
1223 avl_swr( rootp );
1224 (*rootp)->balance = 1;
1225 (*rootp)->right->balance = -1;
1226 return False;
1227 case -1:
1228 avl_swr( rootp );
1229 (*rootp)->balance = 0;
1230 (*rootp)->right->balance = 0;
1231 return True;
1232 case 1:
1233 break;
1234 default:
1235 assert(0);
1236 }
1237 avl_swl( &((*rootp)->left) );
1238 avl_swr( rootp );
1239 avl_nasty( *rootp );
1240 return True;
1241 }
1242 }
1243 else {
1244 assert(cmpres == 0);
1245 assert((*rootp)==a);
1246 return avl_removeroot_wrk(rootp, kCmp);
1247 }
1248 return 0;
1249 }
1250
1251 /* Remove the root of the AVL tree *rootp.
1252 * Warning: dumps core if *rootp is empty
1253 */
1254 static
avl_removeroot_wrk(AvlNode ** rootp,Word (* kCmp)(Word,Word))1255 Bool avl_removeroot_wrk ( AvlNode** rootp,
1256 Word(*kCmp)(Word,Word) )
1257 {
1258 Bool ch;
1259 AvlNode* a;
1260 if (!(*rootp)->left) {
1261 if (!(*rootp)->right) {
1262 (*rootp) = 0;
1263 return True;
1264 }
1265 (*rootp) = (*rootp)->right;
1266 return True;
1267 }
1268 if (!(*rootp)->right) {
1269 (*rootp) = (*rootp)->left;
1270 return True;
1271 }
1272 if ((*rootp)->balance < 0) {
1273 /* remove from the left subtree */
1274 a = (*rootp)->left;
1275 while (a->right) a = a->right;
1276 } else {
1277 /* remove from the right subtree */
1278 a = (*rootp)->right;
1279 while (a->left) a = a->left;
1280 }
1281 ch = avl_remove_wrk(rootp, a, kCmp);
1282 a->left = (*rootp)->left;
1283 a->right = (*rootp)->right;
1284 a->balance = (*rootp)->balance;
1285 (*rootp) = a;
1286 if(a->balance == 0) return ch;
1287 return False;
1288 }
1289
1290 static
avl_find_node(AvlNode * t,Word k,Word (* kCmp)(Word,Word))1291 AvlNode* avl_find_node ( AvlNode* t, Word k, Word(*kCmp)(Word,Word) )
1292 {
1293 Word cmpres;
1294 while (True) {
1295 if (t == NULL) return NULL;
1296 cmpres = kCmp(t->key, k);
1297 if (cmpres > 0) t = t->left; else
1298 if (cmpres < 0) t = t->right; else
1299 return t;
1300 }
1301 }
1302
1303 // Clear the iterator stack.
stackClear(WordFM * fm)1304 static void stackClear(WordFM* fm)
1305 {
1306 Int i;
1307 assert(fm);
1308 for (i = 0; i < WFM_STKMAX; i++) {
1309 fm->nodeStack[i] = NULL;
1310 fm->numStack[i] = 0;
1311 }
1312 fm->stackTop = 0;
1313 }
1314
1315 // Push onto the iterator stack.
stackPush(WordFM * fm,AvlNode * n,Int i)1316 static inline void stackPush(WordFM* fm, AvlNode* n, Int i)
1317 {
1318 assert(fm->stackTop < WFM_STKMAX);
1319 assert(1 <= i && i <= 3);
1320 fm->nodeStack[fm->stackTop] = n;
1321 fm-> numStack[fm->stackTop] = i;
1322 fm->stackTop++;
1323 }
1324
1325 // Pop from the iterator stack.
stackPop(WordFM * fm,AvlNode ** n,Int * i)1326 static inline Bool stackPop(WordFM* fm, AvlNode** n, Int* i)
1327 {
1328 assert(fm->stackTop <= WFM_STKMAX);
1329
1330 if (fm->stackTop > 0) {
1331 fm->stackTop--;
1332 *n = fm->nodeStack[fm->stackTop];
1333 *i = fm-> numStack[fm->stackTop];
1334 assert(1 <= *i && *i <= 3);
1335 fm->nodeStack[fm->stackTop] = NULL;
1336 fm-> numStack[fm->stackTop] = 0;
1337 return True;
1338 } else {
1339 return False;
1340 }
1341 }
1342
1343 static
avl_dopy(AvlNode * nd,Word (* dopyK)(Word),Word (* dopyV)(Word),void * (alloc_nofail)(SizeT))1344 AvlNode* avl_dopy ( AvlNode* nd,
1345 Word(*dopyK)(Word),
1346 Word(*dopyV)(Word),
1347 void*(alloc_nofail)(SizeT) )
1348 {
1349 AvlNode* nyu;
1350 if (! nd)
1351 return NULL;
1352 nyu = alloc_nofail(sizeof(AvlNode));
1353 assert(nyu);
1354
1355 nyu->left = nd->left;
1356 nyu->right = nd->right;
1357 nyu->balance = nd->balance;
1358
1359 /* Copy key */
1360 if (dopyK) {
1361 nyu->key = dopyK( nd->key );
1362 if (nd->key != 0 && nyu->key == 0)
1363 return NULL; /* oom in key dcopy */
1364 } else {
1365 /* copying assumedly unboxed keys */
1366 nyu->key = nd->key;
1367 }
1368
1369 /* Copy val */
1370 if (dopyV) {
1371 nyu->val = dopyV( nd->val );
1372 if (nd->val != 0 && nyu->val == 0)
1373 return NULL; /* oom in val dcopy */
1374 } else {
1375 /* copying assumedly unboxed vals */
1376 nyu->val = nd->val;
1377 }
1378
1379 /* Copy subtrees */
1380 if (nyu->left) {
1381 nyu->left = avl_dopy( nyu->left, dopyK, dopyV, alloc_nofail );
1382 if (! nyu->left)
1383 return NULL;
1384 }
1385 if (nyu->right) {
1386 nyu->right = avl_dopy( nyu->right, dopyK, dopyV, alloc_nofail );
1387 if (! nyu->right)
1388 return NULL;
1389 }
1390
1391 return nyu;
1392 }
1393
1394 /* --- Public interface functions --- */
1395
1396 /* Initialise a WordFM. */
initFM(WordFM * fm,void * (* alloc_nofail)(SizeT),void (* dealloc)(void *),Word (* kCmp)(Word,Word))1397 void initFM ( WordFM* fm,
1398 void* (*alloc_nofail)( SizeT ),
1399 void (*dealloc)(void*),
1400 Word (*kCmp)(Word,Word) )
1401 {
1402 fm->root = 0;
1403 fm->kCmp = kCmp;
1404 fm->alloc_nofail = alloc_nofail;
1405 fm->dealloc = dealloc;
1406 fm->stackTop = 0;
1407 }
1408
1409 /* Allocate and Initialise a WordFM. */
newFM(void * (* alloc_nofail)(SizeT),void (* dealloc)(void *),Word (* kCmp)(Word,Word))1410 WordFM* newFM( void* (*alloc_nofail)( SizeT ),
1411 void (*dealloc)(void*),
1412 Word (*kCmp)(Word,Word) )
1413 {
1414 WordFM* fm = alloc_nofail(sizeof(WordFM));
1415 assert(fm);
1416 initFM(fm, alloc_nofail, dealloc, kCmp);
1417 return fm;
1418 }
1419
avl_free(AvlNode * nd,void (* kFin)(Word),void (* vFin)(Word),void (* dealloc)(void *))1420 static void avl_free ( AvlNode* nd,
1421 void(*kFin)(Word),
1422 void(*vFin)(Word),
1423 void(*dealloc)(void*) )
1424 {
1425 if (!nd)
1426 return;
1427 if (nd->left)
1428 avl_free(nd->left, kFin, vFin, dealloc);
1429 if (nd->right)
1430 avl_free(nd->right, kFin, vFin, dealloc);
1431 if (kFin)
1432 kFin( nd->key );
1433 if (vFin)
1434 vFin( nd->val );
1435 memset(nd, 0, sizeof(AvlNode));
1436 dealloc(nd);
1437 }
1438
1439 /* Free up the FM. If kFin is non-NULL, it is applied to keys
1440 before the FM is deleted; ditto with vFin for vals. */
deleteFM(WordFM * fm,void (* kFin)(Word),void (* vFin)(Word))1441 void deleteFM ( WordFM* fm, void(*kFin)(Word), void(*vFin)(Word) )
1442 {
1443 void(*dealloc)(void*) = fm->dealloc;
1444 avl_free( fm->root, kFin, vFin, dealloc );
1445 memset(fm, 0, sizeof(WordFM) );
1446 dealloc(fm);
1447 }
1448
1449 /* Add (k,v) to fm. */
addToFM(WordFM * fm,Word k,Word v)1450 void addToFM ( WordFM* fm, Word k, Word v )
1451 {
1452 MaybeWord oldV;
1453 AvlNode* node;
1454 node = fm->alloc_nofail( sizeof(struct _AvlNode) );
1455 node->key = k;
1456 node->val = v;
1457 oldV.b = False;
1458 oldV.w = 0;
1459 avl_insert_wrk( &fm->root, &oldV, node, fm->kCmp );
1460 //if (oldV.b && fm->vFin)
1461 // fm->vFin( oldV.w );
1462 if (oldV.b)
1463 free(node);
1464 }
1465
1466 // Delete key from fm, returning associated val if found
delFromFM(WordFM * fm,Word * oldV,Word key)1467 Bool delFromFM ( WordFM* fm, /*OUT*/Word* oldV, Word key )
1468 {
1469 AvlNode* node = avl_find_node( fm->root, key, fm->kCmp );
1470 if (node) {
1471 avl_remove_wrk( &fm->root, node, fm->kCmp );
1472 if (oldV)
1473 *oldV = node->val;
1474 fm->dealloc(node);
1475 return True;
1476 } else {
1477 return False;
1478 }
1479 }
1480
1481 // Look up in fm, assigning found val at spec'd address
lookupFM(WordFM * fm,Word * valP,Word key)1482 Bool lookupFM ( WordFM* fm, /*OUT*/Word* valP, Word key )
1483 {
1484 AvlNode* node = avl_find_node( fm->root, key, fm->kCmp );
1485 if (node) {
1486 if (valP)
1487 *valP = node->val;
1488 return True;
1489 } else {
1490 return False;
1491 }
1492 }
1493
sizeFM(WordFM * fm)1494 Word sizeFM ( WordFM* fm )
1495 {
1496 // Hmm, this is a bad way to do this
1497 return fm->root ? size_avl_nonNull( fm->root ) : 0;
1498 }
1499
1500 // set up FM for iteration
initIterFM(WordFM * fm)1501 void initIterFM ( WordFM* fm )
1502 {
1503 assert(fm);
1504 stackClear(fm);
1505 if (fm->root)
1506 stackPush(fm, fm->root, 1);
1507 }
1508
1509 // get next key/val pair. Will assert if fm has been modified
1510 // or looked up in since initIterFM was called.
nextIterFM(WordFM * fm,Word * pKey,Word * pVal)1511 Bool nextIterFM ( WordFM* fm, /*OUT*/Word* pKey, /*OUT*/Word* pVal )
1512 {
1513 Int i = 0;
1514 AvlNode* n = NULL;
1515
1516 assert(fm);
1517
1518 // This in-order traversal requires each node to be pushed and popped
1519 // three times. These could be avoided by updating nodes in-situ on the
1520 // top of the stack, but the push/pop cost is so small that it's worth
1521 // keeping this loop in this simpler form.
1522 while (stackPop(fm, &n, &i)) {
1523 switch (i) {
1524 case 1:
1525 stackPush(fm, n, 2);
1526 if (n->left) stackPush(fm, n->left, 1);
1527 break;
1528 case 2:
1529 stackPush(fm, n, 3);
1530 if (pKey) *pKey = n->key;
1531 if (pVal) *pVal = n->val;
1532 return True;
1533 case 3:
1534 if (n->right) stackPush(fm, n->right, 1);
1535 break;
1536 default:
1537 assert(0);
1538 }
1539 }
1540
1541 // Stack empty, iterator is exhausted, return NULL
1542 return False;
1543 }
1544
1545 // clear the I'm iterating flag
doneIterFM(WordFM * fm)1546 void doneIterFM ( WordFM* fm )
1547 {
1548 }
1549
dopyFM(WordFM * fm,Word (* dopyK)(Word),Word (* dopyV)(Word))1550 WordFM* dopyFM ( WordFM* fm, Word(*dopyK)(Word), Word(*dopyV)(Word) )
1551 {
1552 WordFM* nyu;
1553
1554 /* can't clone the fm whilst iterating on it */
1555 assert(fm->stackTop == 0);
1556
1557 nyu = fm->alloc_nofail( sizeof(WordFM) );
1558 assert(nyu);
1559
1560 *nyu = *fm;
1561
1562 fm->stackTop = 0;
1563 memset(fm->nodeStack, 0, sizeof(fm->nodeStack));
1564 memset(fm->numStack, 0, sizeof(fm->numStack));
1565
1566 if (nyu->root) {
1567 nyu->root = avl_dopy( nyu->root, dopyK, dopyV, fm->alloc_nofail );
1568 if (! nyu->root)
1569 return NULL;
1570 }
1571
1572 return nyu;
1573 }
1574
1575 //------------------------------------------------------------------//
1576 //--- end WordFM ---//
1577 //--- Implementation ---//
1578 //------------------------------------------------------------------//
1579
1580 /*--------------------------------------------------------------------*/
1581 /*--- end cg_merge.c ---*/
1582 /*--------------------------------------------------------------------*/
1583