Smali to Dalvik OpCode

Helper script to translate parts of the smali code of an application to the corresponding Dalvik opcodes, leaving wildcards for the variable parts of the code (variables, parameters, etc.). The output is formatted to be easily transformed in a LuckyPatcher patch.

Dovrai installare un'estensione come Tampermonkey, Greasemonkey o Violentmonkey per installare questo script.

You will need to install an extension such as Tampermonkey to install this script.

Dovrai installare un'estensione come Tampermonkey o Violentmonkey per installare questo script.

Dovrai installare un'estensione come Tampermonkey o Userscripts per installare questo script.

Dovrai installare un'estensione come ad esempio Tampermonkey per installare questo script.

Dovrai installare un gestore di script utente per installare questo script.

(Ho già un gestore di script utente, lasciamelo installare!)

Dovrai installare un'estensione come ad esempio Stylus per installare questo stile.

Dovrai installare un'estensione come ad esempio Stylus per installare questo stile.

Dovrai installare un'estensione come ad esempio Stylus per installare questo stile.

Dovrai installare un'estensione per la gestione degli stili utente per installare questo stile.

Dovrai installare un'estensione per la gestione degli stili utente per installare questo stile.

Dovrai installare un'estensione per la gestione degli stili utente per installare questo stile.

(Ho già un gestore di stile utente, lasciamelo installare!)

// ==UserScript==
// @name        Smali to Dalvik OpCode
// @namespace   StephenP
// @match       http://pallergabor.uw.hu/androidblog/dalvik_opcodes.html*
// @match       https://quantiti.github.io/dalvik-opcodes/
// @grant       none
// @version     1.2.1
// @author      StephenP
// @contributionURL https://buymeacoffee.com/stephenp_greasyfork
// @description Helper script to translate parts of the smali code of an application to the corresponding Dalvik opcodes, leaving wildcards for the variable parts of the code (variables, parameters, etc.). The output is formatted to be easily transformed in a LuckyPatcher patch.
// ==/UserScript==
const table=document.getElementsByTagName("TABLE")[0]
const names=["nop","move","move/from16","move/16","move-wide","move-wide/from16","move-wide/16","move-object","move-object/from16","move-object/16","move-result","move-result-wide","move-result-object","move-exception","return-void","return","return-wide","return-object","const/4","const/16","const","const/high16","const-wide/16","const-wide/32","const-wide","const-wide/high16","const-string","const-string-jumbo","const-class","monitor-enter","monitor-exit","check-cast","instance-of","array-length","new-instance","new-array","filled-new-array","filled-new-array-range","fill-array-data","throw","goto","goto/16","goto/32","packed-switch","sparse-switch","cmpl-float","cmpg-float","cmpl-double","cmpg-double","cmp-long","if-eq","if-ne","if-lt","if-ge","if-gt","if-le","if-eqz","if-nez","if-ltz","if-gez","if-gtz","if-lez","unused_3E","unused_3F","unused_40","unused_41","unused_42","unused_43","aget","aget-wide","aget-object","aget-boolean","aget-byte","aget-char","aget-short","aput","aput-wide","aput-object","aput-boolean","aput-byte","aput-char","aput-short","iget","iget-wide","iget-object","iget-boolean","iget-byte","iget-char","iget-short","iput","iput-wide","iput-object","iput-boolean","iput-byte","iput-char","iput-short","sget","sget-wide","sget-object","sget-boolean","sget-byte","sget-char","sget-short","sput","sput-wide","sput-object","sput-boolean","sput-byte","sput-char","sput-short","invoke-virtual","invoke-super","invoke-direct","invoke-static","invoke-interface","unused_73","invoke-virtual/range","invoke-super/range","invoke-direct/range","invoke-static/range","invoke-interface-range","unused_79","unused_7A","neg-int","not-int","neg-long","not-long","neg-float","neg-double","int-to-long","int-to-float","int-to-double","long-to-int","long-to-float","long-to-double","float-to-int","float-to-long","float-to-double","double-to-int","double-to-long","double-to-float","int-to-byte","int-to-char","int-to-short","add-int","sub-int","mul-int","div-int","rem-int","and-int","or-int","xor-int","shl-int","shr-int","ushr-int","add-long","sub-long","mul-long","div-long","rem-long","and-long","or-long","xor-long","shl-long","shr-long","ushr-long","add-float","sub-float","mul-float","div-float","rem-float","add-double","sub-double","mul-double","div-double","rem-double","add-int/2addr","sub-int/2addr","mul-int/2addr","div-int/2addr","rem-int/2addr","and-int/2addr","or-int/2addr","xor-int/2addr","shl-int/2addr","shr-int/2addr","ushr-int/2addr","add-long/2addr","sub-long/2addr","mul-long/2addr","div-long/2addr","rem-long/2addr","and-long/2addr","or-long/2addr","xor-long/2addr","shl-long/2addr","shr-long/2addr","ushr-long/2addr","add-float/2addr","sub-float/2addr","mul-float/2addr","div-float/2addr","rem-float/2addr","add-double/2addr","sub-double/2addr","mul-double/2addr","div-double/2addr","rem-double/2addr","add-int/lit16","sub-int/lit16","mul-int/lit16","div-int/lit16","rem-int/lit16","and-int/lit16","or-int/lit16","xor-int/lit16","add-int/lit8","sub-int/lit8","mul-int/lit8","div-int/lit8","rem-int/lit8","and-int/lit8","or-int/lit8","xor-int/lit8","shl-int/lit8","shr-int/lit8","ushr-int/lit8","unused_E3","unused_E4","unused_E5","unused_E6","unused_E7","unused_E8","unused_E9","unused_EA","unused_EB","unused_EC","unused_ED","execute-inline","unused_EF","invoke-direct-empty","unused_F1","iget-quick","iget-wide-quick","iget-object-quick","iput-quick","iput-wide-quick","iput-object-quick","invoke-virtual-quick","invoke-virtual-quick/range","invoke-super-quick","invoke-super-quick/range","unused_FC","unused_FD","unused_FE","unused_FF"];
const cmdLength=[2,2,4,0,2,4,0,2,4,6,2,2,2,2,2,2,2,2,2,4,6,4,4,6,10,4,4,0,4,2,2,4,4,2,4,4,6,6,6,2,2,4,0,6,6,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,0,0,0,0,0,0,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,2,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,6,6,6,6,6,0,6,6,6,6,6,0,0,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,0,0,0,0,0,0,0,0,0,0,0,6,0,6,0,4,4,4,4,4,4,6,6,6,6,0,0,0,0];

var st=document.createElement("STYLE");
const styleStr=`
fieldset>button{
  margin-left: 1em;
  margin-right: 1em;
  padding-top: 0.5em;
  padding-bottom: 0.5em;
}
#container{
  flex-direction: row;
  border: 5px solid red;
}
textarea{
  width: 50%;
}
@media (max-width: 1220px) {
  #container {
    flex-direction: column;
    border: 5px solid blue;
  }
  textarea{
    width: 100%;
    font-size: 2em;
  }
`;
st.innerHTML=styleStr;
document.getElementsByTagName("HEAD")[0].appendChild(st);
var container=document.createElement("DIV");
container.style.display="flex";
container.style.flexDirection="row";
container.style.flexWrap="wrap";
table.parentNode.insertBefore(container,table)
var smali=document.createElement("TEXTAREA");
smali.style.minWidth="600px";
smali.style.height="300px";
smali.id="smali";
smali.placeholder="Paste here the original smali code.";
container.appendChild(smali);
var dalvik=document.createElement("TEXTAREA");
dalvik.style.minWidth="600px";
dalvik.style.height="300px";
dalvik.readOnly=true
dalvik.id="dalvik";
dalvik.placeholder="The dalvik bytecode will appear here.";
container.appendChild(dalvik);
var codeVersion=document.createElement("FIELDSET");
var original=document.createElement("INPUT");
original.id="isOriginal";
original.type="radio";
original.name="codeVersion";
original.checked=true;
var labelO=document.createElement("LABEL");
labelO.innerText="Original";
labelO.for="isOriginal";
var patched=document.createElement("INPUT");
patched.id="isPatched";
patched.type="radio";
patched.name="codeVersion";
patched.checked=false;
var labelP=document.createElement("LABEL");
labelP.innerText="Patched";
labelP.for="isPatched";
codeVersion.appendChild(original);
codeVersion.appendChild(labelO);
codeVersion.appendChild(patched);
codeVersion.appendChild(labelP);
var convert=document.createElement("BUTTON");
convert.innerText="CONVERT";
convert.addEventListener("click",smali2davik);
codeVersion.appendChild(convert);
table.parentNode.insertBefore(codeVersion,table);

function smali2davik(){
  var hasPreciseLength=true;
  var preciseLength=0;
  var output="";
  var lines=document.getElementById("smali").value.split("\n");
  var instructions=[];
  for(let i=0;i<lines.length;i++){
    let line=lines[i].trim().split(" ")[0];
    if(line.length>0){
      if((line[0]!==":")&&(line[0]!==".")){
        instructions.push(line);
      }
    }
  }
  console.log(instructions);
  for(let instr of instructions){
    let i=names.indexOf(instr);
    if(i>=0){
      let wholeInstr=" "+i.toString(16).padStart(2, "0");
      for(let j=0;j<cmdLength[i]-1;j++){
        wholeInstr+=" ??";
      }
      if(cmdLength[i]==0){
        wholeInstr+=" ■Unknown bytecode length for instruction \'"+instr+"\'■";
        hasPreciseLength=false;
      }
      console.log(wholeInstr);
      output+=wholeInstr;
    }
    else{
      output+=(" ■Instruction \'"+instr+"\' not found■");
      hasPreciseLength=false;
    }
  }
  if(hasPreciseLength){
    preciseLength=output.length/3;
  }
  var lineHeader="original";
  if(document.getElementById("isPatched").checked){
    lineHeader="replaced"
  }
  output="{\""+lineHeader+"\":\""+output.trim()+"\"}";
  if(hasPreciseLength){
    dalvik.value=output+"\n\nTotal length: "+preciseLength+" bytes.";
  }
  else{
    dalvik.value=output;
  }
}