JDK-8189659 : Anonymous subtype of raw type causes VerifyError
  • Type: Bug
  • Component: tools
  • Sub-Component: javac
  • Affected Version: 9
  • Priority: P3
  • Status: Resolved
  • Resolution: Fixed
  • OS: generic
  • CPU: x86_64
  • Submitted: 2017-10-18
  • Updated: 2018-11-27
  • Resolved: 2017-11-21
The Version table provides details related to the release that this issue/RFE will be addressed.

Unresolved : Release in which this issue/RFE will be addressed.
Resolved: Release in which this issue/RFE has been resolved.
Fixed : Release in which this issue/RFE has been fixed. The release containing this fix may be available for download as an Early Access Release or a General Availability Release.

To download the current JDK release, click here.
JDK 10
10 b33Fixed
Related Reports
Duplicate :  
Description
FULL PRODUCT VERSION :
java version "9"
Java(TM) SE Runtime Environment (build 9+178)
Java HotSpot(TM) 64-Bit Server VM (build 9+178, mixed mode)

ADDITIONAL OS VERSION INFORMATION :
Linux 4.9.0-3-amd64 #1 SMP Debian 4.9.30-2+deb9u3 (2017-08-06) x86_64 GNU/Linux


A DESCRIPTION OF THE PROBLEM :
If you have an anonym implementation of a generic class without using the diamond operators the verifier will not use the generic type of the class, but will try to verify using Object, which will fail because it cannot be assigned to the generic type of the class.

REGRESSION.  Last worked in version 8u151

STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
Run the provided test code on java 9.

EXPECTED VERSUS ACTUAL BEHAVIOR :
EXPECTED -
Successful execution
ACTUAL -
VerifyError

ERROR MESSAGES/STACK TRACES THAT OCCUR :
9
Exception in thread "main" java.lang.VerifyError: Bad type on operand stack
Exception Details:
  Location:
    verifiertest/GenericClass$1.compareTo(Ljava/lang/Object;)I @2: invokespecial
  Reason:
    Type 'java/lang/Object' (current frame, stack[1]) is not assignable to 'verifiertest/GenericClass'
  Current Frame:
    bci: @2
    flags: { }
    locals: { 'verifiertest/GenericClass$1', 'java/lang/Object' }
    stack: { 'verifiertest/GenericClass$1', 'java/lang/Object' }
  Bytecode:
    0000000: 2a2b b700 05ac                         

	at verifiertest.GenericClass.<clinit>(GenericClass.java:16)

REPRODUCIBILITY :
This bug can be reproduced always.

---------- BEGIN SOURCE ----------
public class GenericClass<_J extends GenericClass> implements Comparable<_J>  {
    static {
        System.err.println(System.getProperty("java.version"));
    }
    public static GenericClass EMPTY = new GenericClass() {
        
        void something() {
            System.out.println("This is something");
        }
    };
    public GenericClass() {
    }
    @Override
    public int compareTo(_J o) {
        return 0;
    }
    public static void main(String[] args) {
        GenericClass generic = GenericClass.EMPTY;
    }
}

---------- END SOURCE ----------

CUSTOMER SUBMITTED WORKAROUND :
One workaround: change the parameter type of compareTo to Object to trick the verifier
Other workaround: use a named class instead of the anonym implementation


Comments
This issue is reproducible in 9 and 10. This regression started from 9 ea b139. ==10 ea b26== Exception in thread "main" java.lang.VerifyError: Bad type on operand stack Exception Details: Location: GenericClass$1.compareTo(Ljava/lang/Object;)I @2: invokespecial Reason: Type 'java/lang/Object' (current frame, stack[1]) is not assignable to 'GenericClass' Current Frame: bci: @2 flags: { } locals: { 'GenericClass$1', 'java/lang/Object' } stack: { 'GenericClass$1', 'java/lang/Object' } Bytecode: 0000000: 2a2b b700 05ac at GenericClass.<clinit>(GenericClass.java:5) ==
19-10-2017