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=003c073de8ea12a101e9542cd4dab9259a8ae946;hb=118cd0216b0c6b0ec1a01689ec2025a13e090861;hp=b87819c34337a0435b02a3f98209f73a29aef7fe;hpb=5cb0787412ab63a3aa5dcc044511e1ce569662cf;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 b87819c343..003c073de8 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 @@ -7,14 +7,16 @@ */ package org.opendaylight.controller.cluster.databroker.actors.dds; +import static com.google.common.base.Preconditions.checkState; +import static java.util.Objects.requireNonNull; + import com.google.common.annotations.Beta; -import com.google.common.base.Preconditions; -import java.util.Collection; +import com.google.common.base.MoreObjects; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.atomic.AtomicReferenceFieldUpdater; -import java.util.function.Function; -import javax.annotation.Nullable; +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; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier; @@ -41,19 +43,20 @@ public abstract class AbstractClientHandle e private static final AtomicReferenceFieldUpdater STATE_UPDATER = AtomicReferenceFieldUpdater.newUpdater(AbstractClientHandle.class, State.class, "state"); - private final TransactionIdentifier transactionId; - private final AbstractClientHistory parent; + private final @NonNull TransactionIdentifier transactionId; + private final @NonNull AbstractClientHistory parent; private volatile State state = new State<>(); // Hidden to prevent outside instantiation AbstractClientHandle(final AbstractClientHistory parent, final TransactionIdentifier transactionId) { - this.transactionId = Preconditions.checkNotNull(transactionId); - this.parent = Preconditions.checkNotNull(parent); + this.transactionId = requireNonNull(transactionId); + this.parent = requireNonNull(parent); } @Override - public final TransactionIdentifier getIdentifier() { + // Non-final for mocking + public TransactionIdentifier getIdentifier() { return transactionId; } @@ -62,7 +65,8 @@ public abstract class AbstractClientHandle e * * @return True if this transaction became closed during this call */ - public final boolean abort() { + // Non-final for mocking + public boolean abort() { if (commonAbort()) { parent.onTransactionAbort(this); return true; @@ -72,12 +76,13 @@ public abstract class AbstractClientHandle e } private boolean commonAbort() { - final Collection toClose = ensureClosed(); + final Map toClose = ensureClosed(); if (toClose == null) { return false; } - toClose.forEach(AbstractProxyTransaction::abort); + toClose.values().forEach(AbstractProxyTransaction::abort); + parent.onTransactionShardsBound(transactionId, toClose.keySet()); return true; } @@ -91,29 +96,38 @@ public abstract class AbstractClientHandle e * Make sure this snapshot is closed. If it became closed as the effect of this call, return a collection of * {@link AbstractProxyTransaction} handles which need to be closed, too. * - * @return null if this snapshot has already been closed, otherwise a collection of proxies, which need to be + * @return null if this snapshot has already been closed, otherwise a State with of proxies, which need to be * closed, too. */ - @Nullable final Collection ensureClosed() { - @SuppressWarnings("unchecked") - final State local = STATE_UPDATER.getAndSet(this, null); - return local == null ? null : local.values(); + final @Nullable Map ensureClosed() { + // 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 : 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; - Preconditions.checkState(local != null, "Transaction %s is closed", transactionId); + checkState(local != null, "Transaction %s is closed", transactionId); return local; } + + @Override + public final String toString() { + return MoreObjects.toStringHelper(this).omitNullValues().add("identifier", transactionId).add("state", state) + .toString(); + } }