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

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

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 java.util.ArrayList;
import java.util.List;

import org.jf.dexlib2.iface.instruction.Instruction;
import org.jf.dexlib2.iface.instruction.SwitchElement;
import org.jf.dexlib2.iface.instruction.formats.SparseSwitchPayload;

import soot.IntType;
import soot.Local;
import soot.Unit;
import soot.dexpler.DexBody;
import soot.dexpler.IDalvikTyper;
import soot.dexpler.typing.DalvikTyper;
import soot.jimple.IntConstant;
import soot.jimple.Jimple;
import soot.jimple.LookupSwitchStmt;
import soot.jimple.Stmt;

public class SparseSwitchInstruction extends SwitchInstruction {

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

  @Override
  protected Stmt switchStatement(DexBody body, Instruction targetData, Local key) {
    SparseSwitchPayload i = (SparseSwitchPayload) targetData;
    List seList = i.getSwitchElements();

    // the default target always follows the switch statement
    int defaultTargetAddress = codeAddress + instruction.getCodeUnits();
    Unit defaultTarget = body.instructionAtAddress(defaultTargetAddress).getUnit();

    List lookupValues = new ArrayList();
    List targets = new ArrayList();
    for (SwitchElement se : seList) {
      lookupValues.add(IntConstant.v(se.getKey()));
      int offset = se.getOffset();
      targets.add(body.instructionAtAddress(codeAddress + offset).getUnit());
    }
    LookupSwitchStmt switchStmt = Jimple.v().newLookupSwitchStmt(key, lookupValues, targets, defaultTarget);
    setUnit(switchStmt);
    addTags(switchStmt);

    if (IDalvikTyper.ENABLE_DVKTYPER) {
      DalvikTyper.v().setType(switchStmt.getKeyBox(), IntType.v(), true);
    }

    return switchStmt;
  }

  @Override
  public void computeDataOffsets(DexBody body) {
    // System.out.println("class of instruction: "+ instruction.getClass());
    // int offset = ((OffsetInstruction) instruction).getCodeOffset();
    // int targetAddress = codeAddress + offset;
    // Instruction targetData = body.instructionAtAddress(targetAddress).instruction;
    // SparseSwitchPayload ssInst = (SparseSwitchPayload) targetData;
    // List targetAddresses = ssInst.getSwitchElements();
    // int size = targetAddresses.size() * 2; // @ are on 32bits
    //
    // // From org.jf.dexlib.Code.Format.SparseSwitchDataPseudoInstruction we learn
    // // that there are 2 bytes after the magic number that we have to jump.
    // // 2 bytes to jump = address + 1
    // //
    // // out.writeByte(0x00); // magic
    // // out.writeByte(0x02); // number
    // // out.writeShort(targets.length); // 2 bytes
    // // out.writeInt(firstKey);
    //
    // setDataFirstByte (targetAddress + 1);
    // setDataLastByte (targetAddress + 1 + size - 1);
    // setDataSize (size);
    //
    // ByteArrayAnnotatedOutput out = new ByteArrayAnnotatedOutput();
    // ssInst.write(out, targetAddress);
    //
    // byte[] outa = out.getArray();
    // byte[] data = new byte[outa.length-2];
    // for (int i=2; i




© 2015 - 2024 Weber Informatics LLC | Privacy Policy