xref: /aosp_15_r20/hardware/interfaces/tv/tuner/1.0/default/Dvr.cpp (revision 4d7e907c777eeecc4c5bd7cf640a754fac206ff7)
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