Remove unused emitter templates.
[libfirm] / ir / be / mips / mips_spec.pl
index 61664c4..df91ab4 100644 (file)
@@ -1,85 +1,9 @@
 # Creation: 2006/02/13
 # $Id$
-# This is a template specification for the Firm-Backend
 
 # the cpu architecture (ia32, ia64, mips, sparc, ppc, ...)
 
 $arch = "mips";
-$new_emit_syntax = 1;
-
-# The node description is done as a perl hash initializer with the
-# following structure:
-#
-# %nodes = (
-#
-# <op-name> => {
-#   op_flags  => "N|L|C|X|I|F|Y|H|c|K",
-#   "arity"     => "0|1|2|3 ... |variable|dynamic|any",
-#   "state"     => "floats|pinned|mem_pinned|exc_pinned",
-#   "args"      => [
-#                    { "type" => "type 1", "name" => "name 1" },
-#                    { "type" => "type 2", "name" => "name 2" },
-#                    ...
-#                  ],
-#   "comment"   => "any comment for constructor",
-#   reg_req   => { in => [ "reg_class|register" ], out => [ "reg_class|register|in_rX" ] },
-#   "cmp_attr"  => "c source code for comparing node attributes",
-#   emit      => "emit code with templates",
-#   "rd_constructor" => "c source code which constructs an ir_node"
-# },
-#
-# ... # (all nodes you need to describe)
-#
-# ); # close the %nodes initializer
-
-# op_flags: flags for the operation, OPTIONAL (default is "N")
-# the op_flags correspond to the firm irop_flags:
-#   N   irop_flag_none
-#   L   irop_flag_labeled
-#   C   irop_flag_commutative
-#   X   irop_flag_cfopcode
-#   I   irop_flag_ip_cfopcode
-#   F   irop_flag_fragile
-#   Y   irop_flag_forking
-#   H   irop_flag_highlevel
-#   c   irop_flag_constlike
-#   K   irop_flag_keep
-#
-#   R   rematerializeable
-#   N   not spillable
-#   I   ignore for register allocation
-#
-# state: state of the operation, OPTIONAL (default is "floats")
-#
-# arity: arity of the operation, MUST NOT BE OMITTED
-#
-# args:  the OPTIONAL arguments of the node constructor (debug, irg and block
-#        are always the first 3 arguments and are always autmatically
-#        created)
-#        If this key is missing the following arguments will be created:
-#        for i = 1 .. arity: ir_node *op_i
-#        ir_mode *mode
-#
-# outs:  if a node defines more than one output, the names of the projections
-#        nodes having outs having automatically the mode mode_T
-#
-# comment: OPTIONAL comment for the node constructor
-#
-# rd_constructor: for every operation there will be a
-#      new_rd_<arch>_<op-name> function with the arguments from above
-#      which creates the ir_node corresponding to the defined operation
-#      you can either put the complete source code of this function here
-#
-#      This key is OPTIONAL. If omitted, the following constructor will
-#      be created:
-#      if (!op_<arch>_<op-name>) assert(0);
-#      for i = 1 to arity
-#         set in[i] = op_i
-#      done
-#      res = new_ir_node(db, irg, block, op_<arch>_<op-name>, mode, arity, in)
-#      return res
-#
-# NOTE: rd_constructor and args are only optional if and only if arity is 0,1,2 or 3
 
 # register types:
 #   0 - no special type
@@ -89,7 +13,7 @@ $new_emit_syntax = 1;
 # NOTE: Last entry of each class is the largest Firm-Mode a register can hold\
 %reg_classes = (
        "gp" => [
-               { name => "zero", type => 4+2 },  # always zero
+               { name => "zero", type => 4 },  # always zero
                { name => "at", type => 4 }, # reserved for assembler
                { name => "v0", realname => "2", type => 1 }, # first return value
                { name => "v1", realname => "3", type => 1 }, # second return value
@@ -121,23 +45,44 @@ $new_emit_syntax = 1;
                { name => "sp", type => 4 }, # stack pointer
                { name => "fp", type => 4 }, # frame pointer
                { name => "ra", type => 2+1 }, # return address
+               { name => "gp_NOREG", realname => "!NOREG_INVALID!", type => 4 | 8 | 16 }, #dummy register for immediate nodes
                { mode => "mode_Iu" }
        ],
 ); # %reg_classes
 
 %emit_templates = (
-    S0 => "${arch}_emit_source_register(env, node, 0);",
-    S1 => "${arch}_emit_source_register(env, node, 1);",
-    S2 => "${arch}_emit_source_register(env, node, 2);",
-    D0 => "${arch}_emit_dest_register(env, node, 0);",
-    D1 => "${arch}_emit_dest_register(env, node, 1);",
-    D2 => "${arch}_emit_dest_register(env, node, 2);",
-       C  => "${arch}_emit_immediate(env, node);",
-       JumpTarget => "${arch}_emit_jump_target(env, node);",
-       JumpTarget1 => "${arch}_emit_jump_target_proj(env, node, 1);",
-       JumpOrFallthrough => "${arch}_emit_jump_or_fallthrough(env, node, 0);",
+    S0  => "${arch}_emit_source_register(node, 0);",
+    S1  => "${arch}_emit_source_register(node, 1);",
+       SI1 => "${arch}_emit_source_register_or_immediate(node, 1);",
+    D0  => "${arch}_emit_dest_register(node, 0);",
+       A0  => "${arch}_emit_load_store_address(node, 0);",
+       I   => "${arch}_emit_immediate_suffix(node, 1);",
+       C   => "${arch}_emit_immediate(node);",
+       JumpTarget => "${arch}_emit_jump_target(node);",
+       JumpTarget1 => "${arch}_emit_jump_target_proj(node, 1);",
+       JumpOrFallthrough => "${arch}_emit_jump_or_fallthrough(node, 0);",
 );
 
+$default_attr_type = "mips_attr_t";
+$default_copy_attr = "mips_copy_attr";
+
+$mode_gp = "mode_Iu";
+
+%init_attr = (
+       mips_attr_t            => "\tinit_mips_attributes(res, flags, in_reqs, exec_units, n_res);",
+
+       mips_immediate_attr_t  => "\tinit_mips_attributes(res, flags, in_reqs, exec_units, n_res);\n".
+                                "\tinit_mips_immediate_attributes(res, imm_type, entity, val);",
+
+       mips_load_store_attr_t => "\tinit_mips_attributes(res, flags, in_reqs, exec_units, n_res);\n".
+                                "\tinit_mips_load_store_attributes(res, entity, offset);",
+);
+
+%compare_attr = (
+       mips_attr_t            => "mips_compare_nodes_attr",
+       mips_immediate_attr_t  => "mips_compare_immediate_attr",
+       mips_load_store_attr_t => "mips_compare_load_store_attr",
+);
 
 #--------------------------------------------------#
 #                        _                         #
@@ -152,6 +97,15 @@ $new_emit_syntax = 1;
 
 %nodes = (
 
+Immediate => {
+       state     => "pinned",
+       op_flags  => "c",
+       reg_req   => { out => [ "gp_NOREG:I" ] },
+       attr      => "mips_immediate_type_t imm_type, ir_entity *entity, long val",
+       attr_type => "mips_immediate_attr_t",
+       mode      => $mode_gp,
+},
+
 #-----------------------------------------------------------------#
 #  _       _                                         _            #
 # (_)     | |                                       | |           #
@@ -166,219 +120,154 @@ $new_emit_syntax = 1;
 # commutative operations
 
 addu => {
-       op_flags  => "C",
+       op_flags  => "R",
        reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '. addu %D0, %S0, %S1',
-       mode      => "mode_Iu",
-},
-
-addiu => {
-       reg_req   => { in => [ "gp" ], out => [ "gp" ] },
-       emit      => '. addiu %D0, %S0, %C',
-       cmp_attr  => 'return attr_a->tv != attr_b->tv;',
-       mode      => "mode_Iu",
+       ins       => [ "left", "right" ],
+       emit      => '. add%I%.u %D0, %S0, %SI1',
+       mode      => $mode_gp,
 },
 
 and => {
-       op_flags  => "C",
+       op_flags  => "R",
        reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '. and %D0, %S0, %S1',
-       mode      => "mode_Iu",
-},
-
-andi => {
-       reg_req   => { in => [ "gp" ], out => [ "gp" ] },
-       emit      => '. andi %D0, %S0, %C',
-       cmp_attr  => 'return attr_a->tv != attr_b->tv;',
-       mode      => "mode_Iu",
+       ins       => [ "left", "right" ],
+       emit      => '. and%I %D0, %S0, %SI1',
+       mode      => $mode_gp,
 },
 
 div => {
-       reg_req   => { in => [ "gp", "gp" ], out => [ "none" ] },
+       reg_req   => { in => [ "gp", "gp" ], out => [ "none", "none" ] },
+       ins       => [ "left", "right" ],
+       outs      => [ "lohi", "M" ],
        emit      => '. div %S0, %S1',
        mode      => "mode_M",
 },
 
 divu => {
-       reg_req   => { in => [ "gp", "gp" ], out => [ "none" ] },
+       reg_req   => { in => [ "gp", "gp" ], out => [ "none", "none" ] },
+       ins       => [ "left", "right" ],
+       outs      => [ "lohi", "M" ],
        emit      => '. divu %S0, %S1',
        mode      => "mode_M",
 },
 
 mult => {
-       op_flags  => "C",
        reg_req   => { in => [ "gp", "gp" ], out => [ "none" ] },
+       ins       => [ "left", "right" ],
        emit      => '. mult %S0, %S1',
        mode      => "mode_M"
 },
 
 multu => {
-       op_flags  => "C",
        reg_req   => { in => [ "gp", "gp" ], out => [ "none" ] },
+       ins       => [ "left", "right" ],
        emit      => '. multu %S0, %S1',
        mode      => "mode_M",
 },
 
 nor => {
-       op_flags  => "C",
+       op_flags  => "R",
        reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '. nor %D0, %S0, %S1',
-       mode      => "mode_Iu"
-},
-
-not => {
-       reg_req   => { in => [ "gp" ], out => [ "gp" ] },
-       emit      => '. nor %D0, %S0, $zero',
-       mode      => "mode_Iu"
+       emit      => '. nor%I %D0, %S0, %SI1',
+       mode      => $mode_gp
 },
 
 or => {
-       op_flags  => "C",
+       op_flags  => "R",
        reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '. or %D0, %S0, %S1',
-       mode      => "mode_Iu"
+       ins       => [ "left", "right" ],
+       emit      => '. or%I %D0, %S0, %SI1',
+       mode      => $mode_gp
 },
 
-ori => {
-       reg_req   => { in => [ "gp" ], out => [ "gp" ] },
-       emit      => '. ori %D0, %S0, %C',
-       cmp_attr  => 'return attr_a->tv != attr_b->tv;',
-       mode      => "mode_Iu"
-},
-
-sl => {
+sll => {
+       op_flags  => "R",
        reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '
-       if (mode_is_signed(get_irn_mode(node))) {
-               . sal %D0, %S0, %S1
-       } else {
-               . sll %D0, %S0, %S1
-       }
-',
-       mode      => "mode_Iu",
-},
-
-sli => {
-       reg_req   => { in => [ "gp" ], out => [ "gp" ] },
-       emit      => '
-       if (mode_is_signed(get_irn_mode(node))) {
-               . sal %D0, %S0, %C
-       } else {
-               . sll %D0, %S0, %C
-       }
-',
-       mode      => "mode_Iu",
+       ins       => [ "left", "right" ],
+       emit      => '. sll %D0, %S0, %SI1',
+       mode      => $mode_gp,
 },
 
 sra => {
+       op_flags  => "R",
        reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '. sra %D0, %S0, %S1',
-       mode      => "mode_Iu"
+       ins       => [ "left", "right" ],
+       emit      => '. sra %D0, %S0, %SI1',
+       mode      => $mode_gp
 },
 
-srai => {
-       reg_req   => { in => [ "gp" ], out => [ "gp" ] },
-       emit      => '. sra %D0, %S0, %C',
-       cmp_attr  => 'return attr_a->tv != attr_b->tv;',
-       mode      => "mode_Iu",
-},
-
-sr => {
+srl => {
+       op_flags  => "R",
        reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '
-       if (mode_is_signed(get_irn_mode(node))) {
-               . sra %D0, %S0, %S1
-       } else {
-               . srl %D0, %S0, %S1
-       }
-',
-       mode      => "mode_Iu",
-},
-
-sri => {
-       reg_req   => { in => [ "gp" ], out => [ "gp" ] },
-       emit      => '
-       if (mode_is_signed(get_irn_mode(node))) {
-               . sra %D0, %S0, %C
-       } else {
-               . srl %D0, %S0, %C
-       }
-',
-       mode      => "mode_Iu"
+       ins       => [ "left", "right" ],
+       emit      => '. srl %D0, %S0, %SI1',
+       mode      => $mode_gp,
 },
 
-srlv => {
-       reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '. srlv %D0, %S0, %S1',
-       mode      => "mode_Iu"
-},
-
-sllv => {
-       reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '. sllv %D0, %S0, %S1',
-       mode      => "mode_Iu"
-},
-
-sub => {
+subu => {
+       op_flags  => "R",
        reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
+       ins       => [ "left", "right" ],
        emit      => '. subu %D0, %S0, %S1',
-       mode      => "mode_Iu"
-},
-
-subuzero => {
-       reg_req   => { in => [ "gp" ], out => [ "gp" ] },
-       emit      => '. subu %D0, $zero, %S0',
-       mode      => "mode_Iu",
+       mode      => $mode_gp
 },
 
 xor => {
+       op_flags  => "R",
        reg_req   => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit      => '. xor %D0, %S0, %S1',
-       mode      => "mode_Iu",
+       ins       => [ "left", "right" ],
+       emit      => '. xor%I %D0, %S0, %SI1',
+       mode      => $mode_gp,
 },
 
-xori => {
+seb => {
+       op_flags  => "R",
        reg_req   => { in => [ "gp" ], out => [ "gp" ] },
-       emit      => '. xori %D0, %S0, %C',
-       cmp_attr  => 'return attr_a->tv != attr_b->tv;',
-       mode      => "mode_Iu",
+       ins       => [ "val" ],
+       emit      => '.seb %D0, %S0',
+       mode      => $mode_gp,
 },
 
-#   ____                _              _
-#  / ___|___  _ __  ___| |_ __ _ _ __ | |_ ___
-# | |   / _ \| '_ \/ __| __/ _` | '_ \| __/ __|
-# | |__| (_) | | | \__ \ || (_| | | | | |_\__ \
-#  \____\___/|_| |_|___/\__\__,_|_| |_|\__|___/
-#
+seh => {
+       op_flags  => "R",
+       reg_req   => { in => [ "gp" ], out => [ "gp" ] },
+       ins       => [ "val" ],
+       emit      => '.seh %D0, %S0',
+       mode      => $mode_gp,
+},
 
-# load upper imediate
 lui => {
        op_flags  => "c",
+       irn_flags => "R",
        reg_req   => { out => [ "gp" ] },
-       emit      => '. lui %D0, %C',
-       cmp_attr  => 'return attr_a->tv != attr_b->tv;',
-       mode      => "mode_Iu",
+       emit      => '.lui %D0, %C',
+       attr_type => "mips_immediate_attr_t",
+       attr      => "mips_immediate_type_t imm_type, ir_entity *entity, long val",
+       mode      => $mode_gp,
 },
 
 mflo => {
+       irn_flags => "R",
        reg_req   => { in => [ "none" ], out => [ "gp" ] },
+       ins       => [ "lohi" ],
        emit      => '. mflo %D0',
-       mode      => "mode_Iu"
+       mode      => $mode_gp
 },
 
 mfhi => {
+       irn_flags => "R",
        reg_req   => { in => [ "none" ], out => [ "gp" ] },
+       ins       => [ "lohi" ],
        emit      => '. mfhi %D0',
-       mode      => "mode_Iu"
+       mode      => $mode_gp
 },
 
 zero => {
        state     => "pinned",
        op_flags  => "c",
-       irn_flags => "I",
-       reg_req   => { out => [ "zero" ] },
+       reg_req   => { out => [ "zero:I" ] },
        emit      => '',
-       mode      => "mode_Iu"
+       mode      => $mode_gp
 },
 
 #
@@ -391,27 +280,17 @@ zero => {
 #
 
 slt => {
+       op_flags => "R",
        reg_req => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit    => '. slt %D0, %S0, %S1',
-       mode    => "mode_Iu",
+       emit    => '. slt%I %D0, %S0, %SI1',
+       mode    => $mode_gp,
 },
 
 sltu => {
-       reg_req => { in => [ "gp", "gp" ], out => [ "gp" ] },
-       emit    => '. sltu %D0, %S0, %S1',
-       mode    => "mode_Iu",
-},
-
-slti => {
-       reg_req => { in => [ "gp" ], out => [ "gp" ] },
-       emit    => '. slti %D0, %S0, %C',
-       mode    => "mode_Iu",
-},
-
-sltiu => {
-       reg_req => { in => [ "gp" ], out => [ "gp" ] },
-       emit    => '. slti %D0, %S0, %C',
-       mode    => "mode_Iu",
+       op_flags => "R",
+       reg_req  => { in => [ "gp", "gp" ], out => [ "gp" ] },
+       emit     => '. slt%I%.u %D0, %S0, %SI1',
+       mode     => $mode_gp,
 },
 
 beq => {
@@ -474,73 +353,91 @@ SwitchJump => {
 #
 
 lw => {
-       op_flags => "L|F",
-       state    => "exc_pinned",
-       reg_req  => { in => [ "none", "gp" ], out => [ "gp", "none" ] },
-       outs     => [ "res", "M" ],
-       emit     => '. lw %D0, %C(%S1)',
+       op_flags  => "L|F",
+       state     => "exc_pinned",
+       reg_req   => { in => [ "gp", "none" ], out => [ "gp", "none" ] },
+       ins       => [ "ptr", "mem" ],
+       outs      => [ "res", "M" ],
+       emit      => '. lw %D0, %A0',
+       attr_type => "mips_load_store_attr_t",
+       attr      => "ir_entity *entity, long offset",
 },
 
 lh => {
-       op_flags => "L|F",
-       state    => "exc_pinned",
-       reg_req  => { in => [ "none", "gp" ], out => [ "gp", "none" ] },
-       outs     => [ "res", "M" ],
-       emit     => '. lh %D0, %C(%S1)',
+       op_flags  => "L|F",
+       state     => "exc_pinned",
+       reg_req   => { in => [ "gp", "none" ], out => [ "gp", "none" ] },
+       ins       => [ "ptr", "mem" ],
+       outs      => [ "res", "M" ],
+       emit      => '. lh %D0, %A0',
+       attr_type => "mips_load_store_attr_t",
+       attr      => "ir_entity *entity, long offset",
 },
 
 lhu => {
-       op_flags => "L|F",
-       state    => "exc_pinned",
-       reg_req  => { in => [ "none", "gp" ], out => [ "gp", "none" ] },
-       outs     => [ "res", "M" ],
-       emit     => '. lhu %D0, %C(%S1)',
+       op_flags  => "L|F",
+       state     => "exc_pinned",
+       reg_req   => { in => [ "gp", "none" ], out => [ "gp", "none" ] },
+       ins       => [ "ptr", "mem" ],
+       outs      => [ "res", "M" ],
+       emit      => '. lhu %D0, %A0',
+       attr_type => "mips_load_store_attr_t",
+       attr      => "ir_entity *entity, long offset",
 },
 
 lb => {
-       op_flags => "L|F",
-       state    => "exc_pinned",
-       reg_req  => { in => [ "none", "gp" ], out => [ "gp", "none" ] },
-       outs     => [ "res", "M" ],
-       emit     => '. lb %D0, %C(%S1)',
+       op_flags  => "L|F",
+       state     => "exc_pinned",
+       reg_req   => { in => [ "gp", "none" ], out => [ "gp", "none" ] },
+       ins       => [ "ptr", "mem" ],
+       outs      => [ "res", "M" ],
+       emit      => '. lb %D0, %A0',
+       attr_type => "mips_load_store_attr_t",
+       attr      => "ir_entity *entity, long offset",
 },
 
 lbu => {
-       op_flags => "L|F",
-       state    => "exc_pinned",
-       reg_req  => { in => [ "none", "gp" ], out => [ "gp", "none" ] },
-       outs     => [ "res", "M" ],
-       emit     => '. lbu %D0, %C(%S1)',
+       op_flags  => "L|F",
+       state     => "exc_pinned",
+       reg_req   => { in => [ "gp", "none" ], out => [ "gp", "none" ] },
+       ins       => [ "ptr", "mem" ],
+       outs      => [ "res", "M" ],
+       emit      => '. lbu %D0, %A0',
+       attr_type => "mips_load_store_attr_t",
+       attr      => "ir_entity *entity, long offset",
 },
 
 sw => {
-       op_flags => "L|F",
-       state    => "exc_pinned",
-       reg_req  => { in => [ "none", "gp", "gp" ], out => [ "none" ] },
-       emit     => '. sw %S2, %C(%S1)',
-       mode     => 'mode_M',
+       op_flags  => "L|F",
+       state     => "exc_pinned",
+       reg_req   => { in => [ "gp", "gp", "none" ], out => [ "none" ] },
+       ins       => [ "ptr", "val", "mem" ],
+       emit      => '. sw %S1, %A0',
+       mode      => 'mode_M',
+       attr_type => "mips_load_store_attr_t",
+       attr      => "ir_entity *entity, long offset",
 },
 
 sh => {
-       op_flags => "L|F",
-       state    => "exc_pinned",
-       reg_req  => { in => [ "none", "gp", "gp" ], out => [ "none" ] },
-       emit     => '. sh %S2, %C(%S1)',
-       mode     => 'mode_M',
+       op_flags  => "L|F",
+       state     => "exc_pinned",
+       reg_req   => { in => [ "gp", "gp", "none" ], out => [ "none" ] },
+       ins       => [ "ptr", "val", "mem" ],
+       emit      => '. sh %S1, %A0',
+       mode      => 'mode_M',
+       attr_type => "mips_load_store_attr_t",
+       attr      => "ir_entity *entity, long offset",
 },
 
 sb => {
-       op_flags => "L|F",
-       state    => "exc_pinned",
-       reg_req  => { in => [ "none", "gp", "gp" ], out => [ "none" ] },
-       emit     => '. sb %S2, %C(%S1)',
-       mode     => 'mode_M',
-},
-
-move => {
-       reg_req  => { in => [ "gp" ], out => [ "gp" ] },
-       emit     => '. move %D0, %S0',
-       mode     => "mode_Iu"
+       op_flags  => "L|F",
+       state     => "exc_pinned",
+       reg_req   => { in => [ "gp", "gp", "none" ], out => [ "none" ] },
+       ins       => [ "ptr", "val", "mem" ],
+       emit      => '. sb %S1, %A0',
+       mode      => 'mode_M',
+       attr_type => "mips_load_store_attr_t",
+       attr      => "ir_entity *entity, long offset",
 },
 
 #