libcore: Check, that a pointer to a char array is passed to LC_OPT_ENT_STR().
[libfirm] / ir / be / bestat.c
1 /*
2  * Copyright (C) 1995-2011 University of Karlsruhe.  All right reserved.
3  *
4  * This file is part of libFirm.
5  *
6  * This file may be distributed and/or modified under the terms of the
7  * GNU General Public License version 2 as published by the Free Software
8  * Foundation and appearing in the file LICENSE.GPL included in the
9  * packaging of this file.
10  *
11  * Licensees holding valid libFirm Professional Edition licenses may use
12  * this file in accordance with the libFirm Commercial License.
13  * Agreement provided with the Software.
14  *
15  * This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
16  * WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17  * PURPOSE.
18  */
19
20 /**
21  * @file
22  * @brief       Provides several statistic functions for the backend.
23  * @author      Christian Wuerdig, Matthias Braun
24  */
25 #include "config.h"
26
27 #include <time.h>
28
29 #include "irnode_t.h"
30 #include "irprintf.h"
31 #include "irgwalk.h"
32 #include "irhooks.h"
33 #include "execfreq.h"
34 #include "firmstat_t.h"
35 #include "irtools.h"
36 #include "error.h"
37 #include "statev_t.h"
38
39 #include "bearch.h"
40 #include "beirg.h"
41 #include "bestat.h"
42 #include "belive_t.h"
43 #include "besched.h"
44 #include "benode.h"
45
46
47
48 typedef struct pressure_walker_env_t pressure_walker_env_t;
49 struct pressure_walker_env_t {
50         ir_graph *irg;
51         be_lv_t  *lv;
52         double    insn_count;
53         double    regpressure;
54         size_t    max_pressure;
55         const arch_register_class_t *cls;
56 };
57
58 static void check_reg_pressure_class(pressure_walker_env_t *env,
59                                      ir_node *block,
60                                      const arch_register_class_t *cls)
61 {
62         ir_graph    *irg  = env->irg;
63         ir_nodeset_t live_nodes;
64         size_t       max_live;
65
66         ir_nodeset_init(&live_nodes);
67         be_liveness_end_of_block(env->lv, cls, block, &live_nodes);
68         max_live = ir_nodeset_size(&live_nodes);
69         env->regpressure += max_live;
70
71         sched_foreach_reverse(block, irn) {
72                 size_t cnt;
73
74                 if (is_Phi(irn))
75                         break;
76
77                 be_liveness_transfer(cls, irn, &live_nodes);
78                 cnt      = ir_nodeset_size(&live_nodes);
79                 max_live = cnt < max_live ? max_live : cnt;
80                 env->regpressure += cnt;
81                 env->insn_count++;
82         }
83
84         if (max_live > env->max_pressure)
85                 env->max_pressure = max_live;
86
87         stat_be_block_regpressure(irg, block, max_live, cls->name);
88         ir_nodeset_destroy(&live_nodes);
89 }
90
91 static void stat_reg_pressure_block(ir_node *block, void *data)
92 {
93         pressure_walker_env_t *env = (pressure_walker_env_t*)data;
94
95         check_reg_pressure_class(env, block, env->cls);
96 }
97
98 void be_do_stat_reg_pressure(ir_graph *irg, const arch_register_class_t *cls)
99 {
100         pressure_walker_env_t  env;
101         double                 average_pressure;
102
103         env.irg          = irg;
104         env.insn_count   = 0;
105         env.max_pressure = 0;
106         env.regpressure  = 0;
107         be_assure_live_sets(irg);
108         env.lv           = be_get_irg_liveness(irg);
109         env.cls          = cls;
110
111         /* Collect register pressure information for each block */
112         irg_block_walk_graph(irg, stat_reg_pressure_block, NULL, &env);
113
114         average_pressure = env.regpressure / env.insn_count;
115         stat_ev_dbl("bechordal_average_register_pressure", average_pressure);
116         stat_ev_dbl("bechordal_maximum_register_pressure", env.max_pressure);
117 }
118
119
120
121
122 typedef struct estimate_irg_costs_env_t {
123         double costs;
124 } estimate_irg_costs_env_t;
125
126 static void estimate_block_costs(ir_node *block, void *data)
127 {
128         estimate_irg_costs_env_t *env = (estimate_irg_costs_env_t*)data;
129         double costs = 0.0;
130
131         sched_foreach(block, node) {
132                 costs += arch_get_op_estimated_cost(node);
133         }
134
135         env->costs += costs * get_block_execfreq(block);
136 }
137
138 double be_estimate_irg_costs(ir_graph *irg)
139 {
140         estimate_irg_costs_env_t env;
141         env.costs = 0.0;
142
143         irg_block_walk_graph(irg, estimate_block_costs, NULL, &env);
144
145         return env.costs;
146 }
147
148
149
150 static void node_stat_walker(ir_node *irn, void *data)
151 {
152         be_node_stats_t *const stats = (be_node_stats_t*)data;
153
154         /* if the node is a normal phi */
155         if (is_Phi(irn)) {
156                 if (get_irn_mode(irn) == mode_M) {
157                         (*stats)[BE_STAT_MEM_PHIS]++;
158                 } else {
159                         (*stats)[BE_STAT_PHIS]++;
160                 }
161         } else if (be_is_Perm(irn)) {
162                 (*stats)[BE_STAT_PERMS]++;
163         } else if (be_is_Copy(irn)) {
164                 (*stats)[BE_STAT_COPIES]++;
165         }
166 }
167
168 void be_collect_node_stats(be_node_stats_t *new_stats, ir_graph *irg)
169 {
170         memset(new_stats, 0, sizeof(*new_stats));
171         irg_walk_graph(irg, NULL, node_stat_walker, new_stats);
172 }
173
174 void be_subtract_node_stats(be_node_stats_t *stats, be_node_stats_t *sub)
175 {
176         int i;
177         for (i = 0; i < BE_STAT_COUNT; ++i) {
178                 (*stats)[i] -= (*sub)[i];
179         }
180 }
181
182 void be_copy_node_stats(be_node_stats_t *dest, be_node_stats_t *src)
183 {
184         memcpy(dest, src, sizeof(be_node_stats_t));
185 }
186
187 static const char *get_stat_name(enum be_stat_tag_t tag)
188 {
189         switch (tag) {
190         case BE_STAT_PHIS:     return "phis";
191         case BE_STAT_MEM_PHIS: return "mem_phis";
192         case BE_STAT_COPIES:   return "copies";
193         case BE_STAT_PERMS:    return "perms";
194         default:               panic("unknown stat tag found");
195         }
196 }
197
198 void be_emit_node_stats(be_node_stats_t *stats, const char *prefix)
199 {
200         static char   buf[256];
201         be_stat_tag_t i;
202
203         for (i = BE_STAT_FIRST; i < BE_STAT_COUNT; ++i) {
204                 snprintf(buf, sizeof(buf), "%s%s", prefix, get_stat_name(i));
205                 stat_ev_dbl(buf, (*stats)[i]);
206         }
207 }
208
209
210
211 static void insn_count_walker(ir_node *irn, void *data)
212 {
213         unsigned long *cnt = (unsigned long*)data;
214
215         switch (get_irn_opcode(irn)) {
216         case iro_Proj:
217         case iro_Phi:
218         case beo_Start:
219         case iro_End:
220                 break;
221         default:
222                 (*cnt)++;
223         }
224 }
225
226 unsigned long be_count_insns(ir_graph *irg)
227 {
228         unsigned long cnt = 0;
229         irg_walk_graph(irg, insn_count_walker, NULL, &cnt);
230         return cnt;
231 }
232
233 static void block_count_walker(ir_node *node, void *data)
234 {
235         unsigned long *cnt = (unsigned long*)data;
236         if (node == get_irg_end_block(get_irn_irg(node)))
237                 return;
238         (*cnt)++;
239 }
240
241 unsigned long be_count_blocks(ir_graph *irg)
242 {
243         unsigned long cnt = 0;
244         irg_block_walk_graph(irg, block_count_walker, NULL, &cnt);
245         return cnt;
246 }