diff options
author | Andreas Huber <andih@google.com> | 2010-06-07 15:19:40 -0700 |
---|---|---|
committer | Andreas Huber <andih@google.com> | 2010-06-07 15:34:45 -0700 |
commit | cf7b9c7aae758ac0b99833915053c63c2ac46e09 (patch) | |
tree | 2480a2da149f7ba8fe651876a6173947ec8c6a16 /media/libstagefright/rtsp/ASessionDescription.cpp | |
parent | e71d10e7ad55ccbcb0756c007caef1c959090384 (diff) | |
download | frameworks_av-cf7b9c7aae758ac0b99833915053c63c2ac46e09.zip frameworks_av-cf7b9c7aae758ac0b99833915053c63c2ac46e09.tar.gz frameworks_av-cf7b9c7aae758ac0b99833915053c63c2ac46e09.tar.bz2 |
Initial checkin of preliminary rtsp support for stagefright.
Change-Id: I0722aa888098c0c1361c97a4c1b123d910afc207
Diffstat (limited to 'media/libstagefright/rtsp/ASessionDescription.cpp')
-rw-r--r-- | media/libstagefright/rtsp/ASessionDescription.cpp | 270 |
1 files changed, 270 insertions, 0 deletions
diff --git a/media/libstagefright/rtsp/ASessionDescription.cpp b/media/libstagefright/rtsp/ASessionDescription.cpp new file mode 100644 index 0000000..ca4c55e --- /dev/null +++ b/media/libstagefright/rtsp/ASessionDescription.cpp @@ -0,0 +1,270 @@ +/* + * Copyright (C) 2010 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include "ASessionDescription.h" + +#include <media/stagefright/foundation/ADebug.h> +#include <media/stagefright/foundation/AString.h> + +#include <stdlib.h> + +namespace android { + +ASessionDescription::ASessionDescription() + : mIsValid(false) { +} + +ASessionDescription::~ASessionDescription() { +} + +bool ASessionDescription::setTo(const void *data, size_t size) { + mIsValid = parse(data, size); + + if (!mIsValid) { + mTracks.clear(); + mFormats.clear(); + } + + return mIsValid; +} + +bool ASessionDescription::parse(const void *data, size_t size) { + mTracks.clear(); + mFormats.clear(); + + mTracks.push(Attribs()); + mFormats.push(AString("[root]")); + + AString desc((const char *)data, size); + LOG(VERBOSE) << desc; + + size_t i = 0; + for (;;) { + ssize_t eolPos = desc.find("\r\n", i); + if (eolPos < 0) { + break; + } + + AString line(desc, i, eolPos - i); + + if (line.size() < 2 || line.c_str()[1] != '=') { + return false; + } + + switch (line.c_str()[0]) { + case 'v': + { + if (strcmp(line.c_str(), "v=0")) { + return false; + } + break; + } + + case 'a': + case 'b': + { + AString key, value; + + ssize_t colonPos = line.find(":", 2); + if (colonPos < 0) { + key = line; + } else { + key.setTo(line, 0, colonPos); + + if (key == "a=fmtp" || key == "a=rtpmap" + || key == "a=framesize") { + ssize_t spacePos = line.find(" ", colonPos + 1); + if (spacePos < 0) { + return false; + } + + key.setTo(line, 0, spacePos); + + colonPos = spacePos; + } + + value.setTo(line, colonPos + 1, line.size() - colonPos - 1); + } + + key.trim(); + value.trim(); + + LOG(VERBOSE) << "adding '" << key << "' => '" << value << "'"; + + mTracks.editItemAt(mTracks.size() - 1).add(key, value); + break; + } + + case 'm': + { + LOG(VERBOSE) << "new section '" << AString(line, 2, line.size() - 2) << "'"; + + mTracks.push(Attribs()); + mFormats.push(AString(line, 2, line.size() - 2)); + break; + } + } + + i = eolPos + 2; + } + + return true; +} + +bool ASessionDescription::isValid() const { + return mIsValid; +} + +size_t ASessionDescription::countTracks() const { + return mTracks.size(); +} + +void ASessionDescription::getFormat(size_t index, AString *value) const { + CHECK_GE(index, 0u); + CHECK_LT(index, mTracks.size()); + + *value = mFormats.itemAt(index); +} + +bool ASessionDescription::findAttribute( + size_t index, const char *key, AString *value) const { + CHECK_GE(index, 0u); + CHECK_LT(index, mTracks.size()); + + value->clear(); + + const Attribs &track = mTracks.itemAt(index); + ssize_t i = track.indexOfKey(AString(key)); + + if (i < 0) { + return false; + } + + *value = track.valueAt(i); + + return true; +} + +void ASessionDescription::getFormatType( + size_t index, unsigned long *PT, + AString *desc, AString *params) const { + AString format; + getFormat(index, &format); + + char *lastSpacePos = strrchr(format.c_str(), ' '); + CHECK(lastSpacePos != NULL); + + char *end; + unsigned long x = strtoul(lastSpacePos + 1, &end, 10); + CHECK_GT(end, lastSpacePos + 1); + CHECK_EQ(*end, '\0'); + + *PT = x; + + char key[20]; + sprintf(key, "a=rtpmap:%lu", x); + + CHECK(findAttribute(index, key, desc)); + + sprintf(key, "a=fmtp:%lu", x); + if (!findAttribute(index, key, params)) { + params->clear(); + } +} + +void ASessionDescription::getDimensions( + size_t index, unsigned long PT, + int32_t *width, int32_t *height) const { + char key[20]; + sprintf(key, "a=framesize:%lu", PT); + AString value; + CHECK(findAttribute(index, key, &value)); + + const char *s = value.c_str(); + char *end; + *width = strtoul(s, &end, 10); + CHECK_GT(end, s); + CHECK_EQ(*end, '-'); + + s = end + 1; + *height = strtoul(s, &end, 10); + CHECK_GT(end, s); + CHECK_EQ(*end, '\0'); +} + +bool ASessionDescription::getDurationUs(int64_t *durationUs) const { + *durationUs = 0; + + CHECK(mIsValid); + + AString value; + if (!findAttribute(0, "a=range", &value)) { + return false; + } + + if (value == "npt=now-") { + return false; + } + + if (strncmp(value.c_str(), "npt=", 4)) { + return false; + } + + const char *s = value.c_str() + 4; + char *end; + double from = strtod(s, &end); + CHECK_GT(end, s); + CHECK_EQ(*end, '-'); + + s = end + 1; + double to = strtod(s, &end); + CHECK_GT(end, s); + CHECK_EQ(*end, '\0'); + + CHECK_GE(to, from); + + *durationUs = (int64_t)((to - from) * 1E6); + + return true; +} + +// static +void ASessionDescription::ParseFormatDesc( + const char *desc, int32_t *timescale, int32_t *numChannels) { + const char *slash1 = strchr(desc, '/'); + CHECK(slash1 != NULL); + + const char *s = slash1 + 1; + char *end; + unsigned long x = strtoul(s, &end, 10); + CHECK_GT(end, s); + CHECK(*end == '\0' || *end == '/'); + + *timescale = x; + *numChannels = 1; + + if (*end == '/') { + s = end + 1; + unsigned long x = strtoul(s, &end, 10); + CHECK_GT(end, s); + CHECK_EQ(*end, '\0'); + + *numChannels = x; + } +} + +} // namespace android + |