Package org.red5.server.stream

Source Code of org.red5.server.stream.VideoFrameDropper

/*
* RED5 Open Source Flash Server - http://code.google.com/p/red5/
*
* Copyright 2006-2014 by respective authors (see below). All rights reserved.
*
* 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.red5.server.stream;

import org.red5.server.net.rtmp.event.IRTMPEvent;
import org.red5.server.net.rtmp.event.VideoData;
import org.red5.server.net.rtmp.event.VideoData.FrameType;
import org.red5.server.stream.message.RTMPMessage;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
* State machine for video frame dropping in live streams.
* <p>
* We start sending all frame types. Disposable interframes can be dropped any
* time without affecting the current state. If a regular interframe is dropped,
* all future frames up to the next keyframes are dropped as well. Dropped
* keyframes result in only keyframes being sent. If two consecutive keyframes
* have been successfully sent, regular interframes will be sent in the next
* iteration as well. If these frames all went through, disposable interframes
* are sent again.
*
* <p>
* So from highest to lowest bandwidth and back, the states go as follows:
* <ul>
* <li>all frames</li>
* <li>keyframes and interframes</li>
* <li>keyframes</li>
* <li>keyframes and interframes</li>
* <li>all frames</li>
* </ul>
*
* @author The Red5 Project
* @author Joachim Bauch (jojo@struktur.de)
*/
public class VideoFrameDropper implements IFrameDropper {

  protected static Logger log = LoggerFactory.getLogger(VideoFrameDropper.class.getName());

  /** Current state. */
  private int state;

  /** Constructs a new VideoFrameDropper. */
  public VideoFrameDropper() {
    reset();
  }

  /** {@inheritDoc} */
  public void reset() {
    reset(SEND_ALL);
  }

  /** {@inheritDoc} */
  public void reset(int state) {
    this.state = state;
  }

  /** {@inheritDoc} */
  public boolean canSendPacket(RTMPMessage message, long pending) {
    IRTMPEvent packet = message.getBody();
    boolean result = true;
    // We currently only drop video packets.
    if (packet instanceof VideoData) {
      VideoData video = (VideoData) packet;
      FrameType type = video.getFrameType();
      switch (state) {
        case SEND_ALL:
          // All packets will be sent
          break;
        case SEND_INTERFRAMES:
          // Only keyframes and interframes will be sent.
          if (type == FrameType.KEYFRAME) {
            if (pending == 0) {
              // Send all frames from now on.
              state = SEND_ALL;
            }
          } else if (type == FrameType.INTERFRAME) {
          }
          break;
        case SEND_KEYFRAMES:
          // Only keyframes will be sent.
          result = (type == FrameType.KEYFRAME);
          if (result && pending == 0) {
            // Maybe switch back to SEND_INTERFRAMES after the next keyframe
            state = SEND_KEYFRAMES_CHECK;
          }
          break;
        case SEND_KEYFRAMES_CHECK:
          // Only keyframes will be sent.
          result = (type == FrameType.KEYFRAME);
          if (result && pending == 0) {
            // Continue with sending interframes as well
            state = SEND_INTERFRAMES;
          }
          break;
        default:
      }
    }
    return result;
  }

  /** {@inheritDoc} */
  public void dropPacket(RTMPMessage message) {
    IRTMPEvent packet = message.getBody();
    // Only check video packets.
    if (packet instanceof VideoData) {
      VideoData video = (VideoData) packet;
      FrameType type = video.getFrameType();
      switch (state) {
        case SEND_ALL:
          if (type == FrameType.DISPOSABLE_INTERFRAME) {
            // Remain in state, packet is safe to drop.
            return;
          } else if (type == FrameType.INTERFRAME) {
            // Drop all frames until the next keyframe.
            state = SEND_KEYFRAMES;
            return;
          } else if (type == FrameType.KEYFRAME) {
            // Drop all frames until the next keyframe.
            state = SEND_KEYFRAMES;
            return;
          }
          break;
        case SEND_INTERFRAMES:
          if (type == FrameType.INTERFRAME) {
            // Drop all frames until the next keyframe.
            state = SEND_KEYFRAMES_CHECK;
            return;
          } else if (type == FrameType.KEYFRAME) {
            // Drop all frames until the next keyframe.
            state = SEND_KEYFRAMES;
            return;
          }
          break;
        case SEND_KEYFRAMES:
          // Remain in state.
          break;
        case SEND_KEYFRAMES_CHECK:
          if (type == FrameType.KEYFRAME) {
            // Switch back to sending keyframes, but don't move to
            // SEND_INTERFRAMES afterwards.
            state = SEND_KEYFRAMES;
            return;
          }
          break;
        default:
      }
    }
  }

  /** {@inheritDoc} */
  public void sendPacket(RTMPMessage message) {

  }

}
TOP

Related Classes of org.red5.server.stream.VideoFrameDropper

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.