/* * 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.exps; import org.jetbrains.java.decompiler.code.CodeConstants; import org.jetbrains.java.decompiler.main.ClassesProcessor.ClassNode; import org.jetbrains.java.decompiler.main.DecompilerContext; import org.jetbrains.java.decompiler.main.TextBuffer; import org.jetbrains.java.decompiler.main.collectors.BytecodeMappingTracer; import org.jetbrains.java.decompiler.modules.decompiler.ExprProcessor; import org.jetbrains.java.decompiler.modules.decompiler.vars.CheckTypesResult; import org.jetbrains.java.decompiler.struct.StructField; import org.jetbrains.java.decompiler.struct.gen.VarType; import org.jetbrains.java.decompiler.util.InterpreterUtil; import java.util.ArrayList; import java.util.List; import java.util.Set; public class AssignmentExprent extends Exprent { public static final int CONDITION_NONE = -1; private static final String[] OPERATORS = { " += ", // FUNCTION_ADD " -= ", // FUNCTION_SUB " *= ", // FUNCTION_MUL " /= ", // FUNCTION_DIV " &= ", // FUNCTION_AND " |= ", // FUNCTION_OR " ^= ", // FUNCTION_XOR " %= ", // FUNCTION_REM " <<= ", // FUNCTION_SHL " >>= ", // FUNCTION_SHR " >>>= " // FUNCTION_USHR }; private Exprent left; private Exprent right; private int condType = CONDITION_NONE; public AssignmentExprent(Exprent left, Exprent right, Set<Integer> bytecodeOffsets) { super(EXPRENT_ASSIGNMENT); this.left = left; this.right = right; addBytecodeOffsets(bytecodeOffsets); } @Override public VarType getExprType() { return left.getExprType(); } @Override public CheckTypesResult checkExprTypeBounds() { CheckTypesResult result = new CheckTypesResult(); VarType typeLeft = left.getExprType(); VarType typeRight = right.getExprType(); if (typeLeft.typeFamily > typeRight.typeFamily) { result.addMinTypeExprent(right, VarType.getMinTypeInFamily(typeLeft.typeFamily)); } else if (typeLeft.typeFamily < typeRight.typeFamily) { result.addMinTypeExprent(left, typeRight); } else { result.addMinTypeExprent(left, VarType.getCommonSupertype(typeLeft, typeRight)); } return result; } @Override public List<Exprent> getAllExprents() { List<Exprent> lst = new ArrayList<>(); lst.add(left); lst.add(right); return lst; } @Override public Exprent copy() { return new AssignmentExprent(left.copy(), right.copy(), bytecode); } @Override public int getPrecedence() { return 13; } @Override public TextBuffer toJava(int indent, BytecodeMappingTracer tracer) { VarType leftType = left.getExprType(); VarType rightType = right.getExprType(); boolean fieldInClassInit = false, hiddenField = false; if (left.type == Exprent.EXPRENT_FIELD) { // first assignment to a final field. Field name without "this" in front of it FieldExprent field = (FieldExprent)left; ClassNode node = ((ClassNode)DecompilerContext.getProperty(DecompilerContext.CURRENT_CLASS_NODE)); if (node != null) { StructField fd = node.classStruct.getField(field.getName(), field.getDescriptor().descriptorString); if (fd != null) { if (field.isStatic() && fd.hasModifier(CodeConstants.ACC_FINAL)) { fieldInClassInit = true; } if (node.getWrapper() != null && node.getWrapper().getHiddenMembers().contains(InterpreterUtil.makeUniqueKey(fd.getName(), fd.getDescriptor()))) { hiddenField = true; } } } } if (hiddenField) { return new TextBuffer(); } TextBuffer buffer = new TextBuffer(); if (fieldInClassInit) { buffer.append(((FieldExprent)left).getName()); } else { buffer.append(left.toJava(indent, tracer)); } if (right.type == EXPRENT_CONST) { ((ConstExprent) right).adjustConstType(leftType); } TextBuffer res = right.toJava(indent, tracer); if (condType == CONDITION_NONE && !leftType.isSuperset(rightType) && (rightType.equals(VarType.VARTYPE_OBJECT) || leftType.type != CodeConstants.TYPE_OBJECT)) { if (right.getPrecedence() >= FunctionExprent.getPrecedence(FunctionExprent.FUNCTION_CAST)) { res.enclose("(", ")"); } res.prepend("(" + ExprProcessor.getCastTypeName(leftType) + ")"); } buffer.append(condType == CONDITION_NONE ? " = " : OPERATORS[condType]).append(res); tracer.addMapping(bytecode); return buffer; } @Override public void replaceExprent(Exprent oldExpr, Exprent newExpr) { if (oldExpr == left) { left = newExpr; } if (oldExpr == right) { right = newExpr; } } @Override public boolean equals(Object o) { if (o == this) return true; if (o == null || !(o instanceof AssignmentExprent)) return false; AssignmentExprent as = (AssignmentExprent)o; return InterpreterUtil.equalObjects(left, as.getLeft()) && InterpreterUtil.equalObjects(right, as.getRight()) && condType == as.getCondType(); } // ***************************************************************************** // getter and setter methods // ***************************************************************************** public Exprent getLeft() { return left; } public void setLeft(Exprent left) { this.left = left; } public Exprent getRight() { return right; } public void setRight(Exprent right) { this.right = right; } public int getCondType() { return condType; } public void setCondType(int condType) { this.condType = condType; } }