/* * Copyright (c) 2014 Cisco Systems, Inc. and others. All rights reserved. * * This program and the accompanying materials are made available under the * terms of the Eclipse Public License v1.0 which accompanies this distribution, * and is available at http://www.eclipse.org/legal/epl-v10.html */ package org.opendaylight.mdsal.binding.dom.codec.impl; import static java.util.Objects.requireNonNull; import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; import java.lang.invoke.MethodHandles; import java.lang.invoke.VarHandle; import org.eclipse.jdt.annotation.NonNull; import org.opendaylight.mdsal.binding.runtime.api.CompositeRuntimeType; import org.opendaylight.yangtools.yang.binding.InstanceIdentifier.Item; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier; abstract sealed class CommonDataObjectCodecPrototype implements CodecContextSupplier permits AugmentationCodecPrototype, DataObjectCodecPrototype { private static final VarHandle INSTANCE; static { try { INSTANCE = MethodHandles.lookup().findVarHandle(CommonDataObjectCodecPrototype.class, "instance", CommonDataObjectCodecContext.class); } catch (NoSuchFieldException | IllegalAccessException e) { throw new ExceptionInInitializerError(e); } } private final @NonNull T type; private final @NonNull CodecContextFactory factory; private final @NonNull Item bindingArg; // Accessed via INSTANCE @SuppressWarnings("unused") @SuppressFBWarnings(value = "UUF_UNUSED_FIELD", justification = "https://github.com/spotbugs/spotbugs/issues/2749") private volatile CommonDataObjectCodecContext instance; CommonDataObjectCodecPrototype(final Item bindingArg, final T type, final CodecContextFactory factory) { this.bindingArg = requireNonNull(bindingArg); this.type = requireNonNull(type); this.factory = requireNonNull(factory); } final @NonNull T getType() { return type; } final @NonNull CodecContextFactory getFactory() { return factory; } final @NonNull Class getBindingClass() { return bindingArg.getType(); } final @NonNull Item getBindingArg() { return bindingArg; } abstract @NonNull NodeIdentifier getYangArg(); @Override public final CommonDataObjectCodecContext get() { final var existing = (CommonDataObjectCodecContext) INSTANCE.getAcquire(this); return existing != null ? existing : loadInstance(); } private @NonNull CommonDataObjectCodecContext loadInstance() { final var tmp = createInstance(); final var witness = (CommonDataObjectCodecContext) INSTANCE.compareAndExchangeRelease(this, null, tmp); return witness == null ? tmp : witness; } // This method must allow concurrent loading, i.e. nothing in it may have effects outside of the loaded object abstract @NonNull CommonDataObjectCodecContext createInstance(); }