• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright © 2016 Intel Corporation
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining a
5  * copy of this software and associated documentation files (the "Software"),
6  * to deal in the Software without restriction, including without limitation
7  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8  * and/or sell copies of the Software, and to permit persons to whom the
9  * Software is furnished to do so, subject to the following conditions:
10  *
11  * The above copyright notice and this permission notice (including the next
12  * paragraph) shall be included in all copies or substantial portions of the
13  * Software.
14  *
15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
21  * DEALINGS IN THE SOFTWARE.
22  */
23 
24 #include <inttypes.h> /* for PRIx64 macro */
25 #include "ir.h"
26 #include "ir_hierarchical_visitor.h"
27 #include "ir_builder_print_visitor.h"
28 #include "compiler/glsl_types.h"
29 #include "glsl_parser_extras.h"
30 #include "main/macros.h"
31 #include "util/hash_table.h"
32 #include "util/u_string.h"
33 
34 class ir_builder_print_visitor : public ir_hierarchical_visitor {
35 public:
36    ir_builder_print_visitor(FILE *f);
37    virtual ~ir_builder_print_visitor();
38 
39    void indent(void);
40 
41    virtual ir_visitor_status visit(class ir_variable *);
42    virtual ir_visitor_status visit(class ir_dereference_variable *);
43    virtual ir_visitor_status visit(class ir_constant *);
44    virtual ir_visitor_status visit(class ir_loop_jump *);
45 
46    virtual ir_visitor_status visit_enter(class ir_if *);
47 
48    virtual ir_visitor_status visit_enter(class ir_loop *);
49    virtual ir_visitor_status visit_leave(class ir_loop *);
50 
51    virtual ir_visitor_status visit_enter(class ir_function_signature *);
52    virtual ir_visitor_status visit_leave(class ir_function_signature *);
53 
54    virtual ir_visitor_status visit_enter(class ir_expression *);
55 
56    virtual ir_visitor_status visit_enter(class ir_assignment *);
57    virtual ir_visitor_status visit_leave(class ir_assignment *);
58 
59    virtual ir_visitor_status visit_leave(class ir_call *);
60    virtual ir_visitor_status visit_leave(class ir_swizzle *);
61    virtual ir_visitor_status visit_leave(class ir_return *);
62 
63    virtual ir_visitor_status visit_enter(ir_texture *ir);
64 
65 private:
66    void print_with_indent(const char *fmt, ...);
67    void print_without_indent(const char *fmt, ...);
68 
69    void print_without_declaration(const ir_rvalue *ir);
70    void print_without_declaration(const ir_constant *ir);
71    void print_without_declaration(const ir_dereference_variable *ir);
72    void print_without_declaration(const ir_swizzle *ir);
73    void print_without_declaration(const ir_expression *ir);
74 
75    unsigned next_ir_index;
76 
77    /**
78     * Mapping from ir_instruction * -> index used in the generated C code
79     * variable name.
80     */
81    hash_table *index_map;
82 
83    FILE *f;
84 
85    int indentation;
86 };
87 
88 /* An operand is "simple" if it can be compactly printed on one line.
89  */
90 static bool
is_simple_operand(const ir_rvalue * ir,unsigned depth=1)91 is_simple_operand(const ir_rvalue *ir, unsigned depth = 1)
92 {
93    if (depth == 0)
94       return false;
95 
96    switch (ir->ir_type) {
97    case ir_type_dereference_variable:
98       return true;
99 
100    case ir_type_constant: {
101       if (ir->type == glsl_type::uint_type ||
102           ir->type == glsl_type::int_type ||
103           ir->type == glsl_type::float_type ||
104           ir->type == glsl_type::bool_type)
105          return true;
106 
107       const ir_constant *const c = (ir_constant *) ir;
108       ir_constant_data all_zero;
109       memset(&all_zero, 0, sizeof(all_zero));
110 
111       return memcmp(&c->value, &all_zero, sizeof(all_zero)) == 0;
112    }
113 
114    case ir_type_swizzle: {
115       const ir_swizzle *swiz = (ir_swizzle *) ir;
116       return swiz->mask.num_components == 1 &&
117              is_simple_operand(swiz->val, depth);
118    }
119 
120    case ir_type_expression: {
121       const ir_expression *expr = (ir_expression *) ir;
122 
123       for (unsigned i = 0; i < expr->num_operands; i++) {
124          if (!is_simple_operand(expr->operands[i], depth - 1))
125             return false;
126       }
127 
128       return true;
129    }
130 
131    default:
132       return false;
133    }
134 }
135 
136 void
_mesa_print_builder_for_ir(FILE * f,exec_list * instructions)137 _mesa_print_builder_for_ir(FILE *f, exec_list *instructions)
138 {
139    ir_builder_print_visitor v(f);
140    v.run(instructions);
141 }
142 
ir_builder_print_visitor(FILE * f)143 ir_builder_print_visitor::ir_builder_print_visitor(FILE *f)
144    : next_ir_index(1), f(f), indentation(0)
145 {
146    index_map = _mesa_pointer_hash_table_create(NULL);
147 }
148 
~ir_builder_print_visitor()149 ir_builder_print_visitor::~ir_builder_print_visitor()
150 {
151    _mesa_hash_table_destroy(index_map, NULL);
152 }
153 
indent(void)154 void ir_builder_print_visitor::indent(void)
155 {
156    for (int i = 0; i < indentation; i++)
157       fprintf(f, "   ");
158 }
159 
160 void
print_with_indent(const char * fmt,...)161 ir_builder_print_visitor::print_with_indent(const char *fmt, ...)
162 {
163    va_list ap;
164 
165    indent();
166 
167    va_start(ap, fmt);
168    vfprintf(f, fmt, ap);
169    va_end(ap);
170 }
171 
172 void
print_without_indent(const char * fmt,...)173 ir_builder_print_visitor::print_without_indent(const char *fmt, ...)
174 {
175    va_list ap;
176 
177    va_start(ap, fmt);
178    vfprintf(f, fmt, ap);
179    va_end(ap);
180 }
181 
182 void
print_without_declaration(const ir_rvalue * ir)183 ir_builder_print_visitor::print_without_declaration(const ir_rvalue *ir)
184 {
185    switch (ir->ir_type) {
186    case ir_type_dereference_variable:
187       print_without_declaration((ir_dereference_variable *) ir);
188       break;
189    case ir_type_constant:
190       print_without_declaration((ir_constant *) ir);
191       break;
192    case ir_type_swizzle:
193       print_without_declaration((ir_swizzle *) ir);
194       break;
195    case ir_type_expression:
196       print_without_declaration((ir_expression *) ir);
197       break;
198    default:
199       unreachable("Invalid IR type.");
200    }
201 }
202 
203 ir_visitor_status
visit(ir_variable * ir)204 ir_builder_print_visitor::visit(ir_variable *ir)
205 {
206    const unsigned my_index = next_ir_index++;
207 
208    _mesa_hash_table_insert(index_map, ir, (void *)(uintptr_t) my_index);
209 
210    const char *mode_str;
211    switch (ir->data.mode) {
212    case ir_var_auto: mode_str = "ir_var_auto"; break;
213    case ir_var_uniform: mode_str = "ir_var_uniform"; break;
214    case ir_var_shader_storage: mode_str = "ir_var_shader_storage"; break;
215    case ir_var_shader_shared: mode_str = "ir_var_shader_shared"; break;
216    case ir_var_shader_in: mode_str = "ir_var_shader_in"; break;
217    case ir_var_shader_out: mode_str = "ir_var_shader_out"; break;
218    case ir_var_function_in: mode_str = "ir_var_function_in"; break;
219    case ir_var_function_out: mode_str = "ir_var_function_out"; break;
220    case ir_var_function_inout: mode_str = "ir_var_function_inout"; break;
221    case ir_var_const_in: mode_str = "ir_var_const_in"; break;
222    case ir_var_system_value: mode_str = "ir_var_system_value"; break;
223    case ir_var_temporary: mode_str = "ir_var_temporary"; break;
224    default:
225       unreachable("Invalid variable mode");
226    }
227 
228    if (ir->data.mode == ir_var_temporary) {
229       print_with_indent("ir_variable *const r%04X = body.make_temp(glsl_type::%s_type, \"%s\");\n",
230                         my_index,
231                         ir->type->name,
232                         ir->name);
233    } else {
234       print_with_indent("ir_variable *const r%04X = new(mem_ctx) ir_variable(glsl_type::%s_type, \"%s\", %s);\n",
235                         my_index,
236                         ir->type->name,
237                         ir->name,
238                         mode_str);
239 
240       switch (ir->data.mode) {
241       case ir_var_function_in:
242       case ir_var_function_out:
243       case ir_var_function_inout:
244       case ir_var_const_in:
245          print_with_indent("sig_parameters.push_tail(r%04X);\n", my_index);
246          break;
247       default:
248          print_with_indent("body.emit(r%04X);\n", my_index);
249          break;
250       }
251    }
252 
253    return visit_continue;
254 }
255 
256 void
print_without_declaration(const ir_dereference_variable * ir)257 ir_builder_print_visitor::print_without_declaration(const ir_dereference_variable *ir)
258 {
259    const struct hash_entry *const he =
260       _mesa_hash_table_search(index_map, ir->var);
261 
262    print_without_indent("r%04X", (unsigned)(uintptr_t) he->data);
263 }
264 
265 ir_visitor_status
visit(ir_dereference_variable * ir)266 ir_builder_print_visitor::visit(ir_dereference_variable *ir)
267 {
268    const struct hash_entry *const he =
269       _mesa_hash_table_search(index_map, ir->var);
270 
271    if (he != NULL)
272       _mesa_hash_table_insert(index_map, ir, he->data);
273 
274    return visit_continue;
275 }
276 
277 ir_visitor_status
visit_enter(ir_function_signature * ir)278 ir_builder_print_visitor::visit_enter(ir_function_signature *ir)
279 {
280    if (!ir->is_defined)
281       return visit_continue_with_parent;
282 
283    print_with_indent("ir_function_signature *\n"
284                      "%s(void *mem_ctx, builtin_available_predicate avail)\n"
285                      "{\n",
286                      ir->function_name());
287    indentation++;
288    print_with_indent("ir_function_signature *const sig =\n");
289    print_with_indent("   new(mem_ctx) ir_function_signature(glsl_type::%s_type, avail);\n",
290                      ir->return_type->name);
291 
292    print_with_indent("ir_factory body(&sig->body, mem_ctx);\n");
293    print_with_indent("sig->is_defined = true;\n\n");
294 
295    if (!ir->parameters.is_empty())
296       print_with_indent("exec_list sig_parameters;\n\n");
297 
298    return visit_continue;
299 }
300 
301 ir_visitor_status
visit_leave(ir_function_signature * ir)302 ir_builder_print_visitor::visit_leave(ir_function_signature *ir)
303 {
304    if (!ir->parameters.is_empty())
305       print_with_indent("sig->replace_parameters(&sig_parameters);\n");
306 
307    print_with_indent("return sig;\n");
308    indentation--;
309    print_with_indent("}\n");
310    return visit_continue;
311 }
312 
313 void
print_without_declaration(const ir_constant * ir)314 ir_builder_print_visitor::print_without_declaration(const ir_constant *ir)
315 {
316   if (ir->type->is_scalar()) {
317       switch (ir->type->base_type) {
318       case GLSL_TYPE_UINT:
319          print_without_indent("body.constant(%uu)", ir->value.u[0]);
320          return;
321       case GLSL_TYPE_INT:
322          print_without_indent("body.constant(int(%d))", ir->value.i[0]);
323          return;
324       case GLSL_TYPE_FLOAT:
325          print_without_indent("body.constant(%ff)", ir->value.f[0]);
326          return;
327       case GLSL_TYPE_BOOL:
328          print_without_indent("body.constant(%s)",
329                               ir->value.i[0] != 0 ? "true" : "false");
330          return;
331       default:
332          break;
333       }
334    }
335 
336    ir_constant_data all_zero;
337    memset(&all_zero, 0, sizeof(all_zero));
338 
339    if (memcmp(&ir->value, &all_zero, sizeof(all_zero)) == 0) {
340       print_without_indent("ir_constant::zero(mem_ctx, glsl_type::%s_type)",
341                            ir->type->name);
342    }
343 }
344 
345 ir_visitor_status
visit(ir_constant * ir)346 ir_builder_print_visitor::visit(ir_constant *ir)
347 {
348    const unsigned my_index = next_ir_index++;
349 
350    _mesa_hash_table_insert(index_map, ir, (void *)(uintptr_t) my_index);
351 
352    if (ir->type == glsl_type::uint_type ||
353        ir->type == glsl_type::int_type ||
354        ir->type == glsl_type::float_type ||
355        ir->type == glsl_type::bool_type) {
356       print_with_indent("ir_constant *const r%04X = ", my_index);
357       print_without_declaration(ir);
358       print_without_indent(";\n");
359       return visit_continue;
360    }
361 
362    ir_constant_data all_zero;
363    memset(&all_zero, 0, sizeof(all_zero));
364 
365    if (memcmp(&ir->value, &all_zero, sizeof(all_zero)) == 0) {
366       print_with_indent("ir_constant *const r%04X = ", my_index);
367       print_without_declaration(ir);
368       print_without_indent(";\n");
369    } else {
370       print_with_indent("ir_constant_data r%04X_data;\n", my_index);
371       print_with_indent("memset(&r%04X_data, 0, sizeof(ir_constant_data));\n",
372                         my_index);
373       for (unsigned i = 0; i < 16; i++) {
374          switch (ir->type->base_type) {
375          case GLSL_TYPE_UINT:
376             if (ir->value.u[i] != 0)
377                print_with_indent("r%04X_data.u[%u] = %u;\n",
378                                     my_index, i, ir->value.u[i]);
379             break;
380          case GLSL_TYPE_INT:
381             if (ir->value.i[i] != 0)
382                print_with_indent("r%04X_data.i[%u] = %i;\n",
383                                     my_index, i, ir->value.i[i]);
384             break;
385          case GLSL_TYPE_FLOAT:
386             if (ir->value.u[i] != 0)
387                print_with_indent("r%04X_data.u[%u] = 0x%08x; /* %f */\n",
388                                     my_index,
389                                     i,
390                                     ir->value.u[i],
391                                     ir->value.f[i]);
392             break;
393          case GLSL_TYPE_DOUBLE: {
394             uint64_t v;
395 
396             STATIC_ASSERT(sizeof(double) == sizeof(uint64_t));
397 
398             memcpy(&v, &ir->value.d[i], sizeof(v));
399             if (v != 0)
400                print_with_indent("r%04X_data.u64[%u] = 0x%016" PRIx64 "; /* %g */\n",
401                                     my_index, i, v, ir->value.d[i]);
402             break;
403          }
404          case GLSL_TYPE_UINT64:
405             if (ir->value.u64[i] != 0)
406                print_with_indent("r%04X_data.u64[%u] = %" PRIu64 ";\n",
407                                     my_index,
408                                     i,
409                                     ir->value.u64[i]);
410             break;
411          case GLSL_TYPE_INT64:
412             if (ir->value.i64[i] != 0)
413                print_with_indent("r%04X_data.i64[%u] = %" PRId64 ";\n",
414                                     my_index,
415                                     i,
416                                     ir->value.i64[i]);
417             break;
418          case GLSL_TYPE_BOOL:
419             if (ir->value.u[i] != 0)
420                print_with_indent("r%04X_data.u[%u] = 1;\n", my_index, i);
421             break;
422          default:
423             unreachable("Invalid constant type");
424          }
425       }
426 
427       print_with_indent("ir_constant *const r%04X = new(mem_ctx) ir_constant(glsl_type::%s_type, &r%04X_data);\n",
428                         my_index,
429                         ir->type->name,
430                         my_index);
431    }
432 
433    return visit_continue;
434 }
435 
436 void
print_without_declaration(const ir_swizzle * ir)437 ir_builder_print_visitor::print_without_declaration(const ir_swizzle *ir)
438 {
439    const struct hash_entry *const he =
440       _mesa_hash_table_search(index_map, ir->val);
441 
442    if (ir->mask.num_components == 1) {
443       static const char swiz[4] = { 'x', 'y', 'z', 'w' };
444 
445       if (is_simple_operand(ir->val)) {
446          print_without_indent("swizzle_%c(", swiz[ir->mask.x]);
447          print_without_declaration(ir->val);
448          print_without_indent(")");
449       } else {
450          assert(he);
451          print_without_indent("swizzle_%c(r%04X)",
452                               swiz[ir->mask.x],
453                               (unsigned)(uintptr_t) he->data);
454       }
455    } else {
456       static const char swiz[4] = { 'X', 'Y', 'Z', 'W' };
457 
458       assert(he);
459       print_without_indent("swizzle(r%04X, MAKE_SWIZZLE4(SWIZZLE_%c, SWIZZLE_%c, SWIZZLE_%c, SWIZZLE_%c), %u)",
460                            (unsigned)(uintptr_t) he->data,
461                            swiz[ir->mask.x],
462                            swiz[ir->mask.y],
463                            swiz[ir->mask.z],
464                            swiz[ir->mask.w],
465                            ir->mask.num_components);
466    }
467 }
468 
469 ir_visitor_status
visit_leave(ir_swizzle * ir)470 ir_builder_print_visitor::visit_leave(ir_swizzle *ir)
471 {
472    const unsigned my_index = next_ir_index++;
473 
474    _mesa_hash_table_insert(index_map, ir, (void *)(uintptr_t) my_index);
475 
476    print_with_indent("ir_swizzle *const r%04X = ", my_index);
477    print_without_declaration(ir);
478    print_without_indent(";\n");
479 
480    return visit_continue;
481 }
482 
483 ir_visitor_status
visit_enter(ir_assignment * ir)484 ir_builder_print_visitor::visit_enter(ir_assignment *ir)
485 {
486    ir_expression *const rhs_expr = ir->rhs->as_expression();
487 
488    if (!is_simple_operand(ir->rhs) && rhs_expr == NULL)
489       return visit_continue;
490 
491    if (rhs_expr != NULL) {
492       const unsigned num_op = rhs_expr->num_operands;
493 
494       for (unsigned i = 0; i < num_op; i++) {
495          if (is_simple_operand(rhs_expr->operands[i]))
496             continue;
497 
498          rhs_expr->operands[i]->accept(this);
499       }
500    }
501 
502    ir_visitor_status s;
503 
504    this->in_assignee = true;
505    s = ir->lhs->accept(this);
506    this->in_assignee = false;
507    if (s != visit_continue)
508       return (s == visit_continue_with_parent) ? visit_continue : s;
509 
510    const struct hash_entry *const he_lhs =
511       _mesa_hash_table_search(index_map, ir->lhs);
512 
513    print_with_indent("body.emit(assign(r%04X, ",
514                      (unsigned)(uintptr_t) he_lhs->data);
515    print_without_declaration(ir->rhs);
516    print_without_indent(", 0x%02x));\n\n", ir->write_mask);
517 
518    return visit_continue_with_parent;
519 }
520 
521 ir_visitor_status
visit_leave(ir_assignment * ir)522 ir_builder_print_visitor::visit_leave(ir_assignment *ir)
523 {
524    const struct hash_entry *const he_lhs =
525       _mesa_hash_table_search(index_map, ir->lhs);
526 
527    const struct hash_entry *const he_rhs =
528       _mesa_hash_table_search(index_map, ir->rhs);
529 
530    assert(ir->lhs && ir->rhs);
531 
532    print_with_indent("body.emit(assign(r%04X, r%04X, 0x%02x));\n\n",
533                      (unsigned)(uintptr_t) he_lhs->data,
534                      (unsigned)(uintptr_t) he_rhs->data,
535                      ir->write_mask);
536 
537    return visit_continue;
538 }
539 
540 void
print_without_declaration(const ir_expression * ir)541 ir_builder_print_visitor::print_without_declaration(const ir_expression *ir)
542 {
543    const unsigned num_op = ir->num_operands;
544 
545    static const char *const arity[] = {
546       "", "unop", "binop", "triop", "quadop"
547    };
548 
549    switch (ir->operation) {
550    case ir_unop_neg:
551    case ir_binop_add:
552    case ir_binop_sub:
553    case ir_binop_mul:
554    case ir_binop_imul_high:
555    case ir_binop_less:
556    case ir_binop_gequal:
557    case ir_binop_equal:
558    case ir_binop_nequal:
559    case ir_binop_lshift:
560    case ir_binop_rshift:
561    case ir_binop_bit_and:
562    case ir_binop_bit_xor:
563    case ir_binop_bit_or:
564    case ir_binop_logic_and:
565    case ir_binop_logic_xor:
566    case ir_binop_logic_or:
567       print_without_indent("%s(",
568                            ir_expression_operation_enum_strings[ir->operation]);
569       break;
570    default:
571       print_without_indent("expr(ir_%s_%s, ",
572                            arity[num_op],
573                            ir_expression_operation_enum_strings[ir->operation]);
574       break;
575    }
576 
577    for (unsigned i = 0; i < num_op; i++) {
578       if (is_simple_operand(ir->operands[i]))
579          print_without_declaration(ir->operands[i]);
580       else {
581          const struct hash_entry *const he =
582             _mesa_hash_table_search(index_map, ir->operands[i]);
583 
584          print_without_indent("r%04X", (unsigned)(uintptr_t) he->data);
585       }
586 
587       if (i < num_op - 1)
588          print_without_indent(", ");
589    }
590 
591    print_without_indent(")");
592 }
593 
594 ir_visitor_status
visit_enter(ir_expression * ir)595 ir_builder_print_visitor::visit_enter(ir_expression *ir)
596 {
597    const unsigned num_op = ir->num_operands;
598 
599    for (unsigned i = 0; i < num_op; i++) {
600       if (is_simple_operand(ir->operands[i]))
601          continue;
602 
603       ir->operands[i]->accept(this);
604    }
605 
606    const unsigned my_index = next_ir_index++;
607 
608    _mesa_hash_table_insert(index_map, ir, (void *)(uintptr_t) my_index);
609 
610    print_with_indent("ir_expression *const r%04X = ", my_index);
611    print_without_declaration(ir);
612    print_without_indent(";\n");
613 
614    return visit_continue_with_parent;
615 }
616 
617 ir_visitor_status
visit_enter(ir_if * ir)618 ir_builder_print_visitor::visit_enter(ir_if *ir)
619 {
620    const unsigned my_index = next_ir_index++;
621 
622    print_with_indent("/* IF CONDITION */\n");
623 
624    ir_visitor_status s = ir->condition->accept(this);
625    if (s != visit_continue)
626       return (s == visit_continue_with_parent) ? visit_continue : s;
627 
628    const struct hash_entry *const he =
629       _mesa_hash_table_search(index_map, ir->condition);
630 
631    print_with_indent("ir_if *f%04X = new(mem_ctx) ir_if(operand(r%04X).val);\n",
632                      my_index,
633                      (unsigned)(uintptr_t) he->data);
634    print_with_indent("exec_list *const f%04X_parent_instructions = body.instructions;\n\n",
635                      my_index);
636 
637    indentation++;
638    print_with_indent("/* THEN INSTRUCTIONS */\n");
639    print_with_indent("body.instructions = &f%04X->then_instructions;\n\n",
640                      my_index);
641 
642    if (s != visit_continue_with_parent) {
643       s = visit_list_elements(this, &ir->then_instructions);
644       if (s == visit_stop)
645         return s;
646    }
647 
648    print_without_indent("\n");
649 
650    if (!ir->else_instructions.is_empty()) {
651       print_with_indent("/* ELSE INSTRUCTIONS */\n");
652       print_with_indent("body.instructions = &f%04X->else_instructions;\n\n",
653               my_index);
654 
655       if (s != visit_continue_with_parent) {
656          s = visit_list_elements(this, &ir->else_instructions);
657          if (s == visit_stop)
658             return s;
659       }
660 
661       print_without_indent("\n");
662    }
663 
664    indentation--;
665 
666    print_with_indent("body.instructions = f%04X_parent_instructions;\n",
667                      my_index);
668    print_with_indent("body.emit(f%04X);\n\n",
669                      my_index);
670    print_with_indent("/* END IF */\n\n");
671 
672    return visit_continue_with_parent;
673 }
674 
675 ir_visitor_status
visit_leave(ir_return * ir)676 ir_builder_print_visitor::visit_leave(ir_return *ir)
677 {
678    const struct hash_entry *const he =
679       _mesa_hash_table_search(index_map, ir->value);
680 
681    print_with_indent("body.emit(ret(r%04X));\n\n",
682                      (unsigned)(uintptr_t) he->data);
683 
684    return visit_continue;
685 }
686 
687 ir_visitor_status
visit_enter(ir_texture * ir)688 ir_builder_print_visitor::visit_enter(ir_texture *ir)
689 {
690    print_with_indent("\nUnsupported IR is encountered: texture functions are not supported. Exiting.\n");
691 
692    return visit_stop;
693 }
694 
695 ir_visitor_status
visit_leave(ir_call * ir)696 ir_builder_print_visitor::visit_leave(ir_call *ir)
697 {
698    const unsigned my_index = next_ir_index++;
699 
700    print_without_indent("\n");
701    print_with_indent("/* CALL %s */\n", ir->callee_name());
702    print_with_indent("exec_list r%04X_parameters;\n", my_index);
703 
704    foreach_in_list(ir_dereference_variable, param, &ir->actual_parameters) {
705       const struct hash_entry *const he =
706          _mesa_hash_table_search(index_map, param);
707 
708       print_with_indent("r%04X_parameters.push_tail(operand(r%04X).val);\n",
709                         my_index,
710                         (unsigned)(uintptr_t) he->data);
711    }
712 
713    char return_deref_string[32];
714    if (ir->return_deref) {
715       const struct hash_entry *const he =
716          _mesa_hash_table_search(index_map, ir->return_deref);
717 
718       snprintf(return_deref_string, sizeof(return_deref_string),
719                "operand(r%04X).val", (unsigned)(uintptr_t) he->data);
720    } else {
721       strcpy(return_deref_string, "NULL");
722    }
723 
724    print_with_indent("body.emit(new(mem_ctx) ir_call(shader->symbols->get_function(\"%s\"),\n",
725                      ir->callee_name());
726    print_with_indent("                               %s, &r%04X_parameters);\n\n",
727                      return_deref_string,
728                      my_index);
729    return visit_continue;
730 }
731 
732 ir_visitor_status
visit_enter(ir_loop * ir)733 ir_builder_print_visitor::visit_enter(ir_loop *ir)
734 {
735    const unsigned my_index = next_ir_index++;
736 
737    _mesa_hash_table_insert(index_map, ir, (void *)(uintptr_t) my_index);
738 
739    print_with_indent("/* LOOP BEGIN */\n");
740    print_with_indent("ir_loop *f%04X = new(mem_ctx) ir_loop();\n", my_index);
741    print_with_indent("exec_list *const f%04X_parent_instructions = body.instructions;\n\n",
742                      my_index);
743 
744    indentation++;
745 
746    print_with_indent("body.instructions = &f%04X->body_instructions;\n\n",
747                      my_index);
748 
749    return visit_continue;
750 }
751 
752 ir_visitor_status
visit_leave(ir_loop * ir)753 ir_builder_print_visitor::visit_leave(ir_loop *ir)
754 {
755    const struct hash_entry *const he =
756       _mesa_hash_table_search(index_map, ir);
757 
758    indentation--;
759 
760    print_with_indent("/* LOOP END */\n\n");
761    print_with_indent("body.instructions = f%04X_parent_instructions;\n",
762                      (unsigned)(uintptr_t) he->data);
763    print_with_indent("body.emit(f%04X);\n\n",
764                      (unsigned)(uintptr_t) he->data);
765 
766    return visit_continue;
767 }
768 
769 ir_visitor_status
visit(ir_loop_jump * ir)770 ir_builder_print_visitor::visit(ir_loop_jump *ir)
771 {
772    print_with_indent("body.emit(new(mem_ctx) ir_loop_jump(ir_loop_jump::jump_%s));\n\n",
773                      ir->is_break() ? "break" : "continue");
774    return visit_continue;
775 }
776