org.apache.pekko.remote.artery.EnvelopeBufferPool.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of pekko-remote_2.12 Show documentation
Show all versions of pekko-remote_2.12 Show documentation
Apache Pekko is a toolkit for building highly concurrent, distributed, and resilient message-driven applications for Java and Scala.
The newest version!
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* license agreements; and to You under the Apache License, version 2.0:
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* This file is part of the Apache Pekko project, which was derived from Akka.
*/
/*
* Copyright (C) 2009-2022 Lightbend Inc.
*/
package org.apache.pekko.remote.artery
import java.nio.{ ByteBuffer, ByteOrder }
import org.agrona.concurrent.{ ManyToManyConcurrentArrayQueue, UnsafeBuffer }
import org.apache.pekko
import pekko.actor.ActorRef
import pekko.actor.InternalActorRef
import pekko.io.DirectByteBufferPool
import pekko.remote.artery.compress.{ CompressionTable, InboundCompressions, NoInboundCompressions }
import pekko.serialization.Serialization
import pekko.util.{ OptionVal, Unsafe }
/**
* INTERNAL API
*/
private[remote] class OutOfBuffersException extends RuntimeException("Out of usable ByteBuffers")
/**
* INTERNAL API
*/
private[remote] class EnvelopeBufferPool(maximumPayload: Int, maximumBuffers: Int) {
private val availableBuffers = new ManyToManyConcurrentArrayQueue[EnvelopeBuffer](maximumBuffers)
def acquire(): EnvelopeBuffer = {
val buf = availableBuffers.poll()
if (buf ne null) {
buf.byteBuffer.clear()
buf
} else {
val newBuf = new EnvelopeBuffer(ByteBuffer.allocateDirect(maximumPayload))
newBuf.byteBuffer.order(ByteOrder.LITTLE_ENDIAN)
newBuf
}
}
def release(buffer: EnvelopeBuffer) = {
// only reuse direct buffers, e.g. not those wrapping ByteString
if (buffer.byteBuffer.isDirect && !availableBuffers.offer(buffer)) buffer.tryCleanDirectByteBuffer()
}
}
/** INTERNAL API */
private[remote] final class ByteFlag(val mask: Byte) extends AnyVal {
def isEnabled(byteFlags: Byte): Boolean = (byteFlags.toInt & mask) != 0
override def toString = s"ByteFlag(${ByteFlag.binaryLeftPad(mask)})"
}
/**
* INTERNAL API
*/
private[remote] object ByteFlag {
def binaryLeftPad(byte: Byte): String = {
val string = Integer.toBinaryString(byte)
val pad = "0" * (8 - string.length) // leftPad
pad + string
}
}
/**
* INTERNAL API
*
* The strategy if the header format must be changed in an incompatible way is:
* - In the end we only want to support one header format, the latest, but during
* a rolling upgrade period we must support two versions in at least one Pekko patch
* release.
* - When supporting two version the outbound messages must still be encoded with old
* version. The Decoder on the receiving side must understand both versions.
* - Create a new copy of the header encoding/decoding logic (issue #24553: we should refactor to make that easier).
* - Bump `ArteryTransport.HighestVersion` and keep `ArterySettings.Version` as the old version.
* - Make sure `Decoder` picks the right parsing logic based on the version field in the incoming frame.
* - Release Akka, e.g. 2.5.13
* - Later, remove the old header parsing logic and bump the `ArterySettings.Version` to the same as
* `ArteryTransport.HighestVersion` again.
* - Release Akka, e.g. 2.5.14, and announce that all nodes in the cluster must first be on version
* 2.5.13 before upgrading to 2.5.14. That means that it is not supported to do a rolling upgrade
* from 2.5.12 directly to 2.5.14.
*/
private[remote] object EnvelopeBuffer {
val TagTypeMask = 0xFF000000
val TagValueMask = 0x0000FFFF
// Flags (1 byte allocated for them)
val MetadataPresentFlag = new ByteFlag(0x1)
val VersionOffset = 0 // Byte
val FlagsOffset = 1 // Byte
val ActorRefCompressionTableVersionOffset = 2 // Byte
val ClassManifestCompressionTableVersionOffset = 3 // Byte
val UidOffset = 4 // Long
val SerializerOffset = 12 // Int
val SenderActorRefTagOffset = 16 // Int
val RecipientActorRefTagOffset = 20 // Int
val ClassManifestTagOffset = 24 // Int
// EITHER metadata followed by literals directly OR literals directly in this spot.
// Mode depends on the `MetadataPresentFlag`.
val MetadataContainerAndLiteralSectionOffset = 28 // Int
}
/** INTERNAL API */
private[remote] object HeaderBuilder {
// We really only use the Header builder on one "side" or the other, thus in order to avoid having to split its impl
// we inject no-op compression's of the "other side".
def in(compression: InboundCompressions): HeaderBuilder =
new HeaderBuilderImpl(compression, CompressionTable.empty[ActorRef], CompressionTable.empty[String])
def out(): HeaderBuilder =
new HeaderBuilderImpl(NoInboundCompressions, CompressionTable.empty[ActorRef], CompressionTable.empty[String])
final val DeadLettersCode = -1
}
/**
* INTERNAL API
*/
private[remote] sealed trait HeaderBuilder {
def setVersion(v: Byte): Unit
def version: Byte
def setFlags(v: Byte): Unit
def flags: Byte
def flag(byteFlag: ByteFlag): Boolean
def setFlag(byteFlag: ByteFlag): Unit
def clearFlag(byteFlag: ByteFlag): Unit
def inboundActorRefCompressionTableVersion: Byte
def inboundClassManifestCompressionTableVersion: Byte
def useOutboundCompression(on: Boolean): Unit
def outboundActorRefCompression: CompressionTable[ActorRef]
def setOutboundActorRefCompression(table: CompressionTable[ActorRef]): Unit
def outboundClassManifestCompression: CompressionTable[String]
def setOutboundClassManifestCompression(table: CompressionTable[String]): Unit
def setUid(u: Long): Unit
def uid: Long
def setSenderActorRef(ref: ActorRef): Unit
/**
* Retrive the compressed ActorRef by the compressionId carried by this header.
* Returns `None` if ActorRef was not compressed, and then the literal [[senderActorRefPath]] should be used.
*/
def senderActorRef(originUid: Long): OptionVal[ActorRef]
/**
* Retrive the raw literal actor path, instead of using the compressed value.
* Returns `None` if ActorRef was compressed (!). To obtain the path in such case call [[senderActorRef]] and extract the path from it directly.
*/
def senderActorRefPath: OptionVal[String]
def setNoSender(): Unit
def isNoSender: Boolean
def setNoRecipient(): Unit
def isNoRecipient: Boolean
def setRecipientActorRef(ref: ActorRef): Unit
/**
* Retrive the compressed ActorRef by the compressionId carried by this header.
* Returns `None` if ActorRef was not compressed, and then the literal [[recipientActorRefPath]] should be used.
*/
def recipientActorRef(originUid: Long): OptionVal[ActorRef]
/**
* Retrive the raw literal actor path, instead of using the compressed value.
* Returns `None` if ActorRef was compressed (!). To obtain the path in such case call [[recipientActorRefPath]] and extract the path from it directly.
*/
def recipientActorRefPath: OptionVal[String]
def setSerializer(serializer: Int): Unit
def serializer: Int
def setManifest(manifest: String): Unit
def manifest(originUid: Long): OptionVal[String]
def setRemoteInstruments(instruments: RemoteInstruments): Unit
/**
* Reset all fields that are related to an outbound message,
* i.e. Encoder calls this as the first thing in onPush.
*/
def resetMessageFields(): Unit
}
/**
* Cache of the serialized path for the most used actor refs, performance critical as applied once for every
* outgoing message. Depends on the Serialization#withTransportInformation thread local to be set for serialization.
*
* INTERNAL API
*/
private[remote] final class SerializationFormatCache
extends LruBoundedCache[ActorRef, String](capacity = 1024, evictAgeThreshold = 600) {
override protected def compute(ref: ActorRef): String = Serialization.serializedActorPath(ref)
// Not calling ref.hashCode since it does a path.hashCode if ActorCell.undefinedUid is encountered.
// Refs with ActorCell.undefinedUid will now collide all the time, but this is not a usual scenario anyway.
override protected def hash(ref: ActorRef): Int = ref.path.uid
override protected def isKeyCacheable(k: ActorRef): Boolean =
// "temp" only for one request-response interaction so don't cache
!InternalActorRef.isTemporaryRef(k)
override protected def isCacheable(v: String): Boolean = true
}
/**
* INTERNAL API
*/
private[remote] final class HeaderBuilderImpl(
inboundCompression: InboundCompressions,
var _outboundActorRefCompression: CompressionTable[ActorRef],
var _outboundClassManifestCompression: CompressionTable[String])
extends HeaderBuilder {
import HeaderBuilder.DeadLettersCode
private[this] val toSerializationFormat: SerializationFormatCache = new SerializationFormatCache
// Fields only available for EnvelopeBuffer
var _version: Byte = 0
var _flags: Byte = 0
var _uid: Long = 0
var _inboundActorRefCompressionTableVersion: Byte = 0
var _inboundClassManifestCompressionTableVersion: Byte = 0
var _useOutboundCompression: Boolean = true
var _senderActorRef: String = null
var _senderActorRefIdx: Int = -1
var _recipientActorRef: String = null
var _recipientActorRefIdx: Int = -1
var _serializer: Int = 0
var _manifest: String = null
var _manifestIdx: Int = -1
var _remoteInstruments: OptionVal[RemoteInstruments] = OptionVal.None
override def resetMessageFields(): Unit = {
// some fields must not be reset because they are set only once from the Encoder,
// which owns the HeaderBuilder instance. Those are never changed.
// version, uid, streamId
_flags = 0
_senderActorRef = null
_senderActorRefIdx = -1
_recipientActorRef = null
_recipientActorRefIdx = -1
_serializer = 0
_manifest = null
_manifestIdx = -1
_remoteInstruments = OptionVal.None
}
override def setVersion(v: Byte) = _version = v
override def version = _version
override def setFlags(v: Byte) = _flags = v
override def flags = _flags
override def flag(byteFlag: ByteFlag): Boolean = (_flags.toInt & byteFlag.mask) != 0
override def setFlag(byteFlag: ByteFlag): Unit =
_flags = (flags | byteFlag.mask).toByte
override def clearFlag(byteFlag: ByteFlag): Unit =
_flags = (flags & ~byteFlag.mask).toByte
override def setUid(uid: Long) = _uid = uid
override def uid: Long = _uid
override def inboundActorRefCompressionTableVersion: Byte = _inboundActorRefCompressionTableVersion
override def inboundClassManifestCompressionTableVersion: Byte = _inboundClassManifestCompressionTableVersion
def useOutboundCompression(on: Boolean): Unit =
_useOutboundCompression = on
def setOutboundActorRefCompression(table: CompressionTable[ActorRef]): Unit = {
_outboundActorRefCompression = table
}
override def outboundActorRefCompression: CompressionTable[ActorRef] = _outboundActorRefCompression
def setOutboundClassManifestCompression(table: CompressionTable[String]): Unit = {
_outboundClassManifestCompression = table
}
def outboundClassManifestCompression: CompressionTable[String] = _outboundClassManifestCompression
/**
* Note that Serialization.currentTransportInformation must be set when calling this method,
* because it's using `Serialization.serializedActorPath`
*/
override def setSenderActorRef(ref: ActorRef): Unit = {
if (_useOutboundCompression) {
_senderActorRefIdx = outboundActorRefCompression.compress(ref)
if (_senderActorRefIdx == -1) _senderActorRef = Serialization.serializedActorPath(ref)
} else
_senderActorRef = Serialization.serializedActorPath(ref)
}
override def setNoSender(): Unit = {
_senderActorRef = null
_senderActorRefIdx = DeadLettersCode
}
override def isNoSender: Boolean =
(_senderActorRef eq null) && _senderActorRefIdx == DeadLettersCode
override def senderActorRef(originUid: Long): OptionVal[ActorRef] = {
// we treat deadLetters as always present, but not included in table
if ((_senderActorRef eq null) && !isNoSender)
inboundCompression.decompressActorRef(originUid, inboundActorRefCompressionTableVersion, _senderActorRefIdx)
else OptionVal.None
}
def senderActorRefPath: OptionVal[String] =
OptionVal(_senderActorRef)
def setNoRecipient(): Unit = {
_recipientActorRef = null
_recipientActorRefIdx = DeadLettersCode
}
def isNoRecipient: Boolean =
(_recipientActorRef eq null) && _recipientActorRefIdx == DeadLettersCode
/**
* Note that Serialization.currentTransportInformation must be set when calling this method,
* because it's using `Serialization.serializedActorPath`
*/
def setRecipientActorRef(ref: ActorRef): Unit = {
if (_useOutboundCompression) {
_recipientActorRefIdx = outboundActorRefCompression.compress(ref)
if (_recipientActorRefIdx == -1) _recipientActorRef = toSerializationFormat.getOrCompute(ref)
} else
_recipientActorRef = toSerializationFormat.getOrCompute(ref)
}
def recipientActorRef(originUid: Long): OptionVal[ActorRef] = {
// we treat deadLetters as always present, but not included in table
if ((_recipientActorRef eq null) && !isNoRecipient)
inboundCompression.decompressActorRef(originUid, inboundActorRefCompressionTableVersion, _recipientActorRefIdx)
else OptionVal.None
}
def recipientActorRefPath: OptionVal[String] =
OptionVal(_recipientActorRef)
override def setSerializer(serializer: Int): Unit = {
_serializer = serializer
}
override def serializer: Int =
_serializer
override def setManifest(manifest: String): Unit = {
if (_useOutboundCompression) {
_manifestIdx = outboundClassManifestCompression.compress(manifest)
if (_manifestIdx == -1) _manifest = manifest
} else
_manifest = manifest
}
override def manifest(originUid: Long): OptionVal[String] = {
if (_manifest ne null) OptionVal.Some(_manifest)
else {
inboundCompression.decompressClassManifest(originUid, inboundClassManifestCompressionTableVersion, _manifestIdx)
}
}
override def setRemoteInstruments(instruments: RemoteInstruments): Unit = {
_remoteInstruments = OptionVal(instruments)
}
override def toString =
"HeaderBuilderImpl(" +
"version:" + version + ", " +
"flags:" + ByteFlag.binaryLeftPad(flags) + ", " +
"UID:" + uid + ", " +
"_senderActorRef:" + _senderActorRef + ", " +
"_senderActorRefIdx:" + _senderActorRefIdx + ", " +
"_recipientActorRef:" + _recipientActorRef + ", " +
"_recipientActorRefIdx:" + _recipientActorRefIdx + ", " +
"_serializer:" + _serializer + ", " +
"_manifest:" + _manifest + ", " +
"_manifestIdx:" + _manifestIdx + ")"
}
/**
* INTERNAL API
*/
private[remote] final class EnvelopeBuffer(val byteBuffer: ByteBuffer) {
import EnvelopeBuffer._
val aeronBuffer = new UnsafeBuffer(byteBuffer)
private var literalChars = new Array[Char](64)
private var literalBytes = new Array[Byte](64)
// The streamId is only used for TCP transport. It is not part of the ordinary envelope header, but included in the
// frame header that is parsed by the TcpFraming stage.
private var _streamId: Int = -1
def streamId: Int =
if (_streamId != -1) _streamId
else throw new IllegalStateException("StreamId was not set")
def setStreamId(newStreamId: Int): Unit = _streamId = newStreamId
def writeHeader(h: HeaderBuilder): Unit = writeHeader(h, null)
def writeHeader(h: HeaderBuilder, oe: OutboundEnvelope): Unit = {
val header = h.asInstanceOf[HeaderBuilderImpl]
byteBuffer.clear()
// Write fixed length parts
byteBuffer.put(VersionOffset, header.version)
byteBuffer.put(FlagsOffset, header.flags)
// compression table version numbers
byteBuffer.put(ActorRefCompressionTableVersionOffset, header.outboundActorRefCompression.version)
byteBuffer.put(ClassManifestCompressionTableVersionOffset, header.outboundClassManifestCompression.version)
byteBuffer.putLong(UidOffset, header.uid)
byteBuffer.putInt(SerializerOffset, header.serializer)
// maybe write some metadata
// after metadata is written (or not), buffer is at correct position to continue writing literals
byteBuffer.position(MetadataContainerAndLiteralSectionOffset)
if (header._remoteInstruments.isDefined) {
header._remoteInstruments.get.serialize(OptionVal(oe), byteBuffer)
if (byteBuffer.position() != MetadataContainerAndLiteralSectionOffset) {
// we actually wrote some metadata so update the flag field to reflect that
header.setFlag(MetadataPresentFlag)
byteBuffer.put(FlagsOffset, header.flags)
}
}
// Serialize sender
if (header._senderActorRefIdx != -1)
byteBuffer.putInt(SenderActorRefTagOffset, header._senderActorRefIdx | TagTypeMask)
else
writeLiteral(SenderActorRefTagOffset, header._senderActorRef)
// Serialize recipient
if (header._recipientActorRefIdx != -1)
byteBuffer.putInt(RecipientActorRefTagOffset, header._recipientActorRefIdx | TagTypeMask)
else
writeLiteral(RecipientActorRefTagOffset, header._recipientActorRef)
// Serialize class manifest
if (header._manifestIdx != -1)
byteBuffer.putInt(ClassManifestTagOffset, header._manifestIdx | TagTypeMask)
else
writeLiteral(ClassManifestTagOffset, header._manifest)
}
def parseHeader(h: HeaderBuilder): Unit = {
val header = h.asInstanceOf[HeaderBuilderImpl]
// Read fixed length parts
header.setVersion(byteBuffer.get(VersionOffset))
if (header.version > ArteryTransport.HighestVersion)
throw new IllegalArgumentException(
s"Incompatible protocol version [${header.version}], " +
s"highest known version for this node is [${ArteryTransport.HighestVersion}]")
header.setFlags(byteBuffer.get(FlagsOffset))
// compression table versions (stored in the Tag)
header._inboundActorRefCompressionTableVersion = byteBuffer.get(ActorRefCompressionTableVersionOffset)
header._inboundClassManifestCompressionTableVersion = byteBuffer.get(ClassManifestCompressionTableVersionOffset)
header.setUid(byteBuffer.getLong(UidOffset))
header.setSerializer(byteBuffer.getInt(SerializerOffset))
byteBuffer.position(MetadataContainerAndLiteralSectionOffset)
if (header.flag(MetadataPresentFlag)) {
// metadata present, so we need to fast forward to the literals that start right after
val totalMetadataLength = byteBuffer.getInt()
byteBuffer.position(byteBuffer.position() + totalMetadataLength)
}
// Deserialize sender
val senderTag = byteBuffer.getInt(SenderActorRefTagOffset)
if ((senderTag & TagTypeMask) != 0) {
val idx = senderTag & TagValueMask
header._senderActorRef = null
header._senderActorRefIdx = idx
} else {
header._senderActorRef = emptyAsNull(readLiteral())
}
// Deserialize recipient
val recipientTag = byteBuffer.getInt(RecipientActorRefTagOffset)
if ((recipientTag & TagTypeMask) != 0) {
val idx = recipientTag & TagValueMask
header._recipientActorRef = null
header._recipientActorRefIdx = idx
} else {
header._recipientActorRef = emptyAsNull(readLiteral())
}
// Deserialize class manifest
val manifestTag = byteBuffer.getInt(ClassManifestTagOffset)
if ((manifestTag & TagTypeMask) != 0) {
val idx = manifestTag & TagValueMask
header._manifest = null
header._manifestIdx = idx
} else {
header._manifest = readLiteral()
}
}
private def emptyAsNull(s: String): String =
if (s == "") null
else s
private def readLiteral(): String = {
// Up-cast to Int to avoid up-casting 4 times.
val length = byteBuffer.getShort.asInstanceOf[Int]
if (length == 0) ""
else {
ensureLiteralCharsLength(length)
val chars = literalChars
val bytes = literalBytes
byteBuffer.get(bytes, 0, length)
var i = 0
while (i < length) {
// UsAscii
chars(i) = bytes(i).asInstanceOf[Char]
i += 1
}
String.valueOf(chars, 0, length)
}
}
private def writeLiteral(tagOffset: Int, literal: String): Unit = {
val length = if (literal eq null) 0 else literal.length
if (length > 65535)
throw new IllegalArgumentException("Literals longer than 65535 cannot be encoded in the envelope")
byteBuffer.putInt(tagOffset, byteBuffer.position())
byteBuffer.putShort(length.toShort)
if (length > 0) {
ensureLiteralCharsLength(length)
val bytes = literalBytes
Unsafe.copyUSAsciiStrToBytes(literal, bytes)
byteBuffer.put(bytes, 0, length)
}
}
private def ensureLiteralCharsLength(length: Int): Unit = {
if (length > literalChars.length) {
literalChars = new Array[Char](length)
literalBytes = new Array[Byte](length)
}
}
def tryCleanDirectByteBuffer(): Unit = DirectByteBufferPool.tryCleanDirectByteBuffer(byteBuffer)
def copy(): EnvelopeBuffer = {
val p = byteBuffer.position()
byteBuffer.rewind()
val bytes = new Array[Byte](byteBuffer.remaining)
byteBuffer.get(bytes)
val newByteBuffer = ByteBuffer.wrap(bytes).order(ByteOrder.LITTLE_ENDIAN)
newByteBuffer.position(p)
byteBuffer.position(p)
new EnvelopeBuffer(newByteBuffer)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy