All Downloads are FREE. Search and download functionalities are using the official Maven repository.

zio.internal.MutableQueueFieldsPadding Maven / Gradle / Ivy

There is a newer version: 2.1.16
Show newest version
/*
 * Copyright 2018-2024 John A. De Goes and the ZIO Contributors
 *
 * 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.
 */

package zio.internal;

import java.io.Serializable;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;

import zio.internal.MutableConcurrentQueue;

/*
 * NOTE: these classes need to be implemented in Java, because:
 *   1) `head` and `tail` need to be naked protected or public fields
 *   in order to be accessible by `AtomicLongFieldUpdater`.
 *   2) there doesn't seems to be a way to expose naked public or
 *   protected fields in Scala is it generates accessor methods for
 *   those.
 *
 * The classes below provide padding for contended fields in the
 * [[MutableConcurrentQueue]] layout, specifically `head` and `tail`.
 *
 * The layout of a Java object in memory is dynamic and is under
 * control of the JVM which can shuffle fields to optimize the
 * object's footprint. However, when the object is used in a
 * concurrent setting having all fields densely packed may affect
 * performance since certain concurrently modified fields can fall on
 * the same cache-line and be subject to _False Sharing_. In such case
 * we would like to space out _hot_ fields and thus make access more
 * cache-friendly.
 *
 * On x86 one cache-line is 64KiB and we'd need to space out fields by
 * at least *2* cache-lines because modern processors do pre-fetching,
 * i.e. get the requested cache-line and the one after. Thus we need
 * ~128KiB in-between hot fields.
 *
 * One solution could be just adding a bunch of long fields, but if
 * those are defined within the same class as `head` or `tail`, JVM
 * will likely shuffle the longs and mess up the padding.
 *
 * There is a workaround that relies on the fact the fields of a
 * super-class go before fields of a sub-class in the object's
 * layout. So, a properly crafted inheritance chain can guarantee
 * spacing for the hot fields.
 *
 * To illustrate the effect of such padding, here's an output by JOL
 * (truncated):
 *
 * zio.internal.impls.padding.MutableQueueFieldsPadding
 *  OFFSET  SIZE   TYPE DESCRIPTION
 *       0    12        (object header)
 *      12     4        (alignment/padding gap)
 *      16     8   long ClassFieldsPadding.p000
 *      24     8   long ClassFieldsPadding.p001
 *     ........................................
 *     136     8   long ClassFieldsPadding.p015
 *     144     8   long HeadPadding.headCounter
 *     152     8   long PreTailPadding.p100
 *     160     8   long PreTailPadding.p101
 *     ....................................
 *     272     8   long PreTailPadding.p115
 *     280     8   long TailPadding.tailCounter
 *     288     8   long MutableQueueFieldsPadding.p200
 *     296     8   long MutableQueueFieldsPadding.p201
 *     ...............................................
 *     408     8   long MutableQueueFieldsPadding.p215
 * Instance size: 416 bytes
 * Space losses: 4 bytes internal + 0 bytes external = 4 bytes total
 */
public abstract class MutableQueueFieldsPadding extends TailPadding implements Serializable {
    public static final AtomicLongFieldUpdater headUpdater = AtomicLongFieldUpdater.newUpdater(HeadPadding.class, "headCounter");
    public static final AtomicLongFieldUpdater tailUpdater = AtomicLongFieldUpdater.newUpdater(TailPadding.class, "tailCounter");

    protected long p200;
    protected long p201;
    protected long p202;
    protected long p203;
    protected long p204;
    protected long p205;
    protected long p206;
    protected long p207;
    protected long p208;
    protected long p209;
    protected long p210;
    protected long p211;
    protected long p212;
    protected long p213;
    protected long p214;
    protected long p215;
}

// Aux classes below

abstract class ClassFieldsPadding extends MutableConcurrentQueue implements Serializable {
    protected long p000;
    protected long p001;
    protected long p002;
    protected long p003;
    protected long p004;
    protected long p005;
    protected long p006;
    protected long p007;
    protected long p008;
    protected long p009;
    protected long p010;
    protected long p011;
    protected long p012;
    protected long p013;
    protected long p014;
    protected long p015;
}

abstract class HeadPadding extends ClassFieldsPadding implements Serializable {
    protected volatile long headCounter;
}

abstract class PreTailPadding extends HeadPadding implements Serializable {
    protected long p100;
    protected long p101;
    protected long p102;
    protected long p103;
    protected long p104;
    protected long p105;
    protected long p106;
    protected long p107;
    protected long p108;
    protected long p109;
    protected long p110;
    protected long p111;
    protected long p112;
    protected long p113;
    protected long p114;
    protected long p115;
}

abstract class TailPadding extends PreTailPadding implements Serializable {
    protected volatile long tailCounter;
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy