2022-11-08 11:32:46 +03:00
|
|
|
/*
|
|
|
|
* IR - Lightweight JIT Compilation Framework
|
|
|
|
* (IR saver)
|
|
|
|
* Copyright (C) 2022 Zend by Perforce.
|
|
|
|
* Authors: Dmitry Stogov <dmitry@php.net>
|
|
|
|
*/
|
|
|
|
|
2022-04-06 00:19:23 +03:00
|
|
|
#include "ir.h"
|
2022-05-27 13:18:04 +03:00
|
|
|
#include "ir_private.h"
|
2022-04-06 00:19:23 +03:00
|
|
|
|
2023-11-30 21:10:41 +03:00
|
|
|
void ir_print_proto(const ir_ctx *ctx, ir_ref func_proto, FILE *f)
|
|
|
|
{
|
|
|
|
ir_ref j;
|
|
|
|
|
|
|
|
if (func_proto) {
|
|
|
|
const ir_proto_t *proto = (const ir_proto_t *)ir_get_str(ctx, func_proto);
|
|
|
|
|
|
|
|
fprintf(f, "(");
|
|
|
|
if (proto->params_count > 0) {
|
|
|
|
fprintf(f, "%s", ir_type_cname[proto->param_types[0]]);
|
|
|
|
for (j = 1; j < proto->params_count; j++) {
|
|
|
|
fprintf(f, ", %s", ir_type_cname[proto->param_types[j]]);
|
|
|
|
}
|
|
|
|
if (proto->flags & IR_VARARG_FUNC) {
|
|
|
|
fprintf(f, ", ...");
|
|
|
|
}
|
|
|
|
} else if (proto->flags & IR_VARARG_FUNC) {
|
|
|
|
fprintf(f, "...");
|
|
|
|
}
|
|
|
|
fprintf(f, "): %s", ir_type_cname[proto->ret_type]);
|
|
|
|
if (proto->flags & IR_FASTCALL_FUNC) {
|
|
|
|
fprintf(f, " __fastcall");
|
2023-12-06 12:47:03 +03:00
|
|
|
} else if (proto->flags & IR_BUILTIN_FUNC) {
|
|
|
|
fprintf(f, " __builtin");
|
2023-11-30 21:10:41 +03:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fprintf(f, "(): int32_t");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-28 13:18:12 +03:00
|
|
|
void ir_save(const ir_ctx *ctx, FILE *f)
|
2022-04-06 00:19:23 +03:00
|
|
|
{
|
|
|
|
ir_ref i, j, n, ref, *p;
|
|
|
|
ir_insn *insn;
|
|
|
|
uint32_t flags;
|
|
|
|
bool first;
|
|
|
|
|
|
|
|
fprintf(f, "{\n");
|
|
|
|
for (i = IR_UNUSED + 1, insn = ctx->ir_base - i; i < ctx->consts_count; i++, insn--) {
|
|
|
|
fprintf(f, "\t%s c_%d = ", ir_type_cname[insn->type], i);
|
|
|
|
if (insn->op == IR_FUNC) {
|
2023-11-30 21:10:41 +03:00
|
|
|
fprintf(f, "func %s", ir_get_str(ctx, insn->val.name));
|
|
|
|
ir_print_proto(ctx, insn->proto, f);
|
2023-10-18 11:45:57 +03:00
|
|
|
} else if (insn->op == IR_SYM) {
|
2023-11-30 21:10:41 +03:00
|
|
|
fprintf(f, "sym(%s)", ir_get_str(ctx, insn->val.name));
|
2022-06-23 13:14:30 +03:00
|
|
|
} else if (insn->op == IR_FUNC_ADDR) {
|
2023-11-30 21:10:41 +03:00
|
|
|
fprintf(f, "func *");
|
2023-05-22 20:48:42 +08:00
|
|
|
ir_print_const(ctx, insn, f, true);
|
2023-11-30 21:10:41 +03:00
|
|
|
ir_print_proto(ctx, insn->proto, f);
|
2022-04-06 00:19:23 +03:00
|
|
|
} else {
|
2023-05-22 20:48:42 +08:00
|
|
|
ir_print_const(ctx, insn, f, true);
|
2022-04-06 00:19:23 +03:00
|
|
|
}
|
|
|
|
fprintf(f, ";\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = IR_UNUSED + 1, insn = ctx->ir_base + i; i < ctx->insns_count;) {
|
|
|
|
flags = ir_op_flags[insn->op];
|
|
|
|
if (flags & IR_OP_FLAG_CONTROL) {
|
|
|
|
if (!(flags & IR_OP_FLAG_MEM) || insn->type == IR_VOID) {
|
|
|
|
fprintf(f, "\tl_%d = ", i);
|
|
|
|
} else {
|
|
|
|
fprintf(f, "\t%s d_%d, l_%d = ", ir_type_cname[insn->type], i, i);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fprintf(f, "\t");
|
|
|
|
if (flags & IR_OP_FLAG_DATA) {
|
2023-06-21 22:36:36 +03:00
|
|
|
fprintf(f, "%s d_%d = ", ir_type_cname[insn->type], i);
|
2022-04-06 00:19:23 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(f, "%s", ir_op_name[insn->op]);
|
|
|
|
n = ir_operands_count(ctx, insn);
|
2022-06-16 12:31:23 +03:00
|
|
|
if ((insn->op == IR_MERGE || insn->op == IR_LOOP_BEGIN) && n != 2) {
|
2022-04-06 00:19:23 +03:00
|
|
|
fprintf(f, "/%d", n);
|
|
|
|
} else if ((insn->op == IR_CALL || insn->op == IR_TAILCALL) && n != 2) {
|
|
|
|
fprintf(f, "/%d", n - 2);
|
|
|
|
} else if (insn->op == IR_PHI && n != 3) {
|
|
|
|
fprintf(f, "/%d", n - 1);
|
2022-10-21 17:16:25 +03:00
|
|
|
} else if (insn->op == IR_SNAPSHOT) {
|
|
|
|
fprintf(f, "/%d", n - 1);
|
2022-04-06 00:19:23 +03:00
|
|
|
}
|
|
|
|
first = 1;
|
|
|
|
for (j = 1, p = insn->ops + 1; j <= n; j++, p++) {
|
|
|
|
uint32_t opnd_kind = IR_OPND_KIND(flags, j);
|
|
|
|
|
|
|
|
ref = *p;
|
|
|
|
if (ref) {
|
|
|
|
switch (opnd_kind) {
|
|
|
|
case IR_OPND_DATA:
|
|
|
|
if (IR_IS_CONST_REF(ref)) {
|
|
|
|
fprintf(f, "%sc_%d", first ? "(" : ", ", -ref);
|
|
|
|
} else {
|
|
|
|
fprintf(f, "%sd_%d", first ? "(" : ", ", ref);
|
|
|
|
}
|
|
|
|
first = 0;
|
|
|
|
break;
|
|
|
|
case IR_OPND_CONTROL:
|
|
|
|
case IR_OPND_CONTROL_DEP:
|
|
|
|
case IR_OPND_CONTROL_REF:
|
|
|
|
fprintf(f, "%sl_%d", first ? "(" : ", ", ref);
|
|
|
|
first = 0;
|
|
|
|
break;
|
|
|
|
case IR_OPND_STR:
|
|
|
|
fprintf(f, "%s\"%s\"", first ? "(" : ", ", ir_get_str(ctx, ref));
|
|
|
|
first = 0;
|
|
|
|
break;
|
2023-11-30 21:10:41 +03:00
|
|
|
case IR_OPND_PROTO:
|
|
|
|
fprintf(f, "%sfunc ", first ? "(" : ", ");
|
|
|
|
ir_print_proto(ctx, ref, f);
|
|
|
|
break;
|
2022-05-24 12:47:39 +03:00
|
|
|
case IR_OPND_PROB:
|
|
|
|
if (ref == 0) {
|
|
|
|
break;
|
|
|
|
}
|
2022-11-08 23:09:35 +03:00
|
|
|
IR_FALLTHROUGH;
|
2022-04-06 00:19:23 +03:00
|
|
|
case IR_OPND_NUM:
|
|
|
|
fprintf(f, "%s%d", first ? "(" : ", ", ref);
|
|
|
|
first = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (opnd_kind == IR_OPND_NUM) {
|
|
|
|
fprintf(f, "%s%d", first ? "(" : ", ", ref);
|
|
|
|
first = 0;
|
2023-11-14 17:12:20 +03:00
|
|
|
} else if (j != n &&
|
|
|
|
(IR_IS_REF_OPND_KIND(opnd_kind) || (opnd_kind == IR_OPND_UNUSED && p[n-j]))) {
|
2022-10-18 15:52:25 +03:00
|
|
|
fprintf(f, "%snull", first ? "(" : ", ");
|
|
|
|
first = 0;
|
2022-04-06 00:19:23 +03:00
|
|
|
}
|
|
|
|
}
|
2022-09-15 15:26:43 +03:00
|
|
|
if (first) {
|
|
|
|
fprintf(f, ";");
|
|
|
|
} else {
|
|
|
|
fprintf(f, ");");
|
|
|
|
}
|
2023-03-21 18:06:47 +03:00
|
|
|
if (((flags & IR_OP_FLAG_DATA) || ((flags & IR_OP_FLAG_MEM) && insn->type != IR_VOID)) && ctx->binding) {
|
2022-09-15 15:26:43 +03:00
|
|
|
ir_ref var = ir_binding_find(ctx, i);
|
|
|
|
if (var) {
|
2023-06-16 02:14:02 +03:00
|
|
|
IR_ASSERT(var < 0);
|
|
|
|
fprintf(f, " # BIND(0x%x);", -var);
|
2022-09-15 15:26:43 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(f, "\n");
|
2023-04-21 13:40:55 +03:00
|
|
|
n = ir_insn_inputs_to_len(n);
|
2022-04-06 00:19:23 +03:00
|
|
|
i += n;
|
|
|
|
insn += n;
|
|
|
|
}
|
|
|
|
fprintf(f, "}\n");
|
|
|
|
}
|