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

io.aeron.logbuffer.TermUnblocker Maven / Gradle / Ivy

There is a newer version: 1.46.7
Show newest version
/*
 * Copyright 2014-2019 Real Logic Ltd.
 *
 * 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.logbuffer;

import org.agrona.concurrent.UnsafeBuffer;

import static io.aeron.logbuffer.FrameDescriptor.*;
import static io.aeron.logbuffer.LogBufferDescriptor.applyDefaultHeader;
import static io.aeron.logbuffer.TermUnblocker.Status.*;
import static io.aeron.protocol.HeaderFlyweight.HDR_TYPE_PAD;

/**
 * Unblocks a term buffer if a publisher has died leaving the log with a partial log entry.
 */
public class TermUnblocker
{
    public enum Status
    {
        /**
         * No action has been taken during operation.
         */
        NO_ACTION,

        /**
         * The term has been unblocked so that the log can progress.
         */
        UNBLOCKED,

        /**
         * The term has been unblocked from the offset until the end of the term.
         */
        UNBLOCKED_TO_END,
    }

    /**
     * Attempt to unblock the current term at the current offset.
     * 
    *
  1. Current position length is > 0, then return
  2. *
  3. Current position length is 0, scan forward by frame alignment until, one of the following: *
      *
    1. reach a non-0 length, unblock up to indicated position (check original frame length for non-0)
    2. *
    3. reach end of term and tail position >= end of term, unblock up to end of term (check original * frame length for non-0) *
    4. *
    5. reach tail position < end of term, do NOT unblock
    6. *
    *
  4. *
* * @param logMetaDataBuffer containing the default headers * @param termBuffer to unblock * @param blockedOffset to unblock at * @param tailOffset to unblock up to * @param termId for the current term. * @return whether unblocking was done, not done, or applied to end of term */ public static Status unblock( final UnsafeBuffer logMetaDataBuffer, final UnsafeBuffer termBuffer, final int blockedOffset, final int tailOffset, final int termId) { Status status = NO_ACTION; int frameLength = frameLengthVolatile(termBuffer, blockedOffset); if (frameLength < 0) { resetHeader(logMetaDataBuffer, termBuffer, blockedOffset, termId, -frameLength); status = UNBLOCKED; } else if (0 == frameLength) { int currentOffset = blockedOffset + FRAME_ALIGNMENT; while (currentOffset < tailOffset) { frameLength = frameLengthVolatile(termBuffer, currentOffset); if (frameLength != 0) { if (scanBackToConfirmZeroed(termBuffer, currentOffset, blockedOffset)) { final int length = currentOffset - blockedOffset; resetHeader(logMetaDataBuffer, termBuffer, blockedOffset, termId, length); status = UNBLOCKED; } break; } currentOffset += FRAME_ALIGNMENT; } if (currentOffset == termBuffer.capacity()) { if (0 == frameLengthVolatile(termBuffer, blockedOffset)) { final int length = currentOffset - blockedOffset; resetHeader(logMetaDataBuffer, termBuffer, blockedOffset, termId, length); status = UNBLOCKED_TO_END; } } } return status; } private static void resetHeader( final UnsafeBuffer logMetaDataBuffer, final UnsafeBuffer termBuffer, final int termOffset, final int termId, final int frameLength) { applyDefaultHeader(logMetaDataBuffer, termBuffer, termOffset); frameType(termBuffer, termOffset, HDR_TYPE_PAD); frameTermOffset(termBuffer, termOffset); frameTermId(termBuffer, termOffset, termId); frameLengthOrdered(termBuffer, termOffset, frameLength); } private static boolean scanBackToConfirmZeroed(final UnsafeBuffer buffer, final int from, final int limit) { int i = from - FRAME_ALIGNMENT; boolean allZeros = true; while (i >= limit) { if (0 != frameLengthVolatile(buffer, i)) { allZeros = false; break; } i -= FRAME_ALIGNMENT; } return allZeros; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy