parent
2e9dfd2437
commit
a62cc3f709
@ -0,0 +1,94 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2000-2017 JetBrains s.r.o. |
||||||
|
* |
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
* you may not use this file except in compliance with the License. |
||||||
|
* You may obtain a copy of the License at |
||||||
|
* |
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
* |
||||||
|
* Unless required by applicable law or agreed to in writing, software |
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
* See the License for the specific language governing permissions and |
||||||
|
* limitations under the License. |
||||||
|
*/ |
||||||
|
package org.jetbrains.java.decompiler.modules.decompiler; |
||||||
|
|
||||||
|
import org.jetbrains.java.decompiler.code.CodeConstants; |
||||||
|
import org.jetbrains.java.decompiler.main.ClassesProcessor; |
||||||
|
import org.jetbrains.java.decompiler.main.DecompilerContext; |
||||||
|
import org.jetbrains.java.decompiler.main.extern.IFernflowerLogger; |
||||||
|
import org.jetbrains.java.decompiler.main.rels.MethodWrapper; |
||||||
|
import org.jetbrains.java.decompiler.modules.decompiler.exps.*; |
||||||
|
import org.jetbrains.java.decompiler.modules.decompiler.sforms.DirectGraph; |
||||||
|
import org.jetbrains.java.decompiler.modules.decompiler.stats.SwitchStatement; |
||||||
|
|
||||||
|
import java.util.ArrayList; |
||||||
|
import java.util.HashMap; |
||||||
|
import java.util.List; |
||||||
|
import java.util.Map; |
||||||
|
|
||||||
|
public class SwitchHelper { |
||||||
|
public static void simplify(SwitchStatement switchStatement) { |
||||||
|
SwitchExprent switchExprent = (SwitchExprent)switchStatement.getHeadexprent(); |
||||||
|
Exprent value = switchExprent.getValue(); |
||||||
|
if (isEnumArray(value)) { |
||||||
|
List<List<Exprent>> caseValues = switchStatement.getCaseValues(); |
||||||
|
Map<Exprent, Exprent> mapping = new HashMap<>(caseValues.size()); |
||||||
|
ArrayExprent array = (ArrayExprent)value; |
||||||
|
ClassesProcessor.ClassNode classNode = |
||||||
|
DecompilerContext.getClassProcessor().getMapRootClasses().get(((FieldExprent)array.getArray()).getClassname()); |
||||||
|
if (classNode != null) { |
||||||
|
MethodWrapper wrapper = classNode.getWrapper().getMethodWrapper(CodeConstants.CLINIT_NAME, "()V"); |
||||||
|
if (wrapper != null) { |
||||||
|
wrapper.getOrBuildGraph().iterateExprents(new DirectGraph.ExprentIterator() { |
||||||
|
@Override |
||||||
|
public int processExprent(Exprent exprent) { |
||||||
|
if (exprent instanceof AssignmentExprent) { |
||||||
|
AssignmentExprent assignment = (AssignmentExprent)exprent; |
||||||
|
Exprent left = assignment.getLeft(); |
||||||
|
if (isEnumArray(left)) { |
||||||
|
mapping.put(assignment.getRight(), ((InvocationExprent)((ArrayExprent)left).getIndex()).getInstance()); |
||||||
|
} |
||||||
|
} |
||||||
|
return 0; |
||||||
|
} |
||||||
|
}); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
List<List<Exprent>> realCaseValues = new ArrayList<>(caseValues.size()); |
||||||
|
for (List<Exprent> caseValue : caseValues) { |
||||||
|
List<Exprent> values = new ArrayList<>(caseValue.size()); |
||||||
|
realCaseValues.add(values); |
||||||
|
for (Exprent exprent : caseValue) { |
||||||
|
if (exprent == null) { |
||||||
|
values.add(null); |
||||||
|
} |
||||||
|
else { |
||||||
|
Exprent realConst = mapping.get(exprent); |
||||||
|
if (realConst == null) { |
||||||
|
DecompilerContext.getLogger() |
||||||
|
.writeMessage("Unable to simplify switch on enum: " + exprent + " not found, available: " + mapping, |
||||||
|
IFernflowerLogger.Severity.ERROR); |
||||||
|
return; |
||||||
|
} |
||||||
|
values.add(realConst.copy()); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
caseValues.clear(); |
||||||
|
caseValues.addAll(realCaseValues); |
||||||
|
switchExprent.replaceExprent(value, ((InvocationExprent)array.getIndex()).getInstance().copy()); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private static boolean isEnumArray(Exprent exprent) { |
||||||
|
if (exprent instanceof ArrayExprent) { |
||||||
|
Exprent field = ((ArrayExprent)exprent).getArray(); |
||||||
|
return field instanceof FieldExprent && ((FieldExprent)field).getName().startsWith("$SwitchMap"); |
||||||
|
} |
||||||
|
return false; |
||||||
|
} |
||||||
|
} |
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -0,0 +1,13 @@ |
|||||||
|
package pkg; |
||||||
|
|
||||||
|
public class TestSwitchOnStrings { |
||||||
|
public int testSOE(String s) { |
||||||
|
switch (s) { |
||||||
|
case "xxx": |
||||||
|
return 2; |
||||||
|
case "yyy": |
||||||
|
return 1; |
||||||
|
} |
||||||
|
return 0; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,13 @@ |
|||||||
|
package pkg; |
||||||
|
|
||||||
|
public class TestSwitchOnStrings { |
||||||
|
public int testSOE(String s) { |
||||||
|
switch (s) { |
||||||
|
case "xxx": |
||||||
|
return 2; |
||||||
|
case "yyy": |
||||||
|
return 1; |
||||||
|
} |
||||||
|
return 0; |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue