diff --git a/src/java.base/share/classes/java/lang/ScopeLocal.java b/src/java.base/share/classes/java/lang/ScopeLocal.java index a4e95478ac7..a27b477b1d1 100644 --- a/src/java.base/share/classes/java/lang/ScopeLocal.java +++ b/src/java.base/share/classes/java/lang/ScopeLocal.java @@ -1,5 +1,6 @@ /* * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, Red Hat Inc. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -30,6 +31,9 @@ import java.util.concurrent.Callable; import java.util.function.Function; import java.util.function.Supplier; + +import jdk.internal.access.JavaLangAccess; +import jdk.internal.access.SharedSecrets; import jdk.internal.vm.StackableScope; import jdk.internal.vm.annotation.ForceInline; import jdk.internal.vm.annotation.Stable; @@ -69,7 +73,7 @@ * private static final ScopeLocal<Credentials> CREDENTIALS = ScopeLocal.newInstance(); * * Credentials creds = ... - * ScopeLocal.where(CREDENTIALS, creds).run(creds, () -> { + * ScopeLocal.where(CREDENTIALS, creds).run(() -> { * : * Connection connection = connectDatabase(); * : @@ -81,6 +85,21 @@ * } * }</pre> * + * As an alternative to the lambda expression form used above, {@link ScopeLocal} also supports + * a <i>try-with-resources</i> form, which looks like this: + * <pre>{@code} + * try (var unused = ScopeLocal.where(CREDENTIALS, creds).bind()) { + * : + * Connection connection = connectDatabase(); + * : + * } + * }</pre> + * + * Note, however, that this version is <i>insecure</i>: it is up to the application programmer + * to make sure that bindings are closed at the right time. While a <i>try-with-resources</i> + * statement is enough to guarantee this, there is no way to enforce the requirement that this form + * is only used in a <i>try-with-resources</i> statement. + * <p>Also, it is not possible to re-bind an already-bound {@link ScopeLocal} with this <i>try-with-resources</i> binding.</p> * @param <T> the scope local's type * @since 99 */ @@ -88,6 +107,8 @@ public final class ScopeLocal<T> { private final @Stable int hash; + private static final JavaLangAccess JLA = SharedSecrets.getJavaLangAccess(); + public final int hashCode() { return hash; } /** @@ -115,6 +136,9 @@ static class Snapshot { Object find(ScopeLocal<?> key) { for (Snapshot b = this; b != null; b = b.prev) { if (((1 << Cache.primaryIndex(key)) & b.primaryBits) != 0) { + if (b.getClass() != Snapshot.class) { + return b.find(key); + } for (Carrier binding = b.bindings; binding != null; binding = binding.prev) { @@ -180,7 +204,7 @@ private static final <T> Carrier where(ScopeLocal<T> key, T value, * @param key The ScopeLocal to bind a value to * @param value The new value * @param <T> The type of the ScopeLocal - * @return TBD + * @return A new map, consisting of {@code this}. plus a new binding. {@code this} is unchanged. */ public final <T> Carrier where(ScopeLocal<T> key, T value) { return where(key, value, this, primaryBits, secondaryBits); @@ -242,49 +266,28 @@ public final <R> R call(Callable<R> op) throws Exception { Objects.requireNonNull(op); Cache.invalidate(primaryBits | secondaryBits); var prevBindings = addScopeLocalBindings(this, primaryBits); - Throwable ex = null; - R result = null; - var scope = new StackableScope().push(); - boolean atTop; try { - result = op.call(); - } catch (Throwable e) { - ex = e; + return StackableScope.call(op); + } catch (Throwable t) { + Cache.invalidate(); + throw t; } finally { - atTop = scope.popForcefully(); // may block Thread.currentThread().scopeLocalBindings = prevBindings; Cache.invalidate(primaryBits | secondaryBits); } - // re-throw exception if op completed with exception - // throw exception if a structure mismatch was detected - if (ex != null || !atTop) { - if (!atTop) { - var e = new StructureViolationException(); - if (ex == null) { - ex = e; - } else { - ex.addSuppressed(e); - } - } - if (ex instanceof Exception e) - throw e; - if (ex instanceof Error e) - throw e; - assert false; - } - return result; } /** - * Runs a value-returning operation with this some ScopeLocals bound to values. - * If the operation terminates with an exception {@code e}, apply {@code handler} - * to {@code e} and return the result. - * - * @param op the operation to run - * @param handler a function to be applied if the operation completes with an exception - * @param <R> the type of the result of the function - * @return the result - */ + * Runs a value-returning operation with this some ScopeLocals bound to values, + * in the same way as {@code call()}.<p> + * If the operation throws an exception, pass it as a single argument to the {@link Function} + * {@code handler}. {@code handler} must return a value compatible with the type returned by {@code op}. + * </p> + * @param op the operation to run + * @param <R> the type of the result of the function + * @param handler the handler to be applied if {code op} threw an exception + * @return the result. + */ public final <R> R callOrElse(Callable<R> op, Function<? super Exception, ? extends R> handler) { try { @@ -312,50 +315,114 @@ public final void run(Runnable op) { Objects.requireNonNull(op); Cache.invalidate(primaryBits | secondaryBits); var prevBindings = addScopeLocalBindings(this, primaryBits); - Throwable ex = null; - boolean atTop; - var scope = new StackableScope().push(); try { - op.run(); - } catch (Throwable e) { - ex = e; + StackableScope.run(op); + } catch (Throwable t) { + Cache.invalidate(); + throw t; } finally { - atTop = scope.popForcefully(); // may block Thread.currentThread().scopeLocalBindings = prevBindings; Cache.invalidate(primaryBits | secondaryBits); } - // re-throw exception if op completed with exception - // throw exception if a structure mismatch was detected - if (ex != null || !atTop) { - if (!atTop) { - var e = new StructureViolationException(); - if (ex == null) { - ex = e; - } else { - ex.addSuppressed(e); - } - } - if (ex instanceof RuntimeException e) - throw e; - if (ex instanceof Error e) - throw e; - assert false; - } } /* * Add a list of bindings to the current Thread's set of bound values. */ - private final static Snapshot addScopeLocalBindings(Carrier bindings, short primaryBits) { + private static final Snapshot addScopeLocalBindings(Carrier bindings, short primaryBits) { Snapshot prev = getScopeLocalBindings(); var b = new Snapshot(bindings, prev, primaryBits); ScopeLocal.setScopeLocalBindings(b); return prev; } + + /* + * Ensure that none of these bindings is already bound. + */ + void checkNotBound() { + for (Carrier c = this; c != null; c = c.prev) { + if (c.key.isBound()) { + throw new RuntimeException("Scope Local already bound"); + } + } + } + + /** + * Create a try-with-resources ScopeLocal binding to be used within + * a try-with-resources block. + * <p>If any of the {@link ScopeLocal}s bound in this {@link Carrier} are already bound in an outer context, + * throw a {@link RuntimeException}.</p> + * @return a {@link ScopeLocalBinder}. + */ + public ScopeLocalBinder bind() { + checkNotBound(); + return (ScopeLocalBinder)new Binder(this).push(); + } } /** - * Creates a binding for a ScopeLocal instance. + * An @AutoCloseable that's used to bind a {@code ScopeLocal} in a try-with-resources construct. + */ + static final class Binder extends StackableScope implements ScopeLocalBinder { + final Carrier bindings; + final short primaryBits; + final Binder prevBinder; + + Binder(Carrier bindings) { + this.bindings = bindings; + this.prevBinder = innermostBinder(); + this.primaryBits = (short)(bindings.primaryBits + | (prevBinder == null ? 0 : prevBinder.primaryBits)); + } + + static Binder innermostBinder() { + StackableScope headScope = JLA.headStackableScope(Thread.currentThread()); + if (headScope == null) { + headScope = JLA.threadContainer(Thread.currentThread()); + } + if (headScope == null) { + return null; + } + return headScope.innermostScope(Binder.class); + } + + /** + * Close a scope local binding context. + * + * @throws StructureViolationException if {@code this} isn't the current top binding + */ + public void close() throws RuntimeException { + Cache.invalidate(bindings.primaryBits|bindings.secondaryBits); + if (! popForcefully()) { + Cache.invalidate(); + throw new StructureViolationException(); + } + } + + protected boolean tryClose() { + Cache.invalidate(bindings.primaryBits|bindings.secondaryBits); + return true; + } + + static Object find(ScopeLocal<?> key) { + for (Binder b = innermostBinder(); b != null; b = b.prevBinder) { + if (((1 << Cache.primaryIndex(key)) & b.primaryBits) != 0) { + for (Carrier binding = b.bindings; + binding != null; + binding = binding.prev) { + if (binding.getKey() == key) { + Object value = binding.get(); + return value; + } + } + } + } + return Snapshot.NIL; + } + } + + /** + * Create a binding for a ScopeLocal instance. * That {@link Carrier} may be used later to invoke a {@link Callable} or * {@link Runnable} instance. More bindings may be added to the {@link Carrier} * by the {@link Carrier#where(ScopeLocal, Object)} method. @@ -437,8 +504,7 @@ public T get() { @SuppressWarnings("unchecked") private T slowGet() { - var bindings = scopeLocalBindings(); - var value = bindings.find(this); + var value = findBinding(); if (value == Snapshot.NIL) { throw new NoSuchElementException(); } @@ -453,14 +519,18 @@ private T slowGet() { */ @SuppressWarnings("unchecked") public boolean isBound() { - return (scopeLocalBindings().find(this) != Snapshot.NIL); + return findBinding() != Snapshot.NIL; } /** * Return the value of the scope local or NIL if not bound. */ private Object findBinding() { - return scopeLocalBindings().find(this); + Object value = scopeLocalBindings().find(this); + if (value != Snapshot.NIL) { + return value; + } + return Binder.find(this); } /** @@ -501,8 +571,7 @@ public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSuppli } private static Snapshot getScopeLocalBindings() { - Thread currentThread = Thread.currentThread(); - return currentThread.scopeLocalBindings; + return Thread.currentThread().scopeLocalBindings; } private static void setScopeLocalBindings(Snapshot bindings) { diff --git a/src/java.base/share/classes/java/lang/ScopeLocalBinder.java b/src/java.base/share/classes/java/lang/ScopeLocalBinder.java new file mode 100644 index 00000000000..a0229099a19 --- /dev/null +++ b/src/java.base/share/classes/java/lang/ScopeLocalBinder.java @@ -0,0 +1,41 @@ +/* + * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, Red Hat Inc. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package java.lang; + +/** + * The interface for a ScopeLocal try-with-resources binding. + */ +public sealed interface ScopeLocalBinder extends AutoCloseable permits ScopeLocal.Binder { + + /** + * Closes this {@link ScopeLocal} binding. If this binding was not the most recent binding + * created by {@code Carrier.bind()}, throws a {@link StructureViolationException}. + * This method is invoked automatically on objects managed by the try-with-resources statement. + * @throws StructureViolationException if the bindings were not closed in the correct order. + */ + public void close() throws StructureViolationException; +} diff --git a/src/java.base/share/classes/jdk/internal/vm/StackableScope.java b/src/java.base/share/classes/jdk/internal/vm/StackableScope.java index b39bccf592d..1d1a90d1f4c 100644 --- a/src/java.base/share/classes/jdk/internal/vm/StackableScope.java +++ b/src/java.base/share/classes/jdk/internal/vm/StackableScope.java @@ -27,6 +27,8 @@ import jdk.internal.access.JavaLangAccess; import jdk.internal.access.SharedSecrets; +import java.util.concurrent.Callable; + /** * A stackable scope. */ @@ -73,6 +75,109 @@ public StackableScope push() { return this; } + private static void handleThrows(Throwable ex, boolean atTop) { + // re-throw exception if op completed with exception + // throw exception if a structure mismatch was detected + if (ex != null || !atTop) { + if (!atTop) { + var e = new StructureViolationException(); + if (ex == null) { + ex = e; + } else { + ex.addSuppressed(e); + } + } + if (ex instanceof RuntimeException e) + throw e; + if (ex instanceof Error e) + throw e; + assert false; + } + } + + /** + * Call op, wrapped in a {@code StackableScope} + * @param op a Callable + * @param <T> a class + * @return a T + * @throws Exception + */ + public static <T> T call(Callable<T> op) throws Exception { + if (head() == null) { + Throwable ex = null; + T result = null; + try { + result = op.call(); + } catch (Throwable e) { + ex = e; + } finally { + StackableScope head = head(); + if (head != null) { + popAll(); + } + handleThrows(ex, head == null); + } + return result; + } else { + // Slow path + return new StackableScope().doCall(op); + } + } + + private <T> T doCall(Callable<T> op) throws Exception { + Throwable ex = null; + boolean atTop; + T result = null; + var scope = push(); + try { + result = op.call(); + } catch (Throwable e) { + ex = e; + } finally { + atTop = scope.popForcefully(); // may block + } + handleThrows(ex, atTop); + return result; + } + + /** + * Run op, wrapped in a {@code StackableScope} + * @param op a Runnable + */ + public static void run(Runnable op) { + if (head() == null) { + Throwable ex = null; + try { + op.run(); + } catch (Throwable e) { + ex = e; + } finally { + StackableScope head = head(); + if (head != null) { + popAll(); + } + handleThrows(ex, head == null); + } + } else { + // Slow path + new StackableScope().doRun(op); + } + } + + private void doRun(Runnable op) { + Throwable ex = null; + boolean atTop; + var scope = push(); + try { + op.run(); + } catch (Throwable e) { + ex = e; + } finally { + atTop = scope.popForcefully(); // may block + } + handleThrows(ex, atTop); + } + /** * Pops this scope from the current thread's scope stack if the scope is * at the top of stack. @@ -164,7 +269,15 @@ public StackableScope enclosingScope() { * Returns the scope of the given type that encloses this scope. */ public <T extends StackableScope> T enclosingScope(Class<T> type) { - StackableScope current = enclosingScope(); + StackableScope enclosing = enclosingScope(); + if (enclosing != null) { + return enclosing.innermostScope(type); + } + return null; + } + + public <T extends StackableScope> T innermostScope(Class<T> type) { + StackableScope current = this; while (current != null) { if (type.isInstance(current)) { @SuppressWarnings("unchecked") diff --git a/test/micro/org/openjdk/bench/java/lang/ScopeLocals.java b/test/micro/org/openjdk/bench/java/lang/ScopeLocals.java index e4bd9825340..8c2cc3808a4 100644 --- a/test/micro/org/openjdk/bench/java/lang/ScopeLocals.java +++ b/test/micro/org/openjdk/bench/java/lang/ScopeLocals.java @@ -139,6 +139,14 @@ public Object bind_ScopeLocal() throws Exception { return ScopeLocal.where(sl1, 42, this::getClass); } + @Benchmark + @OutputTimeUnit(TimeUnit.NANOSECONDS) + public Object TWR_bind_ScopeLocal() throws Exception { + try (var x = ScopeLocal.where(unbound, 42).bind()) { + return getClass(); + } + } + @Benchmark @OutputTimeUnit(TimeUnit.NANOSECONDS) public Object bind_ThreadLocal() throws Exception { diff --git a/test/micro/org/openjdk/bench/java/lang/ScopeLocalsData.java b/test/micro/org/openjdk/bench/java/lang/ScopeLocalsData.java index f98261c1278..a95ef15bc59 100644 --- a/test/micro/org/openjdk/bench/java/lang/ScopeLocalsData.java +++ b/test/micro/org/openjdk/bench/java/lang/ScopeLocalsData.java @@ -39,6 +39,8 @@ public class ScopeLocalsData { static final ScopeLocal<Integer> sl6 = ScopeLocal.newInstance(); static final ScopeLocal<AtomicInteger> sl_atomicInt = ScopeLocal.newInstance(); + static final ScopeLocal<Integer> unbound = ScopeLocal.newInstance(); + static final ScopeLocal<AtomicReference<Integer>> sl_atomicRef = ScopeLocal.newInstance(); static final ThreadLocal<Integer> tl2 = new ThreadLocal<>();