langtools/src/share/classes/com/sun/tools/javac/comp/ConstFold.java
author jjg
Wed, 21 Sep 2011 21:56:53 -0700
changeset 10637 2ea5fbb913ac
parent 5847 1908176fd6e3
child 14359 d4099818ab70
permissions -rw-r--r--
7092965: javac should not close processorClassLoader before end of compilation Reviewed-by: darcy
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
10
06bc494ca11e Initial load
duke
parents:
diff changeset
     1
/*
5520
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 10
diff changeset
     2
 * Copyright (c) 1999, 2006, Oracle and/or its affiliates. All rights reserved.
10
06bc494ca11e Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
06bc494ca11e Initial load
duke
parents:
diff changeset
     4
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
06bc494ca11e Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
5520
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 10
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
10
06bc494ca11e Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
5520
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 10
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
10
06bc494ca11e Initial load
duke
parents:
diff changeset
    10
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
06bc494ca11e Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
06bc494ca11e Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
06bc494ca11e Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
06bc494ca11e Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
06bc494ca11e Initial load
duke
parents:
diff changeset
    16
 *
06bc494ca11e Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
06bc494ca11e Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
06bc494ca11e Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
06bc494ca11e Initial load
duke
parents:
diff changeset
    20
 *
5520
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 10
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 10
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
86e4b9a9da40 6943119: Rebrand source copyright notices
ohair
parents: 10
diff changeset
    23
 * questions.
10
06bc494ca11e Initial load
duke
parents:
diff changeset
    24
 */
06bc494ca11e Initial load
duke
parents:
diff changeset
    25
06bc494ca11e Initial load
duke
parents:
diff changeset
    26
package com.sun.tools.javac.comp;
06bc494ca11e Initial load
duke
parents:
diff changeset
    27
06bc494ca11e Initial load
duke
parents:
diff changeset
    28
import com.sun.tools.javac.code.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    29
import com.sun.tools.javac.jvm.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    30
import com.sun.tools.javac.util.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    31
06bc494ca11e Initial load
duke
parents:
diff changeset
    32
import com.sun.tools.javac.code.Type.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    33
06bc494ca11e Initial load
duke
parents:
diff changeset
    34
import static com.sun.tools.javac.code.TypeTags.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    35
import static com.sun.tools.javac.jvm.ByteCodes.*;
06bc494ca11e Initial load
duke
parents:
diff changeset
    36
06bc494ca11e Initial load
duke
parents:
diff changeset
    37
/** Helper class for constant folding, used by the attribution phase.
06bc494ca11e Initial load
duke
parents:
diff changeset
    38
 *  This class is marked strictfp as mandated by JLS 15.4.
06bc494ca11e Initial load
duke
parents:
diff changeset
    39
 *
5847
1908176fd6e3 6944312: Potential rebranding issues in openjdk/langtools repository sources
jjg
parents: 5520
diff changeset
    40
 *  <p><b>This is NOT part of any supported API.
1908176fd6e3 6944312: Potential rebranding issues in openjdk/langtools repository sources
jjg
parents: 5520
diff changeset
    41
 *  If you write code that depends on this, you do so at your own risk.
10
06bc494ca11e Initial load
duke
parents:
diff changeset
    42
 *  This code and its internal interfaces are subject to change or
06bc494ca11e Initial load
duke
parents:
diff changeset
    43
 *  deletion without notice.</b>
06bc494ca11e Initial load
duke
parents:
diff changeset
    44
 */
06bc494ca11e Initial load
duke
parents:
diff changeset
    45
strictfp class ConstFold {
06bc494ca11e Initial load
duke
parents:
diff changeset
    46
    protected static final Context.Key<ConstFold> constFoldKey =
06bc494ca11e Initial load
duke
parents:
diff changeset
    47
        new Context.Key<ConstFold>();
06bc494ca11e Initial load
duke
parents:
diff changeset
    48
06bc494ca11e Initial load
duke
parents:
diff changeset
    49
    private Symtab syms;
06bc494ca11e Initial load
duke
parents:
diff changeset
    50
06bc494ca11e Initial load
duke
parents:
diff changeset
    51
    public static ConstFold instance(Context context) {
06bc494ca11e Initial load
duke
parents:
diff changeset
    52
        ConstFold instance = context.get(constFoldKey);
06bc494ca11e Initial load
duke
parents:
diff changeset
    53
        if (instance == null)
06bc494ca11e Initial load
duke
parents:
diff changeset
    54
            instance = new ConstFold(context);
06bc494ca11e Initial load
duke
parents:
diff changeset
    55
        return instance;
06bc494ca11e Initial load
duke
parents:
diff changeset
    56
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
    57
06bc494ca11e Initial load
duke
parents:
diff changeset
    58
    private ConstFold(Context context) {
06bc494ca11e Initial load
duke
parents:
diff changeset
    59
        context.put(constFoldKey, this);
06bc494ca11e Initial load
duke
parents:
diff changeset
    60
06bc494ca11e Initial load
duke
parents:
diff changeset
    61
        syms = Symtab.instance(context);
06bc494ca11e Initial load
duke
parents:
diff changeset
    62
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
    63
06bc494ca11e Initial load
duke
parents:
diff changeset
    64
    static Integer minusOne = -1;
06bc494ca11e Initial load
duke
parents:
diff changeset
    65
    static Integer zero     = 0;
06bc494ca11e Initial load
duke
parents:
diff changeset
    66
    static Integer one      = 1;
06bc494ca11e Initial load
duke
parents:
diff changeset
    67
06bc494ca11e Initial load
duke
parents:
diff changeset
    68
   /** Convert boolean to integer (true = 1, false = 0).
06bc494ca11e Initial load
duke
parents:
diff changeset
    69
    */
06bc494ca11e Initial load
duke
parents:
diff changeset
    70
    private static Integer b2i(boolean b) {
06bc494ca11e Initial load
duke
parents:
diff changeset
    71
        return b ? one : zero;
06bc494ca11e Initial load
duke
parents:
diff changeset
    72
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
    73
    private static int intValue(Object x) { return ((Number)x).intValue(); }
06bc494ca11e Initial load
duke
parents:
diff changeset
    74
    private static long longValue(Object x) { return ((Number)x).longValue(); }
06bc494ca11e Initial load
duke
parents:
diff changeset
    75
    private static float floatValue(Object x) { return ((Number)x).floatValue(); }
06bc494ca11e Initial load
duke
parents:
diff changeset
    76
    private static double doubleValue(Object x) { return ((Number)x).doubleValue(); }
06bc494ca11e Initial load
duke
parents:
diff changeset
    77
06bc494ca11e Initial load
duke
parents:
diff changeset
    78
    /** Fold binary or unary operation, returning constant type reflecting the
06bc494ca11e Initial load
duke
parents:
diff changeset
    79
     *  operations result. Return null if fold failed due to an
06bc494ca11e Initial load
duke
parents:
diff changeset
    80
     *  arithmetic exception.
06bc494ca11e Initial load
duke
parents:
diff changeset
    81
     *  @param opcode    The operation's opcode instruction (usually a byte code),
06bc494ca11e Initial load
duke
parents:
diff changeset
    82
     *                   as entered by class Symtab.
06bc494ca11e Initial load
duke
parents:
diff changeset
    83
     *  @param argtypes  The operation's argument types (a list of length 1 or 2).
06bc494ca11e Initial load
duke
parents:
diff changeset
    84
     *                   Argument types are assumed to have non-null constValue's.
06bc494ca11e Initial load
duke
parents:
diff changeset
    85
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
    86
    Type fold(int opcode, List<Type> argtypes) {
06bc494ca11e Initial load
duke
parents:
diff changeset
    87
        int argCount = argtypes.length();
06bc494ca11e Initial load
duke
parents:
diff changeset
    88
        if (argCount == 1)
06bc494ca11e Initial load
duke
parents:
diff changeset
    89
            return fold1(opcode, argtypes.head);
06bc494ca11e Initial load
duke
parents:
diff changeset
    90
        else if (argCount == 2)
06bc494ca11e Initial load
duke
parents:
diff changeset
    91
            return fold2(opcode, argtypes.head, argtypes.tail.head);
06bc494ca11e Initial load
duke
parents:
diff changeset
    92
        else
06bc494ca11e Initial load
duke
parents:
diff changeset
    93
            throw new AssertionError();
06bc494ca11e Initial load
duke
parents:
diff changeset
    94
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
    95
06bc494ca11e Initial load
duke
parents:
diff changeset
    96
    /** Fold unary operation.
06bc494ca11e Initial load
duke
parents:
diff changeset
    97
     *  @param opcode    The operation's opcode instruction (usually a byte code),
06bc494ca11e Initial load
duke
parents:
diff changeset
    98
     *                   as entered by class Symtab.
06bc494ca11e Initial load
duke
parents:
diff changeset
    99
     *                   opcode's ifeq to ifge are for postprocessing
06bc494ca11e Initial load
duke
parents:
diff changeset
   100
     *                   xcmp; ifxx pairs of instructions.
06bc494ca11e Initial load
duke
parents:
diff changeset
   101
     *  @param operand   The operation's operand type.
06bc494ca11e Initial load
duke
parents:
diff changeset
   102
     *                   Argument types are assumed to have non-null constValue's.
06bc494ca11e Initial load
duke
parents:
diff changeset
   103
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   104
    Type fold1(int opcode, Type operand) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   105
        try {
06bc494ca11e Initial load
duke
parents:
diff changeset
   106
            Object od = operand.constValue();
06bc494ca11e Initial load
duke
parents:
diff changeset
   107
            switch (opcode) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   108
            case nop:
06bc494ca11e Initial load
duke
parents:
diff changeset
   109
                return operand;
06bc494ca11e Initial load
duke
parents:
diff changeset
   110
            case ineg: // unary -
06bc494ca11e Initial load
duke
parents:
diff changeset
   111
                return syms.intType.constType(-intValue(od));
06bc494ca11e Initial load
duke
parents:
diff changeset
   112
            case ixor: // ~
06bc494ca11e Initial load
duke
parents:
diff changeset
   113
                return syms.intType.constType(~intValue(od));
06bc494ca11e Initial load
duke
parents:
diff changeset
   114
            case bool_not: // !
06bc494ca11e Initial load
duke
parents:
diff changeset
   115
                return syms.booleanType.constType(b2i(intValue(od) == 0));
06bc494ca11e Initial load
duke
parents:
diff changeset
   116
            case ifeq:
06bc494ca11e Initial load
duke
parents:
diff changeset
   117
                return syms.booleanType.constType(b2i(intValue(od) == 0));
06bc494ca11e Initial load
duke
parents:
diff changeset
   118
            case ifne:
06bc494ca11e Initial load
duke
parents:
diff changeset
   119
                return syms.booleanType.constType(b2i(intValue(od) != 0));
06bc494ca11e Initial load
duke
parents:
diff changeset
   120
            case iflt:
06bc494ca11e Initial load
duke
parents:
diff changeset
   121
                return syms.booleanType.constType(b2i(intValue(od) < 0));
06bc494ca11e Initial load
duke
parents:
diff changeset
   122
            case ifgt:
06bc494ca11e Initial load
duke
parents:
diff changeset
   123
                return syms.booleanType.constType(b2i(intValue(od) > 0));
06bc494ca11e Initial load
duke
parents:
diff changeset
   124
            case ifle:
06bc494ca11e Initial load
duke
parents:
diff changeset
   125
                return syms.booleanType.constType(b2i(intValue(od) <= 0));
06bc494ca11e Initial load
duke
parents:
diff changeset
   126
            case ifge:
06bc494ca11e Initial load
duke
parents:
diff changeset
   127
                return syms.booleanType.constType(b2i(intValue(od) >= 0));
06bc494ca11e Initial load
duke
parents:
diff changeset
   128
06bc494ca11e Initial load
duke
parents:
diff changeset
   129
            case lneg: // unary -
06bc494ca11e Initial load
duke
parents:
diff changeset
   130
                return syms.longType.constType(new Long(-longValue(od)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   131
            case lxor: // ~
06bc494ca11e Initial load
duke
parents:
diff changeset
   132
                return syms.longType.constType(new Long(~longValue(od)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   133
06bc494ca11e Initial load
duke
parents:
diff changeset
   134
            case fneg: // unary -
06bc494ca11e Initial load
duke
parents:
diff changeset
   135
                return syms.floatType.constType(new Float(-floatValue(od)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   136
06bc494ca11e Initial load
duke
parents:
diff changeset
   137
            case dneg: // ~
06bc494ca11e Initial load
duke
parents:
diff changeset
   138
                return syms.doubleType.constType(new Double(-doubleValue(od)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   139
06bc494ca11e Initial load
duke
parents:
diff changeset
   140
            default:
06bc494ca11e Initial load
duke
parents:
diff changeset
   141
                return null;
06bc494ca11e Initial load
duke
parents:
diff changeset
   142
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   143
        } catch (ArithmeticException e) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   144
            return null;
06bc494ca11e Initial load
duke
parents:
diff changeset
   145
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   146
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   147
06bc494ca11e Initial load
duke
parents:
diff changeset
   148
    /** Fold binary operation.
06bc494ca11e Initial load
duke
parents:
diff changeset
   149
     *  @param opcode    The operation's opcode instruction (usually a byte code),
06bc494ca11e Initial load
duke
parents:
diff changeset
   150
     *                   as entered by class Symtab.
06bc494ca11e Initial load
duke
parents:
diff changeset
   151
     *                   opcode's ifeq to ifge are for postprocessing
06bc494ca11e Initial load
duke
parents:
diff changeset
   152
     *                   xcmp; ifxx pairs of instructions.
06bc494ca11e Initial load
duke
parents:
diff changeset
   153
     *  @param left      The type of the operation's left operand.
06bc494ca11e Initial load
duke
parents:
diff changeset
   154
     *  @param right     The type of the operation's right operand.
06bc494ca11e Initial load
duke
parents:
diff changeset
   155
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   156
    Type fold2(int opcode, Type left, Type right) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   157
        try {
06bc494ca11e Initial load
duke
parents:
diff changeset
   158
            if (opcode > ByteCodes.preMask) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   159
                // we are seeing a composite instruction of the form xcmp; ifxx.
06bc494ca11e Initial load
duke
parents:
diff changeset
   160
                // In this case fold both instructions separately.
06bc494ca11e Initial load
duke
parents:
diff changeset
   161
                Type t1 = fold2(opcode >> ByteCodes.preShift, left, right);
06bc494ca11e Initial load
duke
parents:
diff changeset
   162
                return (t1.constValue() == null) ? t1
06bc494ca11e Initial load
duke
parents:
diff changeset
   163
                    : fold1(opcode & ByteCodes.preMask, t1);
06bc494ca11e Initial load
duke
parents:
diff changeset
   164
            } else {
06bc494ca11e Initial load
duke
parents:
diff changeset
   165
                Object l = left.constValue();
06bc494ca11e Initial load
duke
parents:
diff changeset
   166
                Object r = right.constValue();
06bc494ca11e Initial load
duke
parents:
diff changeset
   167
                switch (opcode) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   168
                case iadd:
06bc494ca11e Initial load
duke
parents:
diff changeset
   169
                    return syms.intType.constType(intValue(l) + intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   170
                case isub:
06bc494ca11e Initial load
duke
parents:
diff changeset
   171
                    return syms.intType.constType(intValue(l) - intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   172
                case imul:
06bc494ca11e Initial load
duke
parents:
diff changeset
   173
                    return syms.intType.constType(intValue(l) * intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   174
                case idiv:
06bc494ca11e Initial load
duke
parents:
diff changeset
   175
                    return syms.intType.constType(intValue(l) / intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   176
                case imod:
06bc494ca11e Initial load
duke
parents:
diff changeset
   177
                    return syms.intType.constType(intValue(l) % intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   178
                case iand:
06bc494ca11e Initial load
duke
parents:
diff changeset
   179
                    return (left.tag == BOOLEAN
06bc494ca11e Initial load
duke
parents:
diff changeset
   180
                      ? syms.booleanType : syms.intType)
06bc494ca11e Initial load
duke
parents:
diff changeset
   181
                      .constType(intValue(l) & intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   182
                case bool_and:
06bc494ca11e Initial load
duke
parents:
diff changeset
   183
                    return syms.booleanType.constType(b2i((intValue(l) & intValue(r)) != 0));
06bc494ca11e Initial load
duke
parents:
diff changeset
   184
                case ior:
06bc494ca11e Initial load
duke
parents:
diff changeset
   185
                    return (left.tag == BOOLEAN
06bc494ca11e Initial load
duke
parents:
diff changeset
   186
                      ? syms.booleanType : syms.intType)
06bc494ca11e Initial load
duke
parents:
diff changeset
   187
                      .constType(intValue(l) | intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   188
                case bool_or:
06bc494ca11e Initial load
duke
parents:
diff changeset
   189
                    return syms.booleanType.constType(b2i((intValue(l) | intValue(r)) != 0));
06bc494ca11e Initial load
duke
parents:
diff changeset
   190
                case ixor:
06bc494ca11e Initial load
duke
parents:
diff changeset
   191
                    return (left.tag == BOOLEAN
06bc494ca11e Initial load
duke
parents:
diff changeset
   192
                      ? syms.booleanType : syms.intType)
06bc494ca11e Initial load
duke
parents:
diff changeset
   193
                      .constType(intValue(l) ^ intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   194
                case ishl: case ishll:
06bc494ca11e Initial load
duke
parents:
diff changeset
   195
                    return syms.intType.constType(intValue(l) << intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   196
                case ishr: case ishrl:
06bc494ca11e Initial load
duke
parents:
diff changeset
   197
                    return syms.intType.constType(intValue(l) >> intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   198
                case iushr: case iushrl:
06bc494ca11e Initial load
duke
parents:
diff changeset
   199
                    return syms.intType.constType(intValue(l) >>> intValue(r));
06bc494ca11e Initial load
duke
parents:
diff changeset
   200
                case if_icmpeq:
06bc494ca11e Initial load
duke
parents:
diff changeset
   201
                    return syms.booleanType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   202
                        b2i(intValue(l) == intValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   203
                case if_icmpne:
06bc494ca11e Initial load
duke
parents:
diff changeset
   204
                    return syms.booleanType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   205
                        b2i(intValue(l) != intValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   206
                case if_icmplt:
06bc494ca11e Initial load
duke
parents:
diff changeset
   207
                    return syms.booleanType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   208
                        b2i(intValue(l) < intValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   209
                case if_icmpgt:
06bc494ca11e Initial load
duke
parents:
diff changeset
   210
                    return syms.booleanType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   211
                        b2i(intValue(l) > intValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   212
                case if_icmple:
06bc494ca11e Initial load
duke
parents:
diff changeset
   213
                    return syms.booleanType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   214
                        b2i(intValue(l) <= intValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   215
                case if_icmpge:
06bc494ca11e Initial load
duke
parents:
diff changeset
   216
                    return syms.booleanType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   217
                        b2i(intValue(l) >= intValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   218
06bc494ca11e Initial load
duke
parents:
diff changeset
   219
                case ladd:
06bc494ca11e Initial load
duke
parents:
diff changeset
   220
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   221
                        new Long(longValue(l) + longValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   222
                case lsub:
06bc494ca11e Initial load
duke
parents:
diff changeset
   223
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   224
                        new Long(longValue(l) - longValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   225
                case lmul:
06bc494ca11e Initial load
duke
parents:
diff changeset
   226
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   227
                        new Long(longValue(l) * longValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   228
                case ldiv:
06bc494ca11e Initial load
duke
parents:
diff changeset
   229
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   230
                        new Long(longValue(l) / longValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   231
                case lmod:
06bc494ca11e Initial load
duke
parents:
diff changeset
   232
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   233
                        new Long(longValue(l) % longValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   234
                case land:
06bc494ca11e Initial load
duke
parents:
diff changeset
   235
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   236
                        new Long(longValue(l) & longValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   237
                case lor:
06bc494ca11e Initial load
duke
parents:
diff changeset
   238
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   239
                        new Long(longValue(l) | longValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   240
                case lxor:
06bc494ca11e Initial load
duke
parents:
diff changeset
   241
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   242
                        new Long(longValue(l) ^ longValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   243
                case lshl: case lshll:
06bc494ca11e Initial load
duke
parents:
diff changeset
   244
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   245
                        new Long(longValue(l) << intValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   246
                case lshr: case lshrl:
06bc494ca11e Initial load
duke
parents:
diff changeset
   247
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   248
                        new Long(longValue(l) >> intValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   249
                case lushr:
06bc494ca11e Initial load
duke
parents:
diff changeset
   250
                    return syms.longType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   251
                        new Long(longValue(l) >>> intValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   252
                case lcmp:
06bc494ca11e Initial load
duke
parents:
diff changeset
   253
                    if (longValue(l) < longValue(r))
06bc494ca11e Initial load
duke
parents:
diff changeset
   254
                        return syms.intType.constType(minusOne);
06bc494ca11e Initial load
duke
parents:
diff changeset
   255
                    else if (longValue(l) > longValue(r))
06bc494ca11e Initial load
duke
parents:
diff changeset
   256
                        return syms.intType.constType(one);
06bc494ca11e Initial load
duke
parents:
diff changeset
   257
                    else
06bc494ca11e Initial load
duke
parents:
diff changeset
   258
                        return syms.intType.constType(zero);
06bc494ca11e Initial load
duke
parents:
diff changeset
   259
                case fadd:
06bc494ca11e Initial load
duke
parents:
diff changeset
   260
                    return syms.floatType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   261
                        new Float(floatValue(l) + floatValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   262
                case fsub:
06bc494ca11e Initial load
duke
parents:
diff changeset
   263
                    return syms.floatType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   264
                        new Float(floatValue(l) - floatValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   265
                case fmul:
06bc494ca11e Initial load
duke
parents:
diff changeset
   266
                    return syms.floatType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   267
                        new Float(floatValue(l) * floatValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   268
                case fdiv:
06bc494ca11e Initial load
duke
parents:
diff changeset
   269
                    return syms.floatType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   270
                        new Float(floatValue(l) / floatValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   271
                case fmod:
06bc494ca11e Initial load
duke
parents:
diff changeset
   272
                    return syms.floatType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   273
                        new Float(floatValue(l) % floatValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   274
                case fcmpg: case fcmpl:
06bc494ca11e Initial load
duke
parents:
diff changeset
   275
                    if (floatValue(l) < floatValue(r))
06bc494ca11e Initial load
duke
parents:
diff changeset
   276
                        return syms.intType.constType(minusOne);
06bc494ca11e Initial load
duke
parents:
diff changeset
   277
                    else if (floatValue(l) > floatValue(r))
06bc494ca11e Initial load
duke
parents:
diff changeset
   278
                        return syms.intType.constType(one);
06bc494ca11e Initial load
duke
parents:
diff changeset
   279
                    else if (floatValue(l) == floatValue(r))
06bc494ca11e Initial load
duke
parents:
diff changeset
   280
                        return syms.intType.constType(zero);
06bc494ca11e Initial load
duke
parents:
diff changeset
   281
                    else if (opcode == fcmpg)
06bc494ca11e Initial load
duke
parents:
diff changeset
   282
                        return syms.intType.constType(one);
06bc494ca11e Initial load
duke
parents:
diff changeset
   283
                    else
06bc494ca11e Initial load
duke
parents:
diff changeset
   284
                        return syms.intType.constType(minusOne);
06bc494ca11e Initial load
duke
parents:
diff changeset
   285
                case dadd:
06bc494ca11e Initial load
duke
parents:
diff changeset
   286
                    return syms.doubleType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   287
                        new Double(doubleValue(l) + doubleValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   288
                case dsub:
06bc494ca11e Initial load
duke
parents:
diff changeset
   289
                    return syms.doubleType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   290
                        new Double(doubleValue(l) - doubleValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   291
                case dmul:
06bc494ca11e Initial load
duke
parents:
diff changeset
   292
                    return syms.doubleType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   293
                        new Double(doubleValue(l) * doubleValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   294
                case ddiv:
06bc494ca11e Initial load
duke
parents:
diff changeset
   295
                    return syms.doubleType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   296
                        new Double(doubleValue(l) / doubleValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   297
                case dmod:
06bc494ca11e Initial load
duke
parents:
diff changeset
   298
                    return syms.doubleType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   299
                        new Double(doubleValue(l) % doubleValue(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   300
                case dcmpg: case dcmpl:
06bc494ca11e Initial load
duke
parents:
diff changeset
   301
                    if (doubleValue(l) < doubleValue(r))
06bc494ca11e Initial load
duke
parents:
diff changeset
   302
                        return syms.intType.constType(minusOne);
06bc494ca11e Initial load
duke
parents:
diff changeset
   303
                    else if (doubleValue(l) > doubleValue(r))
06bc494ca11e Initial load
duke
parents:
diff changeset
   304
                        return syms.intType.constType(one);
06bc494ca11e Initial load
duke
parents:
diff changeset
   305
                    else if (doubleValue(l) == doubleValue(r))
06bc494ca11e Initial load
duke
parents:
diff changeset
   306
                        return syms.intType.constType(zero);
06bc494ca11e Initial load
duke
parents:
diff changeset
   307
                    else if (opcode == dcmpg)
06bc494ca11e Initial load
duke
parents:
diff changeset
   308
                        return syms.intType.constType(one);
06bc494ca11e Initial load
duke
parents:
diff changeset
   309
                    else
06bc494ca11e Initial load
duke
parents:
diff changeset
   310
                        return syms.intType.constType(minusOne);
06bc494ca11e Initial load
duke
parents:
diff changeset
   311
                case if_acmpeq:
06bc494ca11e Initial load
duke
parents:
diff changeset
   312
                    return syms.booleanType.constType(b2i(l.equals(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   313
                case if_acmpne:
06bc494ca11e Initial load
duke
parents:
diff changeset
   314
                    return syms.booleanType.constType(b2i(!l.equals(r)));
06bc494ca11e Initial load
duke
parents:
diff changeset
   315
                case string_add:
06bc494ca11e Initial load
duke
parents:
diff changeset
   316
                    return syms.stringType.constType(
06bc494ca11e Initial load
duke
parents:
diff changeset
   317
                        left.stringValue() + right.stringValue());
06bc494ca11e Initial load
duke
parents:
diff changeset
   318
                default:
06bc494ca11e Initial load
duke
parents:
diff changeset
   319
                    return null;
06bc494ca11e Initial load
duke
parents:
diff changeset
   320
                }
06bc494ca11e Initial load
duke
parents:
diff changeset
   321
            }
06bc494ca11e Initial load
duke
parents:
diff changeset
   322
        } catch (ArithmeticException e) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   323
            return null;
06bc494ca11e Initial load
duke
parents:
diff changeset
   324
        }
06bc494ca11e Initial load
duke
parents:
diff changeset
   325
    }
06bc494ca11e Initial load
duke
parents:
diff changeset
   326
06bc494ca11e Initial load
duke
parents:
diff changeset
   327
    /** Coerce constant type to target type.
06bc494ca11e Initial load
duke
parents:
diff changeset
   328
     *  @param etype      The source type of the coercion,
06bc494ca11e Initial load
duke
parents:
diff changeset
   329
     *                    which is assumed to be a constant type compatble with
06bc494ca11e Initial load
duke
parents:
diff changeset
   330
     *                    ttype.
06bc494ca11e Initial load
duke
parents:
diff changeset
   331
     *  @param ttype      The target type of the coercion.
06bc494ca11e Initial load
duke
parents:
diff changeset
   332
     */
06bc494ca11e Initial load
duke
parents:
diff changeset
   333
     Type coerce(Type etype, Type ttype) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   334
         // WAS if (etype.baseType() == ttype.baseType())
06bc494ca11e Initial load
duke
parents:
diff changeset
   335
         if (etype.tsym.type == ttype.tsym.type)
06bc494ca11e Initial load
duke
parents:
diff changeset
   336
             return etype;
06bc494ca11e Initial load
duke
parents:
diff changeset
   337
         if (etype.tag <= DOUBLE) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   338
             Object n = etype.constValue();
06bc494ca11e Initial load
duke
parents:
diff changeset
   339
             switch (ttype.tag) {
06bc494ca11e Initial load
duke
parents:
diff changeset
   340
             case BYTE:
06bc494ca11e Initial load
duke
parents:
diff changeset
   341
                 return syms.byteType.constType(0 + (byte)intValue(n));
06bc494ca11e Initial load
duke
parents:
diff changeset
   342
             case CHAR:
06bc494ca11e Initial load
duke
parents:
diff changeset
   343
                 return syms.charType.constType(0 + (char)intValue(n));
06bc494ca11e Initial load
duke
parents:
diff changeset
   344
             case SHORT:
06bc494ca11e Initial load
duke
parents:
diff changeset
   345
                 return syms.shortType.constType(0 + (short)intValue(n));
06bc494ca11e Initial load
duke
parents:
diff changeset
   346
             case INT:
06bc494ca11e Initial load
duke
parents:
diff changeset
   347
                 return syms.intType.constType(intValue(n));
06bc494ca11e Initial load
duke
parents:
diff changeset
   348
             case LONG:
06bc494ca11e Initial load
duke
parents:
diff changeset
   349
                 return syms.longType.constType(longValue(n));
06bc494ca11e Initial load
duke
parents:
diff changeset
   350
             case FLOAT:
06bc494ca11e Initial load
duke
parents:
diff changeset
   351
                 return syms.floatType.constType(floatValue(n));
06bc494ca11e Initial load
duke
parents:
diff changeset
   352
             case DOUBLE:
06bc494ca11e Initial load
duke
parents:
diff changeset
   353
                 return syms.doubleType.constType(doubleValue(n));
06bc494ca11e Initial load
duke
parents:
diff changeset
   354
             }
06bc494ca11e Initial load
duke
parents:
diff changeset
   355
         }
06bc494ca11e Initial load
duke
parents:
diff changeset
   356
         return ttype;
06bc494ca11e Initial load
duke
parents:
diff changeset
   357
     }
06bc494ca11e Initial load
duke
parents:
diff changeset
   358
}