X-Git-Url: https://git.opendaylight.org/gerrit/gitweb?a=blobdiff_plain;f=opendaylight%2Fmd-sal%2Fsal-distributed-datastore%2Fsrc%2Fmain%2Fjava%2Forg%2Fopendaylight%2Fcontroller%2Fcluster%2Fdatabroker%2Factors%2Fdds%2FAbstractClientHandle.java;h=842bc00d916ab1eae6c531cb4ceeddc971bdbc4c;hb=42d26918854b1c78896afe703dcceaf281bb5cb2;hp=1246c2f7c740a3a6fa4a9183cf6320d989934cbd;hpb=3859df9beca8f13f1ff2b2744ed3470a1715bec3;p=controller.git diff --git a/opendaylight/md-sal/sal-distributed-datastore/src/main/java/org/opendaylight/controller/cluster/databroker/actors/dds/AbstractClientHandle.java b/opendaylight/md-sal/sal-distributed-datastore/src/main/java/org/opendaylight/controller/cluster/databroker/actors/dds/AbstractClientHandle.java index 1246c2f7c7..842bc00d91 100644 --- a/opendaylight/md-sal/sal-distributed-datastore/src/main/java/org/opendaylight/controller/cluster/databroker/actors/dds/AbstractClientHandle.java +++ b/opendaylight/md-sal/sal-distributed-datastore/src/main/java/org/opendaylight/controller/cluster/databroker/actors/dds/AbstractClientHandle.java @@ -13,10 +13,9 @@ import static java.util.Objects.requireNonNull; import com.google.common.annotations.Beta; import com.google.common.base.MoreObjects; import java.util.Collection; -import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.atomic.AtomicReferenceFieldUpdater; -import java.util.function.Function; +import org.eclipse.jdt.annotation.NonNull; import org.eclipse.jdt.annotation.Nullable; import org.opendaylight.controller.cluster.access.concepts.TransactionIdentifier; import org.opendaylight.yangtools.concepts.Identifiable; @@ -98,22 +97,25 @@ public abstract class AbstractClientHandle e * closed, too. */ final @Nullable Collection ensureClosed() { - @SuppressWarnings("unchecked") - final State local = STATE_UPDATER.getAndSet(this, null); - return local == null ? null : local.values(); + // volatile read and a conditional CAS. This ends up being better in the typical case when we are invoked more + // than once (see ClientBackedTransaction) than performing a STATE_UPDATER.getAndSet(). + final State local = state; + return local != null && STATE_UPDATER.compareAndSet(this, local, null) ? local.values() : null; } - final T ensureProxy(final YangInstanceIdentifier path, final Function createProxy) { - final Map local = getState(); + final T ensureProxy(final YangInstanceIdentifier path) { + final State local = getState(); final Long shard = parent.resolveShardForPath(path); - return local.computeIfAbsent(shard, createProxy); + return local.computeIfAbsent(shard, this::createProxy); } final AbstractClientHistory parent() { return parent; } + abstract @NonNull T createProxy(@NonNull Long shard); + private State getState() { final State local = state; checkState(local != null, "Transaction %s is closed", transactionId);