Fork of the Fernflower decompiler
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
fernflower/src/org/jetbrains/java/decompiler/modules/decompiler/ConcatenationHelper.java

250 lines
7.9 KiB

// Copyright 2000-2020 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license that can be found in the LICENSE file.
package org.jetbrains.java.decompiler.modules.decompiler;
import org.jetbrains.java.decompiler.code.CodeConstants;
import org.jetbrains.java.decompiler.modules.decompiler.exps.*;
import org.jetbrains.java.decompiler.struct.consts.PooledConstant;
import org.jetbrains.java.decompiler.struct.consts.PrimitiveConstant;
import org.jetbrains.java.decompiler.struct.gen.MethodDescriptor;
import org.jetbrains.java.decompiler.struct.gen.VarType;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Set;
public final class ConcatenationHelper {
private static final String builderClass = "java/lang/StringBuilder";
private static final String bufferClass = "java/lang/StringBuffer";
private static final String stringClass = "java/lang/String";
private static final VarType builderType = new VarType(CodeConstants.TYPE_OBJECT, 0, "java/lang/StringBuilder");
private static final VarType bufferType = new VarType(CodeConstants.TYPE_OBJECT, 0, "java/lang/StringBuffer");
public static Exprent contractStringConcat(Exprent expr) {
Exprent exprTmp = null;
VarType cltype = null;
// first quick test
if (expr.type == Exprent.EXPRENT_INVOCATION) {
InvocationExprent iex = (InvocationExprent)expr;
if ("toString".equals(iex.getName())) {
if (builderClass.equals(iex.getClassname())) {
cltype = builderType;
}
else if (bufferClass.equals(iex.getClassname())) {
cltype = bufferType;
}
if (cltype != null) {
exprTmp = iex.getInstance();
}
}
else if ("makeConcatWithConstants".equals(iex.getName())) { // java 9 style
List<Exprent> parameters = extractParameters(iex.getBootstrapArguments(), iex);
if (parameters.size() >= 2) {
return createConcatExprent(parameters, expr.bytecode);
}
}
}
if (exprTmp == null) {
return expr;
}
// iterate in depth, collecting possible operands
List<Exprent> lstOperands = new ArrayList<>();
while (true) {
int found = 0;
switch (exprTmp.type) {
case Exprent.EXPRENT_INVOCATION:
InvocationExprent iex = (InvocationExprent)exprTmp;
if (isAppendConcat(iex, cltype)) {
lstOperands.add(0, iex.getLstParameters().get(0));
exprTmp = iex.getInstance();
found = 1;
}
break;
case Exprent.EXPRENT_NEW:
NewExprent nex = (NewExprent)exprTmp;
if (isNewConcat(nex, cltype)) {
VarType[] params = nex.getConstructor().getDescriptor().params;
if (params.length == 1) {
lstOperands.add(0, nex.getConstructor().getLstParameters().get(0));
}
found = 2;
}
}
if (found == 0) {
return expr;
}
else if (found == 2) {
break;
}
}
int first2str = 0;
int index = 0;
while (index < lstOperands.size() && index < 2) {
if (lstOperands.get(index).getExprType().equals(VarType.VARTYPE_STRING)) {
first2str |= (index + 1);
}
index++;
}
if (first2str == 0) {
lstOperands.add(0, new ConstExprent(VarType.VARTYPE_STRING, "", expr.bytecode));
}
// remove redundant String.valueOf
for (int i = 0; i < lstOperands.size(); i++) {
Exprent rep = removeStringValueOf(lstOperands.get(i));
boolean ok = (i > 1);
if (!ok) {
boolean isstr = rep.getExprType().equals(VarType.VARTYPE_STRING);
ok = isstr || first2str != i + 1;
if (i == 0) {
first2str &= 2;
}
}
if (ok) {
lstOperands.set(i, rep);
}
}
return createConcatExprent(lstOperands, expr.bytecode);
}
private static Exprent createConcatExprent(List<Exprent> lstOperands, Set<Integer> bytecode) {
// build exprent to return
Exprent func = lstOperands.get(0);
for (int i = 1; i < lstOperands.size(); i++) {
func = new FunctionExprent(FunctionExprent.FUNCTION_STR_CONCAT, Arrays.asList(func, lstOperands.get(i)), bytecode);
}
return func;
}
// See StringConcatFactory in jdk sources
private static final char TAG_ARG = '\u0001';
private static final char TAG_CONST = '\u0002';
private static List<Exprent> extractParameters(List<PooledConstant> bootstrapArguments, InvocationExprent expr) {
List<Exprent> parameters = expr.getLstParameters();
if (bootstrapArguments != null) {
PooledConstant constant = bootstrapArguments.get(0);
if (constant.type == CodeConstants.CONSTANT_String) {
String recipe = ((PrimitiveConstant)constant).getString();
List<Exprent> res = new ArrayList<>();
StringBuilder acc = new StringBuilder();
int parameterId = 0;
for (int i = 0; i < recipe.length(); i++) {
char c = recipe.charAt(i);
if (c == TAG_CONST || c == TAG_ARG) {
// Detected a special tag, flush all accumulated characters
// as a constant first:
if (acc.length() > 0) {
res.add(new ConstExprent(VarType.VARTYPE_STRING, acc.toString(), expr.bytecode));
acc.setLength(0);
}
if (c == TAG_CONST) {
// skip for now
}
if (c == TAG_ARG) {
res.add(parameters.get(parameterId++));
}
}
else {
// Not a special characters, this is a constant embedded into
// the recipe itself.
acc.append(c);
}
}
// Flush the remaining characters as constant:
if (acc.length() > 0) {
res.add(new ConstExprent(VarType.VARTYPE_STRING, acc.toString(), expr.bytecode));
}
return res;
}
}
return new ArrayList<>(parameters);
}
private static boolean isAppendConcat(InvocationExprent expr, VarType cltype) {
if ("append".equals(expr.getName())) {
MethodDescriptor md = expr.getDescriptor();
if (md.ret.equals(cltype) && md.params.length == 1) {
VarType param = md.params[0];
switch (param.type) {
case CodeConstants.TYPE_OBJECT:
if (!param.equals(VarType.VARTYPE_STRING) &&
!param.equals(VarType.VARTYPE_OBJECT)) {
break;
}
case CodeConstants.TYPE_BOOLEAN:
case CodeConstants.TYPE_CHAR:
case CodeConstants.TYPE_DOUBLE:
case CodeConstants.TYPE_FLOAT:
case CodeConstants.TYPE_INT:
case CodeConstants.TYPE_LONG:
return true;
default:
}
}
}
return false;
}
private static boolean isNewConcat(NewExprent expr, VarType cltype) {
if (expr.getNewType().equals(cltype)) {
VarType[] params = expr.getConstructor().getDescriptor().params;
return params.length == 0 || params.length == 1 && params[0].equals(VarType.VARTYPE_STRING);
}
return false;
}
private static Exprent removeStringValueOf(Exprent exprent) {
if (exprent.type == Exprent.EXPRENT_INVOCATION) {
InvocationExprent iex = (InvocationExprent)exprent;
if ("valueOf".equals(iex.getName()) && stringClass.equals(iex.getClassname())) {
MethodDescriptor md = iex.getDescriptor();
if (md.params.length == 1) {
VarType param = md.params[0];
switch (param.type) {
case CodeConstants.TYPE_OBJECT:
if (!param.equals(VarType.VARTYPE_OBJECT)) {
break;
}
case CodeConstants.TYPE_BOOLEAN:
case CodeConstants.TYPE_CHAR:
case CodeConstants.TYPE_DOUBLE:
case CodeConstants.TYPE_FLOAT:
case CodeConstants.TYPE_INT:
case CodeConstants.TYPE_LONG:
return iex.getLstParameters().get(0);
}
}
}
}
return exprent;
}
}