Android Open Source - spydroid-ipcamera H264 Packetizer






From Project

Back to project page spydroid-ipcamera.

License

The source code is released under:

GNU General Public License

If you think the Android project spydroid-ipcamera listed in this page is inappropriate, such as containing malicious code/tools or violating the copyright, please email info at java2s dot com, thanks.

Java Source Code

/*
 * Copyright (C) 2011-2014 GUIGUI Simon, fyhertz@gmail.com
 * // w  w  w .  j  a  va 2 s.  c o m
 * This file is part of libstreaming (https://github.com/fyhertz/libstreaming)
 * 
 * Spydroid is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 3 of the License, or
 * (at your option) any later version.
 * 
 * This source code is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 * 
 * You should have received a copy of the GNU General Public License
 * along with this source code; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */

package net.majorkernelpanic.streaming.rtp;

import java.io.IOException;

import android.annotation.SuppressLint;
import android.util.Log;

/**
 * 
 *   RFC 3984.
 *   
 *   H.264 streaming over RTP.
 *   
 *   Must be fed with an InputStream containing H.264 NAL units preceded by their length (4 bytes).
 *   The stream must start with mpeg4 or 3gpp header, it will be skipped.
 *   
 */
public class H264Packetizer extends AbstractPacketizer implements Runnable {

  public final static String TAG = "H264Packetizer";

  private Thread t = null;
  private int naluLength = 0;
  private long delay = 0, oldtime = 0;
  private Statistics stats = new Statistics();
  private byte[] sps = null, pps = null;
  byte[] header = new byte[5];  
  private int count = 0;
  private int streamType = 1;


  public H264Packetizer() {
    super();
    socket.setClockFrequency(90000);
  }

  public void start() {
    if (t == null) {
      t = new Thread(this);
      t.start();
    }
  }

  public void stop() {
    if (t != null) {
      try {
        is.close();
      } catch (IOException e) {}
      t.interrupt();
      try {
        t.join();
      } catch (InterruptedException e) {}
      t = null;
    }
  }

  public void setStreamParameters(byte[] pps, byte[] sps) {
    this.pps = pps;
    this.sps = sps;
  }  

  public void run() {
    long duration = 0, delta2 = 0;
    Log.d(TAG,"H264 packetizer started !");
    stats.reset();
    count = 0;

    if (is instanceof MediaCodecInputStream) {
      streamType = 1;
      socket.setCacheSize(0);
    } else {
      streamType = 0;  
      socket.setCacheSize(400);
    }

    try {
      while (!Thread.interrupted()) {

        oldtime = System.nanoTime();
        // We read a NAL units from the input stream and we send them
        send();
        // We measure how long it took to receive NAL units from the phone
        duration = System.nanoTime() - oldtime;
        
        // Every 3 secondes, we send two packets containing NALU type 7 (sps) and 8 (pps)
        // Those should allow the H264 stream to be decoded even if no SDP was sent to the decoder.        
        delta2 += duration/1000000;
        if (delta2>3000) {
          delta2 = 0;
          if (sps != null) {
            buffer = socket.requestBuffer();
            socket.markNextPacket();
            socket.updateTimestamp(ts);
            System.arraycopy(sps, 0, buffer, rtphl, sps.length);
            super.send(rtphl+sps.length);
          }
          if (pps != null) {
            buffer = socket.requestBuffer();
            socket.updateTimestamp(ts);
            socket.markNextPacket();
            System.arraycopy(pps, 0, buffer, rtphl, pps.length);
            super.send(rtphl+pps.length);
          }          
        }

        stats.push(duration);
        // Computes the average duration of a NAL unit
        delay = stats.average();
        //Log.d(TAG,"duration: "+duration/1000000+" delay: "+delay/1000000);

      }
    } catch (IOException e) {
    } catch (InterruptedException e) {}

    Log.d(TAG,"H264 packetizer stopped !");

  }

  /**
   * Reads a NAL unit in the FIFO and sends it.
   * If it is too big, we split it in FU-A units (RFC 3984).
   */
  @SuppressLint("NewApi")
  private void send() throws IOException, InterruptedException {
    int sum = 1, len = 0, type;

    if (streamType == 0) {
      // NAL units are preceeded by their length, we parse the length
      fill(header,0,5);
      ts += delay;
      naluLength = header[3]&0xFF | (header[2]&0xFF)<<8 | (header[1]&0xFF)<<16 | (header[0]&0xFF)<<24;
      if (naluLength>100000 || naluLength<0) resync();
    } else if (streamType == 1) {
      // NAL units are preceeded with 0x00000001
      fill(header,0,5);
      ts = ((MediaCodecInputStream)is).getLastBufferInfo().presentationTimeUs*1000L;
      //ts += delay;
      naluLength = is.available()+1;
      if (!(header[0]==0 && header[1]==0 && header[2]==0)) {
        // Turns out, the NAL units are not preceeded with 0x00000001
        Log.e(TAG, "NAL units are not preceeded by 0x00000001");
        streamType = 2; 
        return;
      }
    } else {
      // Nothing preceededs the NAL units
      fill(header,0,1);
      header[4] = header[0];
      ts = ((MediaCodecInputStream)is).getLastBufferInfo().presentationTimeUs*1000L;
      //ts += delay;
      naluLength = is.available()+1;
    }

    // Parses the NAL unit type
    type = header[4]&0x1F;

    // The stream already contains NAL unit type 7 or 8, we don't need 
    // to add them to the stream ourselves
    if (type == 7 || type == 8) {
      Log.v(TAG,"SPS or PPS present in the stream.");
      count++;
      if (count>4) {
        sps = null;
        pps = null;
      }
    }

    //Log.d(TAG,"- Nal unit length: " + naluLength + " delay: "+delay/1000000+" type: "+type);

    // Small NAL unit => Single NAL unit 
    if (naluLength<=MAXPACKETSIZE-rtphl-2) {
      buffer = socket.requestBuffer();
      buffer[rtphl] = header[4];
      len = fill(buffer, rtphl+1,  naluLength-1);
      socket.updateTimestamp(ts);
      socket.markNextPacket();
      super.send(naluLength+rtphl);
      //Log.d(TAG,"----- Single NAL unit - len:"+len+" delay: "+delay);
    }
    // Large NAL unit => Split nal unit 
    else {

      // Set FU-A header
      header[1] = (byte) (header[4] & 0x1F);  // FU header type
      header[1] += 0x80; // Start bit
      // Set FU-A indicator
      header[0] = (byte) ((header[4] & 0x60) & 0xFF); // FU indicator NRI
      header[0] += 28;

      while (sum < naluLength) {
        buffer = socket.requestBuffer();
        buffer[rtphl] = header[0];
        buffer[rtphl+1] = header[1];
        socket.updateTimestamp(ts);
        if ((len = fill(buffer, rtphl+2,  naluLength-sum > MAXPACKETSIZE-rtphl-2 ? MAXPACKETSIZE-rtphl-2 : naluLength-sum  ))<0) return; sum += len;
        // Last packet before next NAL
        if (sum >= naluLength) {
          // End bit on
          buffer[rtphl+1] += 0x40;
          socket.markNextPacket();
        }
        super.send(len+rtphl+2);
        // Switch start bit
        header[1] = (byte) (header[1] & 0x7F); 
        //Log.d(TAG,"----- FU-A unit, sum:"+sum);
      }
    }
  }

  private int fill(byte[] buffer, int offset,int length) throws IOException {
    int sum = 0, len;
    while (sum<length) {
      len = is.read(buffer, offset+sum, length-sum);
      if (len<0) {
        throw new IOException("End of stream");
      }
      else sum+=len;
    }
    return sum;
  }

  private void resync() throws IOException {
    int type;

    Log.e(TAG,"Packetizer out of sync ! Let's try to fix that...(NAL length: "+naluLength+")");

    while (true) {

      header[0] = header[1];
      header[1] = header[2];
      header[2] = header[3];
      header[3] = header[4];
      header[4] = (byte) is.read();

      type = header[4]&0x1F;

      if (type == 5 || type == 1) {
        naluLength = header[3]&0xFF | (header[2]&0xFF)<<8 | (header[1]&0xFF)<<16 | (header[0]&0xFF)<<24;
        if (naluLength>0 && naluLength<100000) {
          oldtime = System.nanoTime();
          Log.e(TAG,"A NAL unit may have been found in the bit stream !");
          break;
        }
        if (naluLength==0) {
          Log.e(TAG,"NAL unit with NULL size found...");
        } else if (header[3]==0xFF && header[2]==0xFF && header[1]==0xFF && header[0]==0xFF) {
          Log.e(TAG,"NAL unit with 0xFFFFFFFF size found...");
        }
      }

    }

  }

}




Java Source Code List

net.majorkernelpanic.http.ModAssetServer.java
net.majorkernelpanic.http.ModInternationalization.java
net.majorkernelpanic.http.ModSSL.java
net.majorkernelpanic.http.TinyHttpServer.java
net.majorkernelpanic.spydroid.SpydroidApplication.java
net.majorkernelpanic.spydroid.Utilities.java
net.majorkernelpanic.spydroid.api.CustomHttpServer.java
net.majorkernelpanic.spydroid.api.CustomRtspServer.java
net.majorkernelpanic.spydroid.api.RequestHandler.java
net.majorkernelpanic.spydroid.ui.AboutFragment.java
net.majorkernelpanic.spydroid.ui.HandsetFragment.java
net.majorkernelpanic.spydroid.ui.OptionsActivity.java
net.majorkernelpanic.spydroid.ui.PreviewFragment.java
net.majorkernelpanic.spydroid.ui.SpydroidActivity.java
net.majorkernelpanic.spydroid.ui.TabletFragment.java
net.majorkernelpanic.streaming.MediaStream.java
net.majorkernelpanic.streaming.SessionBuilder.java
net.majorkernelpanic.streaming.Session.java
net.majorkernelpanic.streaming.Stream.java
net.majorkernelpanic.streaming.audio.AACStream.java
net.majorkernelpanic.streaming.audio.AMRNBStream.java
net.majorkernelpanic.streaming.audio.AudioQuality.java
net.majorkernelpanic.streaming.audio.AudioStream.java
net.majorkernelpanic.streaming.exceptions.CameraInUseException.java
net.majorkernelpanic.streaming.exceptions.ConfNotSupportedException.java
net.majorkernelpanic.streaming.exceptions.InvalidSurfaceException.java
net.majorkernelpanic.streaming.exceptions.StorageUnavailableException.java
net.majorkernelpanic.streaming.gl.SurfaceManager.java
net.majorkernelpanic.streaming.gl.SurfaceView.java
net.majorkernelpanic.streaming.gl.TextureManager.java
net.majorkernelpanic.streaming.hw.CodecManager.java
net.majorkernelpanic.streaming.hw.EncoderDebugger.java
net.majorkernelpanic.streaming.hw.NV21Convertor.java
net.majorkernelpanic.streaming.mp4.MP4Config.java
net.majorkernelpanic.streaming.mp4.MP4Parser.java
net.majorkernelpanic.streaming.rtcp.SenderReport.java
net.majorkernelpanic.streaming.rtp.AACADTSPacketizer.java
net.majorkernelpanic.streaming.rtp.AACLATMPacketizer.java
net.majorkernelpanic.streaming.rtp.AMRNBPacketizer.java
net.majorkernelpanic.streaming.rtp.AbstractPacketizer.java
net.majorkernelpanic.streaming.rtp.H263Packetizer.java
net.majorkernelpanic.streaming.rtp.H264Packetizer.java
net.majorkernelpanic.streaming.rtp.MediaCodecInputStream.java
net.majorkernelpanic.streaming.rtp.RtpSocket.java
net.majorkernelpanic.streaming.rtsp.RtspClient.java
net.majorkernelpanic.streaming.rtsp.RtspServer.java
net.majorkernelpanic.streaming.rtsp.UriParser.java
net.majorkernelpanic.streaming.video.CodecManager.java
net.majorkernelpanic.streaming.video.H263Stream.java
net.majorkernelpanic.streaming.video.H264Stream.java
net.majorkernelpanic.streaming.video.VideoQuality.java
net.majorkernelpanic.streaming.video.VideoStream.java