Updated ARM Spec

Joe Darcy joe.darcy at oracle.com
Mon Aug 23 12:45:19 PDT 2010


Rémi Forax wrote:
> Le 23/08/2010 12:29, David Holmes a écrit :
>   
>> Rémi Forax said the following on 08/23/10 20:23:
>>     
>>> Le 23/08/2010 12:09, David Holmes a écrit :
>>>       
>>>> Rémi Forax said the following on 08/23/10 20:06:
>>>>         
>>>>> You can explicitly call setStackTrace() on a shared immutable 
>>>>> exception, hence there is a problem.
>>>>> addSupressedExceptions() will have the same issue.
>>>>>           
>>>> I stand corrected. The Java side of the code couldn't care less 
>>>> about the VM level backtrace - it just installs a Java array. :(
>>>>
>>>>         
>>>>> I think the fix is to:
>>>>> - silently discard the stack trace taken as argument of 
>>>>> setStackTrace() if the cause is 'null' (or not 'this')
>>>>> - silently don't register suppressed-exceptions if cause is 'null'
>>>>>           
>>>> I think the Java code should be able to recognize a Throwable 
>>>> instance that should not allow these things to be set, and simply 
>>>> ignore the request. However that is a change in the spec for 
>>>> setStackTrace.
>>>>         
>>> Yes, setStackTrace() can 'not succeed'.
>>> And the way to recognize such Throwable instance is to check is the 
>>> field 'cause' is null.
>>>       
>> Except if someone invokes setCause(null).
>>     
>
> setCause(null) should set the field cause to this.
> For Throwable(null) and Throwable(message, null), null should also be 
> escaped.
>
> The other solution is to use the field suppressedExceptions with the 
> invariant that if
> it's null, it's an immutable shared exception.
>
>   
>> By the way, probing further, the non-OOME shared pre-allocated 
>> instances do not guard themselves against fillInStackTrace being 
>> invoked upon them. That is arguably an additional bug. I say arguably 
>> simply because in the circumstances under which those other exceptions 
>> are thrown, having a misleading stacktrace is probably the least of 
>> the application's worries.
>>     
>
> yes, you right.
>
>   

To recap, Throwable objects have several pieces of mutable state:

* the cause; this is a write-at-most-once value that can be set set via 
a constructor, like Throwable(Throwable cause) or Throwable(String 
message, Throwable cause), or set after construction via the 
Throwable.initCause method.  (Setting the cause to null prevents future 
assignment.)

* the stack trace; set by fillInStackTrace() or 
setStackTrace(StackTraceElement[] stackTrace) -- there doesn't seem to 
be any API prohibition against setting the stack trace multiple times

* suppressed exceptions : new functionality added to support 
try-with-resources statements/ARM blocks.  Modified via calling 
addSuppressedException

Focusing just on suppressed exceptions, to support the JVM reusing 
exception objects, the Throwable API should have some idiom to indicate 
suppressed exception information should *not* be recorded. Logically 
this amount to having the list of exceptions be a zero-length list which 
just discards adds.

I'm hesitant to overload a null cause with discarding suppressed 
exceptions too.  Instead, I propose the following:

* a null value of the suppressedException field indicates 
addSuppressedException is a no-op.

* the suppressedException field is initialized to point to a sentinel 
list in Throwable, something non-null that can be used for == checks.  
With this protocol, the "raw" Throwable objects created by the JVM get 
the addSuppressedException is a no-op behavior for free.

* if the first call to addSuppressedException has this as an argument, 
the suppressedException field is null-ed; otherwise, a list is appended 
to as usual.

Comments?

-Joe




More information about the coin-dev mailing list