blob: cd17965749b532d625e1d7dbd278f5fb82ae2987 [file] [log] [blame]
Andreas Huber4456da52010-11-09 08:57:45 -08001/*
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 "VBRISeeker"
Mark Salyzyna5750e02014-06-18 16:34:45 -070019
20#include <inttypes.h>
21
Andreas Huber4456da52010-11-09 08:57:45 -080022#include <utils/Log.h>
23
Marco Nelissen75226172016-11-16 14:10:52 -080024#include "VBRISeeker.h"
Andreas Huber4456da52010-11-09 08:57:45 -080025
Dongwon Kangd91dc5a2017-10-10 00:07:09 -070026#include <media/stagefright/foundation/avc_utils.h>
Andreas Huber4456da52010-11-09 08:57:45 -080027
28#include <media/stagefright/foundation/ADebug.h>
Dongwon Kang60761282017-10-09 11:16:48 -070029#include <media/stagefright/foundation/ByteUtils.h>
Marco Nelissen2a243f02018-01-30 08:29:57 -080030#include <media/DataSourceBase.h>
Andreas Huber4456da52010-11-09 08:57:45 -080031
32namespace android {
33
34static uint32_t U24_AT(const uint8_t *ptr) {
35 return ptr[0] << 16 | ptr[1] << 8 | ptr[2];
36}
37
38// static
Marco Nelissen3d21ae32018-02-16 08:24:08 -080039VBRISeeker *VBRISeeker::CreateFromSource(
Marco Nelissen2a243f02018-01-30 08:29:57 -080040 DataSourceBase *source, off64_t post_id3_pos) {
James Dongc7fc37a2010-11-16 14:04:54 -080041 off64_t pos = post_id3_pos;
Andreas Huber4456da52010-11-09 08:57:45 -080042
43 uint8_t header[4];
44 ssize_t n = source->readAt(pos, header, sizeof(header));
45 if (n < (ssize_t)sizeof(header)) {
46 return NULL;
47 }
48
49 uint32_t tmp = U32_AT(&header[0]);
50 size_t frameSize;
51 int sampleRate;
Andreas Huber386d6092011-05-19 08:37:39 -070052 if (!GetMPEGAudioFrameSize(tmp, &frameSize, &sampleRate)) {
Andreas Huber4456da52010-11-09 08:57:45 -080053 return NULL;
54 }
55
56 // VBRI header follows 32 bytes after the header _ends_.
57 pos += sizeof(header) + 32;
58
59 uint8_t vbriHeader[26];
60 n = source->readAt(pos, vbriHeader, sizeof(vbriHeader));
61 if (n < (ssize_t)sizeof(vbriHeader)) {
62 return NULL;
63 }
64
65 if (memcmp(vbriHeader, "VBRI", 4)) {
66 return NULL;
67 }
68
69 size_t numFrames = U32_AT(&vbriHeader[14]);
70
71 int64_t durationUs =
Chih-Hung Hsieh3ebfbbc2018-12-11 13:49:18 -080072 numFrames * 1000000LL * (sampleRate >= 32000 ? 1152 : 576) / sampleRate;
Andreas Huber4456da52010-11-09 08:57:45 -080073
Steve Block3856b092011-10-20 11:56:00 +010074 ALOGV("duration = %.2f secs", durationUs / 1E6);
Andreas Huber4456da52010-11-09 08:57:45 -080075
76 size_t numEntries = U16_AT(&vbriHeader[18]);
77 size_t entrySize = U16_AT(&vbriHeader[22]);
78 size_t scale = U16_AT(&vbriHeader[20]);
79
Mark Salyzyna5750e02014-06-18 16:34:45 -070080 ALOGV("%zu entries, scale=%zu, size_per_entry=%zu",
Andreas Huber4456da52010-11-09 08:57:45 -080081 numEntries,
82 scale,
83 entrySize);
84
Marco Nelissen7fdd3642016-11-11 09:20:00 -080085 if (entrySize > 4) {
86 ALOGE("invalid VBRI entry size: %zu", entrySize);
87 return NULL;
88 }
89
Marco Nelissen3d21ae32018-02-16 08:24:08 -080090 VBRISeeker *seeker = new (std::nothrow) VBRISeeker;
Marco Nelissen7fdd3642016-11-11 09:20:00 -080091 if (seeker == NULL) {
92 ALOGW("Couldn't allocate VBRISeeker");
93 return NULL;
94 }
95
Andreas Huber4456da52010-11-09 08:57:45 -080096 size_t totalEntrySize = numEntries * entrySize;
Marco Nelissen7fdd3642016-11-11 09:20:00 -080097 uint8_t *buffer = new (std::nothrow) uint8_t[totalEntrySize];
98 if (!buffer) {
99 ALOGW("Couldn't allocate %zu bytes", totalEntrySize);
Marco Nelissen3d21ae32018-02-16 08:24:08 -0800100 delete seeker;
Marco Nelissen7fdd3642016-11-11 09:20:00 -0800101 return NULL;
102 }
Andreas Huber4456da52010-11-09 08:57:45 -0800103
104 n = source->readAt(pos + sizeof(vbriHeader), buffer, totalEntrySize);
105 if (n < (ssize_t)totalEntrySize) {
106 delete[] buffer;
107 buffer = NULL;
Marco Nelissen3d21ae32018-02-16 08:24:08 -0800108 delete seeker;
Andreas Huber4456da52010-11-09 08:57:45 -0800109 return NULL;
110 }
111
Marco Nelissen9e503852012-03-16 07:56:42 -0700112 seeker->mBasePos = post_id3_pos + frameSize;
Marco Nelissen5fd7d3a2012-06-13 08:51:00 -0700113 // only update mDurationUs if the calculated duration is valid (non zero)
114 // otherwise, leave duration at -1 so that getDuration() and getOffsetForTime()
115 // return false when called, to indicate that this vbri tag does not have the
116 // requested information
117 if (durationUs) {
118 seeker->mDurationUs = durationUs;
119 }
Andreas Huber4456da52010-11-09 08:57:45 -0800120
James Dongc7fc37a2010-11-16 14:04:54 -0800121 off64_t offset = post_id3_pos;
Andreas Huber4456da52010-11-09 08:57:45 -0800122 for (size_t i = 0; i < numEntries; ++i) {
123 uint32_t numBytes;
124 switch (entrySize) {
125 case 1: numBytes = buffer[i]; break;
126 case 2: numBytes = U16_AT(buffer + 2 * i); break;
127 case 3: numBytes = U24_AT(buffer + 3 * i); break;
128 default:
129 {
130 CHECK_EQ(entrySize, 4u);
131 numBytes = U32_AT(buffer + 4 * i); break;
132 }
133 }
134
135 numBytes *= scale;
136
137 seeker->mSegments.push(numBytes);
138
Lajos Molnaree4e1b12015-04-17 13:46:19 -0700139 ALOGV("entry #%zu: %u offset %#016llx", i, numBytes, (long long)offset);
Andreas Huber4456da52010-11-09 08:57:45 -0800140 offset += numBytes;
141 }
142
143 delete[] buffer;
144 buffer = NULL;
145
Steve Blockdf64d152012-01-04 20:05:49 +0000146 ALOGI("Found VBRI header.");
Andreas Huber4456da52010-11-09 08:57:45 -0800147
148 return seeker;
149}
150
151VBRISeeker::VBRISeeker()
152 : mDurationUs(-1) {
153}
154
155bool VBRISeeker::getDuration(int64_t *durationUs) {
156 if (mDurationUs < 0) {
157 return false;
158 }
159
160 *durationUs = mDurationUs;
161
162 return true;
163}
164
James Dongc7fc37a2010-11-16 14:04:54 -0800165bool VBRISeeker::getOffsetForTime(int64_t *timeUs, off64_t *pos) {
Wei Jiae206ba02015-10-02 16:40:49 -0700166 if (mDurationUs < 0 || mSegments.size() == 0) {
Andreas Huber4456da52010-11-09 08:57:45 -0800167 return false;
168 }
169
170 int64_t segmentDurationUs = mDurationUs / mSegments.size();
171
172 int64_t nowUs = 0;
173 *pos = mBasePos;
174 size_t segmentIndex = 0;
175 while (segmentIndex < mSegments.size() && nowUs < *timeUs) {
176 nowUs += segmentDurationUs;
177 *pos += mSegments.itemAt(segmentIndex++);
178 }
179
Lajos Molnaree4e1b12015-04-17 13:46:19 -0700180 ALOGV("getOffsetForTime %lld us => 0x%016llx", (long long)*timeUs, (long long)*pos);
Andreas Huber4456da52010-11-09 08:57:45 -0800181
182 *timeUs = nowUs;
183
184 return true;
185}
186
187} // namespace android
188