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

org.jitsi.impl.neomedia.codec.AbstractCodec2 Maven / Gradle / Ivy

Go to download

libjitsi is an advanced Java media library for secure real-time audio/video communication

The newest version!
/*
 * Copyright @ 2015 Atlassian Pty 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
 *
 *     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 org.jitsi.impl.neomedia.codec;

import java.awt.*;

import javax.media.*;
import javax.media.format.*;

import net.sf.fmj.media.*;

/**
 * Extends FMJ's AbstractCodec to make it even easier to implement a
 * Codec.
 *
 * @author Lyubomir Marinov
 */
public abstract class AbstractCodec2
    extends AbstractCodec
{
    /**
     * The Buffer flag which indicates that the respective
     * Buffer contains audio data which has been decoded as a result of
     * the operation of FEC.
     */
    public static final int BUFFER_FLAG_FEC = (1 << 24);

    /**
     * The Buffer flag which indicates that the respective
     * Buffer contains audio data which has been decoded as a result of
     * the operation of PLC.
     */
    public static final int BUFFER_FLAG_PLC = (1 << 25);

    /**
     * An empty array of Format element type. Explicitly defined to
     * reduce unnecessary allocations.
     */
    public static final Format[] EMPTY_FORMATS = new Format[0];

    /**
     * The maximum number of lost sequence numbers to conceal with packet loss
     * mitigation techniques such as Forward Error Correction (FEC) and Packet
     * Loss Concealment (PLC) when dealing with audio.
     */
    public static final int MAX_AUDIO_SEQUENCE_NUMBERS_TO_PLC = 3;

    /**
     * The maximum (RTP) sequence number value.
     */
    public static final int SEQUENCE_MAX = 65535;

    /**
     * The minimum (RTP) sequence number value.
     */
    public static final int SEQUENCE_MIN = 0;

    /**
     * Calculates the number of sequences which have been lost i.e. which have
     * not been received.
     *
     * @param lastSeqNo the last received sequence number (prior to the current
     * sequence number represented by seqNo.) May be
     * {@link Buffer#SEQUENCE_UNKNOWN}. May be equal to seqNo for the
     * purposes of Codec implementations which repeatedly process one and the
     * same input Buffer multiple times.
     * @param seqNo the current sequence number. May be equal to
     * lastSeqNo for the purposes of Codec implementations which
     * repeatedly process one and the same input Buffer multiple times.
     * @return the number of sequences (between lastSeqNo and
     * seqNo) which have been lost i.e. which have not been received
     */
    public static int calculateLostSeqNoCount(long lastSeqNo, long seqNo)
    {
        if (lastSeqNo == Buffer.SEQUENCE_UNKNOWN)
            return 0;

        int delta = (int) (seqNo - lastSeqNo);

        /*
         * We explicitly allow the same sequence number to be received multiple
         * times for the purposes of Codec implementations which repeatedly
         * process one and the same input Buffer multiple times.
         */
        if (delta == 0)
            return 0;
        else if (delta > 0)
            return delta - 1; // The sequence number has not wrapped yet.
        else
            return delta + SEQUENCE_MAX; // The sequence number has wrapped.
    }

    /**
     * Increments a specific sequence number and makes sure that the result
     * stays within the range of valid RTP sequence number values.
     *
     * @param seqNo the sequence number to increment
     * @return a sequence number which represents an increment over the
     * specified seqNo within the range of valid RTP sequence number
     * values
     */
    public static long incrementSeqNo(long seqNo)
    {
        seqNo++;
        if (seqNo > SEQUENCE_MAX)
            seqNo = SEQUENCE_MIN;
        return seqNo;
    }

    /**
     * Utility to perform format matching.
     *
     * @param in input format
     * @param outs array of output formats
     * @return the first output format that is supported
     */
    public static Format matches(Format in, Format outs[])
    {
        for (Format out : outs)
            if (in.matches(out))
                return out;
        return null;
    }

    public static YUVFormat specialize(YUVFormat yuvFormat, Class dataType)
    {
        Dimension size = yuvFormat.getSize();
        int strideY = yuvFormat.getStrideY();

        if ((strideY == Format.NOT_SPECIFIED) && (size != null))
            strideY = size.width;

        int strideUV = yuvFormat.getStrideUV();

        if ((strideUV == Format.NOT_SPECIFIED)
                && (strideY != Format.NOT_SPECIFIED))
            strideUV = (strideY + 1) / 2;

        int offsetY = yuvFormat.getOffsetY();

        if (offsetY == Format.NOT_SPECIFIED)
            offsetY = 0;

        int offsetU = yuvFormat.getOffsetU();

        if ((offsetU == Format.NOT_SPECIFIED)
                && (strideY != Format.NOT_SPECIFIED)
                && (size != null))
            offsetU = offsetY + strideY * size.height;

        int offsetV = yuvFormat.getOffsetV();

        if ((offsetV == Format.NOT_SPECIFIED)
                && (offsetU != Format.NOT_SPECIFIED)
                && (strideUV != Format.NOT_SPECIFIED)
                && (size != null))
            offsetV = offsetU + strideUV * ((size.height + 1) / 2);

        int maxDataLength
            = ((strideY != Format.NOT_SPECIFIED)
                    && (strideUV != Format.NOT_SPECIFIED))
                    && (size != null)
                ? (strideY * size.height
                        + 2 * strideUV * ((size.height + 1) / 2)
                        + FFmpeg.FF_INPUT_BUFFER_PADDING_SIZE)
                : Format.NOT_SPECIFIED;

        return
            new YUVFormat(
                    size,
                    maxDataLength,
                    (dataType == null) ? yuvFormat.getDataType() : dataType,
                    yuvFormat.getFrameRate(),
                    YUVFormat.YUV_420,
                    strideY, strideUV,
                    offsetY, offsetU, offsetV);
    }

    /**
     * Ensures that the value of the data property of a specific
     * Buffer is an array of bytes whose length is at least a
     * specific number of bytes.
     *
     * @param buffer the Buffer whose data property value is
     * to be validated
     * @param newSize the minimum length of the array of byte which is
     * to be the value of the data property of buffer
     * @param arraycopy true to copy the bytes which are in the
     * value of the data property of buffer at the time of the
     * invocation of the method if the value of the data property of
     * buffer is an array of byte whose length is less than
     * newSize; otherwise, false
     * @return an array of bytes which is the value of the
     * data property of buffer and whose length is at least
     * newSize number of bytes
     */
    public static byte[] validateByteArraySize(
            Buffer buffer,
            int newSize,
            boolean arraycopy)
    {
        Object data = buffer.getData();
        byte[] newBytes;

        if (data instanceof byte[])
        {
            byte[] bytes = (byte[]) data;

            if (bytes.length < newSize)
            {
                newBytes = new byte[newSize];
                buffer.setData(newBytes);
                if (arraycopy)
                {
                    System.arraycopy(bytes, 0, newBytes, 0, bytes.length);
                }
                else
                {
                    buffer.setLength(0);
                    buffer.setOffset(0);
                }
            }
            else
            {
                newBytes = bytes;
            }
        }
        else
        {
            newBytes = new byte[newSize];
            buffer.setData(newBytes);
            buffer.setLength(0);
            buffer.setOffset(0);
        }
        return newBytes;
    }

    /**
     * The bitmap/flag mask of optional features supported by this
     * AbstractCodec2 such as {@link #BUFFER_FLAG_FEC} and
     * {@link #BUFFER_FLAG_PLC}.
     */
    protected int features;

    private final Class formatClass;

    /**
     * The total input length processed by all invocations of
     * {@link #process(Buffer,Buffer)}. Introduced for the purposes of debugging
     * at the time of this writing.
     */
    private long inLenProcessed;

    /**
     * The name of this PlugIn.
     */
    private final String name;

    /**
     * The total output length processed by all invocations of
     * {@link #process(Buffer,Buffer)}. Introduced for the purposes of debugging
     * at the time of this writing.
     */
    private long outLenProcessed;

    private final Format[] supportedOutputFormats;

    /**
     * Initializes a new AbstractCodec2 instance with a specific
     * PlugIn name, a specific Class of input and output
     * Formats and a specific list of Formats supported as
     * output.
     *
     * @param name the PlugIn name of the new instance
     * @param formatClass the Class of input and output
     * Formats supported by the new instance
     * @param supportedOutputFormats the list of Formats supported by
     * the new instance as output
     */
    protected AbstractCodec2(
            String name,
            Class formatClass,
            Format[] supportedOutputFormats)
    {
        this.formatClass = formatClass;
        this.name = name;
        this.supportedOutputFormats = supportedOutputFormats;

        /*
         * An Effect is a Codec that does not modify the Format of the data, it
         * modifies the contents.
         */
        if (this instanceof Effect)
            inputFormats = this.supportedOutputFormats;
    }

    @Override
    public void close()
    {
        if (!opened)
            return;

        doClose();

        opened = false;
        super.close();
    }

    protected void discardOutputBuffer(Buffer outputBuffer)
    {
        outputBuffer.setDiscard(true);
    }

    protected abstract void doClose();

    /**
     * Opens this Codec and acquires the resources that it needs to
     * operate. A call to {@link PlugIn#open()} on this instance will result in
     * a call to doOpen only if {@link AbstractCodec#opened} is
     * false. All required input and/or output formats are assumed to
     * have been set on this Codec before doOpen is called.
     *
     * @throws ResourceUnavailableException if any of the resources that this
     * Codec needs to operate cannot be acquired
     */
    protected abstract void doOpen()
        throws ResourceUnavailableException;

    protected abstract int doProcess(Buffer inBuf, Buffer outBuf);

    /**
     * Gets the Formats which are supported by this Codec as
     * output when the input is in a specific Format.
     *
     * @param inputFormat the Format of the input for which the
     * supported output Formats are to be returned
     * @return an array of Formats supported by this Codec as
     * output when the input is in the specified inputFormat
     */
    protected Format[] getMatchingOutputFormats(Format inputFormat)
    {
        /*
         * An Effect is a Codec that does not modify the Format of the data, it
         * modifies the contents.
         */
        if (this instanceof Effect)
            return new Format[] { inputFormat };

        return
            (supportedOutputFormats == null)
                ? EMPTY_FORMATS
                : supportedOutputFormats.clone();
    }

    @Override
    public String getName()
    {
        return (name == null) ? super.getName() : name;
    }

    /**
     * Implements {@link AbstractCodec#getSupportedOutputFormats(Format)}.
     *
     * @param inputFormat input format
     * @return array of supported output format
     * @see AbstractCodec#getSupportedOutputFormats(Format)
     */
    @Override
    public Format[] getSupportedOutputFormats(Format inputFormat)
    {
        if (inputFormat == null)
            return supportedOutputFormats;

        if (!formatClass.isInstance(inputFormat)
                || (matches(inputFormat, inputFormats) == null))
            return EMPTY_FORMATS;

        return getMatchingOutputFormats(inputFormat);
    }

    /**
     * Opens this PlugIn software or hardware component and acquires
     * the resources that it needs to operate. All required input and/or output
     * formats have to be set on this PlugIn before open is
     * called. Buffers should not be passed into this PlugIn without
     * first calling open.
     *
     * @throws ResourceUnavailableException if any of the resources that this
     * PlugIn needs to operate cannot be acquired
     * @see AbstractPlugIn#open()
     */
    @Override
    public void open()
        throws ResourceUnavailableException
    {
        if (opened)
            return;

        doOpen();

        opened = true;
        super.open();
    }

    /**
     * Implements AbstractCodec#process(Buffer, Buffer).
     *
     * @param inBuf
     * @param outBuf
     * @return BUFFER_PROCESSED_OK if the specified inBuff was
     * successfully processed or BUFFER_PROCESSED_FAILED if the
     * specified was not successfully processed
     * @see AbstractCodec#process(Buffer, Buffer)
     */
    @Override
    public int process(Buffer inBuf, Buffer outBuf)
    {
        if (!checkInputBuffer(inBuf))
            return BUFFER_PROCESSED_FAILED;
        if (isEOM(inBuf))
        {
            propagateEOM(outBuf);
            return BUFFER_PROCESSED_OK;
        }
        if (inBuf.isDiscard())
        {
            discardOutputBuffer(outBuf);
            return BUFFER_PROCESSED_OK;
        }

        int process;
        int inLenProcessed = inBuf.getLength();

        // Buffer.FLAG_SILENCE is set only when the intention is to drop the
        // specified input Buffer but to note that it has not been lost. The
        // latter is usually necessary if this AbstractCodec2 does Forward Error
        // Correction (FEC) and/or Packet Loss Concealment (PLC) and may cause
        // noticeable artifacts otherwise.
        if ((((BUFFER_FLAG_FEC | BUFFER_FLAG_PLC) & features) == 0)
                && ((Buffer.FLAG_SILENCE & inBuf.getFlags()) != 0))
        {
            process = OUTPUT_BUFFER_NOT_FILLED;
        }
        else
        {
            process = doProcess(inBuf, outBuf);
        }

        // Keep track of additional information for the purposes of debugging.
        if ((process & INPUT_BUFFER_NOT_CONSUMED) != 0)
            inLenProcessed -= inBuf.getLength();
        if (inLenProcessed < 0)
            inLenProcessed = 0;

        int outLenProcessed;

        if (((process & BUFFER_PROCESSED_FAILED) != 0)
                || ((process & OUTPUT_BUFFER_NOT_FILLED)) != 0)
        {
            outLenProcessed = 0;
        }
        else
        {
            outLenProcessed = outBuf.getLength();
            if (outLenProcessed < 0)
                outLenProcessed = 0;
        }

        this.inLenProcessed += inLenProcessed;
        this.outLenProcessed += outLenProcessed;

        return process;
    }

    @Override
    public Format setInputFormat(Format format)
    {
        if (!formatClass.isInstance(format)
                || (matches(format, inputFormats) == null))
            return null;

        return super.setInputFormat(format);
    }

    @Override
    public Format setOutputFormat(Format format)
    {
        if (!formatClass.isInstance(format)
                || (matches(format, getMatchingOutputFormats(inputFormat))
                        == null))
            return null;

        return super.setOutputFormat(format);
    }

    /**
     * Updates the format, length and offset of a
     * specific output Buffer to specific values.
     *
     * @param outputBuffer the output Buffer to update the properties
     * of
     * @param format the Format to set on outputBuffer
     * @param length the length to set on outputBuffer
     * @param offset the offset to set on outputBuffer
     */
    protected void updateOutput(
            Buffer outputBuffer,
            Format format, int length, int offset)
    {
        outputBuffer.setFormat(format);
        outputBuffer.setLength(length);
        outputBuffer.setOffset(offset);
    }

    protected short[] validateShortArraySize(Buffer buffer, int newSize)
    {
        Object data = buffer.getData();
        short[] newShorts;

        if (data instanceof short[])
        {
            short[] shorts = (short[]) data;

            if (shorts.length >= newSize)
                return shorts;

            newShorts = new short[newSize];
            System.arraycopy(shorts, 0, newShorts, 0, shorts.length);
        }
        else
        {
            newShorts = new short[newSize];
            buffer.setLength(0);
            buffer.setOffset(0);
        }

        buffer.setData(newShorts);
        return newShorts;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy