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

io.aeron.protocol.StatusMessageFlyweight Maven / Gradle / Ivy

/*
 * Copyright 2014-2021 Real Logic Limited.
 *
 * 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
 *
 * https://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 io.aeron.protocol;

import io.aeron.exceptions.AeronException;
import org.agrona.concurrent.UnsafeBuffer;

import java.nio.ByteBuffer;
import java.nio.ByteOrder;

import static java.nio.ByteOrder.LITTLE_ENDIAN;
import static org.agrona.BitUtil.SIZE_OF_LONG;

/**
 * Flyweight for a Status Message Frame.
 * 

* * Status Message wiki page. */ public class StatusMessageFlyweight extends HeaderFlyweight { /** * Length of the Status Message Frame */ public static final int HEADER_LENGTH = 36; /** * Publisher should send SETUP frame */ public static final short SEND_SETUP_FLAG = 0x80; /** * Offset in the frame at which the session-id field begins. */ private static final int SESSION_ID_FIELD_OFFSET = 8; /** * Offset in the frame at which the stream-id field begins. */ private static final int STREAM_ID_FIELD_OFFSET = 12; /** * Offset in the frame at which the consumption term-id field begins. */ private static final int CONSUMPTION_TERM_ID_FIELD_OFFSET = 16; /** * Offset in the frame at which the consumption term-offset field begins. */ private static final int CONSUMPTION_TERM_OFFSET_FIELD_OFFSET = 20; /** * Offset in the frame at which the receiver window length field begins. */ private static final int RECEIVER_WINDOW_FIELD_OFFSET = 24; /** * Offset in the frame at which the receiver-id field begins. */ private static final int RECEIVER_ID_FIELD_OFFSET = 28; /** * Offset in the frame at which the ASF field begins. */ private static final int APP_SPECIFIC_FEEDBACK_FIELD_OFFSET = 36; /** * Offset in the frame at which the gtag field begins. */ private static final int GROUP_TAG_FIELD_OFFSET = APP_SPECIFIC_FEEDBACK_FIELD_OFFSET; /** * Default constructor which can later be used to wrap a frame. */ public StatusMessageFlyweight() { } /** * Construct the flyweight over a frame. * * @param buffer containing the frame. */ public StatusMessageFlyweight(final ByteBuffer buffer) { super(buffer); } /** * Construct the flyweight over a frame. * * @param buffer containing the frame. */ public StatusMessageFlyweight(final UnsafeBuffer buffer) { super(buffer); } /** * The session-id for the stream. * * @return session-id for the stream. */ public int sessionId() { return getInt(SESSION_ID_FIELD_OFFSET, LITTLE_ENDIAN); } /** * Set the session-id of the stream. * * @param sessionId field value. * @return this for a fluent API. */ public StatusMessageFlyweight sessionId(final int sessionId) { putInt(SESSION_ID_FIELD_OFFSET, sessionId, LITTLE_ENDIAN); return this; } /** * The stream-id for the stream. * * @return the session-id for the stream. */ public int streamId() { return getInt(STREAM_ID_FIELD_OFFSET, LITTLE_ENDIAN); } /** * Set the session-id for the stream. * * @param streamId field value. * @return this for a fluent API. */ public StatusMessageFlyweight streamId(final int streamId) { putInt(STREAM_ID_FIELD_OFFSET, streamId, LITTLE_ENDIAN); return this; } /** * The highest consumption offset within the term. * * @return the highest consumption offset within the term. */ public int consumptionTermOffset() { return getInt(CONSUMPTION_TERM_OFFSET_FIELD_OFFSET, LITTLE_ENDIAN); } /** * Set the highest consumption offset within the term. * * @param termOffset field value. * @return this for a fluent API. */ public StatusMessageFlyweight consumptionTermOffset(final int termOffset) { putInt(CONSUMPTION_TERM_OFFSET_FIELD_OFFSET, termOffset, LITTLE_ENDIAN); return this; } /** * The highest consumption term-id. * * @return highest consumption term-id. */ public int consumptionTermId() { return getInt(CONSUMPTION_TERM_ID_FIELD_OFFSET, LITTLE_ENDIAN); } /** * Set the highest consumption term-id. * * @param termId field value. * @return this for a fluent API. */ public StatusMessageFlyweight consumptionTermId(final int termId) { putInt(CONSUMPTION_TERM_ID_FIELD_OFFSET, termId, LITTLE_ENDIAN); return this; } /** * The receiver window length they will accept. * * @return receiver window length they will accept. */ public int receiverWindowLength() { return getInt(RECEIVER_WINDOW_FIELD_OFFSET, LITTLE_ENDIAN); } /** * Set the receiver window length they will accept. * * @param receiverWindowLength field value. * @return this for a fluent API. */ public StatusMessageFlyweight receiverWindowLength(final int receiverWindowLength) { putInt(RECEIVER_WINDOW_FIELD_OFFSET, receiverWindowLength, LITTLE_ENDIAN); return this; } /** * Identifier for the receiver to distinguish them for FlowControl strategies. * * @return identifier for the receiver to distinguish them for FlowControl strategies. */ public long receiverId() { return getLongUnaligned(RECEIVER_ID_FIELD_OFFSET); } /** * Identifier for the receiver to distinguish them for FlowControl strategies. * * @param id for the receiver to distinguish them for FlowControl strategies. * @return this for a fluent API. */ public StatusMessageFlyweight receiverId(final long id) { return putLongUnaligned(RECEIVER_ID_FIELD_OFFSET, id); } /** * The length of the Application Specific Feedback (or gtag). * * @return length, in bytes, of the Application Specific Feedback (or gtag). */ public int asfLength() { return (frameLength() - HEADER_LENGTH); } /** * The group tag (if present) from the Status Message. * * @return the group tag value or 0 if not present. */ public long groupTag() { final int frameLength = frameLength(); if (frameLength > HEADER_LENGTH) { if (frameLength > (HEADER_LENGTH + SIZE_OF_LONG)) { throw new AeronException( "SM has longer application specific feedback (" + (frameLength - HEADER_LENGTH) + ") than gtag"); } return getLongUnaligned(GROUP_TAG_FIELD_OFFSET); } return 0; } /** * Set the Receiver Group Tag for the Status Message. * * @param groupTag value to set if not null. * @return this for a fluent API. */ public StatusMessageFlyweight groupTag(final Long groupTag) { if (null != groupTag) { frameLength(HEADER_LENGTH + SIZE_OF_LONG); putLongUnaligned(GROUP_TAG_FIELD_OFFSET, groupTag); } else { // make sure to explicitly set the frameLength in case of previous tags used. frameLength(HEADER_LENGTH); } return this; } /** * Return the field offset within the flyweight for the group tag field. * * @return offset of group tag field */ public static int groupTagFieldOffset() { return GROUP_TAG_FIELD_OFFSET; } /** * Get long value from a field that is not aligned on an 8 byte boundary. * * @param offset of the field to get. * @return value of field. */ public long getLongUnaligned(final int offset) { final long value; if (ByteOrder.nativeOrder() == LITTLE_ENDIAN) { value = (((long)getByte(offset + 7)) << 56) | (((long)getByte(offset + 6) & 0xFF) << 48) | (((long)getByte(offset + 5) & 0xFF) << 40) | (((long)getByte(offset + 4) & 0xFF) << 32) | (((long)getByte(offset + 3) & 0xFF) << 24) | (((long)getByte(offset + 2) & 0xFF) << 16) | (((long)getByte(offset + 1) & 0xFF) << 8) | (((long)getByte(offset) & 0xFF)); } else { value = (((long)getByte(offset)) << 56) | (((long)getByte(offset + 1) & 0xFF) << 48) | (((long)getByte(offset + 2) & 0xFF) << 40) | (((long)getByte(offset + 3) & 0xFF) << 32) | (((long)getByte(offset + 4) & 0xFF) << 24) | (((long)getByte(offset + 5) & 0xFF) << 16) | (((long)getByte(offset + 6) & 0xFF) << 8) | (((long)getByte(offset + 7) & 0xFF)); } return value; } /** * Set long value into a field that is not aligned on an 8 byte boundary. * * @param offset of the field to put. * @param value of the field to put. * @return this for fluent API. */ public StatusMessageFlyweight putLongUnaligned(final int offset, final long value) { if (ByteOrder.nativeOrder() == LITTLE_ENDIAN) { putByte(offset + 7, (byte)(value >> 56)); putByte(offset + 6, (byte)(value >> 48)); putByte(offset + 5, (byte)(value >> 40)); putByte(offset + 4, (byte)(value >> 32)); putByte(offset + 3, (byte)(value >> 24)); putByte(offset + 2, (byte)(value >> 16)); putByte(offset + 1, (byte)(value >> 8)); putByte(offset, (byte)(value)); } else { putByte(offset, (byte)(value >> 56)); putByte(offset + 1, (byte)(value >> 48)); putByte(offset + 2, (byte)(value >> 40)); putByte(offset + 3, (byte)(value >> 32)); putByte(offset + 4, (byte)(value >> 24)); putByte(offset + 5, (byte)(value >> 16)); putByte(offset + 6, (byte)(value >> 8)); putByte(offset + 7, (byte)(value)); } return this; } /** * {@inheritDoc} */ public String toString() { return "STATUS{" + "frame-length=" + frameLength() + " version=" + version() + " flags=" + String.valueOf(flagsToChars(flags())) + " type=" + headerType() + " session-id=" + sessionId() + " stream-id=" + streamId() + " consumption-term-id=" + consumptionTermId() + " consumption-term-offset=" + consumptionTermOffset() + " receiver-window-length=" + receiverWindowLength() + "}"; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy