xref: /aosp_15_r20/hardware/interfaces/tv/tuner/1.1/default/Dvr.cpp (revision 4d7e907c777eeecc4c5bd7cf640a754fac206ff7)
1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker  * Copyright 2020 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<V1_0::IFilter> & filter)61*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::attachFilter(const sp<V1_0::IFilter>& filter) {
62*4d7e907cSAndroid Build Coastguard Worker     ALOGV("%s", __FUNCTION__);
63*4d7e907cSAndroid Build Coastguard Worker 
64*4d7e907cSAndroid Build Coastguard Worker     uint64_t filterId;
65*4d7e907cSAndroid Build Coastguard Worker     Result status;
66*4d7e907cSAndroid Build Coastguard Worker 
67*4d7e907cSAndroid Build Coastguard Worker     sp<V1_1::IFilter> filter_v1_1 = V1_1::IFilter::castFrom(filter);
68*4d7e907cSAndroid Build Coastguard Worker     if (filter_v1_1 != NULL) {
69*4d7e907cSAndroid Build Coastguard Worker         filter_v1_1->getId64Bit([&](Result result, uint64_t id) {
70*4d7e907cSAndroid Build Coastguard Worker             filterId = id;
71*4d7e907cSAndroid Build Coastguard Worker             status = result;
72*4d7e907cSAndroid Build Coastguard Worker         });
73*4d7e907cSAndroid Build Coastguard Worker     } else {
74*4d7e907cSAndroid Build Coastguard Worker         filter->getId([&](Result result, uint32_t id) {
75*4d7e907cSAndroid Build Coastguard Worker             filterId = id;
76*4d7e907cSAndroid Build Coastguard Worker             status = result;
77*4d7e907cSAndroid Build Coastguard Worker         });
78*4d7e907cSAndroid Build Coastguard Worker     }
79*4d7e907cSAndroid Build Coastguard Worker 
80*4d7e907cSAndroid Build Coastguard Worker     if (status != Result::SUCCESS) {
81*4d7e907cSAndroid Build Coastguard Worker         return status;
82*4d7e907cSAndroid Build Coastguard Worker     }
83*4d7e907cSAndroid Build Coastguard Worker 
84*4d7e907cSAndroid Build Coastguard Worker     if (!mDemux->attachRecordFilter(filterId)) {
85*4d7e907cSAndroid Build Coastguard Worker         return Result::INVALID_ARGUMENT;
86*4d7e907cSAndroid Build Coastguard Worker     }
87*4d7e907cSAndroid Build Coastguard Worker 
88*4d7e907cSAndroid Build Coastguard Worker     return Result::SUCCESS;
89*4d7e907cSAndroid Build Coastguard Worker }
90*4d7e907cSAndroid Build Coastguard Worker 
detachFilter(const sp<V1_0::IFilter> & filter)91*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::detachFilter(const sp<V1_0::IFilter>& filter) {
92*4d7e907cSAndroid Build Coastguard Worker     ALOGV("%s", __FUNCTION__);
93*4d7e907cSAndroid Build Coastguard Worker 
94*4d7e907cSAndroid Build Coastguard Worker     uint64_t filterId;
95*4d7e907cSAndroid Build Coastguard Worker     Result status;
96*4d7e907cSAndroid Build Coastguard Worker 
97*4d7e907cSAndroid Build Coastguard Worker     sp<V1_1::IFilter> filter_v1_1 = V1_1::IFilter::castFrom(filter);
98*4d7e907cSAndroid Build Coastguard Worker     if (filter_v1_1 != NULL) {
99*4d7e907cSAndroid Build Coastguard Worker         filter_v1_1->getId64Bit([&](Result result, uint64_t id) {
100*4d7e907cSAndroid Build Coastguard Worker             filterId = id;
101*4d7e907cSAndroid Build Coastguard Worker             status = result;
102*4d7e907cSAndroid Build Coastguard Worker         });
103*4d7e907cSAndroid Build Coastguard Worker     } else {
104*4d7e907cSAndroid Build Coastguard Worker         filter->getId([&](Result result, uint32_t id) {
105*4d7e907cSAndroid Build Coastguard Worker             filterId = id;
106*4d7e907cSAndroid Build Coastguard Worker             status = result;
107*4d7e907cSAndroid Build Coastguard Worker         });
108*4d7e907cSAndroid Build Coastguard Worker     }
109*4d7e907cSAndroid Build Coastguard Worker 
110*4d7e907cSAndroid Build Coastguard Worker     if (status != Result::SUCCESS) {
111*4d7e907cSAndroid Build Coastguard Worker         return status;
112*4d7e907cSAndroid Build Coastguard Worker     }
113*4d7e907cSAndroid Build Coastguard Worker 
114*4d7e907cSAndroid Build Coastguard Worker     if (!mDemux->detachRecordFilter(filterId)) {
115*4d7e907cSAndroid Build Coastguard Worker         return Result::INVALID_ARGUMENT;
116*4d7e907cSAndroid Build Coastguard Worker     }
117*4d7e907cSAndroid Build Coastguard Worker 
118*4d7e907cSAndroid Build Coastguard Worker     return Result::SUCCESS;
119*4d7e907cSAndroid Build Coastguard Worker }
120*4d7e907cSAndroid Build Coastguard Worker 
start()121*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::start() {
122*4d7e907cSAndroid Build Coastguard Worker     ALOGV("%s", __FUNCTION__);
123*4d7e907cSAndroid Build Coastguard Worker     if (mDvrThreadRunning) {
124*4d7e907cSAndroid Build Coastguard Worker         return Result::SUCCESS;
125*4d7e907cSAndroid Build Coastguard Worker     }
126*4d7e907cSAndroid Build Coastguard Worker 
127*4d7e907cSAndroid Build Coastguard Worker     if (!mCallback) {
128*4d7e907cSAndroid Build Coastguard Worker         return Result::NOT_INITIALIZED;
129*4d7e907cSAndroid Build Coastguard Worker     }
130*4d7e907cSAndroid Build Coastguard Worker 
131*4d7e907cSAndroid Build Coastguard Worker     if (!mDvrConfigured) {
132*4d7e907cSAndroid Build Coastguard Worker         return Result::INVALID_STATE;
133*4d7e907cSAndroid Build Coastguard Worker     }
134*4d7e907cSAndroid Build Coastguard Worker 
135*4d7e907cSAndroid Build Coastguard Worker     if (mType == DvrType::PLAYBACK) {
136*4d7e907cSAndroid Build Coastguard Worker         mDvrThreadRunning = true;
137*4d7e907cSAndroid Build Coastguard Worker         mDvrThread = std::thread(&Dvr::playbackThreadLoop, this);
138*4d7e907cSAndroid Build Coastguard Worker     } else if (mType == DvrType::RECORD) {
139*4d7e907cSAndroid Build Coastguard Worker         mRecordStatus = RecordStatus::DATA_READY;
140*4d7e907cSAndroid Build Coastguard Worker         mDemux->setIsRecording(mType == DvrType::RECORD);
141*4d7e907cSAndroid Build Coastguard Worker     }
142*4d7e907cSAndroid Build Coastguard Worker 
143*4d7e907cSAndroid Build Coastguard Worker     // TODO start another thread to send filter status callback to the framework
144*4d7e907cSAndroid Build Coastguard Worker 
145*4d7e907cSAndroid Build Coastguard Worker     return Result::SUCCESS;
146*4d7e907cSAndroid Build Coastguard Worker }
147*4d7e907cSAndroid Build Coastguard Worker 
stop()148*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::stop() {
149*4d7e907cSAndroid Build Coastguard Worker     ALOGV("%s", __FUNCTION__);
150*4d7e907cSAndroid Build Coastguard Worker 
151*4d7e907cSAndroid Build Coastguard Worker     mDvrThreadRunning = false;
152*4d7e907cSAndroid Build Coastguard Worker     if (mDvrThread.joinable()) {
153*4d7e907cSAndroid Build Coastguard Worker         mDvrThread.join();
154*4d7e907cSAndroid Build Coastguard Worker     }
155*4d7e907cSAndroid Build Coastguard Worker     // thread should always be joinable if it is running,
156*4d7e907cSAndroid Build Coastguard Worker     // so it should be safe to assume recording stopped.
157*4d7e907cSAndroid Build Coastguard Worker     mDemux->setIsRecording(false);
158*4d7e907cSAndroid Build Coastguard Worker 
159*4d7e907cSAndroid Build Coastguard Worker     return Result::SUCCESS;
160*4d7e907cSAndroid Build Coastguard Worker }
161*4d7e907cSAndroid Build Coastguard Worker 
flush()162*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::flush() {
163*4d7e907cSAndroid Build Coastguard Worker     ALOGV("%s", __FUNCTION__);
164*4d7e907cSAndroid Build Coastguard Worker 
165*4d7e907cSAndroid Build Coastguard Worker     mRecordStatus = RecordStatus::DATA_READY;
166*4d7e907cSAndroid Build Coastguard Worker 
167*4d7e907cSAndroid Build Coastguard Worker     return Result::SUCCESS;
168*4d7e907cSAndroid Build Coastguard Worker }
169*4d7e907cSAndroid Build Coastguard Worker 
close()170*4d7e907cSAndroid Build Coastguard Worker Return<Result> Dvr::close() {
171*4d7e907cSAndroid Build Coastguard Worker     ALOGV("%s", __FUNCTION__);
172*4d7e907cSAndroid Build Coastguard Worker     stop();
173*4d7e907cSAndroid Build Coastguard Worker     return Result::SUCCESS;
174*4d7e907cSAndroid Build Coastguard Worker }
175*4d7e907cSAndroid Build Coastguard Worker 
createDvrMQ()176*4d7e907cSAndroid Build Coastguard Worker bool Dvr::createDvrMQ() {
177*4d7e907cSAndroid Build Coastguard Worker     ALOGV("%s", __FUNCTION__);
178*4d7e907cSAndroid Build Coastguard Worker 
179*4d7e907cSAndroid Build Coastguard Worker     // Create a synchronized FMQ that supports blocking read/write
180*4d7e907cSAndroid Build Coastguard Worker     unique_ptr<DvrMQ> tmpDvrMQ = unique_ptr<DvrMQ>(new (nothrow) DvrMQ(mBufferSize, true));
181*4d7e907cSAndroid Build Coastguard Worker     if (!tmpDvrMQ->isValid()) {
182*4d7e907cSAndroid Build Coastguard Worker         ALOGW("[Dvr] Failed to create FMQ of DVR");
183*4d7e907cSAndroid Build Coastguard Worker         return false;
184*4d7e907cSAndroid Build Coastguard Worker     }
185*4d7e907cSAndroid Build Coastguard Worker 
186*4d7e907cSAndroid Build Coastguard Worker     mDvrMQ = std::move(tmpDvrMQ);
187*4d7e907cSAndroid Build Coastguard Worker 
188*4d7e907cSAndroid Build Coastguard Worker     if (EventFlag::createEventFlag(mDvrMQ->getEventFlagWord(), &mDvrEventFlag) != OK) {
189*4d7e907cSAndroid Build Coastguard Worker         return false;
190*4d7e907cSAndroid Build Coastguard Worker     }
191*4d7e907cSAndroid Build Coastguard Worker 
192*4d7e907cSAndroid Build Coastguard Worker     return true;
193*4d7e907cSAndroid Build Coastguard Worker }
194*4d7e907cSAndroid Build Coastguard Worker 
getDvrEventFlag()195*4d7e907cSAndroid Build Coastguard Worker EventFlag* Dvr::getDvrEventFlag() {
196*4d7e907cSAndroid Build Coastguard Worker     return mDvrEventFlag;
197*4d7e907cSAndroid Build Coastguard Worker }
198*4d7e907cSAndroid Build Coastguard Worker 
playbackThreadLoop()199*4d7e907cSAndroid Build Coastguard Worker void Dvr::playbackThreadLoop() {
200*4d7e907cSAndroid Build Coastguard Worker     ALOGD("[Dvr] playback threadLoop start.");
201*4d7e907cSAndroid Build Coastguard Worker 
202*4d7e907cSAndroid Build Coastguard Worker     while (mDvrThreadRunning) {
203*4d7e907cSAndroid Build Coastguard Worker         uint32_t efState = 0;
204*4d7e907cSAndroid Build Coastguard Worker         status_t status =
205*4d7e907cSAndroid Build Coastguard Worker                 mDvrEventFlag->wait(static_cast<uint32_t>(DemuxQueueNotifyBits::DATA_READY),
206*4d7e907cSAndroid Build Coastguard Worker                                     &efState, WAIT_TIMEOUT, true /* retry on spurious wake */);
207*4d7e907cSAndroid Build Coastguard Worker         if (status != OK) {
208*4d7e907cSAndroid Build Coastguard Worker             ALOGD("[Dvr] wait for data ready on the playback FMQ");
209*4d7e907cSAndroid Build Coastguard Worker             continue;
210*4d7e907cSAndroid Build Coastguard Worker         }
211*4d7e907cSAndroid Build Coastguard Worker 
212*4d7e907cSAndroid Build Coastguard Worker         // If the both dvr playback and dvr record are created, the playback will be treated as
213*4d7e907cSAndroid Build Coastguard Worker         // the source of the record. isVirtualFrontend set to true would direct the dvr playback
214*4d7e907cSAndroid Build Coastguard Worker         // input to the demux record filters or live broadcast filters.
215*4d7e907cSAndroid Build Coastguard Worker         bool isRecording = mDemux->isRecording();
216*4d7e907cSAndroid Build Coastguard Worker         bool isVirtualFrontend = isRecording;
217*4d7e907cSAndroid Build Coastguard Worker 
218*4d7e907cSAndroid Build Coastguard Worker         if (mDvrSettings.playback().dataFormat == DataFormat::ES) {
219*4d7e907cSAndroid Build Coastguard Worker             if (!processEsDataOnPlayback(isVirtualFrontend, isRecording)) {
220*4d7e907cSAndroid Build Coastguard Worker                 ALOGE("[Dvr] playback es data failed to be filtered. Ending thread");
221*4d7e907cSAndroid Build Coastguard Worker                 break;
222*4d7e907cSAndroid Build Coastguard Worker             }
223*4d7e907cSAndroid Build Coastguard Worker             maySendPlaybackStatusCallback();
224*4d7e907cSAndroid Build Coastguard Worker             continue;
225*4d7e907cSAndroid Build Coastguard Worker         }
226*4d7e907cSAndroid Build Coastguard Worker 
227*4d7e907cSAndroid Build Coastguard Worker         // Our current implementation filter the data and write it into the filter FMQ immediately
228*4d7e907cSAndroid Build Coastguard Worker         // after the DATA_READY from the VTS/framework
229*4d7e907cSAndroid Build Coastguard Worker         // This is for the non-ES data source, real playback use case handling.
230*4d7e907cSAndroid Build Coastguard Worker         if (!readPlaybackFMQ(isVirtualFrontend, isRecording) ||
231*4d7e907cSAndroid Build Coastguard Worker             !startFilterDispatcher(isVirtualFrontend, isRecording)) {
232*4d7e907cSAndroid Build Coastguard Worker             ALOGE("[Dvr] playback data failed to be filtered. Ending thread");
233*4d7e907cSAndroid Build Coastguard Worker             break;
234*4d7e907cSAndroid Build Coastguard Worker         }
235*4d7e907cSAndroid Build Coastguard Worker 
236*4d7e907cSAndroid Build Coastguard Worker         maySendPlaybackStatusCallback();
237*4d7e907cSAndroid Build Coastguard Worker     }
238*4d7e907cSAndroid Build Coastguard Worker 
239*4d7e907cSAndroid Build Coastguard Worker     mDvrThreadRunning = false;
240*4d7e907cSAndroid Build Coastguard Worker     ALOGD("[Dvr] playback thread ended.");
241*4d7e907cSAndroid Build Coastguard Worker }
242*4d7e907cSAndroid Build Coastguard Worker 
maySendPlaybackStatusCallback()243*4d7e907cSAndroid Build Coastguard Worker void Dvr::maySendPlaybackStatusCallback() {
244*4d7e907cSAndroid Build Coastguard Worker     lock_guard<mutex> lock(mPlaybackStatusLock);
245*4d7e907cSAndroid Build Coastguard Worker     int availableToRead = mDvrMQ->availableToRead();
246*4d7e907cSAndroid Build Coastguard Worker     int availableToWrite = mDvrMQ->availableToWrite();
247*4d7e907cSAndroid Build Coastguard Worker 
248*4d7e907cSAndroid Build Coastguard Worker     PlaybackStatus newStatus = checkPlaybackStatusChange(availableToWrite, availableToRead,
249*4d7e907cSAndroid Build Coastguard Worker                                                          mDvrSettings.playback().highThreshold,
250*4d7e907cSAndroid Build Coastguard Worker                                                          mDvrSettings.playback().lowThreshold);
251*4d7e907cSAndroid Build Coastguard Worker     if (mPlaybackStatus != newStatus) {
252*4d7e907cSAndroid Build Coastguard Worker         mCallback->onPlaybackStatus(newStatus);
253*4d7e907cSAndroid Build Coastguard Worker         mPlaybackStatus = newStatus;
254*4d7e907cSAndroid Build Coastguard Worker     }
255*4d7e907cSAndroid Build Coastguard Worker }
256*4d7e907cSAndroid Build Coastguard Worker 
checkPlaybackStatusChange(uint32_t availableToWrite,uint32_t availableToRead,uint32_t highThreshold,uint32_t lowThreshold)257*4d7e907cSAndroid Build Coastguard Worker PlaybackStatus Dvr::checkPlaybackStatusChange(uint32_t availableToWrite, uint32_t availableToRead,
258*4d7e907cSAndroid Build Coastguard Worker                                               uint32_t highThreshold, uint32_t lowThreshold) {
259*4d7e907cSAndroid Build Coastguard Worker     if (availableToWrite == 0) {
260*4d7e907cSAndroid Build Coastguard Worker         return PlaybackStatus::SPACE_FULL;
261*4d7e907cSAndroid Build Coastguard Worker     } else if (availableToRead > highThreshold) {
262*4d7e907cSAndroid Build Coastguard Worker         return PlaybackStatus::SPACE_ALMOST_FULL;
263*4d7e907cSAndroid Build Coastguard Worker     } else if (availableToRead < lowThreshold) {
264*4d7e907cSAndroid Build Coastguard Worker         return PlaybackStatus::SPACE_ALMOST_EMPTY;
265*4d7e907cSAndroid Build Coastguard Worker     } else if (availableToRead == 0) {
266*4d7e907cSAndroid Build Coastguard Worker         return PlaybackStatus::SPACE_EMPTY;
267*4d7e907cSAndroid Build Coastguard Worker     }
268*4d7e907cSAndroid Build Coastguard Worker     return mPlaybackStatus;
269*4d7e907cSAndroid Build Coastguard Worker }
270*4d7e907cSAndroid Build Coastguard Worker 
readPlaybackFMQ(bool isVirtualFrontend,bool isRecording)271*4d7e907cSAndroid Build Coastguard Worker bool Dvr::readPlaybackFMQ(bool isVirtualFrontend, bool isRecording) {
272*4d7e907cSAndroid Build Coastguard Worker     // Read playback data from the input FMQ
273*4d7e907cSAndroid Build Coastguard Worker     int size = mDvrMQ->availableToRead();
274*4d7e907cSAndroid Build Coastguard Worker     int playbackPacketSize = mDvrSettings.playback().packetSize;
275*4d7e907cSAndroid Build Coastguard Worker     vector<uint8_t> dataOutputBuffer;
276*4d7e907cSAndroid Build Coastguard Worker     dataOutputBuffer.resize(playbackPacketSize);
277*4d7e907cSAndroid Build Coastguard Worker     // Dispatch the packet to the PID matching filter output buffer
278*4d7e907cSAndroid Build Coastguard Worker     for (int i = 0; i < size / playbackPacketSize; i++) {
279*4d7e907cSAndroid Build Coastguard Worker         if (!mDvrMQ->read(dataOutputBuffer.data(), playbackPacketSize)) {
280*4d7e907cSAndroid Build Coastguard Worker             return false;
281*4d7e907cSAndroid Build Coastguard Worker         }
282*4d7e907cSAndroid Build Coastguard Worker         if (isVirtualFrontend) {
283*4d7e907cSAndroid Build Coastguard Worker             if (isRecording) {
284*4d7e907cSAndroid Build Coastguard Worker                 mDemux->sendFrontendInputToRecord(dataOutputBuffer);
285*4d7e907cSAndroid Build Coastguard Worker             } else {
286*4d7e907cSAndroid Build Coastguard Worker                 mDemux->startBroadcastTsFilter(dataOutputBuffer);
287*4d7e907cSAndroid Build Coastguard Worker             }
288*4d7e907cSAndroid Build Coastguard Worker         } else {
289*4d7e907cSAndroid Build Coastguard Worker             startTpidFilter(dataOutputBuffer);
290*4d7e907cSAndroid Build Coastguard Worker         }
291*4d7e907cSAndroid Build Coastguard Worker     }
292*4d7e907cSAndroid Build Coastguard Worker 
293*4d7e907cSAndroid Build Coastguard Worker     return true;
294*4d7e907cSAndroid Build Coastguard Worker }
295*4d7e907cSAndroid Build Coastguard Worker 
processEsDataOnPlayback(bool isVirtualFrontend,bool isRecording)296*4d7e907cSAndroid Build Coastguard Worker bool Dvr::processEsDataOnPlayback(bool isVirtualFrontend, bool isRecording) {
297*4d7e907cSAndroid Build Coastguard Worker     // Read ES from the DVR FMQ
298*4d7e907cSAndroid Build Coastguard Worker     // Note that currently we only provides ES with metaData in a specific format to be parsed.
299*4d7e907cSAndroid Build Coastguard Worker     // The ES size should be smaller than the Playback FMQ size to avoid reading truncated data.
300*4d7e907cSAndroid Build Coastguard Worker     int size = mDvrMQ->availableToRead();
301*4d7e907cSAndroid Build Coastguard Worker     vector<uint8_t> dataOutputBuffer;
302*4d7e907cSAndroid Build Coastguard Worker     dataOutputBuffer.resize(size);
303*4d7e907cSAndroid Build Coastguard Worker     if (!mDvrMQ->read(dataOutputBuffer.data(), size)) {
304*4d7e907cSAndroid Build Coastguard Worker         return false;
305*4d7e907cSAndroid Build Coastguard Worker     }
306*4d7e907cSAndroid Build Coastguard Worker 
307*4d7e907cSAndroid Build Coastguard Worker     int metaDataSize = size;
308*4d7e907cSAndroid Build Coastguard Worker     int totalFrames = 0;
309*4d7e907cSAndroid Build Coastguard Worker     int videoEsDataSize = 0;
310*4d7e907cSAndroid Build Coastguard Worker     int audioEsDataSize = 0;
311*4d7e907cSAndroid Build Coastguard Worker     int audioPid = 0;
312*4d7e907cSAndroid Build Coastguard Worker     int videoPid = 0;
313*4d7e907cSAndroid Build Coastguard Worker 
314*4d7e907cSAndroid Build Coastguard Worker     vector<MediaEsMetaData> esMeta;
315*4d7e907cSAndroid Build Coastguard Worker     int videoReadPointer = 0;
316*4d7e907cSAndroid Build Coastguard Worker     int audioReadPointer = 0;
317*4d7e907cSAndroid Build Coastguard Worker     int frameCount = 0;
318*4d7e907cSAndroid Build Coastguard Worker     // Get meta data from the es
319*4d7e907cSAndroid Build Coastguard Worker     for (int i = 0; i < metaDataSize; i++) {
320*4d7e907cSAndroid Build Coastguard Worker         switch (dataOutputBuffer[i]) {
321*4d7e907cSAndroid Build Coastguard Worker             case 'm':
322*4d7e907cSAndroid Build Coastguard Worker                 metaDataSize = 0;
323*4d7e907cSAndroid Build Coastguard Worker                 getMetaDataValue(i, dataOutputBuffer.data(), metaDataSize);
324*4d7e907cSAndroid Build Coastguard Worker                 videoReadPointer = metaDataSize;
325*4d7e907cSAndroid Build Coastguard Worker                 continue;
326*4d7e907cSAndroid Build Coastguard Worker             case 'l':
327*4d7e907cSAndroid Build Coastguard Worker                 getMetaDataValue(i, dataOutputBuffer.data(), totalFrames);
328*4d7e907cSAndroid Build Coastguard Worker                 esMeta.resize(totalFrames);
329*4d7e907cSAndroid Build Coastguard Worker                 continue;
330*4d7e907cSAndroid Build Coastguard Worker             case 'V':
331*4d7e907cSAndroid Build Coastguard Worker                 getMetaDataValue(i, dataOutputBuffer.data(), videoEsDataSize);
332*4d7e907cSAndroid Build Coastguard Worker                 audioReadPointer = metaDataSize + videoEsDataSize;
333*4d7e907cSAndroid Build Coastguard Worker                 continue;
334*4d7e907cSAndroid Build Coastguard Worker             case 'A':
335*4d7e907cSAndroid Build Coastguard Worker                 getMetaDataValue(i, dataOutputBuffer.data(), audioEsDataSize);
336*4d7e907cSAndroid Build Coastguard Worker                 continue;
337*4d7e907cSAndroid Build Coastguard Worker             case 'p':
338*4d7e907cSAndroid Build Coastguard Worker                 if (dataOutputBuffer[++i] == 'a') {
339*4d7e907cSAndroid Build Coastguard Worker                     getMetaDataValue(i, dataOutputBuffer.data(), audioPid);
340*4d7e907cSAndroid Build Coastguard Worker                 } else if (dataOutputBuffer[i] == 'v') {
341*4d7e907cSAndroid Build Coastguard Worker                     getMetaDataValue(i, dataOutputBuffer.data(), videoPid);
342*4d7e907cSAndroid Build Coastguard Worker                 }
343*4d7e907cSAndroid Build Coastguard Worker                 continue;
344*4d7e907cSAndroid Build Coastguard Worker             case 'v':
345*4d7e907cSAndroid Build Coastguard Worker             case 'a':
346*4d7e907cSAndroid Build Coastguard Worker                 if (dataOutputBuffer[i + 1] != ',') {
347*4d7e907cSAndroid Build Coastguard Worker                     ALOGE("[Dvr] Invalid format meta data.");
348*4d7e907cSAndroid Build Coastguard Worker                     return false;
349*4d7e907cSAndroid Build Coastguard Worker                 }
350*4d7e907cSAndroid Build Coastguard Worker                 esMeta[frameCount] = {
351*4d7e907cSAndroid Build Coastguard Worker                         .isAudio = dataOutputBuffer[i] == 'a' ? true : false,
352*4d7e907cSAndroid Build Coastguard Worker                 };
353*4d7e907cSAndroid Build Coastguard Worker                 i += 5;  // Move to Len
354*4d7e907cSAndroid Build Coastguard Worker                 getMetaDataValue(i, dataOutputBuffer.data(), esMeta[frameCount].len);
355*4d7e907cSAndroid Build Coastguard Worker                 if (esMeta[frameCount].isAudio) {
356*4d7e907cSAndroid Build Coastguard Worker                     esMeta[frameCount].startIndex = audioReadPointer;
357*4d7e907cSAndroid Build Coastguard Worker                     audioReadPointer += esMeta[frameCount].len;
358*4d7e907cSAndroid Build Coastguard Worker                 } else {
359*4d7e907cSAndroid Build Coastguard Worker                     esMeta[frameCount].startIndex = videoReadPointer;
360*4d7e907cSAndroid Build Coastguard Worker                     videoReadPointer += esMeta[frameCount].len;
361*4d7e907cSAndroid Build Coastguard Worker                 }
362*4d7e907cSAndroid Build Coastguard Worker                 i += 4;  // move to PTS
363*4d7e907cSAndroid Build Coastguard Worker                 getMetaDataValue(i, dataOutputBuffer.data(), esMeta[frameCount].pts);
364*4d7e907cSAndroid Build Coastguard Worker                 frameCount++;
365*4d7e907cSAndroid Build Coastguard Worker                 continue;
366*4d7e907cSAndroid Build Coastguard Worker             default:
367*4d7e907cSAndroid Build Coastguard Worker                 continue;
368*4d7e907cSAndroid Build Coastguard Worker         }
369*4d7e907cSAndroid Build Coastguard Worker     }
370*4d7e907cSAndroid Build Coastguard Worker 
371*4d7e907cSAndroid Build Coastguard Worker     if (frameCount != totalFrames) {
372*4d7e907cSAndroid Build Coastguard Worker         ALOGE("[Dvr] Invalid meta data, frameCount=%d, totalFrames reported=%d", frameCount,
373*4d7e907cSAndroid Build Coastguard Worker               totalFrames);
374*4d7e907cSAndroid Build Coastguard Worker         return false;
375*4d7e907cSAndroid Build Coastguard Worker     }
376*4d7e907cSAndroid Build Coastguard Worker 
377*4d7e907cSAndroid Build Coastguard Worker     if (metaDataSize + audioEsDataSize + videoEsDataSize != size) {
378*4d7e907cSAndroid Build Coastguard Worker         ALOGE("[Dvr] Invalid meta data, metaSize=%d, videoSize=%d, audioSize=%d, totolSize=%d",
379*4d7e907cSAndroid Build Coastguard Worker               metaDataSize, videoEsDataSize, audioEsDataSize, size);
380*4d7e907cSAndroid Build Coastguard Worker         return false;
381*4d7e907cSAndroid Build Coastguard Worker     }
382*4d7e907cSAndroid Build Coastguard Worker 
383*4d7e907cSAndroid Build Coastguard Worker     // Read es raw data from the FMQ per meta data built previously
384*4d7e907cSAndroid Build Coastguard Worker     vector<uint8_t> frameData;
385*4d7e907cSAndroid Build Coastguard Worker     map<uint64_t, sp<IFilter>>::iterator it;
386*4d7e907cSAndroid Build Coastguard Worker     int pid = 0;
387*4d7e907cSAndroid Build Coastguard Worker     for (int i = 0; i < totalFrames; i++) {
388*4d7e907cSAndroid Build Coastguard Worker         frameData.resize(esMeta[i].len);
389*4d7e907cSAndroid Build Coastguard Worker         pid = esMeta[i].isAudio ? audioPid : videoPid;
390*4d7e907cSAndroid Build Coastguard Worker         memcpy(frameData.data(), dataOutputBuffer.data() + esMeta[i].startIndex, esMeta[i].len);
391*4d7e907cSAndroid Build Coastguard Worker         // Send to the media filters or record filters
392*4d7e907cSAndroid Build Coastguard Worker         if (!isRecording) {
393*4d7e907cSAndroid Build Coastguard Worker             for (it = mFilters.begin(); it != mFilters.end(); it++) {
394*4d7e907cSAndroid Build Coastguard Worker                 if (pid == mDemux->getFilterTpid(it->first)) {
395*4d7e907cSAndroid Build Coastguard Worker                     mDemux->updateMediaFilterOutput(it->first, frameData,
396*4d7e907cSAndroid Build Coastguard Worker                                                     static_cast<uint64_t>(esMeta[i].pts));
397*4d7e907cSAndroid Build Coastguard Worker                 }
398*4d7e907cSAndroid Build Coastguard Worker             }
399*4d7e907cSAndroid Build Coastguard Worker         } else {
400*4d7e907cSAndroid Build Coastguard Worker             mDemux->sendFrontendInputToRecord(frameData, pid, static_cast<uint64_t>(esMeta[i].pts));
401*4d7e907cSAndroid Build Coastguard Worker         }
402*4d7e907cSAndroid Build Coastguard Worker         startFilterDispatcher(isVirtualFrontend, isRecording);
403*4d7e907cSAndroid Build Coastguard Worker         frameData.clear();
404*4d7e907cSAndroid Build Coastguard Worker     }
405*4d7e907cSAndroid Build Coastguard Worker 
406*4d7e907cSAndroid Build Coastguard Worker     return true;
407*4d7e907cSAndroid Build Coastguard Worker }
408*4d7e907cSAndroid Build Coastguard Worker 
getMetaDataValue(int & index,uint8_t * dataOutputBuffer,int & value)409*4d7e907cSAndroid Build Coastguard Worker void Dvr::getMetaDataValue(int& index, uint8_t* dataOutputBuffer, int& value) {
410*4d7e907cSAndroid Build Coastguard Worker     index += 2;  // Move the pointer across the ":" to the value
411*4d7e907cSAndroid Build Coastguard Worker     while (dataOutputBuffer[index] != ',' && dataOutputBuffer[index] != '\n') {
412*4d7e907cSAndroid Build Coastguard Worker         value = ((dataOutputBuffer[index++] - 48) + value * 10);
413*4d7e907cSAndroid Build Coastguard Worker     }
414*4d7e907cSAndroid Build Coastguard Worker }
415*4d7e907cSAndroid Build Coastguard Worker 
startTpidFilter(vector<uint8_t> data)416*4d7e907cSAndroid Build Coastguard Worker void Dvr::startTpidFilter(vector<uint8_t> data) {
417*4d7e907cSAndroid Build Coastguard Worker     map<uint64_t, sp<IFilter>>::iterator it;
418*4d7e907cSAndroid Build Coastguard Worker     for (it = mFilters.begin(); it != mFilters.end(); it++) {
419*4d7e907cSAndroid Build Coastguard Worker         uint16_t pid = ((data[1] & 0x1f) << 8) | ((data[2] & 0xff));
420*4d7e907cSAndroid Build Coastguard Worker         if (DEBUG_DVR) {
421*4d7e907cSAndroid Build Coastguard Worker             ALOGW("[Dvr] start ts filter pid: %d", pid);
422*4d7e907cSAndroid Build Coastguard Worker         }
423*4d7e907cSAndroid Build Coastguard Worker         if (pid == mDemux->getFilterTpid(it->first)) {
424*4d7e907cSAndroid Build Coastguard Worker             mDemux->updateFilterOutput(it->first, data);
425*4d7e907cSAndroid Build Coastguard Worker         }
426*4d7e907cSAndroid Build Coastguard Worker     }
427*4d7e907cSAndroid Build Coastguard Worker }
428*4d7e907cSAndroid Build Coastguard Worker 
startFilterDispatcher(bool isVirtualFrontend,bool isRecording)429*4d7e907cSAndroid Build Coastguard Worker bool Dvr::startFilterDispatcher(bool isVirtualFrontend, bool isRecording) {
430*4d7e907cSAndroid Build Coastguard Worker     if (isVirtualFrontend) {
431*4d7e907cSAndroid Build Coastguard Worker         if (isRecording) {
432*4d7e907cSAndroid Build Coastguard Worker             return mDemux->startRecordFilterDispatcher();
433*4d7e907cSAndroid Build Coastguard Worker         } else {
434*4d7e907cSAndroid Build Coastguard Worker             return mDemux->startBroadcastFilterDispatcher();
435*4d7e907cSAndroid Build Coastguard Worker         }
436*4d7e907cSAndroid Build Coastguard Worker     }
437*4d7e907cSAndroid Build Coastguard Worker 
438*4d7e907cSAndroid Build Coastguard Worker     map<uint64_t, sp<IFilter>>::iterator it;
439*4d7e907cSAndroid Build Coastguard Worker     // Handle the output data per filter type
440*4d7e907cSAndroid Build Coastguard Worker     for (it = mFilters.begin(); it != mFilters.end(); it++) {
441*4d7e907cSAndroid Build Coastguard Worker         if (mDemux->startFilterHandler(it->first) != Result::SUCCESS) {
442*4d7e907cSAndroid Build Coastguard Worker             return false;
443*4d7e907cSAndroid Build Coastguard Worker         }
444*4d7e907cSAndroid Build Coastguard Worker     }
445*4d7e907cSAndroid Build Coastguard Worker 
446*4d7e907cSAndroid Build Coastguard Worker     return true;
447*4d7e907cSAndroid Build Coastguard Worker }
448*4d7e907cSAndroid Build Coastguard Worker 
writeRecordFMQ(const vector<uint8_t> & data)449*4d7e907cSAndroid Build Coastguard Worker bool Dvr::writeRecordFMQ(const vector<uint8_t>& data) {
450*4d7e907cSAndroid Build Coastguard Worker     lock_guard<mutex> lock(mWriteLock);
451*4d7e907cSAndroid Build Coastguard Worker     if (mRecordStatus == RecordStatus::OVERFLOW) {
452*4d7e907cSAndroid Build Coastguard Worker         ALOGW("[Dvr] stops writing and wait for the client side flushing.");
453*4d7e907cSAndroid Build Coastguard Worker         return true;
454*4d7e907cSAndroid Build Coastguard Worker     }
455*4d7e907cSAndroid Build Coastguard Worker     if (mDvrMQ->write(data.data(), data.size())) {
456*4d7e907cSAndroid Build Coastguard Worker         mDvrEventFlag->wake(static_cast<uint32_t>(DemuxQueueNotifyBits::DATA_READY));
457*4d7e907cSAndroid Build Coastguard Worker         maySendRecordStatusCallback();
458*4d7e907cSAndroid Build Coastguard Worker         return true;
459*4d7e907cSAndroid Build Coastguard Worker     }
460*4d7e907cSAndroid Build Coastguard Worker 
461*4d7e907cSAndroid Build Coastguard Worker     maySendRecordStatusCallback();
462*4d7e907cSAndroid Build Coastguard Worker     return false;
463*4d7e907cSAndroid Build Coastguard Worker }
464*4d7e907cSAndroid Build Coastguard Worker 
maySendRecordStatusCallback()465*4d7e907cSAndroid Build Coastguard Worker void Dvr::maySendRecordStatusCallback() {
466*4d7e907cSAndroid Build Coastguard Worker     lock_guard<mutex> lock(mRecordStatusLock);
467*4d7e907cSAndroid Build Coastguard Worker     int availableToRead = mDvrMQ->availableToRead();
468*4d7e907cSAndroid Build Coastguard Worker     int availableToWrite = mDvrMQ->availableToWrite();
469*4d7e907cSAndroid Build Coastguard Worker 
470*4d7e907cSAndroid Build Coastguard Worker     RecordStatus newStatus = checkRecordStatusChange(availableToWrite, availableToRead,
471*4d7e907cSAndroid Build Coastguard Worker                                                      mDvrSettings.record().highThreshold,
472*4d7e907cSAndroid Build Coastguard Worker                                                      mDvrSettings.record().lowThreshold);
473*4d7e907cSAndroid Build Coastguard Worker     if (mRecordStatus != newStatus) {
474*4d7e907cSAndroid Build Coastguard Worker         mCallback->onRecordStatus(newStatus);
475*4d7e907cSAndroid Build Coastguard Worker         mRecordStatus = newStatus;
476*4d7e907cSAndroid Build Coastguard Worker     }
477*4d7e907cSAndroid Build Coastguard Worker }
478*4d7e907cSAndroid Build Coastguard Worker 
checkRecordStatusChange(uint32_t availableToWrite,uint32_t availableToRead,uint32_t highThreshold,uint32_t lowThreshold)479*4d7e907cSAndroid Build Coastguard Worker RecordStatus Dvr::checkRecordStatusChange(uint32_t availableToWrite, uint32_t availableToRead,
480*4d7e907cSAndroid Build Coastguard Worker                                           uint32_t highThreshold, uint32_t lowThreshold) {
481*4d7e907cSAndroid Build Coastguard Worker     if (availableToWrite == 0) {
482*4d7e907cSAndroid Build Coastguard Worker         return DemuxFilterStatus::OVERFLOW;
483*4d7e907cSAndroid Build Coastguard Worker     } else if (availableToRead > highThreshold) {
484*4d7e907cSAndroid Build Coastguard Worker         return DemuxFilterStatus::HIGH_WATER;
485*4d7e907cSAndroid Build Coastguard Worker     } else if (availableToRead < lowThreshold) {
486*4d7e907cSAndroid Build Coastguard Worker         return DemuxFilterStatus::LOW_WATER;
487*4d7e907cSAndroid Build Coastguard Worker     }
488*4d7e907cSAndroid Build Coastguard Worker     return mRecordStatus;
489*4d7e907cSAndroid Build Coastguard Worker }
490*4d7e907cSAndroid Build Coastguard Worker 
addPlaybackFilter(uint64_t filterId,sp<IFilter> filter)491*4d7e907cSAndroid Build Coastguard Worker bool Dvr::addPlaybackFilter(uint64_t filterId, sp<IFilter> filter) {
492*4d7e907cSAndroid Build Coastguard Worker     mFilters[filterId] = filter;
493*4d7e907cSAndroid Build Coastguard Worker     return true;
494*4d7e907cSAndroid Build Coastguard Worker }
495*4d7e907cSAndroid Build Coastguard Worker 
removePlaybackFilter(uint64_t filterId)496*4d7e907cSAndroid Build Coastguard Worker bool Dvr::removePlaybackFilter(uint64_t filterId) {
497*4d7e907cSAndroid Build Coastguard Worker     mFilters.erase(filterId);
498*4d7e907cSAndroid Build Coastguard Worker     return true;
499*4d7e907cSAndroid Build Coastguard Worker }
500*4d7e907cSAndroid Build Coastguard Worker }  // namespace implementation
501*4d7e907cSAndroid Build Coastguard Worker }  // namespace V1_0
502*4d7e907cSAndroid Build Coastguard Worker }  // namespace tuner
503*4d7e907cSAndroid Build Coastguard Worker }  // namespace tv
504*4d7e907cSAndroid Build Coastguard Worker }  // namespace hardware
505*4d7e907cSAndroid Build Coastguard Worker }  // namespace android
506