// Copyright 2015 The Bazel Authors. All rights reserved. // // Licensed 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 com.google.devtools.build.lib.concurrent; import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe; /** A keyed store of locks. */ @ThreadSafe public interface KeyedLocker { /** * Used to yield access to the implicit locks granted by {@link #writeLock} or {@link #readLock}. */ @ThreadSafe interface AutoUnlocker extends AutoCloseable { /** Exception used to indicate illegal use of {@link AutoUnlocker#close}. */ class IllegalUnlockException extends RuntimeException { public IllegalUnlockException(String msg) { super(msg); } } /** * Closes the {@link AutoUnlocker} instance. If this instance was the last unclosed one * returned by {@link #writeLock} with argument {@code k} owned by the current * thread, then exclusive access to {@code k} is yielded. If this instance was the last unclosed * one returned by {@link #readLock} with argument {@code k}, then a thread can request * exclusive write access using {@link #writeLock} with argument {@code k}. * *

This method may only be called at most once per {@link AutoUnlocker} instance and must * be called by the same thread that acquired the {@link AutoUnlocker} via {@link #writeLock} * or {@link #readLock}. Otherwise, an {@link IllegalUnlockException} is thrown. */ @Override void close(); } /** * Blocks the current thread until it has exclusive access to do things with {@code k} and * returns a {@link AutoUnlocker} instance for yielding the implicit lock. * *

Notably, this means that a thread is allowed to call {@code writeLock(k)} again before * calling {@link AutoUnlocker#close} for the first call to {@code writeLock(k)}. Each call to * {@code #writeLock} will return a different {@link AutoUnlocker} instance. * *

The intended usage is: * *

   * {@code
   * try (AutoUnlocker unlocker = locker.writeLock(k)) {
   *   // Your code here.
   * }
   * }
   * 
* *

Note that the usual caveats about mutexes apply here, e.g. the following may deadlock: * *

   * {@code
   * // Thread A
   * try (AutoUnlocker unlocker = locker.writeLock(k1)) {
   *   // This will deadlock if Thread B already acquired a writeLock for k2.
   *   try (AutoUnlocker unlocker = locker.writeLock(k2)) {
   *   }
   * }
   * // end Thread A
   *
   * // Thread B
   * try (AutoUnlocker unlocker = locker.writeLock(k2)) {
   *   // This will deadlock if Thread A already acquired a writeLock for k1.
   *   try (AutoUnlocker unlocker = locker.writeLock(k1)) {
   *   }
   * }
   * // end Thread B
   * }
   * 
*/ AutoUnlocker writeLock(K key); /** * Blocks the current thread until it has access to read things that have to do with {@code k}. * Multiple threads may acquire simultaneous read locks, so long as there is no thread with a * write lock. * *

As with {@link #writeLock}, the same thread can call {@code readLock(k)} multiple times for * the same k before closing the lock. */ AutoUnlocker readLock(K key); }