1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker * Copyright (C) 2019 The Android Open Source Project
3*4d7e907cSAndroid Build Coastguard Worker *
4*4d7e907cSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*4d7e907cSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*4d7e907cSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*4d7e907cSAndroid Build Coastguard Worker *
8*4d7e907cSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*4d7e907cSAndroid Build Coastguard Worker *
10*4d7e907cSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*4d7e907cSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*4d7e907cSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4d7e907cSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*4d7e907cSAndroid Build Coastguard Worker * limitations under the License.
15*4d7e907cSAndroid Build Coastguard Worker */
16*4d7e907cSAndroid Build Coastguard Worker
17*4d7e907cSAndroid Build Coastguard Worker #define LOG_TAG "[email protected]"
18*4d7e907cSAndroid Build Coastguard Worker
19*4d7e907cSAndroid Build Coastguard Worker #include "Dvr.h"
20*4d7e907cSAndroid Build Coastguard Worker #include <utils/Log.h>
21*4d7e907cSAndroid Build Coastguard Worker
22*4d7e907cSAndroid Build Coastguard Worker namespace android {
23*4d7e907cSAndroid Build Coastguard Worker namespace hardware {
24*4d7e907cSAndroid Build Coastguard Worker namespace tv {
25*4d7e907cSAndroid Build Coastguard Worker namespace tuner {
26*4d7e907cSAndroid Build Coastguard Worker namespace V1_0 {
27*4d7e907cSAndroid Build Coastguard Worker namespace implementation {
28*4d7e907cSAndroid Build Coastguard Worker
29*4d7e907cSAndroid Build Coastguard Worker #define WAIT_TIMEOUT 3000000000
30*4d7e907cSAndroid Build Coastguard Worker
Dvr()31*4d7e907cSAndroid Build Coastguard Worker Dvr::Dvr() {}
32*4d7e907cSAndroid Build Coastguard Worker
Dvr(DvrType type,uint32_t bufferSize,const sp<IDvrCallback> & cb,sp<Demux> demux)33*4d7e907cSAndroid Build Coastguard Worker Dvr::Dvr(DvrType type, uint32_t bufferSize, const sp<IDvrCallback>& cb, sp<Demux> demux) {
34*4d7e907cSAndroid Build Coastguard Worker mType = type;
35*4d7e907cSAndroid Build Coastguard Worker mBufferSize = bufferSize;
36*4d7e907cSAndroid Build Coastguard Worker mCallback = cb;
37*4d7e907cSAndroid Build Coastguard Worker mDemux = demux;
38*4d7e907cSAndroid Build Coastguard Worker }
39*4d7e907cSAndroid Build Coastguard Worker
~Dvr()40*4d7e907cSAndroid Build Coastguard Worker Dvr::~Dvr() {
41*4d7e907cSAndroid Build Coastguard Worker // make sure thread has joined
42*4d7e907cSAndroid Build Coastguard Worker close();
43*4d7e907cSAndroid Build Coastguard Worker }
44*4d7e907cSAndroid Build Coastguard Worker
getQueueDesc(getQueueDesc_cb _hidl_cb)45*4d7e907cSAndroid Build Coastguard Worker Return<void> Dvr::getQueueDesc(getQueueDesc_cb _hidl_cb) {
46*4d7e907cSAndroid Build Coastguard Worker ALOGV("%s", __FUNCTION__);
47*4d7e907cSAndroid Build Coastguard Worker
48*4d7e907cSAndroid Build Coastguard Worker _hidl_cb(Result::SUCCESS, *mDvrMQ->getDesc());
49*4d7e907cSAndroid Build Coastguard Worker return Void();
50*4d7e907cSAndroid Build Coastguard Worker }
51*4d7e907cSAndroid Build Coastguard Worker
configure(const DvrSettings & settings)52*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::configure(const DvrSettings& settings) {
53*4d7e907cSAndroid Build Coastguard Worker ALOGV("%s", __FUNCTION__);
54*4d7e907cSAndroid Build Coastguard Worker
55*4d7e907cSAndroid Build Coastguard Worker mDvrSettings = settings;
56*4d7e907cSAndroid Build Coastguard Worker mDvrConfigured = true;
57*4d7e907cSAndroid Build Coastguard Worker
58*4d7e907cSAndroid Build Coastguard Worker return Result::SUCCESS;
59*4d7e907cSAndroid Build Coastguard Worker }
60*4d7e907cSAndroid Build Coastguard Worker
attachFilter(const sp<IFilter> & filter)61*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::attachFilter(const sp<IFilter>& filter) {
62*4d7e907cSAndroid Build Coastguard Worker ALOGV("%s", __FUNCTION__);
63*4d7e907cSAndroid Build Coastguard Worker
64*4d7e907cSAndroid Build Coastguard Worker uint32_t filterId;
65*4d7e907cSAndroid Build Coastguard Worker Result status;
66*4d7e907cSAndroid Build Coastguard Worker
67*4d7e907cSAndroid Build Coastguard Worker filter->getId([&](Result result, uint32_t id) {
68*4d7e907cSAndroid Build Coastguard Worker filterId = id;
69*4d7e907cSAndroid Build Coastguard Worker status = result;
70*4d7e907cSAndroid Build Coastguard Worker });
71*4d7e907cSAndroid Build Coastguard Worker
72*4d7e907cSAndroid Build Coastguard Worker if (status != Result::SUCCESS) {
73*4d7e907cSAndroid Build Coastguard Worker return status;
74*4d7e907cSAndroid Build Coastguard Worker }
75*4d7e907cSAndroid Build Coastguard Worker
76*4d7e907cSAndroid Build Coastguard Worker // TODO check if the attached filter is a record filter
77*4d7e907cSAndroid Build Coastguard Worker if (!mDemux->attachRecordFilter(filterId)) {
78*4d7e907cSAndroid Build Coastguard Worker return Result::INVALID_ARGUMENT;
79*4d7e907cSAndroid Build Coastguard Worker }
80*4d7e907cSAndroid Build Coastguard Worker
81*4d7e907cSAndroid Build Coastguard Worker return Result::SUCCESS;
82*4d7e907cSAndroid Build Coastguard Worker }
83*4d7e907cSAndroid Build Coastguard Worker
detachFilter(const sp<IFilter> & filter)84*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::detachFilter(const sp<IFilter>& filter) {
85*4d7e907cSAndroid Build Coastguard Worker ALOGV("%s", __FUNCTION__);
86*4d7e907cSAndroid Build Coastguard Worker
87*4d7e907cSAndroid Build Coastguard Worker uint32_t filterId;
88*4d7e907cSAndroid Build Coastguard Worker Result status;
89*4d7e907cSAndroid Build Coastguard Worker
90*4d7e907cSAndroid Build Coastguard Worker filter->getId([&](Result result, uint32_t id) {
91*4d7e907cSAndroid Build Coastguard Worker filterId = id;
92*4d7e907cSAndroid Build Coastguard Worker status = result;
93*4d7e907cSAndroid Build Coastguard Worker });
94*4d7e907cSAndroid Build Coastguard Worker
95*4d7e907cSAndroid Build Coastguard Worker if (status != Result::SUCCESS) {
96*4d7e907cSAndroid Build Coastguard Worker return status;
97*4d7e907cSAndroid Build Coastguard Worker }
98*4d7e907cSAndroid Build Coastguard Worker
99*4d7e907cSAndroid Build Coastguard Worker if (!mDemux->detachRecordFilter(filterId)) {
100*4d7e907cSAndroid Build Coastguard Worker return Result::INVALID_ARGUMENT;
101*4d7e907cSAndroid Build Coastguard Worker }
102*4d7e907cSAndroid Build Coastguard Worker
103*4d7e907cSAndroid Build Coastguard Worker return Result::SUCCESS;
104*4d7e907cSAndroid Build Coastguard Worker }
105*4d7e907cSAndroid Build Coastguard Worker
start()106*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::start() {
107*4d7e907cSAndroid Build Coastguard Worker ALOGV("%s", __FUNCTION__);
108*4d7e907cSAndroid Build Coastguard Worker
109*4d7e907cSAndroid Build Coastguard Worker if (!mCallback) {
110*4d7e907cSAndroid Build Coastguard Worker return Result::NOT_INITIALIZED;
111*4d7e907cSAndroid Build Coastguard Worker }
112*4d7e907cSAndroid Build Coastguard Worker
113*4d7e907cSAndroid Build Coastguard Worker if (!mDvrConfigured) {
114*4d7e907cSAndroid Build Coastguard Worker return Result::INVALID_STATE;
115*4d7e907cSAndroid Build Coastguard Worker }
116*4d7e907cSAndroid Build Coastguard Worker
117*4d7e907cSAndroid Build Coastguard Worker if (mType == DvrType::PLAYBACK) {
118*4d7e907cSAndroid Build Coastguard Worker mDvrThread = std::thread(&Dvr::playbackThreadLoop, this);
119*4d7e907cSAndroid Build Coastguard Worker } else if (mType == DvrType::RECORD) {
120*4d7e907cSAndroid Build Coastguard Worker mRecordStatus = RecordStatus::DATA_READY;
121*4d7e907cSAndroid Build Coastguard Worker mDemux->setIsRecording(mType == DvrType::RECORD);
122*4d7e907cSAndroid Build Coastguard Worker }
123*4d7e907cSAndroid Build Coastguard Worker
124*4d7e907cSAndroid Build Coastguard Worker // TODO start another thread to send filter status callback to the framework
125*4d7e907cSAndroid Build Coastguard Worker
126*4d7e907cSAndroid Build Coastguard Worker return Result::SUCCESS;
127*4d7e907cSAndroid Build Coastguard Worker }
128*4d7e907cSAndroid Build Coastguard Worker
stop()129*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::stop() {
130*4d7e907cSAndroid Build Coastguard Worker ALOGV("%s", __FUNCTION__);
131*4d7e907cSAndroid Build Coastguard Worker
132*4d7e907cSAndroid Build Coastguard Worker mDvrThreadRunning = false;
133*4d7e907cSAndroid Build Coastguard Worker if (mDvrThread.joinable()) {
134*4d7e907cSAndroid Build Coastguard Worker mDvrThread.join();
135*4d7e907cSAndroid Build Coastguard Worker }
136*4d7e907cSAndroid Build Coastguard Worker // thread should always be joinable if it is running,
137*4d7e907cSAndroid Build Coastguard Worker // so it should be safe to assume recording stopped.
138*4d7e907cSAndroid Build Coastguard Worker mDemux->setIsRecording(false);
139*4d7e907cSAndroid Build Coastguard Worker
140*4d7e907cSAndroid Build Coastguard Worker return Result::SUCCESS;
141*4d7e907cSAndroid Build Coastguard Worker }
142*4d7e907cSAndroid Build Coastguard Worker
flush()143*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::flush() {
144*4d7e907cSAndroid Build Coastguard Worker ALOGV("%s", __FUNCTION__);
145*4d7e907cSAndroid Build Coastguard Worker
146*4d7e907cSAndroid Build Coastguard Worker mRecordStatus = RecordStatus::DATA_READY;
147*4d7e907cSAndroid Build Coastguard Worker
148*4d7e907cSAndroid Build Coastguard Worker return Result::SUCCESS;
149*4d7e907cSAndroid Build Coastguard Worker }
150*4d7e907cSAndroid Build Coastguard Worker
close()151*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::close() {
152*4d7e907cSAndroid Build Coastguard Worker ALOGV("%s", __FUNCTION__);
153*4d7e907cSAndroid Build Coastguard Worker stop();
154*4d7e907cSAndroid Build Coastguard Worker return Result::SUCCESS;
155*4d7e907cSAndroid Build Coastguard Worker }
156*4d7e907cSAndroid Build Coastguard Worker
createDvrMQ()157*4d7e907cSAndroid Build Coastguard Worker bool Dvr::createDvrMQ() {
158*4d7e907cSAndroid Build Coastguard Worker ALOGV("%s", __FUNCTION__);
159*4d7e907cSAndroid Build Coastguard Worker
160*4d7e907cSAndroid Build Coastguard Worker // Create a synchronized FMQ that supports blocking read/write
161*4d7e907cSAndroid Build Coastguard Worker unique_ptr<DvrMQ> tmpDvrMQ = unique_ptr<DvrMQ>(new (nothrow) DvrMQ(mBufferSize, true));
162*4d7e907cSAndroid Build Coastguard Worker if (!tmpDvrMQ->isValid()) {
163*4d7e907cSAndroid Build Coastguard Worker ALOGW("[Dvr] Failed to create FMQ of DVR");
164*4d7e907cSAndroid Build Coastguard Worker return false;
165*4d7e907cSAndroid Build Coastguard Worker }
166*4d7e907cSAndroid Build Coastguard Worker
167*4d7e907cSAndroid Build Coastguard Worker mDvrMQ = std::move(tmpDvrMQ);
168*4d7e907cSAndroid Build Coastguard Worker
169*4d7e907cSAndroid Build Coastguard Worker if (EventFlag::createEventFlag(mDvrMQ->getEventFlagWord(), &mDvrEventFlag) != OK) {
170*4d7e907cSAndroid Build Coastguard Worker return false;
171*4d7e907cSAndroid Build Coastguard Worker }
172*4d7e907cSAndroid Build Coastguard Worker
173*4d7e907cSAndroid Build Coastguard Worker return true;
174*4d7e907cSAndroid Build Coastguard Worker }
175*4d7e907cSAndroid Build Coastguard Worker
getDvrEventFlag()176*4d7e907cSAndroid Build Coastguard Worker EventFlag* Dvr::getDvrEventFlag() {
177*4d7e907cSAndroid Build Coastguard Worker return mDvrEventFlag;
178*4d7e907cSAndroid Build Coastguard Worker }
179*4d7e907cSAndroid Build Coastguard Worker
playbackThreadLoop()180*4d7e907cSAndroid Build Coastguard Worker void Dvr::playbackThreadLoop() {
181*4d7e907cSAndroid Build Coastguard Worker ALOGD("[Dvr] playback threadLoop start.");
182*4d7e907cSAndroid Build Coastguard Worker mDvrThreadRunning = true;
183*4d7e907cSAndroid Build Coastguard Worker
184*4d7e907cSAndroid Build Coastguard Worker while (mDvrThreadRunning) {
185*4d7e907cSAndroid Build Coastguard Worker uint32_t efState = 0;
186*4d7e907cSAndroid Build Coastguard Worker status_t status =
187*4d7e907cSAndroid Build Coastguard Worker mDvrEventFlag->wait(static_cast<uint32_t>(DemuxQueueNotifyBits::DATA_READY),
188*4d7e907cSAndroid Build Coastguard Worker &efState, WAIT_TIMEOUT, true /* retry on spurious wake */);
189*4d7e907cSAndroid Build Coastguard Worker if (status != OK) {
190*4d7e907cSAndroid Build Coastguard Worker ALOGD("[Dvr] wait for data ready on the playback FMQ");
191*4d7e907cSAndroid Build Coastguard Worker continue;
192*4d7e907cSAndroid Build Coastguard Worker }
193*4d7e907cSAndroid Build Coastguard Worker
194*4d7e907cSAndroid Build Coastguard Worker if (mDvrSettings.playback().dataFormat == DataFormat::ES) {
195*4d7e907cSAndroid Build Coastguard Worker if (!processEsDataOnPlayback(false /*isVirtualFrontend*/, false /*isRecording*/)) {
196*4d7e907cSAndroid Build Coastguard Worker ALOGE("[Dvr] playback es data failed to be filtered. Ending thread");
197*4d7e907cSAndroid Build Coastguard Worker break;
198*4d7e907cSAndroid Build Coastguard Worker }
199*4d7e907cSAndroid Build Coastguard Worker maySendPlaybackStatusCallback();
200*4d7e907cSAndroid Build Coastguard Worker }
201*4d7e907cSAndroid Build Coastguard Worker // Our current implementation filter the data and write it into the filter FMQ immediately
202*4d7e907cSAndroid Build Coastguard Worker // after the DATA_READY from the VTS/framework
203*4d7e907cSAndroid Build Coastguard Worker if (!readPlaybackFMQ(false /*isVirtualFrontend*/, false /*isRecording*/) ||
204*4d7e907cSAndroid Build Coastguard Worker !startFilterDispatcher(false /*isVirtualFrontend*/, false /*isRecording*/)) {
205*4d7e907cSAndroid Build Coastguard Worker ALOGE("[Dvr] playback data failed to be filtered. Ending thread");
206*4d7e907cSAndroid Build Coastguard Worker break;
207*4d7e907cSAndroid Build Coastguard Worker }
208*4d7e907cSAndroid Build Coastguard Worker
209*4d7e907cSAndroid Build Coastguard Worker maySendPlaybackStatusCallback();
210*4d7e907cSAndroid Build Coastguard Worker }
211*4d7e907cSAndroid Build Coastguard Worker
212*4d7e907cSAndroid Build Coastguard Worker mDvrThreadRunning = false;
213*4d7e907cSAndroid Build Coastguard Worker ALOGD("[Dvr] playback thread ended.");
214*4d7e907cSAndroid Build Coastguard Worker }
215*4d7e907cSAndroid Build Coastguard Worker
maySendPlaybackStatusCallback()216*4d7e907cSAndroid Build Coastguard Worker void Dvr::maySendPlaybackStatusCallback() {
217*4d7e907cSAndroid Build Coastguard Worker lock_guard<mutex> lock(mPlaybackStatusLock);
218*4d7e907cSAndroid Build Coastguard Worker int availableToRead = mDvrMQ->availableToRead();
219*4d7e907cSAndroid Build Coastguard Worker int availableToWrite = mDvrMQ->availableToWrite();
220*4d7e907cSAndroid Build Coastguard Worker
221*4d7e907cSAndroid Build Coastguard Worker PlaybackStatus newStatus = checkPlaybackStatusChange(availableToWrite, availableToRead,
222*4d7e907cSAndroid Build Coastguard Worker mDvrSettings.playback().highThreshold,
223*4d7e907cSAndroid Build Coastguard Worker mDvrSettings.playback().lowThreshold);
224*4d7e907cSAndroid Build Coastguard Worker if (mPlaybackStatus != newStatus) {
225*4d7e907cSAndroid Build Coastguard Worker mCallback->onPlaybackStatus(newStatus);
226*4d7e907cSAndroid Build Coastguard Worker mPlaybackStatus = newStatus;
227*4d7e907cSAndroid Build Coastguard Worker }
228*4d7e907cSAndroid Build Coastguard Worker }
229*4d7e907cSAndroid Build Coastguard Worker
checkPlaybackStatusChange(uint32_t availableToWrite,uint32_t availableToRead,uint32_t highThreshold,uint32_t lowThreshold)230*4d7e907cSAndroid Build Coastguard Worker PlaybackStatus Dvr::checkPlaybackStatusChange(uint32_t availableToWrite, uint32_t availableToRead,
231*4d7e907cSAndroid Build Coastguard Worker uint32_t highThreshold, uint32_t lowThreshold) {
232*4d7e907cSAndroid Build Coastguard Worker if (availableToWrite == 0) {
233*4d7e907cSAndroid Build Coastguard Worker return PlaybackStatus::SPACE_FULL;
234*4d7e907cSAndroid Build Coastguard Worker } else if (availableToRead > highThreshold) {
235*4d7e907cSAndroid Build Coastguard Worker return PlaybackStatus::SPACE_ALMOST_FULL;
236*4d7e907cSAndroid Build Coastguard Worker } else if (availableToRead < lowThreshold) {
237*4d7e907cSAndroid Build Coastguard Worker return PlaybackStatus::SPACE_ALMOST_EMPTY;
238*4d7e907cSAndroid Build Coastguard Worker } else if (availableToRead == 0) {
239*4d7e907cSAndroid Build Coastguard Worker return PlaybackStatus::SPACE_EMPTY;
240*4d7e907cSAndroid Build Coastguard Worker }
241*4d7e907cSAndroid Build Coastguard Worker return mPlaybackStatus;
242*4d7e907cSAndroid Build Coastguard Worker }
243*4d7e907cSAndroid Build Coastguard Worker
readPlaybackFMQ(bool isVirtualFrontend,bool isRecording)244*4d7e907cSAndroid Build Coastguard Worker bool Dvr::readPlaybackFMQ(bool isVirtualFrontend, bool isRecording) {
245*4d7e907cSAndroid Build Coastguard Worker // Read playback data from the input FMQ
246*4d7e907cSAndroid Build Coastguard Worker int size = mDvrMQ->availableToRead();
247*4d7e907cSAndroid Build Coastguard Worker int playbackPacketSize = mDvrSettings.playback().packetSize;
248*4d7e907cSAndroid Build Coastguard Worker vector<uint8_t> dataOutputBuffer;
249*4d7e907cSAndroid Build Coastguard Worker dataOutputBuffer.resize(playbackPacketSize);
250*4d7e907cSAndroid Build Coastguard Worker // Dispatch the packet to the PID matching filter output buffer
251*4d7e907cSAndroid Build Coastguard Worker for (int i = 0; i < size / playbackPacketSize; i++) {
252*4d7e907cSAndroid Build Coastguard Worker if (!mDvrMQ->read(dataOutputBuffer.data(), playbackPacketSize)) {
253*4d7e907cSAndroid Build Coastguard Worker return false;
254*4d7e907cSAndroid Build Coastguard Worker }
255*4d7e907cSAndroid Build Coastguard Worker if (isVirtualFrontend) {
256*4d7e907cSAndroid Build Coastguard Worker if (isRecording) {
257*4d7e907cSAndroid Build Coastguard Worker mDemux->sendFrontendInputToRecord(dataOutputBuffer);
258*4d7e907cSAndroid Build Coastguard Worker } else {
259*4d7e907cSAndroid Build Coastguard Worker mDemux->startBroadcastTsFilter(dataOutputBuffer);
260*4d7e907cSAndroid Build Coastguard Worker }
261*4d7e907cSAndroid Build Coastguard Worker } else {
262*4d7e907cSAndroid Build Coastguard Worker startTpidFilter(dataOutputBuffer);
263*4d7e907cSAndroid Build Coastguard Worker }
264*4d7e907cSAndroid Build Coastguard Worker }
265*4d7e907cSAndroid Build Coastguard Worker
266*4d7e907cSAndroid Build Coastguard Worker return true;
267*4d7e907cSAndroid Build Coastguard Worker }
268*4d7e907cSAndroid Build Coastguard Worker
processEsDataOnPlayback(bool isVirtualFrontend,bool isRecording)269*4d7e907cSAndroid Build Coastguard Worker bool Dvr::processEsDataOnPlayback(bool isVirtualFrontend, bool isRecording) {
270*4d7e907cSAndroid Build Coastguard Worker // Read ES from the DVR FMQ
271*4d7e907cSAndroid Build Coastguard Worker // Note that currently we only provides ES with metaData in a specific format to be parsed.
272*4d7e907cSAndroid Build Coastguard Worker // The ES size should be smaller than the Playback FMQ size to avoid reading truncated data.
273*4d7e907cSAndroid Build Coastguard Worker int size = mDvrMQ->availableToRead();
274*4d7e907cSAndroid Build Coastguard Worker vector<uint8_t> dataOutputBuffer;
275*4d7e907cSAndroid Build Coastguard Worker dataOutputBuffer.resize(size);
276*4d7e907cSAndroid Build Coastguard Worker if (!mDvrMQ->read(dataOutputBuffer.data(), size)) {
277*4d7e907cSAndroid Build Coastguard Worker return false;
278*4d7e907cSAndroid Build Coastguard Worker }
279*4d7e907cSAndroid Build Coastguard Worker
280*4d7e907cSAndroid Build Coastguard Worker int metaDataSize = size;
281*4d7e907cSAndroid Build Coastguard Worker int totalFrames = 0;
282*4d7e907cSAndroid Build Coastguard Worker int videoEsDataSize = 0;
283*4d7e907cSAndroid Build Coastguard Worker int audioEsDataSize = 0;
284*4d7e907cSAndroid Build Coastguard Worker int audioPid = 0;
285*4d7e907cSAndroid Build Coastguard Worker int videoPid = 0;
286*4d7e907cSAndroid Build Coastguard Worker
287*4d7e907cSAndroid Build Coastguard Worker vector<MediaEsMetaData> esMeta;
288*4d7e907cSAndroid Build Coastguard Worker int videoReadPointer = 0;
289*4d7e907cSAndroid Build Coastguard Worker int audioReadPointer = 0;
290*4d7e907cSAndroid Build Coastguard Worker int frameCount = 0;
291*4d7e907cSAndroid Build Coastguard Worker // Get meta data from the es
292*4d7e907cSAndroid Build Coastguard Worker for (int i = 0; i < metaDataSize; i++) {
293*4d7e907cSAndroid Build Coastguard Worker switch (dataOutputBuffer[i]) {
294*4d7e907cSAndroid Build Coastguard Worker case 'm':
295*4d7e907cSAndroid Build Coastguard Worker metaDataSize = 0;
296*4d7e907cSAndroid Build Coastguard Worker getMetaDataValue(i, dataOutputBuffer.data(), metaDataSize);
297*4d7e907cSAndroid Build Coastguard Worker videoReadPointer = metaDataSize;
298*4d7e907cSAndroid Build Coastguard Worker continue;
299*4d7e907cSAndroid Build Coastguard Worker case 'l':
300*4d7e907cSAndroid Build Coastguard Worker getMetaDataValue(i, dataOutputBuffer.data(), totalFrames);
301*4d7e907cSAndroid Build Coastguard Worker esMeta.resize(totalFrames);
302*4d7e907cSAndroid Build Coastguard Worker continue;
303*4d7e907cSAndroid Build Coastguard Worker case 'V':
304*4d7e907cSAndroid Build Coastguard Worker getMetaDataValue(i, dataOutputBuffer.data(), videoEsDataSize);
305*4d7e907cSAndroid Build Coastguard Worker audioReadPointer = metaDataSize + videoEsDataSize;
306*4d7e907cSAndroid Build Coastguard Worker continue;
307*4d7e907cSAndroid Build Coastguard Worker case 'A':
308*4d7e907cSAndroid Build Coastguard Worker getMetaDataValue(i, dataOutputBuffer.data(), audioEsDataSize);
309*4d7e907cSAndroid Build Coastguard Worker continue;
310*4d7e907cSAndroid Build Coastguard Worker case 'p':
311*4d7e907cSAndroid Build Coastguard Worker if (dataOutputBuffer[++i] == 'a') {
312*4d7e907cSAndroid Build Coastguard Worker getMetaDataValue(i, dataOutputBuffer.data(), audioPid);
313*4d7e907cSAndroid Build Coastguard Worker } else if (dataOutputBuffer[i] == 'v') {
314*4d7e907cSAndroid Build Coastguard Worker getMetaDataValue(i, dataOutputBuffer.data(), videoPid);
315*4d7e907cSAndroid Build Coastguard Worker }
316*4d7e907cSAndroid Build Coastguard Worker continue;
317*4d7e907cSAndroid Build Coastguard Worker case 'v':
318*4d7e907cSAndroid Build Coastguard Worker case 'a':
319*4d7e907cSAndroid Build Coastguard Worker if (dataOutputBuffer[i + 1] != ',') {
320*4d7e907cSAndroid Build Coastguard Worker ALOGE("[Dvr] Invalid format meta data.");
321*4d7e907cSAndroid Build Coastguard Worker return false;
322*4d7e907cSAndroid Build Coastguard Worker }
323*4d7e907cSAndroid Build Coastguard Worker esMeta[frameCount] = {
324*4d7e907cSAndroid Build Coastguard Worker .isAudio = dataOutputBuffer[i] == 'a' ? true : false,
325*4d7e907cSAndroid Build Coastguard Worker };
326*4d7e907cSAndroid Build Coastguard Worker i += 5; // Move to Len
327*4d7e907cSAndroid Build Coastguard Worker getMetaDataValue(i, dataOutputBuffer.data(), esMeta[frameCount].len);
328*4d7e907cSAndroid Build Coastguard Worker if (esMeta[frameCount].isAudio) {
329*4d7e907cSAndroid Build Coastguard Worker esMeta[frameCount].startIndex = audioReadPointer;
330*4d7e907cSAndroid Build Coastguard Worker audioReadPointer += esMeta[frameCount].len;
331*4d7e907cSAndroid Build Coastguard Worker } else {
332*4d7e907cSAndroid Build Coastguard Worker esMeta[frameCount].startIndex = videoReadPointer;
333*4d7e907cSAndroid Build Coastguard Worker videoReadPointer += esMeta[frameCount].len;
334*4d7e907cSAndroid Build Coastguard Worker }
335*4d7e907cSAndroid Build Coastguard Worker i += 4; // move to PTS
336*4d7e907cSAndroid Build Coastguard Worker getMetaDataValue(i, dataOutputBuffer.data(), esMeta[frameCount].pts);
337*4d7e907cSAndroid Build Coastguard Worker frameCount++;
338*4d7e907cSAndroid Build Coastguard Worker continue;
339*4d7e907cSAndroid Build Coastguard Worker default:
340*4d7e907cSAndroid Build Coastguard Worker continue;
341*4d7e907cSAndroid Build Coastguard Worker }
342*4d7e907cSAndroid Build Coastguard Worker }
343*4d7e907cSAndroid Build Coastguard Worker
344*4d7e907cSAndroid Build Coastguard Worker if (frameCount != totalFrames) {
345*4d7e907cSAndroid Build Coastguard Worker ALOGE("[Dvr] Invalid meta data, frameCount=%d, totalFrames reported=%d", frameCount,
346*4d7e907cSAndroid Build Coastguard Worker totalFrames);
347*4d7e907cSAndroid Build Coastguard Worker return false;
348*4d7e907cSAndroid Build Coastguard Worker }
349*4d7e907cSAndroid Build Coastguard Worker
350*4d7e907cSAndroid Build Coastguard Worker if (metaDataSize + audioEsDataSize + videoEsDataSize != size) {
351*4d7e907cSAndroid Build Coastguard Worker ALOGE("[Dvr] Invalid meta data, metaSize=%d, videoSize=%d, audioSize=%d, totolSize=%d",
352*4d7e907cSAndroid Build Coastguard Worker metaDataSize, videoEsDataSize, audioEsDataSize, size);
353*4d7e907cSAndroid Build Coastguard Worker return false;
354*4d7e907cSAndroid Build Coastguard Worker }
355*4d7e907cSAndroid Build Coastguard Worker
356*4d7e907cSAndroid Build Coastguard Worker // Read es raw data from the FMQ per meta data built previously
357*4d7e907cSAndroid Build Coastguard Worker vector<uint8_t> frameData;
358*4d7e907cSAndroid Build Coastguard Worker map<uint32_t, sp<IFilter>>::iterator it;
359*4d7e907cSAndroid Build Coastguard Worker int pid = 0;
360*4d7e907cSAndroid Build Coastguard Worker for (int i = 0; i < totalFrames; i++) {
361*4d7e907cSAndroid Build Coastguard Worker frameData.resize(esMeta[i].len);
362*4d7e907cSAndroid Build Coastguard Worker pid = esMeta[i].isAudio ? audioPid : videoPid;
363*4d7e907cSAndroid Build Coastguard Worker memcpy(frameData.data(), dataOutputBuffer.data() + esMeta[i].startIndex, esMeta[i].len);
364*4d7e907cSAndroid Build Coastguard Worker // Send to the media filter
365*4d7e907cSAndroid Build Coastguard Worker if (isVirtualFrontend && isRecording) {
366*4d7e907cSAndroid Build Coastguard Worker // TODO validate record
367*4d7e907cSAndroid Build Coastguard Worker mDemux->sendFrontendInputToRecord(frameData);
368*4d7e907cSAndroid Build Coastguard Worker } else {
369*4d7e907cSAndroid Build Coastguard Worker for (it = mFilters.begin(); it != mFilters.end(); it++) {
370*4d7e907cSAndroid Build Coastguard Worker if (pid == mDemux->getFilterTpid(it->first)) {
371*4d7e907cSAndroid Build Coastguard Worker mDemux->updateMediaFilterOutput(it->first, frameData,
372*4d7e907cSAndroid Build Coastguard Worker static_cast<uint64_t>(esMeta[i].pts));
373*4d7e907cSAndroid Build Coastguard Worker startFilterDispatcher(isVirtualFrontend, isRecording);
374*4d7e907cSAndroid Build Coastguard Worker }
375*4d7e907cSAndroid Build Coastguard Worker }
376*4d7e907cSAndroid Build Coastguard Worker }
377*4d7e907cSAndroid Build Coastguard Worker }
378*4d7e907cSAndroid Build Coastguard Worker
379*4d7e907cSAndroid Build Coastguard Worker return true;
380*4d7e907cSAndroid Build Coastguard Worker }
381*4d7e907cSAndroid Build Coastguard Worker
getMetaDataValue(int & index,uint8_t * dataOutputBuffer,int & value)382*4d7e907cSAndroid Build Coastguard Worker void Dvr::getMetaDataValue(int& index, uint8_t* dataOutputBuffer, int& value) {
383*4d7e907cSAndroid Build Coastguard Worker index += 2; // Move the pointer across the ":" to the value
384*4d7e907cSAndroid Build Coastguard Worker while (dataOutputBuffer[index] != ',' && dataOutputBuffer[index] != '\n') {
385*4d7e907cSAndroid Build Coastguard Worker value = ((dataOutputBuffer[index++] - 48) + value * 10);
386*4d7e907cSAndroid Build Coastguard Worker }
387*4d7e907cSAndroid Build Coastguard Worker }
388*4d7e907cSAndroid Build Coastguard Worker
startTpidFilter(vector<uint8_t> data)389*4d7e907cSAndroid Build Coastguard Worker void Dvr::startTpidFilter(vector<uint8_t> data) {
390*4d7e907cSAndroid Build Coastguard Worker map<uint32_t, sp<IFilter>>::iterator it;
391*4d7e907cSAndroid Build Coastguard Worker for (it = mFilters.begin(); it != mFilters.end(); it++) {
392*4d7e907cSAndroid Build Coastguard Worker uint16_t pid = ((data[1] & 0x1f) << 8) | ((data[2] & 0xff));
393*4d7e907cSAndroid Build Coastguard Worker if (DEBUG_DVR) {
394*4d7e907cSAndroid Build Coastguard Worker ALOGW("[Dvr] start ts filter pid: %d", pid);
395*4d7e907cSAndroid Build Coastguard Worker }
396*4d7e907cSAndroid Build Coastguard Worker if (pid == mDemux->getFilterTpid(it->first)) {
397*4d7e907cSAndroid Build Coastguard Worker mDemux->updateFilterOutput(it->first, data);
398*4d7e907cSAndroid Build Coastguard Worker }
399*4d7e907cSAndroid Build Coastguard Worker }
400*4d7e907cSAndroid Build Coastguard Worker }
401*4d7e907cSAndroid Build Coastguard Worker
startFilterDispatcher(bool isVirtualFrontend,bool isRecording)402*4d7e907cSAndroid Build Coastguard Worker bool Dvr::startFilterDispatcher(bool isVirtualFrontend, bool isRecording) {
403*4d7e907cSAndroid Build Coastguard Worker if (isVirtualFrontend) {
404*4d7e907cSAndroid Build Coastguard Worker if (isRecording) {
405*4d7e907cSAndroid Build Coastguard Worker return mDemux->startRecordFilterDispatcher();
406*4d7e907cSAndroid Build Coastguard Worker } else {
407*4d7e907cSAndroid Build Coastguard Worker return mDemux->startBroadcastFilterDispatcher();
408*4d7e907cSAndroid Build Coastguard Worker }
409*4d7e907cSAndroid Build Coastguard Worker }
410*4d7e907cSAndroid Build Coastguard Worker
411*4d7e907cSAndroid Build Coastguard Worker map<uint32_t, sp<IFilter>>::iterator it;
412*4d7e907cSAndroid Build Coastguard Worker // Handle the output data per filter type
413*4d7e907cSAndroid Build Coastguard Worker for (it = mFilters.begin(); it != mFilters.end(); it++) {
414*4d7e907cSAndroid Build Coastguard Worker if (mDemux->startFilterHandler(it->first) != Result::SUCCESS) {
415*4d7e907cSAndroid Build Coastguard Worker return false;
416*4d7e907cSAndroid Build Coastguard Worker }
417*4d7e907cSAndroid Build Coastguard Worker }
418*4d7e907cSAndroid Build Coastguard Worker
419*4d7e907cSAndroid Build Coastguard Worker return true;
420*4d7e907cSAndroid Build Coastguard Worker }
421*4d7e907cSAndroid Build Coastguard Worker
writeRecordFMQ(const vector<uint8_t> & data)422*4d7e907cSAndroid Build Coastguard Worker bool Dvr::writeRecordFMQ(const vector<uint8_t>& data) {
423*4d7e907cSAndroid Build Coastguard Worker lock_guard<mutex> lock(mWriteLock);
424*4d7e907cSAndroid Build Coastguard Worker if (mRecordStatus == RecordStatus::OVERFLOW) {
425*4d7e907cSAndroid Build Coastguard Worker ALOGW("[Dvr] stops writing and wait for the client side flushing.");
426*4d7e907cSAndroid Build Coastguard Worker return true;
427*4d7e907cSAndroid Build Coastguard Worker }
428*4d7e907cSAndroid Build Coastguard Worker if (mDvrMQ->write(data.data(), data.size())) {
429*4d7e907cSAndroid Build Coastguard Worker mDvrEventFlag->wake(static_cast<uint32_t>(DemuxQueueNotifyBits::DATA_READY));
430*4d7e907cSAndroid Build Coastguard Worker maySendRecordStatusCallback();
431*4d7e907cSAndroid Build Coastguard Worker return true;
432*4d7e907cSAndroid Build Coastguard Worker }
433*4d7e907cSAndroid Build Coastguard Worker
434*4d7e907cSAndroid Build Coastguard Worker maySendRecordStatusCallback();
435*4d7e907cSAndroid Build Coastguard Worker return false;
436*4d7e907cSAndroid Build Coastguard Worker }
437*4d7e907cSAndroid Build Coastguard Worker
maySendRecordStatusCallback()438*4d7e907cSAndroid Build Coastguard Worker void Dvr::maySendRecordStatusCallback() {
439*4d7e907cSAndroid Build Coastguard Worker lock_guard<mutex> lock(mRecordStatusLock);
440*4d7e907cSAndroid Build Coastguard Worker int availableToRead = mDvrMQ->availableToRead();
441*4d7e907cSAndroid Build Coastguard Worker int availableToWrite = mDvrMQ->availableToWrite();
442*4d7e907cSAndroid Build Coastguard Worker
443*4d7e907cSAndroid Build Coastguard Worker RecordStatus newStatus = checkRecordStatusChange(availableToWrite, availableToRead,
444*4d7e907cSAndroid Build Coastguard Worker mDvrSettings.record().highThreshold,
445*4d7e907cSAndroid Build Coastguard Worker mDvrSettings.record().lowThreshold);
446*4d7e907cSAndroid Build Coastguard Worker if (mRecordStatus != newStatus) {
447*4d7e907cSAndroid Build Coastguard Worker mCallback->onRecordStatus(newStatus);
448*4d7e907cSAndroid Build Coastguard Worker mRecordStatus = newStatus;
449*4d7e907cSAndroid Build Coastguard Worker }
450*4d7e907cSAndroid Build Coastguard Worker }
451*4d7e907cSAndroid Build Coastguard Worker
checkRecordStatusChange(uint32_t availableToWrite,uint32_t availableToRead,uint32_t highThreshold,uint32_t lowThreshold)452*4d7e907cSAndroid Build Coastguard Worker RecordStatus Dvr::checkRecordStatusChange(uint32_t availableToWrite, uint32_t availableToRead,
453*4d7e907cSAndroid Build Coastguard Worker uint32_t highThreshold, uint32_t lowThreshold) {
454*4d7e907cSAndroid Build Coastguard Worker if (availableToWrite == 0) {
455*4d7e907cSAndroid Build Coastguard Worker return DemuxFilterStatus::OVERFLOW;
456*4d7e907cSAndroid Build Coastguard Worker } else if (availableToRead > highThreshold) {
457*4d7e907cSAndroid Build Coastguard Worker return DemuxFilterStatus::HIGH_WATER;
458*4d7e907cSAndroid Build Coastguard Worker } else if (availableToRead < lowThreshold) {
459*4d7e907cSAndroid Build Coastguard Worker return DemuxFilterStatus::LOW_WATER;
460*4d7e907cSAndroid Build Coastguard Worker }
461*4d7e907cSAndroid Build Coastguard Worker return mRecordStatus;
462*4d7e907cSAndroid Build Coastguard Worker }
463*4d7e907cSAndroid Build Coastguard Worker
addPlaybackFilter(uint32_t filterId,sp<IFilter> filter)464*4d7e907cSAndroid Build Coastguard Worker bool Dvr::addPlaybackFilter(uint32_t filterId, sp<IFilter> filter) {
465*4d7e907cSAndroid Build Coastguard Worker mFilters[filterId] = filter;
466*4d7e907cSAndroid Build Coastguard Worker return true;
467*4d7e907cSAndroid Build Coastguard Worker }
468*4d7e907cSAndroid Build Coastguard Worker
removePlaybackFilter(uint32_t filterId)469*4d7e907cSAndroid Build Coastguard Worker bool Dvr::removePlaybackFilter(uint32_t filterId) {
470*4d7e907cSAndroid Build Coastguard Worker mFilters.erase(filterId);
471*4d7e907cSAndroid Build Coastguard Worker return true;
472*4d7e907cSAndroid Build Coastguard Worker }
473*4d7e907cSAndroid Build Coastguard Worker } // namespace implementation
474*4d7e907cSAndroid Build Coastguard Worker } // namespace V1_0
475*4d7e907cSAndroid Build Coastguard Worker } // namespace tuner
476*4d7e907cSAndroid Build Coastguard Worker } // namespace tv
477*4d7e907cSAndroid Build Coastguard Worker } // namespace hardware
478*4d7e907cSAndroid Build Coastguard Worker } // namespace android
479