/sip/WEB-INF/classes/org/red5/server/webapp/sip/RTPStreamSender.java

http://red5phone.googlecode.com/ · Java · 527 lines · 275 code · 139 blank · 113 comment · 40 complexity · 624caa38397d14b9a5325c41ea0a7f2e MD5 · raw file

  1. package org.red5.server.webapp.sip;
  2. import local.net.RtpPacket;
  3. import local.net.RtpSocket;
  4. import java.net.InetAddress;
  5. import java.net.DatagramSocket;
  6. import java.util.Vector;
  7. import org.slf4j.Logger;
  8. import org.red5.logging.Red5LoggerFactory;
  9. import org.red5.codecs.SIPCodec;
  10. import org.red5.codecs.asao.*;
  11. import local.media.G711;
  12. public class RTPStreamSender {
  13. protected static Logger log = Red5LoggerFactory.getLogger( RTPStreamSender.class, "sip" );
  14. public static int RTP_HEADER_SIZE = 12;
  15. private static final int NELLYMOSER_DECODED_PACKET_SIZE = 256;
  16. private static final int NELLYMOSER_ENCODED_PACKET_SIZE = 64;
  17. RtpSocket rtpSocket = null;
  18. /** Sip codec to be used on audio session */
  19. private SIPCodec sipCodec = null;
  20. boolean socketIsLocal = false;
  21. boolean doSync = true;
  22. private int syncAdj = 0;
  23. private Decoder decoder;
  24. private DecoderMap decoderMap;
  25. private byte[] packetBuffer;
  26. private RtpPacket rtpPacket;
  27. private int startPayloadPos;
  28. private int dtmf2833Type = 101;
  29. private int seqn = 0;
  30. private long time = 0;
  31. // Temporary buffer with received PCM audio from FlashPlayer.
  32. float[] tempBuffer;
  33. // Floats remaining on temporary buffer.
  34. int tempBufferRemaining = 0;
  35. // Encoding buffer used to encode to final codec format;
  36. float[] encodingBuffer;
  37. // Offset of encoding buffer.
  38. int encodingOffset = 0;
  39. // Indicates whether the current asao buffer was processed.
  40. boolean asao_buffer_processed = false;
  41. // Indicates whether the handling buffers have already
  42. // been initialized.
  43. boolean hasInitilializedBuffers = false;
  44. /**
  45. * Constructs a RtpStreamSender.
  46. *
  47. * @param RTMPUser
  48. * the RTMP stream source
  49. * @param do_sync
  50. * whether time synchronization must be performed by the
  51. * RtpStreamSender, or it is performed by the InputStream (e.g.
  52. * the system audio input)
  53. * @param sipCodec
  54. * codec to be used on audio session
  55. * @param dest_addr
  56. * the destination address
  57. * @param dest_port
  58. * the destination port
  59. */
  60. public RTPStreamSender(
  61. RTMPUser rtmpUser,
  62. boolean do_sync,
  63. SIPCodec sipCodec,
  64. String dest_addr,
  65. int dest_port ) {
  66. init( rtmpUser, do_sync, sipCodec, null, dest_addr, dest_port );
  67. }
  68. /**
  69. * Constructs a RtpStreamSender.
  70. *
  71. * @param RTMPUser
  72. * the RTMP stream source
  73. * @param do_sync
  74. * whether time synchronization must be performed by the
  75. * RtpStreamSender, or it is performed by the InputStream (e.g.
  76. * the system audio input)
  77. * @param sipCodec
  78. * codec to be used on audio session
  79. * @param src_port
  80. * the source port
  81. * @param dest_addr
  82. * the destination address
  83. * @param dest_port
  84. * the destination port
  85. */
  86. // public RtpStreamSender(RTMPUser rtmpUser, boolean do_sync, int
  87. // payloadType, long frame_rate, int frame_size, int src_port, String
  88. // dest_addr, int dest_port)
  89. // {
  90. // init( rtmpUser, do_sync, payloadType, frame_rate, frame_size, null, src_port, dest_addr, dest_port);
  91. // }
  92. /**
  93. * Constructs a RtpStreamSender.
  94. *
  95. * @param RTMPUser
  96. * the RTMP stream source
  97. * @param do_sync
  98. * whether time synchronization must be performed by the
  99. * RtpStreamSender, or it is performed by the InputStream (e.g.
  100. * the system audio input)
  101. * @param sipCodec
  102. * codec to be used on audio session
  103. * @param src_socket
  104. * the socket used to send the RTP packet
  105. * @param dest_addr
  106. * the destination address
  107. * @param dest_port
  108. * the thestination port
  109. */
  110. public RTPStreamSender(
  111. RTMPUser rtmpUser,
  112. boolean do_sync,
  113. SIPCodec sipCodec,
  114. DatagramSocket src_socket,
  115. String dest_addr,
  116. int dest_port ) {
  117. init( rtmpUser, do_sync, sipCodec, src_socket, dest_addr, dest_port );
  118. }
  119. /** Inits the RtpStreamSender */
  120. private void init(
  121. RTMPUser rtmpUser,
  122. boolean do_sync,
  123. SIPCodec sipCodec,
  124. DatagramSocket src_socket,
  125. String dest_addr,
  126. int dest_port ) {
  127. rtmpUser.rtpStreamSender = this;
  128. this.sipCodec = sipCodec;
  129. this.doSync = do_sync;
  130. try {
  131. if ( src_socket == null ) {
  132. src_socket = new DatagramSocket();
  133. socketIsLocal = true;
  134. }
  135. rtpSocket = new RtpSocket( src_socket, InetAddress.getByName( dest_addr ), dest_port );
  136. }
  137. catch ( Exception e ) {
  138. e.printStackTrace();
  139. }
  140. }
  141. /** Sets the synchronization adjustment time (in milliseconds). */
  142. public void setSyncAdj( int millisecs ) {
  143. syncAdj = millisecs;
  144. }
  145. public void start() {
  146. packetBuffer = new byte[ sipCodec.getOutgoingEncodedFrameSize() + RTP_HEADER_SIZE ];
  147. rtpPacket = new RtpPacket( packetBuffer, 0 );
  148. rtpPacket.setPayloadType( sipCodec.getCodecId() );
  149. startPayloadPos = rtpPacket.getHeaderLength();
  150. seqn = 0;
  151. time = 0;
  152. println( "start()", "using blocks of " + ( packetBuffer.length - RTP_HEADER_SIZE ) + " bytes." );
  153. decoder = new Decoder();
  154. decoderMap = null;
  155. }
  156. public void queueSipDtmfDigits( String argDigits ) {
  157. byte[] dtmfbuf = new byte[ sipCodec.getOutgoingEncodedFrameSize() + RTP_HEADER_SIZE ];
  158. RtpPacket dtmfpacket = new RtpPacket( dtmfbuf, 0 );
  159. dtmfpacket.setPayloadType( dtmf2833Type );
  160. dtmfpacket.setPayloadLength( sipCodec.getOutgoingEncodedFrameSize() );
  161. byte[] blankbuf = new byte[ sipCodec.getOutgoingEncodedFrameSize() + RTP_HEADER_SIZE ];
  162. RtpPacket blankpacket = new RtpPacket( blankbuf, 0 );
  163. blankpacket.setPayloadType( sipCodec.getCodecId() );
  164. blankpacket.setPayloadLength( sipCodec.getOutgoingEncodedFrameSize() );
  165. for ( int d = 0; d < argDigits.length(); d++ ) {
  166. char digit = argDigits.charAt( d );
  167. if ( digit == '*' ) {
  168. dtmfbuf[ startPayloadPos ] = 10;
  169. }
  170. else if ( digit == '#' ) {
  171. dtmfbuf[ startPayloadPos ] = 11;
  172. }
  173. else if ( digit >= 'A' && digit <= 'D' ) {
  174. dtmfbuf[ startPayloadPos ] = (byte) ( digit - 53 );
  175. }
  176. else {
  177. dtmfbuf[ startPayloadPos ] = (byte) ( digit - 48 );
  178. }
  179. //println( "queueSipDtmfDigits", "Sending digit:" + dtmfbuf[ startPayloadPos ] );
  180. // notice we are bumping times/seqn just like audio packets
  181. try {
  182. // send start event packet 3 times
  183. dtmfbuf[ startPayloadPos + 1 ] = 0; // start event flag
  184. // and volume
  185. dtmfbuf[ startPayloadPos + 2 ] = 1; // duration 8 bits
  186. dtmfbuf[ startPayloadPos + 3 ] = -32; // duration 8 bits
  187. for ( int r = 0; r < 3; r++ ) {
  188. dtmfpacket.setSequenceNumber( seqn++ );
  189. dtmfpacket.setTimestamp( sipCodec.getOutgoingDecodedFrameSize() );
  190. doRtpDelay();
  191. rtpSocketSend( dtmfpacket );
  192. }
  193. // send end event packet 3 times
  194. dtmfbuf[ startPayloadPos + 1 ] = -128; // end event flag
  195. dtmfbuf[ startPayloadPos + 2 ] = 3; // duration 8 bits
  196. dtmfbuf[ startPayloadPos + 3 ] = 116; // duration 8 bits
  197. for ( int r = 0; r < 3; r++ ) {
  198. dtmfpacket.setSequenceNumber( seqn++ );
  199. dtmfpacket.setTimestamp(sipCodec.getOutgoingDecodedFrameSize() );
  200. doRtpDelay();
  201. rtpSocketSend( dtmfpacket );
  202. }
  203. // send 200 ms of blank packets
  204. for ( int r = 0; r < 200 / sipCodec.getOutgoingPacketization(); r++ ) {
  205. blankpacket.setSequenceNumber( seqn++ );
  206. blankpacket.setTimestamp(sipCodec.getOutgoingDecodedFrameSize() );
  207. doRtpDelay();
  208. rtpSocketSend( blankpacket );
  209. }
  210. }
  211. catch ( Exception e ) {
  212. println( "queueSipDtmfDigits", e.getLocalizedMessage() );
  213. }
  214. }
  215. }
  216. /** Fill the buffer of RtpPacket with necessary data. */
  217. private int fillRtpPacketBuffer(byte[] asaoBuffer) {
  218. boolean isBufferFilled = false;
  219. int copyingSize = 0;
  220. int finalCopySize = 0;
  221. byte[] codedBuffer = new byte[ sipCodec.getOutgoingEncodedFrameSize() ];
  222. try {
  223. //println( "fillRtpPacketBuffer",
  224. // "packetBuffer.length = " + packetBuffer.length
  225. // + ", asaoBuffer.length = " + asaoBuffer.length
  226. // + ", tempBuffer.length = " + tempBuffer.length
  227. // + ", encodingOffset = " + encodingOffset
  228. // + ", tempBufferRemaining = " + tempBufferRemaining + "." );
  229. if ( ( tempBufferRemaining + encodingOffset ) >= sipCodec.getOutgoingDecodedFrameSize() ) {
  230. copyingSize = encodingBuffer.length - encodingOffset;
  231. BufferUtils.floatBufferIndexedCopy(
  232. encodingBuffer,
  233. encodingOffset,
  234. tempBuffer,
  235. tempBuffer.length - tempBufferRemaining,
  236. copyingSize );
  237. encodingOffset = sipCodec.getOutgoingDecodedFrameSize();
  238. tempBufferRemaining -= copyingSize;
  239. finalCopySize = sipCodec.getOutgoingDecodedFrameSize();
  240. //println( "fillRtpPacketBuffer", "Simple copy of " + copyingSize + " bytes." );
  241. }
  242. else {
  243. if ( tempBufferRemaining > 0 ) {
  244. BufferUtils.floatBufferIndexedCopy(
  245. encodingBuffer,
  246. encodingOffset,
  247. tempBuffer,
  248. tempBuffer.length - tempBufferRemaining,
  249. tempBufferRemaining );
  250. encodingOffset += tempBufferRemaining;
  251. finalCopySize += tempBufferRemaining;
  252. tempBufferRemaining = 0;
  253. //println( "fillRtpPacketBuffer",
  254. // "tempBufferRemaining copied -> "
  255. // + "encodingOffset = " + encodingOffset
  256. // + ", tempBufferRemaining = " + tempBufferRemaining + "." );
  257. }
  258. // Decode new asao packet.
  259. asao_buffer_processed = true;
  260. ByteStream audioStream = new ByteStream( asaoBuffer, 1, NELLYMOSER_ENCODED_PACKET_SIZE );
  261. decoderMap = decoder.decode( decoderMap, audioStream.bytes, 1, tempBuffer, 0 );
  262. //tempBuffer = ResampleUtils.normalize(tempBuffer, 256); // normalise volume
  263. tempBufferRemaining = tempBuffer.length;
  264. //println( "fillRtpPacketBuffer",
  265. // "Decoded pcm " + tempBuffer.length + " floats." );
  266. if ( tempBuffer.length <= 0 ) {
  267. println( "fillRtpPacketBuffer", "Asao decoder Error." );
  268. }
  269. // Try to complete the encodingBuffer with necessary data.
  270. //println( "fillRtpPacketBuffer",
  271. // "New buffer processed -> "
  272. // + "finalCopySize = " + finalCopySize
  273. // + ", encodingOffset = " + encodingOffset
  274. // + ", tempBufferRemaining = " + tempBufferRemaining + "." );
  275. if ( ( encodingOffset + tempBufferRemaining ) > sipCodec.getOutgoingDecodedFrameSize() ) {
  276. copyingSize = encodingBuffer.length - encodingOffset;
  277. }
  278. else {
  279. copyingSize = tempBufferRemaining;
  280. }
  281. //println( "fillRtpPacketBuffer", "CopyingSize = " + copyingSize + "." );
  282. BufferUtils.floatBufferIndexedCopy(
  283. encodingBuffer,
  284. encodingOffset,
  285. tempBuffer,
  286. 0,
  287. copyingSize );
  288. encodingOffset += copyingSize;
  289. tempBufferRemaining -= copyingSize;
  290. finalCopySize += copyingSize;
  291. }
  292. if (encodingOffset == encodingBuffer.length)
  293. {
  294. isBufferFilled = true;
  295. int encodedBytes = sipCodec.pcmToCodec( encodingBuffer, codedBuffer );
  296. //println( "fillRtpPacketBuffer",
  297. // "encodedBytes = " + encodedBytes +
  298. // ", outgoingEncodedFrameSize = " + sipCodec.getOutgoingEncodedFrameSize() + "." );
  299. if ( encodedBytes == sipCodec.getOutgoingEncodedFrameSize() ) {
  300. BufferUtils.byteBufferIndexedCopy( packetBuffer,
  301. RTP_HEADER_SIZE, codedBuffer, 0, codedBuffer.length );
  302. }
  303. else {
  304. //println( "fillRtpPacketBuffer", "Failure encoding buffer." );
  305. }
  306. }
  307. //println( "fillRtpPacketBuffer",
  308. // "finalCopySize = " + finalCopySize
  309. // + ", isBufferFilled = " + isBufferFilled
  310. // + ", encodingOffset = " + encodingOffset
  311. // + ", tempBufferRemaining = " + tempBufferRemaining + "." );
  312. }
  313. catch ( Exception e ) {
  314. e.printStackTrace();
  315. }
  316. return finalCopySize;
  317. }
  318. public void send( byte[] asaoBuffer, int offset, int num ) {
  319. if ( rtpSocket == null ) {
  320. return;
  321. }
  322. asao_buffer_processed = false;
  323. if ( !hasInitilializedBuffers ) {
  324. tempBuffer = new float[ NELLYMOSER_DECODED_PACKET_SIZE ];
  325. encodingBuffer = new float[ sipCodec.getOutgoingDecodedFrameSize() ];
  326. hasInitilializedBuffers = true;
  327. }
  328. //println( "send",
  329. // "asaoBuffer.length = [" + asaoBuffer.length + "], offset = ["
  330. // + offset + "], num = [" + num + "]." );
  331. if ( num > 0 ) {
  332. do {
  333. int encodedBytes = fillRtpPacketBuffer( asaoBuffer );
  334. //println( "send", sipCodec.getCodecName() + " encoded " + encodedBytes + " bytes." );
  335. if ( encodedBytes == 0 ) {
  336. break;
  337. }
  338. if ( encodingOffset == sipCodec.getOutgoingDecodedFrameSize() ) {
  339. //println( "send", "Seding packet with " + encodedBytes + " bytes." );
  340. try {
  341. rtpPacket.setSequenceNumber( seqn++ );
  342. rtpPacket.setTimestamp( time );
  343. rtpPacket.setPayloadLength( sipCodec.getOutgoingEncodedFrameSize() );
  344. rtpSocketSend( rtpPacket );
  345. }
  346. catch ( Exception e ) {
  347. println( "send", sipCodec.getCodecName() + " encoder error." );
  348. }
  349. encodingOffset = 0;
  350. }
  351. //println( "send", "asao_buffer_processed = ["
  352. // + asao_buffer_processed + "] ." );
  353. }
  354. while ( !asao_buffer_processed );
  355. }
  356. else if ( num < 0 ) {
  357. println( "send", "Closing" );
  358. }
  359. }
  360. public void halt() {
  361. DatagramSocket socket = rtpSocket.getDatagramSocket();
  362. rtpSocket.close();
  363. if ( socketIsLocal && socket != null ) {
  364. socket.close();
  365. }
  366. rtpSocket = null;
  367. println( "halt", "Terminated" );
  368. }
  369. private void doRtpDelay() {
  370. try {
  371. Thread.sleep( sipCodec.getOutgoingPacketization() - 2 );
  372. }
  373. catch ( Exception e ) {
  374. }
  375. }
  376. private synchronized void rtpSocketSend(RtpPacket rtpPacket) {
  377. try {
  378. rtpSocket.send( rtpPacket );
  379. time += sipCodec.getOutgoingDecodedFrameSize();
  380. }
  381. catch ( Exception e ) {
  382. }
  383. }
  384. private static void println( String method, String message ) {
  385. log.debug( "RTPStreamSender - " + method + " -> " + message );
  386. System.out.println( "RTPStreamSender - " + method + " -> " + message );
  387. }
  388. }