Review request for JDK-8013357: Javac accepts erroneous binary comparison operations
Maurizio Cimadamore
maurizio.cimadamore at oracle.com
Thu May 30 02:03:48 PDT 2013
On 29/05/13 20:53, Eric McCorkle wrote:
> Hello,
>
> Please review my latest patch for this problem. Since my earlier review
> request, several things have happened:
>
> 1) A CCC has been submitted, reviewed (at length) and approved.
>
> 2) I've confirmed the correct behavior wrt object-primitive comparisons.
>
> 3) I've fixed some errors in our tests, which would not compile when
> javac enforces the correct type rules.
>
> 4) This change has been flagged for a release note.
>
> The webrev is here:
> http://cr.openjdk.java.net/~emc/8013357/webrev.01/
Hi,
couple of comments:
*) I would be wary of adding Number to the symtab which will load
symbols eagerly; j.l.Number is not available on all platforms, so there
will be issues when using javac in cross compilation environment,
compiling against a J2ME rt.jar; the usual trick is to call Symbtab.
synthesizeEmptyInterfaceIfMissing (see for other code in Symtab that
does that for a reference).
*) The logic for checking equality seems questionable/overly complex
(expanded in meta code below):
boolean isEqualityComparable (Type s, Type t)
if (<both primitives>) {
return s <: t //1
} else if (<both references>) {
return s castable to t //2
} else if (source >= 1.5) {
if (<t is primitive>) {
return (<s is j.l.Number>) ? <t.isNumeric() : (unbox(s) <: t) //3
} else {
return (<t is j.l.Number>) ? <s.isNumeric() : (unbox(t) <: s) //4
}
}
The first thing that strikes me is that in places (1, 2, 3, 4) we are
not being symmetric; i.e.
isEqualityComparable(int, long) = true
isEqualityComparable(long, int) = false
The second thing that worries me is that the spec says something like:
*) if both are numeric (i.e. primitives) and one is convertible to
numeric (i.e. unboxable), then a numeric comparison is performed
I think this means that the code should do some check to see whether
numeric comparison is applicable, (i.e. eventually unboxing) and then
calling appropriate routine to do the comparison test.
Also, restating a point raised by Alex, Number == primitive is not
allowed as Number is not convertible to a numeric type; I think the test
should be something like:
boolean isEqualityComparable(Type s, Type t)
if (<both primitives>) {
if (source >= 1.5) {
s = types.unboxedTypeOrType(s);
t = types.unboxedTypeOrType(t);
}
return s <: t || t <: s;
} else if (<both references>) {
return (s castable to t) || (t castable to s);
}
Maurizio
-------------- next part --------------
An HTML attachment was scrubbed...
URL: http://mail.openjdk.java.net/pipermail/compiler-dev/attachments/20130530/c8bc4660/attachment.html
More information about the compiler-dev
mailing list