X-Git-Url: https://git.opendaylight.org/gerrit/gitweb?a=blobdiff_plain;f=opendaylight%2Fmd-sal%2Fsal-dom-broker%2Fsrc%2Fmain%2Fjava%2Forg%2Fopendaylight%2Fcontroller%2Fmd%2Fsal%2Fdom%2Fstore%2Fimpl%2FMutableDataTree.java;h=b711163b465d3ca07914da3d3472347ab854533e;hb=84248dac9ed8aa37e996e39429c8aa8ece473eaf;hp=f252744876f0b8da325523fdcce62c8bad3a7552;hpb=721b580748cb93b3dac952ff1f111d0ab0da0c79;p=controller.git diff --git a/opendaylight/md-sal/sal-dom-broker/src/main/java/org/opendaylight/controller/md/sal/dom/store/impl/MutableDataTree.java b/opendaylight/md-sal/sal-dom-broker/src/main/java/org/opendaylight/controller/md/sal/dom/store/impl/MutableDataTree.java index f252744876..b711163b46 100644 --- a/opendaylight/md-sal/sal-dom-broker/src/main/java/org/opendaylight/controller/md/sal/dom/store/impl/MutableDataTree.java +++ b/opendaylight/md-sal/sal-dom-broker/src/main/java/org/opendaylight/controller/md/sal/dom/store/impl/MutableDataTree.java @@ -10,6 +10,7 @@ package org.opendaylight.controller.md.sal.dom.store.impl; import static com.google.common.base.Preconditions.checkState; import java.util.Map.Entry; +import java.util.concurrent.atomic.AtomicBoolean; import org.opendaylight.controller.md.sal.dom.store.impl.tree.NodeModification; import org.opendaylight.controller.md.sal.dom.store.impl.tree.StoreMetadataNode; @@ -17,21 +18,24 @@ import org.opendaylight.controller.md.sal.dom.store.impl.tree.TreeNodeUtils; import org.opendaylight.yangtools.yang.data.api.InstanceIdentifier; import org.opendaylight.yangtools.yang.data.api.InstanceIdentifier.PathArgument; import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode; +import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNodeContainer; import org.opendaylight.yangtools.yang.data.impl.schema.NormalizedNodeUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import com.google.common.base.Optional; +import com.google.common.base.Preconditions; +/* + * FIXME: the thread safety of concurrent write/delete/read/seal operations + * needs to be evaluated. + */ class MutableDataTree { - - private static final Logger log = LoggerFactory.getLogger(MutableDataTree.class); - - final DataAndMetadataSnapshot snapshot; - final NodeModification rootModification; - final ModificationApplyOperation strategyTree; - - private boolean sealed = false; + private static final Logger LOG = LoggerFactory.getLogger(MutableDataTree.class); + private final AtomicBoolean sealed = new AtomicBoolean(); + private final ModificationApplyOperation strategyTree; + private final DataAndMetadataSnapshot snapshot; + private final NodeModification rootModification; private MutableDataTree(final DataAndMetadataSnapshot snapshot, final ModificationApplyOperation strategyTree) { this.snapshot = snapshot; @@ -44,6 +48,24 @@ class MutableDataTree { resolveModificationFor(path).write(value); } + public void merge(final InstanceIdentifier path, final NormalizedNode data) { + checkSealed(); + mergeImpl(resolveModificationFor(path),data); + } + + private void mergeImpl(final OperationWithModification op,final NormalizedNode data) { + + if(data instanceof NormalizedNodeContainer) { + @SuppressWarnings({ "rawtypes", "unchecked" }) + NormalizedNodeContainer> dataContainer = (NormalizedNodeContainer) data; + for(NormalizedNode child : dataContainer.getValue()) { + PathArgument childId = child.getIdentifier(); + mergeImpl(op.forChild(childId), child); + } + } + op.merge(data); + } + public void delete(final InstanceIdentifier path) { checkSealed(); resolveModificationFor(path).delete(); @@ -58,7 +80,6 @@ class MutableDataTree { return NormalizedNodeUtils.findNode(modification.getKey(), data, path); } return Optional.absent(); - } private Optional resolveSnapshot( @@ -78,13 +99,13 @@ class MutableDataTree { return resolveModificationStrategy(path).apply(modification, modification.getOriginal(), StoreUtils.increase(snapshot.getMetadataTree().getSubtreeVersion())); } catch (Exception e) { - log.error("Could not create snapshot for {}", path,e); + LOG.error("Could not create snapshot for {}:{}", path,modification,e); throw e; } } private ModificationApplyOperation resolveModificationStrategy(final InstanceIdentifier path) { - log.trace("Resolving modification apply strategy for {}", path); + LOG.trace("Resolving modification apply strategy for {}", path); return TreeNodeUtils.findNodeChecked(strategyTree, path); } @@ -103,12 +124,13 @@ class MutableDataTree { } public void seal() { - sealed = true; + final boolean success = sealed.compareAndSet(false, true); + Preconditions.checkState(success, "Attempted to seal an already-sealed Data Tree."); rootModification.seal(); } private void checkSealed() { - checkState(!sealed, "Data Tree is sealed. No further modifications allowed."); + checkState(!sealed.get(), "Data Tree is sealed. No further modifications allowed."); } protected NodeModification getRootModification() { @@ -119,6 +141,4 @@ class MutableDataTree { public String toString() { return "MutableDataTree [modification=" + rootModification + "]"; } - - }