blob: 3ee2f37d6934d574167e5f692dd67c0d7d80a87d [file] [log] [blame]
/*
* Copyright 2000-2013 JetBrains s.r.o.
*
* 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.
*/
/*
* @author max
*/
package com.intellij.util.containers;
import com.intellij.openapi.util.Comparing;
import gnu.trove.THashMap;
import org.jetbrains.annotations.NotNull;
import java.util.Collection;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
public class LockPoolSynchronizedMap<K, V> extends THashMap<K, V> implements ConcurrentMap<K, V> {
private final Lock r;
private final Lock w;
private static final StripedLockHolder<ReentrantReadWriteLock> LOCKS = new StripedLockHolder<ReentrantReadWriteLock>(ReentrantReadWriteLock.class) {
@NotNull
@Override
protected ReentrantReadWriteLock create() {
return new ReentrantReadWriteLock();
}
};
{
final ReentrantReadWriteLock mutex = LOCKS.allocateLock();
r = mutex.readLock();
w = mutex.writeLock();
}
public LockPoolSynchronizedMap() {
}
public LockPoolSynchronizedMap(final int initialCapacity) {
super(initialCapacity);
}
public LockPoolSynchronizedMap(final int initialCapacity, final float loadFactor) {
super(initialCapacity, loadFactor);
}
@Override
public int size() {
r.lock();
try {
return super.size();
}
finally {
r.unlock();
}
}
@Override
public boolean isEmpty() {
r.lock();
try {
return super.isEmpty();
}
finally {
r.unlock();
}
}
@Override
public boolean containsKey(Object key) {
r.lock();
try {
return super.containsKey(key);
}
finally {
r.unlock();
}
}
@Override
public boolean containsValue(Object value) {
r.lock();
try {
return super.containsValue(value);
}
finally {
r.unlock();
}
}
@Override
public V get(Object key) {
r.lock();
try {
return super.get(key);
}
finally {
r.unlock();
}
}
@Override
public V put(K key, V value) {
w.lock();
try {
return super.put(key, value);
}
finally {
w.unlock();
}
}
@Override
public V remove(Object key) {
w.lock();
try {
return super.remove(key);
}
finally {
w.unlock();
}
}
@Override
public void putAll(@NotNull Map<? extends K, ? extends V> map) {
w.lock();
try {
super.putAll(map);
}
finally {
w.unlock();
}
}
@Override
public void clear() {
w.lock();
try {
super.clear();
}
finally {
w.unlock();
}
}
@Override
public LockPoolSynchronizedMap<K, V> clone() {
r.lock();
try {
return (LockPoolSynchronizedMap<K,V>)super.clone();
}
finally {
r.unlock();
}
}
@NotNull
@Override
public Set<K> keySet() {
r.lock();
try {
return super.keySet();
}
finally {
r.unlock();
}
}
@NotNull
@Override
public Set<Map.Entry<K, V>> entrySet() {
r.lock();
try {
return super.entrySet();
}
finally {
r.unlock();
}
}
@NotNull
@Override
public Collection<V> values() {
r.lock();
try {
return super.values();
}
finally {
r.unlock();
}
}
@Override
public boolean replace(@NotNull K key, @NotNull V oldValue, @NotNull V newValue) {
w.lock();
try {
V prev = get(key);
if (!Comparing.equal(oldValue, prev)) {
return false;
}
put(key, newValue);
return true;
}
finally {
w.unlock();
}
}
@Override
public V replace(@NotNull K key, @NotNull V newValue) {
w.lock();
try {
V prev = get(key);
put(key, newValue);
return prev;
}
finally {
w.unlock();
}
}
@Override
public V putIfAbsent(@NotNull K key, V value) {
w.lock();
try {
V prev = get(key);
if (prev == null) {
put(key, value);
return value;
}
else {
return prev;
}
}
finally {
w.unlock();
}
}
@Override
public boolean remove(@NotNull Object key, Object oldValue) {
w.lock();
try {
V currentValue = get(key);
return Comparing.equal(oldValue, currentValue) && super.remove(key) != null;
}
finally {
w.unlock();
}
}
}