All Downloads are FREE. Search and download functionalities are using the official Maven repository.

soot.dexpler.instructions.ConstInstruction Maven / Gradle / Ivy

There is a newer version: 4.6.0
Show newest version
package soot.dexpler.instructions;

/*-
 * #%L
 * Soot - a J*va Optimization Framework
 * %%
 * Copyright (C) 2012 Michael Markert, Frank Hartmann
 * 
 * (c) 2012 University of Luxembourg - Interdisciplinary Centre for
 * Security Reliability and Trust (SnT) - All rights reserved
 * Alexandre Bartel
 * 
 * %%
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation, either version 2.1 of the
 * License, or (at your option) any later version.
 * 
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Lesser Public License for more details.
 * 
 * You should have received a copy of the GNU General Lesser Public
 * License along with this program.  If not, see
 * .
 * #L%
 */

import org.jf.dexlib2.Opcode;
import org.jf.dexlib2.iface.instruction.Instruction;
import org.jf.dexlib2.iface.instruction.NarrowLiteralInstruction;
import org.jf.dexlib2.iface.instruction.OneRegisterInstruction;
import org.jf.dexlib2.iface.instruction.WideLiteralInstruction;

import soot.dexpler.DexBody;
import soot.dexpler.IDalvikTyper;
import soot.dexpler.typing.DalvikTyper;
import soot.dexpler.typing.UntypedConstant;
import soot.dexpler.typing.UntypedIntOrFloatConstant;
import soot.dexpler.typing.UntypedLongOrDoubleConstant;
import soot.jimple.AssignStmt;
import soot.jimple.Constant;
import soot.jimple.IntConstant;
import soot.jimple.Jimple;
import soot.jimple.LongConstant;

public class ConstInstruction extends DexlibAbstractInstruction {

  public ConstInstruction(Instruction instruction, int codeAdress) {
    super(instruction, codeAdress);
  }

  @Override
  public void jimplify(DexBody body) {
    int dest = ((OneRegisterInstruction) instruction).getRegisterA();

    Constant cst = getConstant(dest, body);
    AssignStmt assign = Jimple.v().newAssignStmt(body.getRegisterLocal(dest), cst);
    setUnit(assign);
    addTags(assign);
    body.add(assign);

    if (IDalvikTyper.ENABLE_DVKTYPER) {
      if (cst instanceof UntypedConstant) {
        DalvikTyper.v().addConstraint(assign.getLeftOpBox(), assign.getRightOpBox());
      } else {
        DalvikTyper.v().setType(assign.getLeftOpBox(), cst.getType(), false);
      }
    }
  }

  /**
   * Return the literal constant for this instruction.
   *
   * @param register
   *          the register number to fill
   * @param body
   *          the body containing the instruction
   */
  private Constant getConstant(int dest, DexBody body) {

    long literal = 0;

    if (instruction instanceof WideLiteralInstruction) {
      literal = ((WideLiteralInstruction) instruction).getWideLiteral();
    } else if (instruction instanceof NarrowLiteralInstruction) {
      literal = ((NarrowLiteralInstruction) instruction).getNarrowLiteral();
    } else {
      throw new RuntimeException("literal error: expected narrow or wide literal.");
    }

    // floats are handled later in DexBody by calling DexNumtransformer
    Opcode opcode = instruction.getOpcode();
    switch (opcode) {
      case CONST:
      case CONST_4:
      case CONST_16:
        if (IDalvikTyper.ENABLE_DVKTYPER) {
          return UntypedIntOrFloatConstant.v((int) literal);
        } else {
          return IntConstant.v((int) literal);
        }

      case CONST_HIGH16:
        if (IDalvikTyper.ENABLE_DVKTYPER) {
          //
          // return UntypedIntOrFloatConstant.v((int)literal<<16).toFloatConstant();
          // seems that dexlib correctly puts the 16bits into the topmost bits.
          //
          return UntypedIntOrFloatConstant.v((int) literal);// .toFloatConstant();
        } else {
          return IntConstant.v((int) literal);
        }

      case CONST_WIDE_HIGH16:
        if (IDalvikTyper.ENABLE_DVKTYPER) {
          // return UntypedLongOrDoubleConstant.v((long)literal<<48).toDoubleConstant();
          // seems that dexlib correctly puts the 16bits into the topmost bits.
          //
          return UntypedLongOrDoubleConstant.v(literal);// .toDoubleConstant();
        } else {
          return LongConstant.v(literal);
        }

      case CONST_WIDE:
      case CONST_WIDE_16:
      case CONST_WIDE_32:
        if (IDalvikTyper.ENABLE_DVKTYPER) {
          return UntypedLongOrDoubleConstant.v(literal);
        } else {
          return LongConstant.v(literal);
        }
      default:
        throw new IllegalArgumentException("Expected a const or a const-wide instruction, got neither.");
    }
  }

  @Override
  boolean overridesRegister(int register) {
    OneRegisterInstruction i = (OneRegisterInstruction) instruction;
    int dest = i.getRegisterA();
    return register == dest;
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy