Note that the mouse-over doesn't work with Label:{}. Need to find out why.
[p5sagit/Devel-Size.git] / memnodes.pl
1 #!/bin/env perl
2
3 use strict;
4 use warnings;
5
6 use DBI qw(looks_like_number);
7 use DBD::SQLite;
8 use JSON::XS;
9
10 use Getopt::Long;
11
12 GetOptions(
13     'json!' => \my $opt_json,
14     'dot!' => \my $opt_dot,
15     'db=s'  => \my $opt_db,
16     'verbose|v!' => \my $opt_verbose,
17     'debug|d!' => \my $opt_debug,
18     'showid!' => \my $opt_showid,
19 ) or exit 1;
20
21 my $j = JSON::XS->new->ascii->pretty(0);
22
23 my ($dbh, $node_ins_sth);
24 if ($opt_db) {
25     $dbh = DBI->connect("dbi:SQLite:dbname=$opt_db","","", {
26         RaiseError => 1, PrintError => 0, AutoCommit => 0
27     });
28     $dbh->do("PRAGMA synchronous = OFF");
29     $dbh->do("DROP TABLE IF EXISTS node");
30     $dbh->do(q{
31         CREATE TABLE node (
32             id integer primary key,
33             name text,
34             title text,
35             type integer,
36             depth integer,
37             parent_id integer,
38
39             self_size integer,
40             kids_size integer,
41             kids_node_count integer,
42             child_ids text,
43             attr_json text,
44             leaves_json text
45         )
46     });
47     $node_ins_sth = $dbh->prepare(q{
48         INSERT INTO node VALUES (?,?,?,?,?,?,  ?,?,?,?,?,?)
49     });
50 }
51
52 my @stack;
53 my %seqn2node;
54
55 use HTML::Entities qw(encode_entities);;
56 my $dotnode = sub {
57     my $name = encode_entities(shift);
58     $name =~ s/"/\\"/g;
59     return '"'.$name.'"';
60 };
61
62 print "memnodes = [" if $opt_json;
63
64 if ($opt_dot) {
65     print "digraph {\n"; # }
66     print "graph [overlap=false]\n"; # target="???", URL="???"
67 }
68
69 sub fmt_size {
70     my $size = shift;
71     my $kb = $size / 1024;
72     return $size if $kb < 5;
73     return sprintf "%.1fKb", $kb if $kb < 1000;
74     return sprintf "%.1fMb", $kb/1024;
75 }
76
77
78 sub enter_node {
79     my $x = shift;
80     if ($opt_json) {
81         print "    " x $x->{depth};
82         print qq({ "id": "$x->{id}", "name": "$x->{name}", "depth":$x->{depth}, "children":[ \n);
83     }
84     if ($opt_dot) {
85         #printf $fh qq{\tn%d [ %s ]\n}, $x->{id}, $dotnode->($x->{name});
86         #print qq({ "id": "$x->{id}", "name": "$x->{name}", "depth":$x->{depth}, "children":[ \n);
87     }
88     return;
89 }
90
91 sub leave_node {
92     my $x = shift;
93     delete $seqn2node{$x->{id}};
94
95     my $self_size = 0; $self_size += $_  for values %{$x->{leaves}};
96     $x->{self_size} = $self_size;
97
98     my $parent = $stack[-1];
99     if ($parent) {
100         # link to parent
101         $x->{parent_id} = $parent->{id};
102         # accumulate into parent
103         $parent->{kids_node_count} += 1 + ($x->{kids_node_count}||0);
104         $parent->{kids_size} += $self_size + $x->{kids_size};
105         push @{$parent->{child_id}}, $x->{id};
106     }
107     # output
108     # ...
109     if ($opt_json) {
110         print "\t" x $x->{depth};
111         my $size = $self_size + $x->{kids_size};
112         print qq(], "data":{ "\$area": $size } },\n);
113     }
114     if ($opt_dot) {
115         printf "// n%d parent=%s(type=%s)\n", $x->{id},
116                 $parent ? $parent->{id} : "",
117                 $parent ? $parent->{type} : ""
118             if 0;
119         if ($x->{type} != 2) {
120             my $name = $x->{title} ? "\"$x->{title}\" $x->{name}" : $x->{name};
121
122             if ($x->{kids_size}) {
123                 $name .= sprintf " %s+%s=%s", fmt_size($x->{self_size}), fmt_size($x->{kids_size}), fmt_size($x->{self_size}+$x->{kids_size});
124             }
125             else {
126                 $name .= sprintf " +%s", fmt_size($x->{self_size});
127             }
128             $name .= " $x->{id}" if $opt_showid;
129
130             my @node_attr = (
131                 sprintf("label=%s", $dotnode->($name)),
132                 "id=$x->{id}",
133             );
134             my @link_attr;
135             #if ($x->{name} eq 'hek') { push @node_attr, "shape=point"; push @node_attr, "labelfontsize=6"; }
136             if ($parent) { # probably a link
137                 my $parent_id = $parent->{id};
138                 my @link_attr = ("id=$parent_id");
139                 if ($parent->{type} == 2) { # link
140                     (my $link_name = $parent->{name}) =~ s/->$//;
141                     push @link_attr, (sprintf "label=%s", $dotnode->($link_name));
142                     $parent_id = ($stack[-2]||die "panic")->{id};
143                 }
144                 printf qq{n%d -> n%d [%s];\n},
145                     $parent_id, $x->{id}, join(",", @link_attr);
146             }
147             printf qq{n%d [ %s ];\n}, $x->{id}, join(",", @node_attr);
148         }
149
150     }
151     if ($dbh) {
152         my $attr_json = $j->encode($x->{attr});
153         my $leaves_json = $j->encode($x->{leaves});
154         $node_ins_sth->execute(
155             $x->{id}, $x->{name}, $x->{title}, $x->{type}, $x->{depth}, $x->{parent_id},
156             $x->{self_size}, $x->{kids_size}, $x->{kids_node_count},
157             $x->{child_id} ? join(",", @{$x->{child_id}}) : undef,
158             $attr_json, $leaves_json,
159         );
160         # XXX attribs
161     }
162     return;
163 }
164
165
166 while (<>) {
167     chomp;
168     my ($type, $id, $val, $name, $extra) = split / /, $_, 5;
169     if ($type =~ s/^-//) {     # Node type ($val is depth)
170         while ($val < @stack) {
171             leave_node(my $x = pop @stack);
172             warn "N $id d$val ends $x->{id} d$x->{depth}: size $x->{self_size}+$x->{kids_size}\n"
173                 if $opt_verbose;
174         }
175         die 1 if $stack[$val];
176         my $node = $stack[$val] = { id => $id, type => $type, name => $name, extra => $extra, attr => {}, leaves => {}, depth => $val, self_size=>0, kids_size=>0 };
177         enter_node($node);
178         $seqn2node{$id} = $node;
179     }
180     elsif ($type eq "L") {  # Leaf name and memory size
181         my $node = $seqn2node{$id} || die;
182         $node->{leaves}{$name} += $val;
183     }
184     elsif (looks_like_number($type)) {  # Attribute type, name and value
185         my $node = $seqn2node{$id} || die;
186         my $attr = $node->{attr} || die;
187         if ($type == 1) { # NPattr_NAME
188             warn "Node $id already has attribute $type:$name (value $attr->{$type}{$name})\n"
189                 if exists $attr->{$type}{$name};
190             $attr->{$type}{$name} = $val || $id;
191             warn "A \@$id: '$name' $val\n";
192             $node->{title} = $name if $type == 1 and !$val;
193         }
194         elsif (2 <= $type and $type <= 4) { # NPattr_PAD*
195             warn "Node $id already has attribute $type:$name (value $attr->{$type}[$val])\n"
196                 if defined $attr->{$type}[$val];
197             $attr->{$type}[$val] = $name;
198         }
199         else {
200             warn "Invalid attribute type '$type' on line $. ($_)";
201         }
202     }
203     else {
204         warn "Invalid type '$type' on line $. ($_)";
205         next;
206     }
207     $dbh->commit if $dbh and $id % 10_000 == 0;
208 }
209
210 my $x;
211 while (@stack > 1) {
212     leave_node($x = pop @stack) while @stack;
213     warn "EOF ends $x->{id} d$x->{depth}: size $x->{self_size}+$x->{kids_size}\n";
214 }
215 print " ];\n" if $opt_json;
216 print "}\n" if $opt_dot;
217
218 $dbh->commit if $dbh;
219
220 use Data::Dumper;
221 warn Dumper(\$x);
222 warn Dumper(\%seqn2node);
223
224 =for
225 SV(PVAV) fill=1/1       [#1 @0] 
226 :   +64 sv =64 
227 :   +16 av_max =80 
228 :   AVelem->        [#2 @1] 
229 :   :   SV(RV)      [#3 @2] 
230 :   :   :   +24 sv =104 
231 :   :   :   RV->        [#4 @3] 
232 :   :   :   :   SV(PVAV) fill=-1/-1     [#5 @4] 
233 :   :   :   :   :   +64 sv =168 
234 :   AVelem->        [#6 @1] 
235 :   :   SV(IV)      [#7 @2] 
236 :   :   :   +24 sv =192 
237 192 at -e line 1.
238 =cut
239 __DATA__
240 N 1 0 SV(PVAV) fill=1/1
241 L 1 64 sv
242 L 1 16 av_max
243 N 2 1 AVelem->
244 N 3 2 SV(RV)
245 L 3 24 sv
246 N 4 3 RV->
247 N 5 4 SV(PVAV) fill=-1/-1
248 L 5 64 sv
249 N 6 1 AVelem->
250 N 7 2 SV(IV)
251 L 7 24 sv