From e44ba9905e410be06b69152fff5950094c6651bf Mon Sep 17 00:00:00 2001 From: "Egor.Ushakov" Date: Wed, 26 Apr 2017 21:11:10 +0300 Subject: [PATCH] IDEA-127533 int field is displayed as char --- .../java/decompiler/main/ClassWriter.java | 5 + .../modules/decompiler/ExprProcessor.java | 4 + .../decompiler/exps/AssignmentExprent.java | 6 +- .../modules/decompiler/exps/ConstExprent.java | 15 + .../decompiler/exps/FunctionExprent.java | 13 + .../java/decompiler/SingleClassesTest.java | 1 + testData/classes/pkg/TestConstType.class | Bin 0 -> 2120 bytes testData/results/TestClassLoop.dec | 4 +- testData/results/TestConstType.dec | 337 ++++++++++++++++++ testData/src/pkg/TestConstType.java | 106 ++++++ 10 files changed, 488 insertions(+), 3 deletions(-) create mode 100644 testData/classes/pkg/TestConstType.class create mode 100644 testData/results/TestConstType.dec create mode 100644 testData/src/pkg/TestConstType.java diff --git a/src/org/jetbrains/java/decompiler/main/ClassWriter.java b/src/org/jetbrains/java/decompiler/main/ClassWriter.java index 5646407..59b744c 100644 --- a/src/org/jetbrains/java/decompiler/main/ClassWriter.java +++ b/src/org/jetbrains/java/decompiler/main/ClassWriter.java @@ -442,6 +442,11 @@ public class ClassWriter { } else { buffer.append(" = "); + + if (initializer.type == Exprent.EXPRENT_CONST) { + ((ConstExprent) initializer).adjustConstType(fieldType); + } + // FIXME: special case field initializer. Can map to more than one method (constructor) and bytecode intruction. buffer.append(initializer.toJava(indent, tracer)); } diff --git a/src/org/jetbrains/java/decompiler/modules/decompiler/ExprProcessor.java b/src/org/jetbrains/java/decompiler/modules/decompiler/ExprProcessor.java index 04e3c26..5d41ff5 100644 --- a/src/org/jetbrains/java/decompiler/modules/decompiler/ExprProcessor.java +++ b/src/org/jetbrains/java/decompiler/modules/decompiler/ExprProcessor.java @@ -885,6 +885,10 @@ public class ExprProcessor implements CodeConstants { if (quote) buffer.append('('); + if (exprent.type == Exprent.EXPRENT_CONST) { + ((ConstExprent) exprent).adjustConstType(leftType); + } + buffer.append(exprent.toJava(indent, tracer)); if (quote) buffer.append(')'); diff --git a/src/org/jetbrains/java/decompiler/modules/decompiler/exps/AssignmentExprent.java b/src/org/jetbrains/java/decompiler/modules/decompiler/exps/AssignmentExprent.java index 957f3f5..4374ff8 100644 --- a/src/org/jetbrains/java/decompiler/modules/decompiler/exps/AssignmentExprent.java +++ b/src/org/jetbrains/java/decompiler/modules/decompiler/exps/AssignmentExprent.java @@ -1,5 +1,5 @@ /* - * Copyright 2000-2014 JetBrains s.r.o. + * 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. @@ -138,6 +138,10 @@ public class AssignmentExprent extends Exprent { 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 && diff --git a/src/org/jetbrains/java/decompiler/modules/decompiler/exps/ConstExprent.java b/src/org/jetbrains/java/decompiler/modules/decompiler/exps/ConstExprent.java index f649d80..9e89f6c 100644 --- a/src/org/jetbrains/java/decompiler/modules/decompiler/exps/ConstExprent.java +++ b/src/org/jetbrains/java/decompiler/modules/decompiler/exps/ConstExprent.java @@ -367,6 +367,21 @@ public class ConstExprent extends Exprent { this.constType = constType; } + public void adjustConstType(VarType expectedType) { + // BYTECHAR and SHORTCHAR => CHAR in the CHAR context + if (expectedType.equals(VarType.VARTYPE_CHAR) && + (constType.equals(VarType.VARTYPE_BYTECHAR) || constType.equals(VarType.VARTYPE_SHORTCHAR))) { + if (getIntValue() != 0) { + setConstType(VarType.VARTYPE_CHAR); + } + } + // CHAR => INT in the INT context + else if (expectedType.equals(VarType.VARTYPE_INT) && + constType.equals(VarType.VARTYPE_CHAR)) { + setConstType(VarType.VARTYPE_INT); + } + } + public Object getValue() { return value; } diff --git a/src/org/jetbrains/java/decompiler/modules/decompiler/exps/FunctionExprent.java b/src/org/jetbrains/java/decompiler/modules/decompiler/exps/FunctionExprent.java index 6662965..0cc0025 100644 --- a/src/org/jetbrains/java/decompiler/modules/decompiler/exps/FunctionExprent.java +++ b/src/org/jetbrains/java/decompiler/modules/decompiler/exps/FunctionExprent.java @@ -459,7 +459,20 @@ public class FunctionExprent extends Exprent { .append(wrapOperandString(lstOperands.get(1), true, indent, tracer)); } + // try to determine more accurate type for 'char' literals if (funcType >= FUNCTION_EQ) { + if (funcType <= FUNCTION_LE) { + Exprent left = lstOperands.get(0); + Exprent right = lstOperands.get(1); + + if (right.type == EXPRENT_CONST) { + ((ConstExprent) right).adjustConstType(left.getExprType()); + } + else if (left.type == EXPRENT_CONST) { + ((ConstExprent) left).adjustConstType(right.getExprType()); + } + } + return wrapOperandString(lstOperands.get(0), false, indent, tracer) .append(OPERATORS[funcType - FUNCTION_EQ + 11]) .append(wrapOperandString(lstOperands.get(1), true, indent, tracer)); diff --git a/test/org/jetbrains/java/decompiler/SingleClassesTest.java b/test/org/jetbrains/java/decompiler/SingleClassesTest.java index 3a213c4..51521b4 100644 --- a/test/org/jetbrains/java/decompiler/SingleClassesTest.java +++ b/test/org/jetbrains/java/decompiler/SingleClassesTest.java @@ -113,6 +113,7 @@ public class SingleClassesTest { @Test public void testLambdaParams() { doTest("pkg/TestLambdaParams"); } @Test public void testInterfaceMethods() { doTest("pkg/TestInterfaceMethods"); } //@Test public void testUnionType() { doTest("pkg/TestUnionType"); } //TODO: fix + @Test public void testConstType() { doTest("pkg/TestConstType"); } private void doTest(String testFile, String... companionFiles) { ConsoleDecompiler decompiler = fixture.getDecompiler(); diff --git a/testData/classes/pkg/TestConstType.class b/testData/classes/pkg/TestConstType.class new file mode 100644 index 0000000000000000000000000000000000000000..57d4fec11e4920937a29e6a9143552ed071e4635 GIT binary patch literal 2120 zcmZuyO-vkR7=DIjciCm2vrD0+P(j3&p9*McZTV?|1=S4@8Q<~ z+VN@-^@s%U9Qp+Yf`q{!F5$93G=Nty6vQw_1g?01J6{mK8}mT>1ML02DbOnr7q}&` zBXCdPGl8!>Fd&=15ynYYrD=6r%P;o(%w zOiY>SxQx|gp}iE(Xeb+AU0!URv@&)imCV?ao2yo*20dfha*eWC*bor|1t}gVnXt{+ z@`$;*e~ZLLt5)Edv0yPC#Rtz)7i@3zbprnt;;;qhr&a>PzBcOvf;p5?1i&zGiv57lsb z6ol-DCL$4=IO^lNMmCy0ht^~SAE2Pg^8lXv&WUDXsgyQbA)g?;IK`3fp1})f;Ybse z)qQU{7J*Y$_qYPbMge(^t%~h7TYZyJLOXX6a1Dwr55!gl?+1!dO-pqsrst)c4>>j~ zcr))+Hl_C>Nop4Zbv0`0uiQ-blhZGx%LyE(S z%M>40T&}o6@e##G6;~=grnpLRwc_K7pCKP)VI9XQQw?)n%kOO+j<8;9Sg|LW^ET%7 zJbS&Y(6C$f%lpE5BZDuh}pYOvA?9;-%{fr zc+*elz|ZK!FX+n4_@i9L&vV^<8y+$flJRj*)eiJ3V`~RKV$BO=Hu5JXvf&!%K)6=cBUvMw%~eYa%Nn>y>|t^D;TR&6Vu0ZdGOWvVG0N2;KKzAY zG_RV*bJaA?V7gQ_`Qa@sHH=aYyK%vdLn|iBTRW@X!U^`}&FW+8V2iMgu*v)&+YH-j zwpq3whSz-;HO3aPGF(F}W27WKTubzY>$ 8 +10 <-> 10 +11 <-> 11 +14 <-> 13 +17 <-> 16 +20 <-> 19 +21 <-> 20 +24 <-> 23 +25 <-> 24 +27 <-> 27 +31 <-> 31 +32 <-> 32 +34 <-> 33 +35 <-> 34 +36 <-> 35 +37 <-> 36 +38 <-> 37 +39 <-> 38 +40 <-> 39 +41 <-> 40 +42 <-> 41 +43 <-> 42 +44 <-> 43 +45 <-> 44 +46 <-> 45 +47 <-> 46 +48 <-> 47 +49 <-> 48 +50 <-> 49 +51 <-> 50 +52 <-> 51 +53 <-> 52 +54 <-> 53 +55 <-> 54 +56 <-> 55 +57 <-> 56 +58 <-> 57 +59 <-> 58 +60 <-> 59 +63 <-> 62 +64 <-> 63 +66 <-> 64 +67 <-> 65 +69 <-> 66 +70 <-> 67 +72 <-> 68 +73 <-> 69 +75 <-> 71 +76 <-> 71 +79 <-> 71 +85 <-> 77 +87 <-> 88 +88 <-> 89 +90 <-> 79 +91 <-> 80 +93 <-> 91 +94 <-> 92 +96 <-> 85 +97 <-> 86 +99 <-> 82 +100 <-> 83 +102 <-> 94 +104 <-> 97 diff --git a/testData/src/pkg/TestConstType.java b/testData/src/pkg/TestConstType.java new file mode 100644 index 0000000..d4c0192 --- /dev/null +++ b/testData/src/pkg/TestConstType.java @@ -0,0 +1,106 @@ +package pkg; + +public class TestConstType { + private char lineBreak = '\n'; + private char zero = 0; + + public void setLineBreak(char os) { + switch (os) { + case 'u': + lineBreak = '\r'; + break; + + case 'w': + lineBreak = '\n'; + break; + } + } + + public void init() { + setLineBreak('w'); + } + + public String convertIndentation(String text) { + if (text.charAt(0) == '\t') { + text = text.replace('\t', ' '); + } + return text; + } + + public void printalot() { + System.out.println('a'); + System.out.println('\t'); + + System.out.println(0); + System.out.println(65); + System.out.println(120); + System.out.println(32760); + System.out.println(32761); + System.out.println(35000); + System.out.println(50000); + System.out.println(128000); + System.out.println(60793); + System.out.println(60737); + System.out.println(60777); + System.out.println(60785); + System.out.println(60835); + System.out.println(60843); + System.out.println(60851); + System.out.println(60859); + System.out.println(1048576); + System.out.println(49152); + System.out.println(44100); + System.out.println(44101); + System.out.println(44102); + System.out.println(44103); + System.out.println(60000); + System.out.println(64000); + System.out.println(65000); + System.out.println(45000); + } + + public char guessType(int val) { + if (0 <= val && val <= 127) { + return 'X'; + } + else if (-128 <= val && val <= 127) { + return 'B'; + } + else if (128 <= val && val <= 32767) { + return 'Y'; + } + else if (-32768 <= val && val <= 32767) { + return 'S'; + } + else if (32768 <= val && val <= 0xFFFF) { + return 'C'; + } + else { + return 'I'; + } + } + + public int getTypeMaxValue(char type) { + int maxValue; + switch (type) { + case 'X': + maxValue = 128; + break; + case 'B': + maxValue = 127; + break; + case 'Y': + maxValue = 32768; + break; + case 'S': + maxValue = 32767; + break; + case 'C': + maxValue = 0xFFFF; + break; + default: + maxValue = Integer.MAX_VALUE; + } + return maxValue; + } +}