rfc2250.cpp
上传用户:sun1608
上传日期:2007-02-02
资源大小:6116k
文件大小:4k
源码类别:

流媒体/Mpeg4/MP4

开发平台:

Visual C++

  1. /*
  2.  * The contents of this file are subject to the Mozilla Public
  3.  * License Version 1.1 (the "License"); you may not use this file
  4.  * except in compliance with the License. You may obtain a copy of
  5.  * the License at http://www.mozilla.org/MPL/
  6.  * 
  7.  * Software distributed under the License is distributed on an "AS
  8.  * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
  9.  * implied. See the License for the specific language governing
  10.  * rights and limitations under the License.
  11.  * 
  12.  * The Original Code is MPEG4IP.
  13.  * 
  14.  * The Initial Developer of the Original Code is Cisco Systems Inc.
  15.  * Portions created by Cisco Systems Inc. are
  16.  * Copyright (C) Cisco Systems Inc. 2000-2002.  All Rights Reserved.
  17.  * 
  18.  * Contributor(s): 
  19.  * Dave Mackie dmackie@cisco.com
  20.  */
  21. /* 
  22.  * Notes:
  23.  *  - file formatted with tabstops == 4 spaces 
  24.  */
  25. #include <mp4av_common.h>
  26. extern "C" bool MP4AV_Rfc2250Hinter(
  27. MP4FileHandle mp4File, 
  28. MP4TrackId mediaTrackId, 
  29. bool interleave,
  30. u_int16_t maxPayloadSize)
  31. {
  32. // RFC 2250 doesn't support interleaving
  33. if (interleave) {
  34. return false;
  35. }
  36. MP4TrackId hintTrackId =
  37. MP4AddHintTrack(mp4File, mediaTrackId);
  38. if (hintTrackId == MP4_INVALID_TRACK_ID) {
  39. return false;
  40. }
  41. u_int32_t numSamples =
  42. MP4GetTrackNumberOfSamples(mp4File, mediaTrackId);
  43. if (numSamples == 0) {
  44. return false;
  45. }
  46. MP4Duration sampleDuration = 
  47. MP4GetTrackFixedSampleDuration(mp4File, mediaTrackId);
  48. if (sampleDuration == MP4_INVALID_DURATION) {
  49. return false;
  50. }
  51. u_int8_t payloadNumber = 0; // use dynamic payload number
  52. MP4SetHintTrackRtpPayload(mp4File, hintTrackId, 
  53. "MPA", &payloadNumber, 0);
  54. u_int16_t bytesThisHint = 0;
  55. u_int16_t samplesThisHint = 0;
  56. MP4AddRtpHint(mp4File, hintTrackId);
  57. MP4AddRtpPacket(mp4File, hintTrackId, true);
  58. for (MP4SampleId sampleId = 1; sampleId <= numSamples; sampleId++) {
  59. u_int32_t sampleSize = 
  60. MP4GetSampleSize(mp4File, mediaTrackId, sampleId);
  61. if (samplesThisHint > 0) {
  62. if (bytesThisHint + sampleSize <= maxPayloadSize) {
  63. // add the mp3 frame to current hint
  64. MP4AddRtpSampleData(mp4File, hintTrackId,
  65. sampleId, 0, sampleSize);
  66. samplesThisHint++;
  67. bytesThisHint += sampleSize;
  68. continue;
  69. } else {
  70. // write out current hint
  71. MP4WriteRtpHint(mp4File, hintTrackId, 
  72. samplesThisHint * sampleDuration);
  73. // start a new hint 
  74. samplesThisHint = 0;
  75. bytesThisHint = 0;
  76. MP4AddRtpHint(mp4File, hintTrackId);
  77. MP4AddRtpPacket(mp4File, hintTrackId, true);
  78. // fall thru
  79. }
  80. }
  81. if (sampleSize + 4 <= maxPayloadSize) {
  82. // add rfc 2250 payload header
  83. static u_int32_t zero32 = 0;
  84. MP4AddRtpImmediateData(mp4File, hintTrackId,
  85. (u_int8_t*)&zero32, sizeof(zero32));
  86. // add the mp3 frame to current hint
  87. MP4AddRtpSampleData(mp4File, hintTrackId,
  88. sampleId, 0, sampleSize);
  89. bytesThisHint += (4 + sampleSize);
  90. } else {
  91. // jumbo frame, need to fragment it
  92. u_int16_t sampleOffset = 0;
  93. while (sampleOffset < sampleSize) {
  94. u_int16_t fragLength = 
  95. MIN(sampleSize - sampleOffset, maxPayloadSize) - 4;
  96. u_int8_t payloadHeader[4];
  97. payloadHeader[0] = payloadHeader[1] = 0;
  98. payloadHeader[2] = (sampleOffset >> 8);
  99. payloadHeader[3] = sampleOffset & 0xFF;
  100. MP4AddRtpImmediateData(mp4File, hintTrackId,
  101. (u_int8_t*)&payloadHeader, sizeof(payloadHeader));
  102. MP4AddRtpSampleData(mp4File, hintTrackId,
  103. sampleId, sampleOffset, fragLength);
  104. sampleOffset += fragLength;
  105. // if we're not at the last fragment
  106. if (sampleOffset < sampleSize) {
  107. MP4AddRtpPacket(mp4File, hintTrackId, false);
  108. }
  109. }
  110. // lie to ourselves so as to force next frame to output 
  111. // our hint as is, and start a new hint for itself
  112. bytesThisHint = maxPayloadSize;
  113. }
  114. samplesThisHint = 1;
  115. }
  116. // write out current (final) hint
  117. MP4WriteRtpHint(mp4File, hintTrackId, 
  118. samplesThisHint * sampleDuration);
  119. return true;
  120. }