1 /*
2  * Copyright (C) 2010 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 //#define LOG_NDEBUG 0
18 #define LOG_TAG "ASessionDescription"
19 #include <utils/Log.h>
20 
21 #include <media/stagefright/rtsp/ASessionDescription.h>
22 
23 #include <media/stagefright/foundation/ADebug.h>
24 #include <media/stagefright/foundation/AString.h>
25 
26 #include <stdlib.h>
27 
28 namespace android {
29 
30 constexpr unsigned kDefaultAs = 960; // kbps?
31 
ASessionDescription()32 ASessionDescription::ASessionDescription()
33     : mIsValid(false) {
34 }
35 
~ASessionDescription()36 ASessionDescription::~ASessionDescription() {
37 }
38 
setTo(const void * data,size_t size)39 bool ASessionDescription::setTo(const void *data, size_t size) {
40     mIsValid = parse(data, size);
41 
42     if (!mIsValid) {
43         mTracks.clear();
44         mFormats.clear();
45     }
46 
47     return mIsValid;
48 }
49 
parse(const void * data,size_t size)50 bool ASessionDescription::parse(const void *data, size_t size) {
51     mTracks.clear();
52     mFormats.clear();
53 
54     mTracks.push(Attribs());
55     mFormats.push(AString("[root]"));
56 
57     AString desc((const char *)data, size);
58 
59     size_t i = 0;
60     for (;;) {
61         ssize_t eolPos = desc.find("\n", i);
62 
63         if (eolPos < 0) {
64             break;
65         }
66 
67         AString line;
68         if ((size_t)eolPos > i && desc.c_str()[eolPos - 1] == '\r') {
69             // We accept both '\n' and '\r\n' line endings, if it's
70             // the latter, strip the '\r' as well.
71             line.setTo(desc, i, eolPos - i - 1);
72         } else {
73             line.setTo(desc, i, eolPos - i);
74         }
75 
76         if (line.empty()) {
77             i = eolPos + 1;
78             continue;
79         }
80 
81         if (line.size() < 2 || line.c_str()[1] != '=') {
82             return false;
83         }
84 
85         ALOGV("%s", line.c_str());
86 
87         switch (line.c_str()[0]) {
88             case 'v':
89             {
90                 if (strcmp(line.c_str(), "v=0")) {
91                     return false;
92                 }
93                 break;
94             }
95 
96             case 'a':
97             case 'b':
98             {
99                 AString key, value;
100 
101                 ssize_t colonPos = line.find(":", 2);
102                 if (colonPos < 0) {
103                     key = line;
104                 } else {
105                     key.setTo(line, 0, colonPos);
106 
107                     if (key == "a=fmtp" || key == "a=rtpmap"
108                             || key == "a=framesize" || key == "a=extmap") {
109                         ssize_t spacePos = line.find(" ", colonPos + 1);
110                         if (spacePos < 0) {
111                             return false;
112                         }
113 
114                         key.setTo(line, 0, spacePos);
115 
116                         colonPos = spacePos;
117                     }
118 
119                     value.setTo(line, colonPos + 1, line.size() - colonPos - 1);
120                 }
121 
122                 key.trim();
123                 value.trim();
124 
125                 ALOGV("adding '%s' => '%s'", key.c_str(), value.c_str());
126 
127                 mTracks.editItemAt(mTracks.size() - 1).add(key, value);
128                 break;
129             }
130 
131             case 'm':
132             {
133                 ALOGV("new section '%s'",
134                      AString(line, 2, line.size() - 2).c_str());
135 
136                 mTracks.push(Attribs());
137                 mFormats.push(AString(line, 2, line.size() - 2));
138                 break;
139             }
140 
141             default:
142             {
143                 AString key, value;
144 
145                 ssize_t equalPos = line.find("=");
146                 /* The condition 'if (line.size() < 2 || line.c_str()[1] != '=')' a few lines above
147                  * ensures '=' is at position 1.  However for robustness we do the following check.
148                  */
149                 if (equalPos < 0) {
150                     return false;
151                 }
152 
153                 key = AString(line, 0, equalPos + 1);
154                 value = AString(line, equalPos + 1, line.size() - equalPos - 1);
155 
156                 key.trim();
157                 value.trim();
158 
159                 ALOGV("adding '%s' => '%s'", key.c_str(), value.c_str());
160 
161                 mTracks.editItemAt(mTracks.size() - 1).add(key, value);
162                 break;
163             }
164         }
165 
166         i = eolPos + 1;
167     }
168 
169     return true;
170 }
171 
isValid() const172 bool ASessionDescription::isValid() const {
173     return mIsValid;
174 }
175 
countTracks() const176 size_t ASessionDescription::countTracks() const {
177     return mTracks.size();
178 }
179 
getFormat(size_t index,AString * value) const180 void ASessionDescription::getFormat(size_t index, AString *value) const {
181     CHECK_GE(index, 0u);
182     CHECK_LT(index, mTracks.size());
183 
184     *value = mFormats.itemAt(index);
185 }
186 
findAttribute(size_t index,const char * key,AString * value) const187 bool ASessionDescription::findAttribute(
188         size_t index, const char *key, AString *value) const {
189     CHECK_GE(index, 0u);
190     CHECK_LT(index, mTracks.size());
191 
192     value->clear();
193 
194     const Attribs &track = mTracks.itemAt(index);
195     ssize_t i = track.indexOfKey(AString(key));
196 
197     if (i < 0) {
198         return false;
199     }
200 
201     *value = track.valueAt(i);
202 
203     return true;
204 }
205 
getCvoExtMap(size_t index,int32_t * cvoExtMap) const206 bool ASessionDescription::getCvoExtMap(
207         size_t index, int32_t *cvoExtMap) const {
208     CHECK_GE(index, 0u);
209     CHECK_LT(index, mTracks.size());
210 
211     AString key, value;
212     *cvoExtMap = 0;
213 
214     const Attribs &track = mTracks.itemAt(index);
215     for (size_t i = 0; i < track.size(); i++) {
216         value = track.valueAt(i);
217         if (value.size() > 0 && strcmp(value.c_str(), "urn:3gpp:video-orientation") == 0) {
218             key = track.keyAt(i);
219             break;
220         }
221     }
222 
223     if (key.size() > 0) {
224         const char *colonPos = strrchr(key.c_str(), ':');
225         colonPos++;
226         *cvoExtMap = atoi(colonPos);
227         return true;
228     }
229 
230     return false;
231 }
232 
getFormatType(size_t index,unsigned long * PT,AString * desc,AString * params) const233 void ASessionDescription::getFormatType(
234         size_t index, unsigned long *PT,
235         AString *desc, AString *params) const {
236     AString format;
237     getFormat(index, &format);
238 
239     const char *lastSpacePos = strrchr(format.c_str(), ' ');
240     CHECK(lastSpacePos != NULL);
241 
242     char *end;
243     unsigned long x = strtoul(lastSpacePos + 1, &end, 10);
244     CHECK_GT(end, lastSpacePos + 1);
245     CHECK_EQ(*end, '\0');
246 
247     *PT = x;
248 
249     char key[20];
250     snprintf(key, sizeof(key), "a=rtpmap:%lu", x);
251     if (findAttribute(index, key, desc)) {
252         snprintf(key, sizeof(key), "a=fmtp:%lu", x);
253         if (!findAttribute(index, key, params)) {
254             params->clear();
255         }
256     } else {
257         desc->clear();
258         params->clear();
259     }
260 }
261 
getDimensions(size_t index,unsigned long PT,int32_t * width,int32_t * height) const262 bool ASessionDescription::getDimensions(
263         size_t index, unsigned long PT,
264         int32_t *width, int32_t *height) const {
265     *width = 0;
266     *height = 0;
267 
268     char key[20];
269     snprintf(key, sizeof(key), "a=framesize:%lu", PT);
270     AString value;
271     if (!findAttribute(index, key, &value)) {
272         return false;
273     }
274 
275     const char *s = value.c_str();
276     char *end;
277     *width = strtoul(s, &end, 10);
278     CHECK_GT(end, s);
279     CHECK_EQ(*end, '-');
280 
281     s = end + 1;
282     *height = strtoul(s, &end, 10);
283     CHECK_GT(end, s);
284     CHECK_EQ(*end, '\0');
285 
286     return true;
287 }
288 
getDurationUs(int64_t * durationUs) const289 bool ASessionDescription::getDurationUs(int64_t *durationUs) const {
290     *durationUs = 0;
291 
292     CHECK(mIsValid);
293 
294     AString value;
295     if (!findAttribute(0, "a=range", &value)) {
296         return false;
297     }
298 
299     if (strncmp(value.c_str(), "npt=", 4) && strncmp(value.c_str(), "npt:", 4)) {
300         return false;
301     }
302 
303     float from, to;
304     if (!parseNTPRange(value.c_str() + 4, &from, &to)) {
305         return false;
306     }
307 
308     *durationUs = (int64_t)((to - from) * 1E6);
309 
310     return true;
311 }
312 
313 // static
ParseFormatDesc(const char * desc,int32_t * timescale,int32_t * numChannels)314 void ASessionDescription::ParseFormatDesc(
315         const char *desc, int32_t *timescale, int32_t *numChannels) {
316     const char *slash1 = strchr(desc, '/');
317     CHECK(slash1 != NULL);
318 
319     const char *s = slash1 + 1;
320     char *end;
321     unsigned long x = strtoul(s, &end, 10);
322     CHECK_GT(end, s);
323     CHECK(*end == '\0' || *end == '/');
324 
325     *timescale = x;
326     *numChannels = 1;
327 
328     if (*end == '/') {
329         s = end + 1;
330         unsigned long x = strtoul(s, &end, 10);
331         CHECK_GT(end, s);
332         CHECK_EQ(*end, '\0');
333 
334         *numChannels = x;
335     }
336 }
337 
338 // static
parseNTPRange(const char * s,float * npt1,float * npt2)339 bool ASessionDescription::parseNTPRange(
340         const char *s, float *npt1, float *npt2) {
341     if (s[0] == '-') {
342         return false;  // no start time available.
343     }
344 
345     if (!strncmp("now", s, 3)) {
346         return false;  // no absolute start time available
347     }
348 
349     char *end;
350     *npt1 = strtof(s, &end);
351 
352     if (end == s || *end != '-') {
353         // Failed to parse float or trailing "dash".
354         return false;
355     }
356 
357     s = end + 1;  // skip the dash.
358 
359     if (*s == '\0') {
360         *npt2 = FLT_MAX;  // open ended.
361         return true;
362     }
363 
364     if (!strncmp("now", s, 3)) {
365         return false;  // no absolute end time available
366     }
367 
368     *npt2 = strtof(s, &end);
369 
370     if (end == s || *end != '\0') {
371         return false;
372     }
373 
374     return *npt2 > *npt1;
375 }
376 
377 // static
SDPStringFactory(AString & sdp,const char * ip,bool isAudio,unsigned port,unsigned payloadType,unsigned as,const char * codec,const char * fmtp,int32_t width,int32_t height,int32_t cvoExtMap)378 void ASessionDescription::SDPStringFactory(AString &sdp,
379         const char *ip, bool isAudio, unsigned port, unsigned payloadType,
380         unsigned as, const char *codec, const char *fmtp,
381         int32_t width, int32_t height, int32_t cvoExtMap)
382 {
383     bool isIPv4 = (AString(ip).find("::") == -1) ? true : false;
384     sdp.clear();
385     sdp.append("v=0\r\n");
386 
387     sdp.append("a=range:npt=now-\r\n");
388 
389     sdp.append("m=");
390     sdp.append(isAudio ? "audio " : "video ");
391     sdp.append(port);
392     sdp.append(" RTP/AVP ");
393     sdp.append(payloadType);
394     sdp.append("\r\n");
395 
396     sdp.append("c= IN IP");
397     if (isIPv4) {
398         sdp.append("4 ");
399     } else {
400         sdp.append("6 ");
401     }
402     sdp.append(ip);
403     sdp.append("\r\n");
404 
405     sdp.append("b=AS:");
406     sdp.append(as > 0 ? as : kDefaultAs);
407     sdp.append("\r\n");
408 
409     sdp.append("a=rtpmap:");
410     sdp.append(payloadType);
411     sdp.append(" ");
412     sdp.append(codec);
413     sdp.append("/");
414     sdp.append(isAudio ? "8000" : "90000");
415     sdp.append("\r\n");
416 
417     if (fmtp != NULL) {
418         sdp.append("a=fmtp:");
419         sdp.append(payloadType);
420         sdp.append(" ");
421         sdp.append(fmtp);
422         sdp.append("\r\n");
423     }
424 
425     if (!isAudio && width > 0 && height > 0) {
426         sdp.append("a=framesize:");
427         sdp.append(payloadType);
428         sdp.append(" ");
429         sdp.append(width);
430         sdp.append("-");
431         sdp.append(height);
432         sdp.append("\r\n");
433     }
434 
435     if (cvoExtMap > 0) {
436         sdp.append("a=extmap:");
437         sdp.append(cvoExtMap);
438         sdp.append(" ");
439         sdp.append("urn:3gpp:video-orientation");
440         sdp.append("\r\n");
441     }
442 
443     ALOGV("SDPStringFactory => %s", sdp.c_str());
444 }
445 
446 }  // namespace android
447 
448