/* Copyright (C) 2006 Mandriva Conectiva S.A. Copyright (C) 2006 Arnaldo Carvalho de Melo This program is free software; you can redistribute it and/or modify it under the terms of version 2 of the GNU General Public License as published by the Free Software Foundation. */ #include #include #include #include #include #include #include "classes.h" static struct option long_options[] = { { "help", no_argument, NULL, 'h' }, { NULL, 0, NULL, 0, } }; static unsigned int total_cus_changed; static unsigned int total_nr_functions_changed; static unsigned long total_function_bytes_added; static unsigned long total_function_bytes_removed; static void usage(void) { fprintf(stderr, "usage: codiff [options] \n" " where: \n" " -h, --help usage options\n"); } static void diff_function(struct cu *cu, struct cu *new_cu, struct class *function) { struct class *new_function; assert(function->tag == DW_TAG_subprogram); if (function->inlined) return; new_function = cu__find_class_by_name(new_cu, function->name); if (new_function != NULL) { function->diff = (class__function_size(new_function) - class__function_size(function)); if (function->diff != 0) { const size_t len = strlen(function->name); if (len > cu->max_len_changed_function) cu->max_len_changed_function = len; ++cu->nr_functions_changed; if (function->diff > 0) cu->function_bytes_added += function->diff; else cu->function_bytes_removed += -function->diff; } } } static int diff_iterator(struct cu *cu, struct class *class, void *new_cu) { switch (class->tag) { case DW_TAG_subprogram: diff_function(cu, new_cu, class); break; } return 0; } static int cu_diff_iterator(struct cu *cu, void *new_cus) { struct cu *new_cu = cus__find_cu_by_name(new_cus, cu->name); if (new_cu != NULL) return cu__for_each_class(cu, diff_iterator, new_cu); return 0; } static void show_diffs_function(struct class *class, struct cu *cu) { if (class->diff != 0) printf(" %-*.*s | %+4d\n", cu->max_len_changed_function, cu->max_len_changed_function, class->name, class->diff); } static int show_diffs_iterator(struct cu *cu, struct class *class, void *new_cu) { switch (class->tag) { case DW_TAG_subprogram: show_diffs_function(class, cu); break; } return 0; } static int cu_show_diffs_iterator(struct cu *cu, void *cookie) { static int first_cu_printed; if (cu->nr_functions_changed == 0) return 0; if (first_cu_printed) putchar('\n'); else first_cu_printed = 1; ++total_cus_changed; total_nr_functions_changed += cu->nr_functions_changed; printf("%s:\n", cu->name); cu__for_each_class(cu, show_diffs_iterator, NULL); printf("%u function%s changed", cu->nr_functions_changed, cu->nr_functions_changed > 1 ? "s" : ""); if (cu->function_bytes_added != 0) { total_function_bytes_added += cu->function_bytes_added; printf(", %u bytes added", cu->function_bytes_added); } if (cu->function_bytes_removed != 0) { total_function_bytes_removed += cu->function_bytes_removed; printf(", %u bytes removed", cu->function_bytes_removed); } putchar('\n'); return 0; } static void print_total_function_diff(const char *filename) { printf("\n%s:\n", filename); printf(" %u function%s changed", total_nr_functions_changed, total_nr_functions_changed > 1 ? "s" : ""); if (total_function_bytes_added != 0) printf(", %lu bytes added", total_function_bytes_added); if (total_function_bytes_removed != 0) printf(", %lu bytes removed", total_function_bytes_removed); putchar('\n'); } int main(int argc, char *argv[]) { int option, option_index; struct cus *old_cus, *new_cus; const char *old_filename, *new_filename; while ((option = getopt_long(argc, argv, "h", long_options, &option_index)) >= 0) switch (option) { case 'h': usage(); return EXIT_SUCCESS; default: usage(); return EXIT_FAILURE; } if (optind < argc) { switch (argc - optind) { case 2: old_filename = argv[optind++]; new_filename = argv[optind++]; break; case 1: default: usage(); return EXIT_FAILURE; } } else { usage(); return EXIT_FAILURE; } old_cus = cus__new(old_filename); new_cus = cus__new(new_filename); if (old_cus == NULL || new_cus == NULL) { fputs("codiff: insufficient memory\n", stderr); return EXIT_FAILURE; } if (cus__load(old_cus) != 0) { fprintf(stderr, "codiff: couldn't load DWARF info from %s\n", old_filename); return EXIT_FAILURE; } if (cus__load(new_cus) != 0) { fprintf(stderr, "codiff: couldn't load DWARF info from %s\n", new_filename); return EXIT_FAILURE; } cus__for_each_cu(old_cus, cu_diff_iterator, new_cus); cus__for_each_cu(old_cus, cu_show_diffs_iterator, NULL); if (total_cus_changed > 1) print_total_function_diff(new_filename); return EXIT_SUCCESS; }