| 1 | /********** | 
|---|
| 2 | This library is free software; you can redistribute it and/or modify it under | 
|---|
| 3 | the terms of the GNU Lesser General Public License as published by the | 
|---|
| 4 | Free Software Foundation; either version 3 of the License, or (at your | 
|---|
| 5 | option) any later version. (See <http://www.gnu.org/copyleft/lesser.html>.) | 
|---|
| 6 |  | 
|---|
| 7 | This library is distributed in the hope that it will be useful, but WITHOUT | 
|---|
| 8 | ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | 
|---|
| 9 | FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License for | 
|---|
| 10 | more details. | 
|---|
| 11 |  | 
|---|
| 12 | You should have received a copy of the GNU Lesser General Public License | 
|---|
| 13 | along with this library; if not, write to the Free Software Foundation, Inc., | 
|---|
| 14 | 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301  USA | 
|---|
| 15 | **********/ | 
|---|
| 16 | // "liveMedia" | 
|---|
| 17 | // Copyright (c) 1996-2020 Live Networks, Inc.  All rights reserved. | 
|---|
| 18 | // RTP sink for Theora video | 
|---|
| 19 | // Implementation | 
|---|
| 20 |  | 
|---|
| 21 | #include "TheoraVideoRTPSink.hh" | 
|---|
| 22 | #include "Base64.hh" | 
|---|
| 23 | #include "VorbisAudioRTPSource.hh" // for parseVorbisOrTheoraConfigStr() | 
|---|
| 24 | #include "VorbisAudioRTPSink.hh" // for generateVorbisOrTheoraConfigStr() | 
|---|
| 25 |  | 
|---|
| 26 | TheoraVideoRTPSink* TheoraVideoRTPSink | 
|---|
| 27 | ::createNew(UsageEnvironment& env, Groupsock* RTPgs, u_int8_t rtpPayloadFormat, | 
|---|
| 28 | u_int8_t* , unsigned , | 
|---|
| 29 | u_int8_t* , unsigned , | 
|---|
| 30 | u_int8_t* , unsigned , | 
|---|
| 31 | u_int32_t identField) { | 
|---|
| 32 | return new TheoraVideoRTPSink(env, RTPgs, | 
|---|
| 33 | rtpPayloadFormat, | 
|---|
| 34 | identificationHeader, identificationHeaderSize, | 
|---|
| 35 | commentHeader, commentHeaderSize, | 
|---|
| 36 | setupHeader, setupHeaderSize, identField); | 
|---|
| 37 | } | 
|---|
| 38 |  | 
|---|
| 39 | TheoraVideoRTPSink* TheoraVideoRTPSink | 
|---|
| 40 | ::createNew(UsageEnvironment& env, Groupsock* RTPgs, u_int8_t rtpPayloadFormat, | 
|---|
| 41 | char const* configStr) { | 
|---|
| 42 | // Begin by decoding and unpacking the configuration string: | 
|---|
| 43 | u_int8_t* ; unsigned ; | 
|---|
| 44 | u_int8_t* ; unsigned ; | 
|---|
| 45 | u_int8_t* ; unsigned ; | 
|---|
| 46 | u_int32_t identField; | 
|---|
| 47 |  | 
|---|
| 48 | parseVorbisOrTheoraConfigStr(configStr, | 
|---|
| 49 | identificationHeader, identificationHeaderSize, | 
|---|
| 50 | commentHeader, commentHeaderSize, | 
|---|
| 51 | setupHeader, setupHeaderSize, | 
|---|
| 52 | identField); | 
|---|
| 53 |  | 
|---|
| 54 | TheoraVideoRTPSink* resultSink | 
|---|
| 55 | = new TheoraVideoRTPSink(env, RTPgs, rtpPayloadFormat, | 
|---|
| 56 | identificationHeader, identificationHeaderSize, | 
|---|
| 57 | commentHeader, commentHeaderSize, | 
|---|
| 58 | setupHeader, setupHeaderSize, | 
|---|
| 59 | identField); | 
|---|
| 60 | delete[] identificationHeader; delete[] commentHeader; delete[] setupHeader; | 
|---|
| 61 |  | 
|---|
| 62 | return resultSink; | 
|---|
| 63 | } | 
|---|
| 64 |  | 
|---|
| 65 | TheoraVideoRTPSink | 
|---|
| 66 | ::TheoraVideoRTPSink(UsageEnvironment& env, Groupsock* RTPgs, u_int8_t rtpPayloadFormat, | 
|---|
| 67 | u_int8_t* , unsigned , | 
|---|
| 68 | u_int8_t* , unsigned , | 
|---|
| 69 | u_int8_t* , unsigned , | 
|---|
| 70 | u_int32_t identField) | 
|---|
| 71 | : VideoRTPSink(env, RTPgs, rtpPayloadFormat, 90000, "THEORA"), | 
|---|
| 72 | fIdent(identField), fFmtpSDPLine(NULL) { | 
|---|
| 73 | static const char *pf_to_str[] = { | 
|---|
| 74 | "YCbCr-4:2:0", | 
|---|
| 75 | "Reserved", | 
|---|
| 76 | "YCbCr-4:2:2", | 
|---|
| 77 | "YCbCr-4:4:4", | 
|---|
| 78 | }; | 
|---|
| 79 |  | 
|---|
| 80 | unsigned width = 1280; // default value | 
|---|
| 81 | unsigned height = 720; // default value | 
|---|
| 82 | unsigned pf = 0; // default value | 
|---|
| 83 | if (identificationHeaderSize >= 42) { | 
|---|
| 84 | // Parse this header to get the "width", "height", "pf" (pixel format), and | 
|---|
| 85 | // 'nominal bitrate' parameters: | 
|---|
| 86 | u_int8_t* p = identificationHeader; // alias | 
|---|
| 87 | width = (p[14]<<16)|(p[15]<<8)|p[16]; | 
|---|
| 88 | height = (p[17]<<16)|(p[18]<<8)|p[19]; | 
|---|
| 89 | pf = (p[41]&0x18)>>3; | 
|---|
| 90 | unsigned nominalBitrate = (p[37]<<16)|(p[38]<<8)|p[39]; | 
|---|
| 91 | if (nominalBitrate > 0) estimatedBitrate() = nominalBitrate/1000; | 
|---|
| 92 | } | 
|---|
| 93 |  | 
|---|
| 94 | // Generate a 'config' string from the supplied configuration headers: | 
|---|
| 95 | char* | 
|---|
| 96 | = generateVorbisOrTheoraConfigStr(identificationHeader, identificationHeaderSize, | 
|---|
| 97 | commentHeader, commentHeaderSize, | 
|---|
| 98 | setupHeader, setupHeaderSize, | 
|---|
| 99 | identField); | 
|---|
| 100 | if (base64PackedHeaders == NULL) return; | 
|---|
| 101 |  | 
|---|
| 102 | // Then use this 'config' string to construct our "a=fmtp:" SDP line: | 
|---|
| 103 | unsigned fmtpSDPLineMaxSize = 200 + strlen(base64PackedHeaders);// 200 => more than enough space | 
|---|
| 104 | fFmtpSDPLine = new char[fmtpSDPLineMaxSize]; | 
|---|
| 105 | sprintf(fFmtpSDPLine, "a=fmtp:%d sampling=%s;width=%u;height=%u;delivery-method=out_band/rtsp;configuration=%s\r\n", rtpPayloadType(), pf_to_str[pf], width, height, base64PackedHeaders); | 
|---|
| 106 | delete[] base64PackedHeaders; | 
|---|
| 107 | } | 
|---|
| 108 |  | 
|---|
| 109 | TheoraVideoRTPSink::~TheoraVideoRTPSink() { | 
|---|
| 110 | delete[] fFmtpSDPLine; | 
|---|
| 111 | } | 
|---|
| 112 |  | 
|---|
| 113 | char const* TheoraVideoRTPSink::auxSDPLine() { | 
|---|
| 114 | return fFmtpSDPLine; | 
|---|
| 115 | } | 
|---|
| 116 |  | 
|---|
| 117 | void TheoraVideoRTPSink | 
|---|
| 118 | ::doSpecialFrameHandling(unsigned fragmentationOffset, | 
|---|
| 119 | unsigned char* frameStart, | 
|---|
| 120 | unsigned numBytesInFrame, | 
|---|
| 121 | struct timeval framePresentationTime, | 
|---|
| 122 | unsigned numRemainingBytes) { | 
|---|
| 123 | // Set the 4-byte "payload header", as defined in http://svn.xiph.org/trunk/theora/doc/draft-ietf-avt-rtp-theora-00.txt | 
|---|
| 124 | u_int8_t [6]; | 
|---|
| 125 |  | 
|---|
| 126 | // The three bytes of the header are our "Ident": | 
|---|
| 127 | header[0] = fIdent>>16; header[1] = fIdent>>8; header[2] = fIdent; | 
|---|
| 128 |  | 
|---|
| 129 | // The final byte contains the "F", "TDT", and "numPkts" fields: | 
|---|
| 130 | u_int8_t F; // Fragment type | 
|---|
| 131 | if (numRemainingBytes > 0) { | 
|---|
| 132 | if (fragmentationOffset > 0) { | 
|---|
| 133 | F = 2<<6; // continuation fragment | 
|---|
| 134 | } else { | 
|---|
| 135 | F = 1<<6; // start fragment | 
|---|
| 136 | } | 
|---|
| 137 | } else { | 
|---|
| 138 | if (fragmentationOffset > 0) { | 
|---|
| 139 | F = 3<<6; // end fragment | 
|---|
| 140 | } else { | 
|---|
| 141 | F = 0<<6; // not fragmented | 
|---|
| 142 | } | 
|---|
| 143 | } | 
|---|
| 144 | u_int8_t const TDT = 0<<4; // Theora Data Type (always a "Raw Theora payload") | 
|---|
| 145 | u_int8_t numPkts = F == 0 ? (numFramesUsedSoFar() + 1): 0; // set to 0 when we're a fragment | 
|---|
| 146 | header[3] = F|TDT|numPkts; | 
|---|
| 147 |  | 
|---|
| 148 | // There's also a 2-byte 'frame-specific' header: The length of the | 
|---|
| 149 | // Theora data: | 
|---|
| 150 | header[4] = numBytesInFrame >>8; | 
|---|
| 151 | header[5] = numBytesInFrame; | 
|---|
| 152 | setSpecialHeaderBytes(header, sizeof(header)); | 
|---|
| 153 |  | 
|---|
| 154 | if (numRemainingBytes == 0) { | 
|---|
| 155 | // This packet contains the last (or only) fragment of the frame. | 
|---|
| 156 | // Set the RTP 'M' ('marker') bit: | 
|---|
| 157 | setMarkerBit(); | 
|---|
| 158 | } | 
|---|
| 159 |  | 
|---|
| 160 | // Important: Also call our base class's doSpecialFrameHandling(), | 
|---|
| 161 | // to set the packet's timestamp: | 
|---|
| 162 | MultiFramedRTPSink::doSpecialFrameHandling(fragmentationOffset, | 
|---|
| 163 | frameStart, numBytesInFrame, | 
|---|
| 164 | framePresentationTime, | 
|---|
| 165 | numRemainingBytes); | 
|---|
| 166 | } | 
|---|
| 167 |  | 
|---|
| 168 | Boolean TheoraVideoRTPSink::frameCanAppearAfterPacketStart(unsigned char const* /*frameStart*/, | 
|---|
| 169 | unsigned /*numBytesInFrame*/) const { | 
|---|
| 170 | // Only one frame per packet: | 
|---|
| 171 | return False; | 
|---|
| 172 | } | 
|---|
| 173 |  | 
|---|
| 174 | unsigned TheoraVideoRTPSink::() const { | 
|---|
| 175 | return 6; | 
|---|
| 176 | } | 
|---|
| 177 |  | 
|---|