
io.aeron.driver.media.DataTransportPoller Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of aeron-driver Show documentation
Show all versions of aeron-driver Show documentation
Efficient reliable UDP unicast, UDP multicast, and IPC transport protocol.
The newest version!
/*
* Copyright 2014-2025 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.driver.media;
import io.aeron.driver.Configuration;
import io.aeron.driver.DriverConductorProxy;
import io.aeron.protocol.DataHeaderFlyweight;
import io.aeron.protocol.RttMeasurementFlyweight;
import io.aeron.protocol.SetupFlyweight;
import org.agrona.BufferUtil;
import org.agrona.CloseHelper;
import org.agrona.ErrorHandler;
import org.agrona.LangUtil;
import org.agrona.collections.ArrayUtil;
import org.agrona.concurrent.UnsafeBuffer;
import org.agrona.nio.TransportPoller;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.ClosedChannelException;
import java.nio.channels.SelectionKey;
import java.util.function.Consumer;
import static io.aeron.logbuffer.FrameDescriptor.frameType;
import static io.aeron.protocol.HeaderFlyweight.*;
import static org.agrona.BitUtil.CACHE_LINE_LENGTH;
/**
* Encapsulates the polling of data {@link UdpChannelTransport}s using whatever means provides the lowest latency.
*/
public final class DataTransportPoller extends UdpTransportPoller
{
private static final ChannelAndTransport[] EMPTY_TRANSPORTS = new ChannelAndTransport[0];
private final ByteBuffer byteBuffer = BufferUtil.allocateDirectAligned(
Configuration.MAX_UDP_PAYLOAD_LENGTH, CACHE_LINE_LENGTH);
private final UnsafeBuffer unsafeBuffer = new UnsafeBuffer(byteBuffer);
private final DataHeaderFlyweight dataMessage = new DataHeaderFlyweight(unsafeBuffer);
private final SetupFlyweight setupMessage = new SetupFlyweight(unsafeBuffer);
private final RttMeasurementFlyweight rttMeasurement = new RttMeasurementFlyweight(unsafeBuffer);
private final Consumer selectorPoller =
(selectionKey) -> poll((ChannelAndTransport)selectionKey.attachment());
private ChannelAndTransport[] channelAndTransports = EMPTY_TRANSPORTS;
private int totalBytesReceived;
/**
* Construct a new {@link TransportPoller} with an {@link ErrorHandler} for logging.
*
* @param errorHandler which can be used to log errors and continue.
*/
public DataTransportPoller(final ErrorHandler errorHandler)
{
super(errorHandler);
}
/**
* {@inheritDoc}
*/
public void close()
{
for (final ChannelAndTransport transport : channelAndTransports)
{
final ReceiveChannelEndpoint receiveChannelEndpoint = transport.channelEndpoint;
receiveChannelEndpoint.closeMultiRcvDestinationTransports(this);
CloseHelper.close(errorHandler, receiveChannelEndpoint);
}
super.close();
}
/**
* {@inheritDoc}
*/
public int pollTransports()
{
totalBytesReceived = 0;
if (channelAndTransports.length <= ITERATION_THRESHOLD)
{
for (final ChannelAndTransport channelAndTransport : channelAndTransports)
{
poll(channelAndTransport);
}
}
else
{
try
{
selector.selectNow(selectorPoller);
}
catch (final IOException ex)
{
errorHandler.onError(ex);
}
}
return totalBytesReceived;
}
/**
* {@inheritDoc}
*/
public SelectionKey registerForRead(final UdpChannelTransport transport)
{
return registerForRead((ReceiveChannelEndpoint)transport, transport, 0);
}
/**
* Register transport for reading with the poller.
*
* @param channelEndpoint to which the transport belongs.
* @param transport new transport to be registered.
* @param transportIndex for the transport in the channel.
* @return {@link SelectionKey} for registration to cancel.
*/
public SelectionKey registerForRead(
final ReceiveChannelEndpoint channelEndpoint, final UdpChannelTransport transport, final int transportIndex)
{
SelectionKey key = null;
try
{
final ChannelAndTransport channelAndTransport = new ChannelAndTransport(
channelEndpoint, transport, transportIndex);
key = transport.receiveDatagramChannel().register(selector, SelectionKey.OP_READ, channelAndTransport);
channelAndTransports = ArrayUtil.add(channelAndTransports, channelAndTransport);
}
catch (final ClosedChannelException ex)
{
LangUtil.rethrowUnchecked(ex);
}
return key;
}
/**
* {@inheritDoc}
*/
public void cancelRead(final UdpChannelTransport transport)
{
cancelRead((ReceiveChannelEndpoint)transport, transport);
}
/**
* Cancel the reading of a given transport.
*
* @param channelEndpoint to which the transport belongs.
* @param transport transport which was previously registered.
*/
public void cancelRead(final ReceiveChannelEndpoint channelEndpoint, final UdpChannelTransport transport)
{
final ChannelAndTransport[] transports = channelAndTransports;
int index = ArrayUtil.UNKNOWN_INDEX;
for (int i = 0, length = transports.length; i < length; i++)
{
if (channelEndpoint == transports[i].channelEndpoint && transport == transports[i].transport)
{
index = i;
break;
}
}
if (index != ArrayUtil.UNKNOWN_INDEX)
{
channelAndTransports = 1 == transports.length ? EMPTY_TRANSPORTS : ArrayUtil.remove(transports, index);
}
}
/**
* Check if any of the registered channels or transports require re-resolution.
*
* @param nowNs as the current time.
* @param conductorProxy for sending re-resolution requests.
*/
public void checkForReResolutions(final long nowNs, final DriverConductorProxy conductorProxy)
{
for (final ChannelAndTransport channelAndTransport : channelAndTransports)
{
channelAndTransport.channelEndpoint.checkForReResolution(nowNs, conductorProxy);
}
}
private void poll(final ChannelAndTransport channelAndTransport)
{
try
{
receive(channelAndTransport);
}
catch (final Exception ex)
{
errorHandler.onError(ex);
}
}
private void receive(final ChannelAndTransport channelAndTransport)
{
final InetSocketAddress srcAddress = channelAndTransport.transport.receive(byteBuffer);
if (null != srcAddress)
{
final int length = byteBuffer.position();
totalBytesReceived += length;
final ReceiveChannelEndpoint channelEndpoint = channelAndTransport.channelEndpoint;
if (channelEndpoint.isValidFrame(unsafeBuffer, length))
{
channelEndpoint.receiveHook(unsafeBuffer, length, srcAddress);
final int frameType = frameType(unsafeBuffer, 0);
if (HDR_TYPE_DATA == frameType || HDR_TYPE_PAD == frameType)
{
channelEndpoint.onDataPacket(
dataMessage, unsafeBuffer, length, srcAddress, channelAndTransport.transportIndex);
}
else if (HDR_TYPE_SETUP == frameType)
{
channelEndpoint.onSetupMessage(
setupMessage, unsafeBuffer, length, srcAddress, channelAndTransport.transportIndex);
}
else if (HDR_TYPE_RTTM == frameType)
{
channelEndpoint.onRttMeasurement(
rttMeasurement, unsafeBuffer, length, srcAddress, channelAndTransport.transportIndex);
}
}
}
}
/**
* {@inheritDoc}
*/
public String toString()
{
return "DataTransportPoller{}";
}
static class ChannelAndTransport
{
final ReceiveChannelEndpoint channelEndpoint;
final UdpChannelTransport transport;
final int transportIndex;
ChannelAndTransport(
final ReceiveChannelEndpoint channelEndpoint, final UdpChannelTransport transport, final int transportIndex)
{
this.channelEndpoint = channelEndpoint;
this.transport = transport;
this.transportIndex = transportIndex;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy