Nest host validation vs NestHost attribute performed by Lookup::defineHiddenClass

Mandy Chung mandy.chung at oracle.com
Wed Sep 25 01:54:11 UTC 2019



On 9/24/19 6:15 PM, David Holmes wrote:
> Hi Mandy,
>
> I think we need to consider the distinction between the runtime 
> nest_host and the statically defined NestHost attribute. If we did 
> eager validation and rejected bad NestHost attributes then the two 
> would always be the same. But for pragmatic reasons we don't do that 
> and only establish the runtime nest_host when an actual access check 
> is needed. If the NestHost H validates then we have H as the runtime 
> nest_host and there's no problem. If the NestHost doesn't validate 
> then the runtime nest_host is logically the class C itself: it 
> implicitly forms a singleton nest.

Right.  You may recall that we discussed previously and decided not to 
ignore the exception in the nest host validation when designing the API.

During a recent discussion Alex and I have with LMF in the picture, it's 
time to revisit this.

>
> Lookup.defineHiddenClass, when trying to inject into a nest, must use 
> the actual runtime nest_host of the lookup class C. This is clear to 
> me as the intent is to give the injected class the same access rights 
> to a nest as C has. So defineHiddenClass needs to trigger nest host 
> validation (ignoring exceptions), if it hasn't been performed yet, and 
> use the resulting runtime nest_host: either H or C depending on the 
> validation result.

Yup.  This revised proposal using the runtime nest host achieves the 
same intent that the nestmate class has the same access rights to C's nest.

Mandy

>
> That's my 2c.
>
> David
>
> On 25/09/2019 10:27 am, Mandy Chung wrote:
>> This is regarding an incompatibility concern when LambdaMetaFactory 
>> and any other framework library converts the use of 
>> `Unsafe::defineAnonymousClass` to `Lookup::defineHiddenClass` when a 
>> caller caller `C` whose nest host `H` indicated by `NestHost` 
>> attribute is "bad".
>>
>> `NestHost` and `NestMembers` attributes are consulted by JVM access 
>> control (JVMS 5.4.4).  If `H` cannot be loaded, or is not in the same 
>> run-time package as `C`, or does not authorize nest membership (`H` 
>> has a `NestMembers attribute but there is no entry named `C`), then a 
>> linkage error occurs.
>>
>> OTOH, `Class::getNestHost` may return a Class that is not the named 
>> class indicated by `NestHost` attribute.  Specifically:
>>
>>     If there is a linkage error accessing the nest host, or if this 
>> class
>>     or interface is not enumerated as a member of the nest by the 
>> nest host,
>>     then it is considered to belong to its own nest and this is returned
>>     as the host.
>>
>> If `C` has a bad nest host indicated by `NestHost` attribute, then 
>> effectively `NestHost` attribute is ignored and `C` is the host of 
>> its own nest as if `C` lacked of `NestHost` attribute. No error 
>> occurs until `C` attempts to access a private member of `H` - 
>> consistency for VM and core reflection.
>>
>> Now `Lookup::defineHiddenClass` can be used to add a hidden class to 
>> a nest of the lookup class.
>>
>> If a framework uses C's full-power lookup object to call 
>> `Lookup::defineHiddenClass` to add a hidden class `X` into C's nest. 
>> What is the nest host of `X`?
>>
>> If C has a good nest host, of course X will have the same nest host 
>> as C.  If C has a bad nest host `H`, the lookup object on C is used 
>> to call `Lookup::defineHiddenClass` - should it fail with illegal 
>> nest host? Should it succeed to be consistent with core reflection 
>> (asC.class::getNestHost returns C - see more)?
>>
>> That is the current proposed behavior of `Lookup::defineHiddenClass` 
>> to validate that the lookup class must be the host of a nest.  If the 
>> lookup class has `NestHost` attribute, it is a member of some other 
>> class's nest and so fails.  The design principle is that the lookup 
>> class and the proper permission together authorizes the lookup object 
>> to add a new member into a nest.  In addition, C's nest host is H and 
>> adding a class with C as the nest host should be disallowed as this 
>> conceptually creates a "nested nest" (H is C's nest host, C is X's 
>> nest host).
>>
>> A framework library can't tell whether C is the class indicated by 
>> C's NestHost attribute. Take an example in practice.  C uses lambda 
>> expression or method reference expression but H cannot be found while 
>> C never accesses H's private member (If C accesses H's private 
>> member, then IAE must occur since it fails the access check). The 
>> first invocation of indy LMF for C class unfortunately throws 
>> BootstrapMethodError because LambdaMetaFactory 
>> calls`Lookup::defineHiddenClass` to define a lambda proxy class for C 
>> and IAE occurs as C is a member of `H` statically.
>>
>> It is expected that this error case should rarely occur but it does 
>> have an incompatibility risk that impacts existing applications.
>>
>> As Class::getNestHost returns C as the nest host, a framework library 
>> calls `defineHiddenClass` with the returned nest host but surprisely 
>> it throws an exception that nothing it can do.
>>
>> If C has a bad nest host `H`, it will not get an error until C 
>> attempts to access a private member of H.  I think we should revisit 
>> this validation and relax `defineHiddenClass` validation to check 
>> that the lookup class is the same class as Class::getNestHost 
>> returns.  The hidden class X has C as the nest host (but NOT H).  The 
>> VM access check will ensure that X will fail when accessing H's 
>> private members.  No nested nest is created since at runtime C is the 
>> host of its own nest and X is added to C's nest (where the static 
>> nestmate relationship is not in effect).
>>
>> Summary:
>>
>> `Lookup::defineHiddenClass` performs the nest host validation but 
>> won't consultthe `NestHost` attribute of the lookup class.
>>
>> Feedback?
>>
>> thanks
>> Mandy




More information about the valhalla-dev mailing list