blob: 938cde838cf7435ac60aaf39492fc39817e6d71e [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.polygene.runtime.composite;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import org.apache.polygene.api.common.AppliesTo;
import org.apache.polygene.api.composite.Composite;
import org.apache.polygene.api.composite.DefaultMethodsFilter;
import org.apache.polygene.api.injection.scope.This;
import static org.apache.polygene.api.util.AccessibleObjects.accessible;
@AppliesTo( { DefaultMethodsFilter.class } )
public class InterfaceDefaultMethodsMixin
implements InvocationHandler
{
// TODO (niclas): We have one instance of this class per mixin, so it seems a bit wasteful to have a ConcurrentHashMap. Maybe a small array 3 elements, which is changed to a Map is run out of space? Tricky concurrency on that, so leave it for later (a.k.a. will forget about it)
private final ConcurrentMap<Method, MethodCallHandler> methodHandleCache = new ConcurrentHashMap<>();
@This
private Composite me;
@Override
public Object invoke( Object proxy, Method method, Object[] args )
throws Throwable
{
if( method.isDefault() )
{
// Call the interface's default method
MethodCallHandler callHandler = forMethod( method );
return callHandler.invoke( proxy, args );
}
// call the composite's method instead.
return method.invoke( me, args );
}
private MethodCallHandler forMethod( Method method )
{
return methodHandleCache.computeIfAbsent( method, this::createMethodCallHandler );
}
private MethodCallHandler createMethodCallHandler( Method method )
{
Class<?> declaringClass = method.getDeclaringClass();
try
{
Constructor<MethodHandles.Lookup> constructor = MethodHandles.Lookup.class.getDeclaredConstructor( Class.class, int.class );
MethodHandles.Lookup lookup = accessible( constructor ).newInstance( declaringClass, MethodHandles.Lookup.PRIVATE );
MethodHandle handle = lookup.unreflectSpecial( method, declaringClass );
return ( proxy, args ) -> handle.bindTo( proxy ).invokeWithArguments( args );
}
catch( IllegalAccessException | NoSuchMethodException | InstantiationException | InvocationTargetException e )
{
throw new RuntimeException( e );
}
}
@FunctionalInterface
private interface MethodCallHandler
{
Object invoke( Object proxy, Object[] args )
throws Throwable;
}
}