Home | History | Annotate | Download | only in rawdex
      1 /*
      2  * Copyright (C) 2014 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package dexfuzz.rawdex;
     18 
     19 import dexfuzz.Log;
     20 import dexfuzz.rawdex.formats.AbstractFormat;
     21 import dexfuzz.rawdex.formats.ContainsConst;
     22 import dexfuzz.rawdex.formats.ContainsPoolIndex;
     23 import dexfuzz.rawdex.formats.ContainsTarget;
     24 import dexfuzz.rawdex.formats.ContainsVRegs;
     25 import dexfuzz.rawdex.formats.Format10t;
     26 import dexfuzz.rawdex.formats.Format10x;
     27 import dexfuzz.rawdex.formats.Format11n;
     28 import dexfuzz.rawdex.formats.Format11x;
     29 import dexfuzz.rawdex.formats.Format12x;
     30 import dexfuzz.rawdex.formats.Format20t;
     31 import dexfuzz.rawdex.formats.Format21c;
     32 import dexfuzz.rawdex.formats.Format21h;
     33 import dexfuzz.rawdex.formats.Format21s;
     34 import dexfuzz.rawdex.formats.Format21t;
     35 import dexfuzz.rawdex.formats.Format22b;
     36 import dexfuzz.rawdex.formats.Format22c;
     37 import dexfuzz.rawdex.formats.Format22s;
     38 import dexfuzz.rawdex.formats.Format22t;
     39 import dexfuzz.rawdex.formats.Format22x;
     40 import dexfuzz.rawdex.formats.Format23x;
     41 import dexfuzz.rawdex.formats.Format30t;
     42 import dexfuzz.rawdex.formats.Format31c;
     43 import dexfuzz.rawdex.formats.Format31i;
     44 import dexfuzz.rawdex.formats.Format31t;
     45 import dexfuzz.rawdex.formats.Format32x;
     46 import dexfuzz.rawdex.formats.Format35c;
     47 import dexfuzz.rawdex.formats.Format3rc;
     48 import dexfuzz.rawdex.formats.Format51l;
     49 
     50 import java.io.IOException;
     51 import java.util.HashMap;
     52 import java.util.Map;
     53 
     54 public class Instruction implements RawDexObject {
     55   // Only used by Format35* instructions
     56   public static class InvokeFormatInfo {
     57     public byte vregD;
     58     public byte vregE;
     59     public byte vregF;
     60     public byte vregG;
     61   }
     62 
     63   // Immutable information about this class of instruction.
     64   public OpcodeInfo info;
     65 
     66   // The raw bytes of the instruction.
     67   // Only used during reading, and writing out is done from the decoded instruction data.
     68   //  Except in the case of the 3 "data" instructions.
     69   public byte[] rawBytes;
     70 
     71   public static final int RAW_TYPE_PACKED_SWITCH_DATA = 1;
     72   public static final int RAW_TYPE_SPARSE_SWITCH_DATA = 2;
     73   public static final int RAW_TYPE_FILL_ARRAY_DATA_DATA = 3;
     74 
     75   public int rawType;
     76   public boolean justRaw;
     77   public int rawSize;
     78 
     79   public long vregA = 0;
     80   public long vregB = 0;
     81   public long vregC = 0;
     82 
     83   public InvokeFormatInfo invokeFormatInfo;
     84 
     85   /**
     86    * Clone an instruction.
     87    */
     88   public Instruction clone() {
     89     Instruction newInsn = new Instruction();
     90     // If we've generated a new instruction, we won't have calculated its raw array.
     91     if (newInsn.rawBytes != null) {
     92       newInsn.rawBytes = new byte[rawBytes.length];
     93       for (int i = 0; i < rawBytes.length; i++) {
     94         newInsn.rawBytes[i] = rawBytes[i];
     95       }
     96     }
     97     newInsn.justRaw = justRaw;
     98     newInsn.rawType = rawType;
     99     newInsn.rawSize = rawSize;
    100 
    101     newInsn.vregA = vregA;
    102     newInsn.vregB = vregB;
    103     newInsn.vregC = vregC;
    104     newInsn.info = info;
    105     if (invokeFormatInfo != null) {
    106       newInsn.invokeFormatInfo = new InvokeFormatInfo();
    107       newInsn.invokeFormatInfo.vregD = invokeFormatInfo.vregD;
    108       newInsn.invokeFormatInfo.vregE = invokeFormatInfo.vregE;
    109       newInsn.invokeFormatInfo.vregF = invokeFormatInfo.vregF;
    110       newInsn.invokeFormatInfo.vregG = invokeFormatInfo.vregG;
    111     }
    112     return newInsn;
    113   }
    114 
    115   @Override
    116   public void read(DexRandomAccessFile file) throws IOException {
    117     // Remember the offset, so after reading the opcode, we can read the whole
    118     // insn into raw_bytes.
    119     long offset = file.getFilePointer();
    120     int opcodeValue = readOpcode(file);
    121     info = getOpcodeInfo(opcodeValue);
    122     if (info == null) {
    123       Log.errorAndQuit("Couldn't find OpcodeInfo for opcode with value: "
    124           + opcodeValue);
    125     }
    126 
    127     rawBytes = new byte[2 * getSize()];
    128     file.seek(offset);
    129     file.read(rawBytes);
    130 
    131     vregA = info.format.getA(rawBytes);
    132     vregB = info.format.getB(rawBytes);
    133     vregC = info.format.getC(rawBytes);
    134 
    135     // Special case for 35* formats.
    136     if (info.format.needsInvokeFormatInfo()) {
    137       invokeFormatInfo = new InvokeFormatInfo();
    138       invokeFormatInfo.vregD = (byte) (rawBytes[4] >> 4);
    139       invokeFormatInfo.vregE = (byte) (rawBytes[5] & 0xf);
    140       invokeFormatInfo.vregF = (byte) (rawBytes[5] >> 4);
    141       invokeFormatInfo.vregG = (byte) (rawBytes[1] & 0xf);
    142     }
    143   }
    144 
    145   @Override
    146   public void write(DexRandomAccessFile file) throws IOException {
    147     if (justRaw) {
    148       // It is the responsibility of the CodeTranslator to make
    149       // sure the raw bytes have been updated.
    150       file.write(rawBytes);
    151     } else {
    152       info.format.writeToFile(file, this);
    153     }
    154   }
    155 
    156   /**
    157    * Get the size of an instruction, in code-words. (Code-words are 16-bits.)
    158    */
    159   public int getSize() {
    160     if (justRaw) {
    161       // It is the responsibility of the CodeTranslator to make sure
    162       // the raw size has been updated.
    163       return rawSize;
    164     }
    165     return info.format.getSize();
    166   }
    167 
    168   private int readOpcode(DexRandomAccessFile file) throws IOException {
    169     short firstCodeWord = file.readUShort();
    170     int opcode = (firstCodeWord & 0xff);
    171     int upperBits = (firstCodeWord & 0xff00) >> 8;
    172     if (opcode == 0x0 && upperBits != 0x0) {
    173       justRaw = true;
    174       rawType = upperBits;
    175       // Need to calculate special sizes.
    176       switch (rawType) {
    177         case RAW_TYPE_PACKED_SWITCH_DATA:
    178           rawSize = (file.readUShort() * 2) + 4;
    179           break;
    180         case RAW_TYPE_SPARSE_SWITCH_DATA:
    181           rawSize = (file.readUShort() * 4) + 2;
    182           break;
    183         case RAW_TYPE_FILL_ARRAY_DATA_DATA:
    184         {
    185           int elementWidth = file.readUShort();
    186           rawSize = ((file.readUInt() * elementWidth + 1) / 2) + 4;
    187           break;
    188         }
    189         default:
    190           Log.errorAndQuit("Unrecognised ident in data-payload instruction: " + rawType);
    191       }
    192     }
    193     return opcode;
    194   }
    195 
    196   @Override
    197   public String toString() {
    198     if (justRaw) {
    199       switch (rawType) {
    200         case RAW_TYPE_PACKED_SWITCH_DATA:
    201           return "PACKED SWITCH DATA";
    202         case RAW_TYPE_SPARSE_SWITCH_DATA:
    203           return "SPARSE SWITCH DATA";
    204         case RAW_TYPE_FILL_ARRAY_DATA_DATA:
    205           return "FILL ARRAY DATA DATA";
    206         default:
    207       }
    208 
    209     }
    210 
    211     String repr = info.name;
    212 
    213     AbstractFormat format = info.format;
    214 
    215     if (invokeFormatInfo != null) {
    216       String vregs = "";
    217 
    218       int numVregs = (int) vregA;
    219 
    220       if (numVregs > 5) {
    221         Log.debug("vA in an 35c invoke was greater than 5? Assuming 5.");
    222         numVregs = 5;
    223       } else if (numVregs < 0) {
    224         Log.debug("vA in an 35c invoke was less than 0? Assuming 0.");
    225         numVregs = 0;
    226       }
    227 
    228       switch (numVregs) {
    229         case 5:
    230           vregs = ", v" + invokeFormatInfo.vregG;
    231           // fallthrough
    232         case 4:
    233           vregs = ", v" + invokeFormatInfo.vregF + vregs;
    234           // fallthrough
    235         case 3:
    236           vregs = ", v" + invokeFormatInfo.vregE + vregs;
    237           // fallthrough
    238         case 2:
    239           vregs = ", v" + invokeFormatInfo.vregD + vregs;
    240           // fallthrough
    241         case 1:
    242           vregs = "v" + vregC + vregs;
    243           break;
    244         default:
    245       }
    246 
    247       repr += "(" + vregs + ")";
    248 
    249       long poolIndex = ((ContainsPoolIndex)format).getPoolIndex(this);
    250       repr += " meth@" + poolIndex;
    251 
    252       return repr;
    253     }
    254 
    255 
    256 
    257     if (format instanceof ContainsVRegs) {
    258       String vregs = "";
    259       switch (((ContainsVRegs)format).getVRegCount()) {
    260         case 3:
    261           vregs = ", v" + vregC;
    262           // fallthrough
    263         case 2:
    264           vregs = ", v" + vregB + vregs;
    265           // fallthrough
    266         case 1:
    267           vregs = "v" + vregA + vregs;
    268           break;
    269         default:
    270           Log.errorAndQuit("Invalid number of vregs reported by a Format.");
    271       }
    272 
    273       repr += " " + vregs;
    274     }
    275     if (format instanceof ContainsConst) {
    276       long constant = ((ContainsConst)format).getConst(this);
    277       repr += " #" + constant;
    278     }
    279     if (format instanceof ContainsPoolIndex) {
    280       long poolIndex = ((ContainsPoolIndex)format).getPoolIndex(this);
    281       repr += " pool@" + poolIndex;
    282     }
    283     if (format instanceof ContainsTarget) {
    284       long target = ((ContainsTarget)format).getTarget(this);
    285       repr += " +" + target;
    286     }
    287 
    288     return repr;
    289   }
    290 
    291   @Override
    292   public void incrementIndex(IndexUpdateKind kind, int insertedIdx) {
    293     // Do nothing.
    294   }
    295 
    296   // STATIC INSTRUCTION CODE
    297   private static Map<Integer,OpcodeInfo> opcode_map_by_int = new HashMap<Integer,OpcodeInfo>();
    298   private static Map<Opcode,OpcodeInfo> opcode_map_by_enum = new HashMap<Opcode,OpcodeInfo>();
    299 
    300   public static OpcodeInfo getOpcodeInfo(Opcode opcode) {
    301     return opcode_map_by_enum.get(opcode);
    302   }
    303 
    304   public static OpcodeInfo getOpcodeInfo(int opcodeValue) {
    305     return opcode_map_by_int.get(opcodeValue);
    306   }
    307 
    308   private static void addOpcodeInfo(Opcode opcode, String name,
    309       int opcodeValue, AbstractFormat fmt) {
    310     OpcodeInfo info = new OpcodeInfo(opcode, name, opcodeValue, fmt);
    311     if (opcode.ordinal() != opcodeValue) {
    312       Log.errorAndQuit(String.format("Opcode: %s (enum ordinal 0x%x) != (value 0x%x)",
    313           opcode.toString(), opcode.ordinal(), opcodeValue));
    314     }
    315     opcode_map_by_int.put(opcodeValue, info);
    316     opcode_map_by_enum.put(opcode, info);
    317   }
    318 
    319   static {
    320     addOpcodeInfo(Opcode.NOP, "nop", 0x00, new Format10x());
    321     addOpcodeInfo(Opcode.MOVE, "move", 0x01, new Format12x());
    322     addOpcodeInfo(Opcode.MOVE_FROM16, "move/from16", 0x02, new Format22x());
    323     addOpcodeInfo(Opcode.MOVE_16, "move/16", 0x03, new Format32x());
    324     addOpcodeInfo(Opcode.MOVE_WIDE, "move-wide", 0x04, new Format12x());
    325     addOpcodeInfo(Opcode.MOVE_WIDE_FROM16, "move-wide/from16", 0x05, new Format22x());
    326     addOpcodeInfo(Opcode.MOVE_WIDE_16, "move-wide/16", 0x06, new Format32x());
    327     addOpcodeInfo(Opcode.MOVE_OBJECT, "move-object", 0x07, new Format12x());
    328     addOpcodeInfo(Opcode.MOVE_OBJECT_FROM16, "move-object/from16", 0x08, new Format22x());
    329     addOpcodeInfo(Opcode.MOVE_OBJECT_16, "move-object/16", 0x09, new Format32x());
    330     addOpcodeInfo(Opcode.MOVE_RESULT, "move-result", 0x0a, new Format11x());
    331     addOpcodeInfo(Opcode.MOVE_RESULT_WIDE, "move-result-wide", 0x0b, new Format11x());
    332     addOpcodeInfo(Opcode.MOVE_RESULT_OBJECT, "move-result-object", 0x0c, new Format11x());
    333     addOpcodeInfo(Opcode.MOVE_EXCEPTION, "move-exception", 0x0d, new Format11x());
    334     addOpcodeInfo(Opcode.RETURN_VOID, "return-void", 0x0e, new Format10x());
    335     addOpcodeInfo(Opcode.RETURN, "return", 0x0f, new Format11x());
    336     addOpcodeInfo(Opcode.RETURN_WIDE, "return-wide", 0x10, new Format11x());
    337     addOpcodeInfo(Opcode.RETURN_OBJECT, "return-object", 0x11, new Format11x());
    338     addOpcodeInfo(Opcode.CONST_4, "const/4", 0x12, new Format11n());
    339     addOpcodeInfo(Opcode.CONST_16, "const/16", 0x13, new Format21s());
    340     addOpcodeInfo(Opcode.CONST, "const", 0x14, new Format31i());
    341     addOpcodeInfo(Opcode.CONST_HIGH16, "const/high16", 0x15, new Format21h());
    342     addOpcodeInfo(Opcode.CONST_WIDE_16, "const-wide/16", 0x16, new Format21s());
    343     addOpcodeInfo(Opcode.CONST_WIDE_32, "const-wide/32", 0x17, new Format31i());
    344     addOpcodeInfo(Opcode.CONST_WIDE, "const-wide", 0x18, new Format51l());
    345     addOpcodeInfo(Opcode.CONST_WIDE_HIGH16, "const-wide/high16", 0x19, new Format21h());
    346     addOpcodeInfo(Opcode.CONST_STRING, "const-string", 0x1a, new Format21c());
    347     addOpcodeInfo(Opcode.CONST_STRING_JUMBO, "const-string/jumbo", 0x1b, new Format31c());
    348     addOpcodeInfo(Opcode.CONST_CLASS, "const-class", 0x1c, new Format21c());
    349     addOpcodeInfo(Opcode.MONITOR_ENTER, "monitor-enter", 0x1d, new Format11x());
    350     addOpcodeInfo(Opcode.MONITOR_EXIT, "monitor-exit", 0x1e, new Format11x());
    351     addOpcodeInfo(Opcode.CHECK_CAST, "check-cast", 0x1f, new Format21c());
    352     addOpcodeInfo(Opcode.INSTANCE_OF, "instance-of", 0x20, new Format22c());
    353     addOpcodeInfo(Opcode.ARRAY_LENGTH, "array-length", 0x21, new Format12x());
    354     addOpcodeInfo(Opcode.NEW_INSTANCE, "new-instance", 0x22, new Format21c());
    355     addOpcodeInfo(Opcode.NEW_ARRAY, "new-array", 0x23, new Format22c());
    356     addOpcodeInfo(Opcode.FILLED_NEW_ARRAY, "filled-new-array", 0x24, new Format35c());
    357     addOpcodeInfo(Opcode.FILLED_NEW_ARRAY_RANGE, "filled-new-array/range",
    358         0x25, new Format3rc());
    359     addOpcodeInfo(Opcode.FILL_ARRAY_DATA, "fill-array-data", 0x26, new Format31t());
    360     addOpcodeInfo(Opcode.THROW, "throw", 0x27, new Format11x());
    361     addOpcodeInfo(Opcode.GOTO, "goto", 0x28, new Format10t());
    362     addOpcodeInfo(Opcode.GOTO_16, "goto/16", 0x29, new Format20t());
    363     addOpcodeInfo(Opcode.GOTO_32, "goto/32", 0x2a, new Format30t());
    364     addOpcodeInfo(Opcode.PACKED_SWITCH, "packed-switch", 0x2b, new Format31t());
    365     addOpcodeInfo(Opcode.SPARSE_SWITCH, "sparse-switch", 0x2c, new Format31t());
    366     addOpcodeInfo(Opcode.CMPL_FLOAT, "cmpl-float", 0x2d, new Format23x());
    367     addOpcodeInfo(Opcode.CMPG_FLOAT, "cmpg-float", 0x2e, new Format23x());
    368     addOpcodeInfo(Opcode.CMPL_DOUBLE, "cmpl-double", 0x2f, new Format23x());
    369     addOpcodeInfo(Opcode.CMPG_DOUBLE, "cmpg-double", 0x30, new Format23x());
    370     addOpcodeInfo(Opcode.CMP_LONG, "cmp-long", 0x31, new Format23x());
    371     addOpcodeInfo(Opcode.IF_EQ, "if-eq", 0x32, new Format22t());
    372     addOpcodeInfo(Opcode.IF_NE, "if-ne", 0x33, new Format22t());
    373     addOpcodeInfo(Opcode.IF_LT, "if-lt", 0x34, new Format22t());
    374     addOpcodeInfo(Opcode.IF_GE, "if-ge", 0x35, new Format22t());
    375     addOpcodeInfo(Opcode.IF_GT, "if-gt", 0x36, new Format22t());
    376     addOpcodeInfo(Opcode.IF_LE, "if-le", 0x37, new Format22t());
    377     addOpcodeInfo(Opcode.IF_EQZ, "if-eqz", 0x38, new Format21t());
    378     addOpcodeInfo(Opcode.IF_NEZ, "if-nez", 0x39, new Format21t());
    379     addOpcodeInfo(Opcode.IF_LTZ, "if-ltz", 0x3a, new Format21t());
    380     addOpcodeInfo(Opcode.IF_GEZ, "if-gez", 0x3b, new Format21t());
    381     addOpcodeInfo(Opcode.IF_GTZ, "if-gtz", 0x3c, new Format21t());
    382     addOpcodeInfo(Opcode.IF_LEZ, "if-lez", 0x3d, new Format21t());
    383     addOpcodeInfo(Opcode.UNUSED_3E, "unused-3e", 0x3e, new Format10x());
    384     addOpcodeInfo(Opcode.UNUSED_3F, "unused-3f", 0x3f, new Format10x());
    385     addOpcodeInfo(Opcode.UNUSED_40, "unused-40", 0x40, new Format10x());
    386     addOpcodeInfo(Opcode.UNUSED_41, "unused-41", 0x41, new Format10x());
    387     addOpcodeInfo(Opcode.UNUSED_42, "unused-42", 0x42, new Format10x());
    388     addOpcodeInfo(Opcode.UNUSED_43, "unused-43", 0x43, new Format10x());
    389     addOpcodeInfo(Opcode.AGET, "aget", 0x44, new Format23x());
    390     addOpcodeInfo(Opcode.AGET_WIDE, "aget-wide", 0x45, new Format23x());
    391     addOpcodeInfo(Opcode.AGET_WIDE, "aget-wide", 0x45, new Format23x());
    392     addOpcodeInfo(Opcode.AGET_OBJECT, "aget-object", 0x46, new Format23x());
    393     addOpcodeInfo(Opcode.AGET_BOOLEAN, "aget-boolean", 0x47, new Format23x());
    394     addOpcodeInfo(Opcode.AGET_BYTE, "aget-byte", 0x48, new Format23x());
    395     addOpcodeInfo(Opcode.AGET_CHAR, "aget-char", 0x49, new Format23x());
    396     addOpcodeInfo(Opcode.AGET_SHORT, "aget-short", 0x4a, new Format23x());
    397     addOpcodeInfo(Opcode.APUT, "aput", 0x4b, new Format23x());
    398     addOpcodeInfo(Opcode.APUT_WIDE, "aput-wide", 0x4c, new Format23x());
    399     addOpcodeInfo(Opcode.APUT_OBJECT, "aput-object", 0x4d, new Format23x());
    400     addOpcodeInfo(Opcode.APUT_BOOLEAN, "aput-boolean", 0x4e, new Format23x());
    401     addOpcodeInfo(Opcode.APUT_BYTE, "aput-byte", 0x4f, new Format23x());
    402     addOpcodeInfo(Opcode.APUT_CHAR, "aput-char", 0x50, new Format23x());
    403     addOpcodeInfo(Opcode.APUT_SHORT, "aput-short", 0x51, new Format23x());
    404     addOpcodeInfo(Opcode.IGET, "iget", 0x52, new Format22c());
    405     addOpcodeInfo(Opcode.IGET_WIDE, "iget-wide", 0x53, new Format22c());
    406     addOpcodeInfo(Opcode.IGET_OBJECT, "iget-object", 0x54, new Format22c());
    407     addOpcodeInfo(Opcode.IGET_BOOLEAN, "iget-boolean", 0x55, new Format22c());
    408     addOpcodeInfo(Opcode.IGET_BYTE, "iget-byte", 0x56, new Format22c());
    409     addOpcodeInfo(Opcode.IGET_CHAR, "iget-char", 0x57, new Format22c());
    410     addOpcodeInfo(Opcode.IGET_SHORT, "iget-short", 0x58, new Format22c());
    411     addOpcodeInfo(Opcode.IPUT, "iput", 0x59, new Format22c());
    412     addOpcodeInfo(Opcode.IPUT_WIDE, "iput-wide", 0x5a, new Format22c());
    413     addOpcodeInfo(Opcode.IPUT_OBJECT, "iput-object", 0x5b, new Format22c());
    414     addOpcodeInfo(Opcode.IPUT_BOOLEAN, "iput-boolean", 0x5c, new Format22c());
    415     addOpcodeInfo(Opcode.IPUT_BYTE, "iput-byte", 0x5d, new Format22c());
    416     addOpcodeInfo(Opcode.IPUT_CHAR, "iput-char", 0x5e, new Format22c());
    417     addOpcodeInfo(Opcode.IPUT_SHORT, "iput-short", 0x5f, new Format22c());
    418     addOpcodeInfo(Opcode.SGET, "sget", 0x60, new Format21c());
    419     addOpcodeInfo(Opcode.SGET_WIDE, "sget-wide", 0x61, new Format21c());
    420     addOpcodeInfo(Opcode.SGET_OBJECT, "sget-object", 0x62, new Format21c());
    421     addOpcodeInfo(Opcode.SGET_BOOLEAN, "sget-boolean", 0x63, new Format21c());
    422     addOpcodeInfo(Opcode.SGET_BYTE, "sget-byte", 0x64, new Format21c());
    423     addOpcodeInfo(Opcode.SGET_CHAR, "sget-char", 0x65, new Format21c());
    424     addOpcodeInfo(Opcode.SGET_SHORT, "sget-short", 0x66, new Format21c());
    425     addOpcodeInfo(Opcode.SPUT, "sput", 0x67, new Format21c());
    426     addOpcodeInfo(Opcode.SPUT_WIDE, "sput-wide", 0x68, new Format21c());
    427     addOpcodeInfo(Opcode.SPUT_OBJECT, "sput-object", 0x69, new Format21c());
    428     addOpcodeInfo(Opcode.SPUT_BOOLEAN, "sput-boolean", 0x6a, new Format21c());
    429     addOpcodeInfo(Opcode.SPUT_BYTE, "sput-byte", 0x6b, new Format21c());
    430     addOpcodeInfo(Opcode.SPUT_CHAR, "sput-char", 0x6c, new Format21c());
    431     addOpcodeInfo(Opcode.SPUT_SHORT, "sput-short", 0x6d, new Format21c());
    432     addOpcodeInfo(Opcode.INVOKE_VIRTUAL, "invoke-virtual", 0x6e, new Format35c());
    433     addOpcodeInfo(Opcode.INVOKE_SUPER, "invoke-super", 0x6f, new Format35c());
    434     addOpcodeInfo(Opcode.INVOKE_DIRECT, "invoke-direct", 0x70, new Format35c());
    435     addOpcodeInfo(Opcode.INVOKE_STATIC, "invoke-static", 0x71, new Format35c());
    436     addOpcodeInfo(Opcode.INVOKE_INTERFACE, "invoke-interface", 0x72, new Format35c());
    437     addOpcodeInfo(Opcode.RETURN_VOID_NO_BARRIER, "return-void-no-barrier", 0x73, new Format10x());
    438     addOpcodeInfo(Opcode.INVOKE_VIRTUAL_RANGE, "invoke-virtual/range", 0x74, new Format3rc());
    439     addOpcodeInfo(Opcode.INVOKE_SUPER_RANGE, "invoke-super/range", 0x75, new Format3rc());
    440     addOpcodeInfo(Opcode.INVOKE_DIRECT_RANGE, "invoke-direct/range", 0x76, new Format3rc());
    441     addOpcodeInfo(Opcode.INVOKE_STATIC_RANGE, "invoke-static/range", 0x77, new Format3rc());
    442     addOpcodeInfo(Opcode.INVOKE_INTERFACE_RANGE, "invoke-interface/range",
    443         0x78, new Format3rc());
    444     addOpcodeInfo(Opcode.UNUSED_79, "unused-79", 0x79, new Format10x());
    445     addOpcodeInfo(Opcode.UNUSED_7A, "unused-7a", 0x7a, new Format10x());
    446     addOpcodeInfo(Opcode.NEG_INT, "neg-int", 0x7b, new Format12x());
    447     addOpcodeInfo(Opcode.NOT_INT, "not-int", 0x7c, new Format12x());
    448     addOpcodeInfo(Opcode.NEG_LONG, "neg-long", 0x7d, new Format12x());
    449     addOpcodeInfo(Opcode.NOT_LONG, "not-long", 0x7e, new Format12x());
    450     addOpcodeInfo(Opcode.NEG_FLOAT, "neg-float", 0x7f, new Format12x());
    451     addOpcodeInfo(Opcode.NEG_DOUBLE, "neg-double", 0x80, new Format12x());
    452     addOpcodeInfo(Opcode.INT_TO_LONG, "int-to-long", 0x81, new Format12x());
    453     addOpcodeInfo(Opcode.INT_TO_FLOAT, "int-to-float", 0x82, new Format12x());
    454     addOpcodeInfo(Opcode.INT_TO_DOUBLE, "int-to-double", 0x83, new Format12x());
    455     addOpcodeInfo(Opcode.LONG_TO_INT, "long-to-int", 0x84, new Format12x());
    456     addOpcodeInfo(Opcode.LONG_TO_FLOAT, "long-to-float", 0x85, new Format12x());
    457     addOpcodeInfo(Opcode.LONG_TO_DOUBLE, "long-to-double", 0x86, new Format12x());
    458     addOpcodeInfo(Opcode.FLOAT_TO_INT, "float-to-int", 0x87, new Format12x());
    459     addOpcodeInfo(Opcode.FLOAT_TO_LONG, "float-to-long", 0x88, new Format12x());
    460     addOpcodeInfo(Opcode.FLOAT_TO_DOUBLE, "float-to-double", 0x89, new Format12x());
    461     addOpcodeInfo(Opcode.DOUBLE_TO_INT, "double-to-int", 0x8a, new Format12x());
    462     addOpcodeInfo(Opcode.DOUBLE_TO_LONG, "double-to-long", 0x8b, new Format12x());
    463     addOpcodeInfo(Opcode.DOUBLE_TO_FLOAT, "double-to-float", 0x8c, new Format12x());
    464     addOpcodeInfo(Opcode.INT_TO_BYTE, "int-to-byte", 0x8d, new Format12x());
    465     addOpcodeInfo(Opcode.INT_TO_CHAR, "int-to-char", 0x8e, new Format12x());
    466     addOpcodeInfo(Opcode.INT_TO_SHORT, "int-to-short", 0x8f, new Format12x());
    467     addOpcodeInfo(Opcode.ADD_INT, "add-int", 0x90, new Format23x());
    468     addOpcodeInfo(Opcode.SUB_INT, "sub-int", 0x91, new Format23x());
    469     addOpcodeInfo(Opcode.MUL_INT, "mul-int", 0x92, new Format23x());
    470     addOpcodeInfo(Opcode.DIV_INT, "div-int", 0x93, new Format23x());
    471     addOpcodeInfo(Opcode.REM_INT, "rem-int", 0x94, new Format23x());
    472     addOpcodeInfo(Opcode.AND_INT, "and-int", 0x95, new Format23x());
    473     addOpcodeInfo(Opcode.OR_INT, "or-int", 0x96, new Format23x());
    474     addOpcodeInfo(Opcode.XOR_INT, "xor-int", 0x97, new Format23x());
    475     addOpcodeInfo(Opcode.SHL_INT, "shl-int", 0x98, new Format23x());
    476     addOpcodeInfo(Opcode.SHR_INT, "shr-int", 0x99, new Format23x());
    477     addOpcodeInfo(Opcode.USHR_INT, "ushr-int", 0x9a, new Format23x());
    478     addOpcodeInfo(Opcode.ADD_LONG, "add-long", 0x9b, new Format23x());
    479     addOpcodeInfo(Opcode.SUB_LONG, "sub-long", 0x9c, new Format23x());
    480     addOpcodeInfo(Opcode.MUL_LONG, "mul-long", 0x9d, new Format23x());
    481     addOpcodeInfo(Opcode.DIV_LONG, "div-long", 0x9e, new Format23x());
    482     addOpcodeInfo(Opcode.REM_LONG, "rem-long", 0x9f, new Format23x());
    483     addOpcodeInfo(Opcode.AND_LONG, "and-long", 0xa0, new Format23x());
    484     addOpcodeInfo(Opcode.OR_LONG, "or-long", 0xa1, new Format23x());
    485     addOpcodeInfo(Opcode.XOR_LONG, "xor-long", 0xa2, new Format23x());
    486     addOpcodeInfo(Opcode.SHL_LONG, "shl-long", 0xa3, new Format23x());
    487     addOpcodeInfo(Opcode.SHR_LONG, "shr-long", 0xa4, new Format23x());
    488     addOpcodeInfo(Opcode.USHR_LONG, "ushr-long", 0xa5, new Format23x());
    489     addOpcodeInfo(Opcode.ADD_FLOAT, "add-float", 0xa6, new Format23x());
    490     addOpcodeInfo(Opcode.SUB_FLOAT, "sub-float", 0xa7, new Format23x());
    491     addOpcodeInfo(Opcode.MUL_FLOAT, "mul-float", 0xa8, new Format23x());
    492     addOpcodeInfo(Opcode.DIV_FLOAT, "div-float", 0xa9, new Format23x());
    493     addOpcodeInfo(Opcode.REM_FLOAT, "rem-float", 0xaa, new Format23x());
    494     addOpcodeInfo(Opcode.ADD_DOUBLE, "add-double", 0xab, new Format23x());
    495     addOpcodeInfo(Opcode.SUB_DOUBLE, "sub-double", 0xac, new Format23x());
    496     addOpcodeInfo(Opcode.MUL_DOUBLE, "mul-double", 0xad, new Format23x());
    497     addOpcodeInfo(Opcode.DIV_DOUBLE, "div-double", 0xae, new Format23x());
    498     addOpcodeInfo(Opcode.REM_DOUBLE, "rem-double", 0xaf, new Format23x());
    499     addOpcodeInfo(Opcode.ADD_INT_2ADDR, "add-int/2addr", 0xb0, new Format12x());
    500     addOpcodeInfo(Opcode.SUB_INT_2ADDR, "sub-int/2addr", 0xb1, new Format12x());
    501     addOpcodeInfo(Opcode.MUL_INT_2ADDR, "mul-int/2addr", 0xb2, new Format12x());
    502     addOpcodeInfo(Opcode.DIV_INT_2ADDR, "div-int/2addr", 0xb3, new Format12x());
    503     addOpcodeInfo(Opcode.REM_INT_2ADDR, "rem-int/2addr", 0xb4, new Format12x());
    504     addOpcodeInfo(Opcode.AND_INT_2ADDR, "and-int/2addr", 0xb5, new Format12x());
    505     addOpcodeInfo(Opcode.OR_INT_2ADDR, "or-int/2addr", 0xb6, new Format12x());
    506     addOpcodeInfo(Opcode.XOR_INT_2ADDR, "xor-int/2addr", 0xb7, new Format12x());
    507     addOpcodeInfo(Opcode.SHL_INT_2ADDR, "shl-int/2addr", 0xb8, new Format12x());
    508     addOpcodeInfo(Opcode.SHR_INT_2ADDR, "shr-int/2addr", 0xb9, new Format12x());
    509     addOpcodeInfo(Opcode.USHR_INT_2ADDR, "ushr-int/2addr", 0xba, new Format12x());
    510     addOpcodeInfo(Opcode.ADD_LONG_2ADDR, "add-long/2addr", 0xbb, new Format12x());
    511     addOpcodeInfo(Opcode.SUB_LONG_2ADDR, "sub-long/2addr", 0xbc, new Format12x());
    512     addOpcodeInfo(Opcode.MUL_LONG_2ADDR, "mul-long/2addr", 0xbd, new Format12x());
    513     addOpcodeInfo(Opcode.DIV_LONG_2ADDR, "div-long/2addr", 0xbe, new Format12x());
    514     addOpcodeInfo(Opcode.REM_LONG_2ADDR, "rem-long/2addr", 0xbf, new Format12x());
    515     addOpcodeInfo(Opcode.AND_LONG_2ADDR, "and-long/2addr", 0xc0, new Format12x());
    516     addOpcodeInfo(Opcode.OR_LONG_2ADDR, "or-long/2addr", 0xc1, new Format12x());
    517     addOpcodeInfo(Opcode.XOR_LONG_2ADDR, "xor-long/2addr", 0xc2, new Format12x());
    518     addOpcodeInfo(Opcode.SHL_LONG_2ADDR, "shl-long/2addr", 0xc3, new Format12x());
    519     addOpcodeInfo(Opcode.SHR_LONG_2ADDR, "shr-long/2addr", 0xc4, new Format12x());
    520     addOpcodeInfo(Opcode.USHR_LONG_2ADDR, "ushr-long/2addr", 0xc5, new Format12x());
    521     addOpcodeInfo(Opcode.ADD_FLOAT_2ADDR, "add-float/2addr", 0xc6, new Format12x());
    522     addOpcodeInfo(Opcode.SUB_FLOAT_2ADDR, "sub-float/2addr", 0xc7, new Format12x());
    523     addOpcodeInfo(Opcode.MUL_FLOAT_2ADDR, "mul-float/2addr", 0xc8, new Format12x());
    524     addOpcodeInfo(Opcode.DIV_FLOAT_2ADDR, "div-float/2addr", 0xc9, new Format12x());
    525     addOpcodeInfo(Opcode.REM_FLOAT_2ADDR, "rem-float/2addr", 0xca, new Format12x());
    526     addOpcodeInfo(Opcode.ADD_DOUBLE_2ADDR, "add-double/2addr", 0xcb, new Format12x());
    527     addOpcodeInfo(Opcode.SUB_DOUBLE_2ADDR, "sub-double/2addr", 0xcc, new Format12x());
    528     addOpcodeInfo(Opcode.MUL_DOUBLE_2ADDR, "mul-double/2addr", 0xcd, new Format12x());
    529     addOpcodeInfo(Opcode.DIV_DOUBLE_2ADDR, "div-double/2addr", 0xce, new Format12x());
    530     addOpcodeInfo(Opcode.REM_DOUBLE_2ADDR, "rem-double/2addr", 0xcf, new Format12x());
    531     addOpcodeInfo(Opcode.ADD_INT_LIT16, "add-int/lit16", 0xd0, new Format22s());
    532     addOpcodeInfo(Opcode.RSUB_INT, "rsub-int", 0xd1, new Format22s());
    533     addOpcodeInfo(Opcode.MUL_INT_LIT16, "mul-int/lit16", 0xd2, new Format22s());
    534     addOpcodeInfo(Opcode.DIV_INT_LIT16, "div-int/lit16", 0xd3, new Format22s());
    535     addOpcodeInfo(Opcode.REM_INT_LIT16, "rem-int/lit16", 0xd4, new Format22s());
    536     addOpcodeInfo(Opcode.AND_INT_LIT16, "and-int/lit16", 0xd5, new Format22s());
    537     addOpcodeInfo(Opcode.OR_INT_LIT16, "or-int/lit16", 0xd6, new Format22s());
    538     addOpcodeInfo(Opcode.XOR_INT_LIT16, "xor-int/lit16", 0xd7, new Format22s());
    539     addOpcodeInfo(Opcode.ADD_INT_LIT8, "add-int/lit8", 0xd8, new Format22b());
    540     addOpcodeInfo(Opcode.RSUB_INT_LIT8, "rsub-int/lit8", 0xd9, new Format22b());
    541     addOpcodeInfo(Opcode.MUL_INT_LIT8, "mul-int/lit8", 0xda, new Format22b());
    542     addOpcodeInfo(Opcode.DIV_INT_LIT8, "div-int/lit8", 0xdb, new Format22b());
    543     addOpcodeInfo(Opcode.REM_INT_LIT8, "rem-int/lit8", 0xdc, new Format22b());
    544     addOpcodeInfo(Opcode.AND_INT_LIT8, "and-int/lit8", 0xdd, new Format22b());
    545     addOpcodeInfo(Opcode.OR_INT_LIT8, "or-int/lit8", 0xde, new Format22b());
    546     addOpcodeInfo(Opcode.XOR_INT_LIT8, "xor-int/lit8", 0xdf, new Format22b());
    547     addOpcodeInfo(Opcode.SHL_INT_LIT8, "shl-int/lit8", 0xe0, new Format22b());
    548     addOpcodeInfo(Opcode.SHR_INT_LIT8, "shr-int/lit8", 0xe1, new Format22b());
    549     addOpcodeInfo(Opcode.USHR_INT_LIT8, "ushr-int/lit8", 0xe2, new Format22b());
    550     addOpcodeInfo(Opcode.IGET_QUICK, "+iget-quick", 0xe3, new Format22c());
    551     addOpcodeInfo(Opcode.IGET_WIDE_QUICK, "+iget-wide-quick", 0xe4, new Format22c());
    552     addOpcodeInfo(Opcode.IGET_OBJECT_QUICK, "+iget-object-quick", 0xe5, new Format22c());
    553     addOpcodeInfo(Opcode.IPUT_QUICK, "+iput-quick", 0xe6, new Format22c());
    554     addOpcodeInfo(Opcode.IPUT_WIDE_QUICK, "+iput-wide-quick", 0xe7, new Format22c());
    555     addOpcodeInfo(Opcode.IPUT_OBJECT_QUICK, "+iput-object-quick", 0xe8, new Format22c());
    556     addOpcodeInfo(Opcode.INVOKE_VIRTUAL_QUICK, "+invoke-virtual-quick", 0xe9, new Format35c());
    557     addOpcodeInfo(Opcode.INVOKE_VIRTUAL_QUICK_RANGE, "+invoke-virtual-quick/range",
    558         0xea, new Format3rc());
    559     addOpcodeInfo(Opcode.IPUT_BOOLEAN_QUICK, "+iput-boolean-quick", 0xeb, new Format22c());
    560     addOpcodeInfo(Opcode.IPUT_BYTE_QUICK, "+iput-byte-quick", 0xec, new Format22c());
    561     addOpcodeInfo(Opcode.IPUT_CHAR_QUICK, "+iput-char-quick", 0xed, new Format22c());
    562     addOpcodeInfo(Opcode.IPUT_SHORT_QUICK, "+iput-short-quick", 0xee, new Format22c());
    563     addOpcodeInfo(Opcode.UNUSED_EF, "unused-ef", 0xef, new Format10x());
    564     addOpcodeInfo(Opcode.UNUSED_F0, "unused-f0", 0xf0, new Format10x());
    565     addOpcodeInfo(Opcode.UNUSED_F1, "unused-f1", 0xf1, new Format10x());
    566     addOpcodeInfo(Opcode.UNUSED_F2, "unused-f2", 0xf2, new Format10x());
    567     addOpcodeInfo(Opcode.UNUSED_F3, "unused-f3", 0xf3, new Format10x());
    568     addOpcodeInfo(Opcode.UNUSED_F4, "unused-f4", 0xf4, new Format10x());
    569     addOpcodeInfo(Opcode.UNUSED_F5, "unused-f5", 0xf5, new Format10x());
    570     addOpcodeInfo(Opcode.UNUSED_F6, "unused-f6", 0xf6, new Format10x());
    571     addOpcodeInfo(Opcode.UNUSED_F7, "unused-f7", 0xf7, new Format10x());
    572     addOpcodeInfo(Opcode.UNUSED_F8, "unused-f8", 0xf8, new Format10x());
    573     addOpcodeInfo(Opcode.UNUSED_F9, "unused-f9", 0xf9, new Format10x());
    574     addOpcodeInfo(Opcode.UNUSED_FA, "unused-fa", 0xfa, new Format10x());
    575     addOpcodeInfo(Opcode.UNUSED_FB, "unused-fb", 0xfb, new Format10x());
    576     addOpcodeInfo(Opcode.UNUSED_FC, "unused-fc", 0xfc, new Format10x());
    577     addOpcodeInfo(Opcode.UNUSED_FD, "unused-fd", 0xfd, new Format10x());
    578     addOpcodeInfo(Opcode.UNUSED_FE, "unused-fe", 0xfe, new Format10x());
    579     addOpcodeInfo(Opcode.UNUSED_FF, "unused-ff", 0xff, new Format10x());
    580     if (opcode_map_by_int.size() != 256) {
    581       Log.errorAndQuit("Incorrect number of bytecodes defined.");
    582     }
    583   }
    584 }
    585