|  | // 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<K> { | 
|  | /** | 
|  | * 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}. | 
|  | * | 
|  | * <p>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. | 
|  | * | 
|  | * <p>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. | 
|  | * | 
|  | * <p>The intended usage is: | 
|  | * | 
|  | * <pre> | 
|  | * {@code | 
|  | * try (AutoUnlocker unlocker = locker.writeLock(k)) { | 
|  | *   // Your code here. | 
|  | * } | 
|  | * } | 
|  | * </pre> | 
|  | * | 
|  | * <p>Note that the usual caveats about mutexes apply here, e.g. the following may deadlock: | 
|  | * | 
|  | * <pre> | 
|  | * {@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 | 
|  | * } | 
|  | * </pre> | 
|  | */ | 
|  | 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. | 
|  | * | 
|  | * <p>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); | 
|  | } |