1 /*
2 * Copyright (c) 2023 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include <fstream>
17 #include <sstream>
18 #include "hitrace_meter.h"
19 #include "hcodec.h"
20 #include "hcodec_log.h"
21 #include "hcodec_dfx.h"
22 #include "hcodec_utils.h"
23
24 namespace OHOS::MediaAVCodec {
25 using namespace std;
26
ScopedTrace(const std::string & value)27 ScopedTrace::ScopedTrace(const std::string &value)
28 {
29 StartTrace(HITRACE_TAG_ZMEDIA, value);
30 }
31
~ScopedTrace()32 ScopedTrace::~ScopedTrace()
33 {
34 FinishTrace(HITRACE_TAG_ZMEDIA);
35 }
36
FuncTracker(std::string value)37 FuncTracker::FuncTracker(std::string value) : value_(std::move(value))
38 {
39 PLOGI("%s >>", value_.c_str());
40 }
41
~FuncTracker()42 FuncTracker::~FuncTracker()
43 {
44 PLOGI("%s <<", value_.c_str());
45 }
46
OnPrintAllBufferOwner(const MsgInfo & msg)47 void HCodec::OnPrintAllBufferOwner(const MsgInfo& msg)
48 {
49 std::chrono::time_point<std::chrono::steady_clock> lastOwnerChangeTime;
50 msg.param->GetValue(KEY_LAST_OWNER_CHANGE_TIME, lastOwnerChangeTime);
51 if ((lastOwnerChangeTime == lastOwnerChangeTime_) &&
52 (circulateWarnPrintedTimes_ < MAX_CIRCULATE_WARN_TIMES)) {
53 HLOGW("buffer circulate stoped");
54 UpdateOwner();
55 PrintAllBufferInfo();
56 circulateWarnPrintedTimes_++;
57 }
58 ParamSP param = make_shared<ParamBundle>();
59 param->SetValue(KEY_LAST_OWNER_CHANGE_TIME, lastOwnerChangeTime_);
60 SendAsyncMsg(MsgWhat::PRINT_ALL_BUFFER_OWNER, param,
61 THREE_SECONDS_IN_US * (circulateWarnPrintedTimes_ + 1));
62 }
63
PrintAllBufferInfo()64 void HCodec::PrintAllBufferInfo()
65 {
66 auto now = chrono::steady_clock::now();
67 PrintAllBufferInfo(true, now);
68 PrintAllBufferInfo(false, now);
69 }
70
PrintAllBufferInfo(bool isInput,std::chrono::time_point<std::chrono::steady_clock> now)71 void HCodec::PrintAllBufferInfo(bool isInput, std::chrono::time_point<std::chrono::steady_clock> now)
72 {
73 const char* inOutStr = isInput ? " in" : "out";
74 bool eos = isInput ? inputPortEos_ : outputPortEos_;
75 uint64_t cnt = isInput ? inTotalCnt_ : outRecord_.totalCnt;
76 const std::array<int, OWNER_CNT>& arr = isInput ? inputOwner_ : outputOwner_;
77 const vector<BufferInfo>& pool = isInput ? inputBufferPool_ : outputBufferPool_;
78
79 std::stringstream s;
80 for (const BufferInfo& info : pool) {
81 int64_t holdMs = chrono::duration_cast<chrono::milliseconds>(now - info.lastOwnerChangeTime).count();
82 s << info.bufferId << ":" << ToString(info.owner) << "(" << holdMs << "), ";
83 }
84 HLOGI("%s: eos=%d, cnt=%" PRIu64 ", %d/%d/%d/%d, %s", inOutStr, eos, cnt,
85 arr[OWNED_BY_US], arr[OWNED_BY_USER], arr[OWNED_BY_OMX], arr[OWNED_BY_SURFACE], s.str().c_str());
86 }
87
OnGetHidumperInfo()88 std::string HCodec::OnGetHidumperInfo()
89 {
90 auto now = chrono::steady_clock::now();
91 std::stringstream s;
92 s << endl;
93 s << " " << compUniqueStr_ << "[" << currState_->GetName() << "]" << endl;
94 s << " " << "------------INPUT-----------" << endl;
95 s << " " << "eos:" << inputPortEos_ << ", etb:" << inTotalCnt_ << endl;
96 for (const BufferInfo& info : inputBufferPool_) {
97 int64_t holdMs = chrono::duration_cast<chrono::milliseconds>(now - info.lastOwnerChangeTime).count();
98 s << " " << "inBufId = " << info.bufferId << ", owner = " << ToString(info.owner)
99 << ", holdMs = " << holdMs << endl;
100 }
101 s << " " << "----------------------------" << endl;
102 s << " " << "------------OUTPUT----------" << endl;
103 s << " " << "eos:" << outputPortEos_ << ", fbd:" << outRecord_.totalCnt << endl;
104 for (const BufferInfo& info : outputBufferPool_) {
105 int64_t holdMs = chrono::duration_cast<chrono::milliseconds>(now - info.lastOwnerChangeTime).count();
106 s << " " << "outBufId = " << info.bufferId << ", owner = " << ToString(info.owner)
107 << ", holdMs = " << holdMs << endl;
108 }
109 s << " " << "----------------------------" << endl;
110 return s.str();
111 }
112
UpdateOwner()113 void HCodec::UpdateOwner()
114 {
115 UpdateOwner(true);
116 UpdateOwner(false);
117 }
118
UpdateOwner(bool isInput)119 void HCodec::UpdateOwner(bool isInput)
120 {
121 std::array<int, OWNER_CNT>& arr = isInput ? inputOwner_ : outputOwner_;
122 const std::array<std::string, OWNER_CNT>& ownerStr = isInput ? inputOwnerStr_ : outputOwnerStr_;
123 const vector<BufferInfo>& pool = isInput ? inputBufferPool_ : outputBufferPool_;
124
125 arr.fill(0);
126 for (const BufferInfo &info : pool) {
127 arr[info.owner]++;
128 }
129 for (uint32_t owner = 0; owner < static_cast<uint32_t>(OWNER_CNT); owner++) {
130 CountTrace(HITRACE_TAG_ZMEDIA, ownerStr[owner], arr[owner]);
131 }
132 }
133
ReduceOwner(bool isInput,BufferOwner owner)134 void HCodec::ReduceOwner(bool isInput, BufferOwner owner)
135 {
136 std::array<int, OWNER_CNT>& arr = isInput ? inputOwner_ : outputOwner_;
137 const std::array<std::string, OWNER_CNT>& ownerStr = isInput ? inputOwnerStr_ : outputOwnerStr_;
138 arr[owner]--;
139 CountTrace(HITRACE_TAG_ZMEDIA, ownerStr[owner], arr[owner]);
140 }
141
PrintStatistic(bool isInput,std::chrono::time_point<std::chrono::steady_clock> now)142 void HCodec::PrintStatistic(bool isInput, std::chrono::time_point<std::chrono::steady_clock> now)
143 {
144 int64_t fromFirstToNow = chrono::duration_cast<chrono::microseconds>(
145 now - (isInput ? firstInTime_ : firstOutTime_)).count();
146 if (fromFirstToNow == 0) {
147 return;
148 }
149 double fps = PRINT_PER_FRAME * US_TO_S / fromFirstToNow;
150 const std::array<int, OWNER_CNT>& arr = isInput ? inputOwner_ : outputOwner_;
151 double aveHoldMs[OWNER_CNT];
152 for (uint32_t owner = 0; owner < static_cast<uint32_t>(OWNER_CNT); owner++) {
153 TotalCntAndCost& holdRecord = isInput ? inputHoldTimeRecord_[owner] : outputHoldTimeRecord_[owner];
154 aveHoldMs[owner] = (holdRecord.totalCnt == 0) ? -1 : (holdRecord.totalCostUs / US_TO_MS / holdRecord.totalCnt);
155 }
156 HLOGI("%s:%.0f; %d/%d/%d/%d, %.0f/%.0f/%.0f/%.0f", (isInput ? " in" : "out"), fps,
157 arr[OWNED_BY_US], arr[OWNED_BY_USER], arr[OWNED_BY_OMX], arr[OWNED_BY_SURFACE],
158 aveHoldMs[OWNED_BY_US], aveHoldMs[OWNED_BY_USER], aveHoldMs[OWNED_BY_OMX], aveHoldMs[OWNED_BY_SURFACE]);
159 }
160
ChangeOwner(BufferInfo & info,BufferOwner newOwner)161 void HCodec::ChangeOwner(BufferInfo& info, BufferOwner newOwner)
162 {
163 debugMode_ ? ChangeOwnerDebug(info, newOwner) : ChangeOwnerNormal(info, newOwner);
164 }
165
ChangeOwnerNormal(BufferInfo & info,BufferOwner newOwner)166 void HCodec::ChangeOwnerNormal(BufferInfo& info, BufferOwner newOwner)
167 {
168 std::array<int, OWNER_CNT>& arr = info.isInput ? inputOwner_ : outputOwner_;
169 const std::array<std::string, OWNER_CNT>& ownerStr = info.isInput ? inputOwnerStr_ : outputOwnerStr_;
170
171 BufferOwner oldOwner = info.owner;
172 auto now = chrono::steady_clock::now();
173 lastOwnerChangeTime_ = now;
174 circulateWarnPrintedTimes_ = 0;
175 arr[oldOwner]--;
176 arr[newOwner]++;
177 CountTrace(HITRACE_TAG_ZMEDIA, ownerStr[oldOwner], arr[oldOwner]);
178 CountTrace(HITRACE_TAG_ZMEDIA, ownerStr[newOwner], arr[newOwner]);
179
180 uint64_t holdUs = static_cast<uint64_t>(
181 chrono::duration_cast<chrono::microseconds>(now - info.lastOwnerChangeTime).count());
182 TotalCntAndCost& holdRecord = info.isInput ? inputHoldTimeRecord_[oldOwner] :
183 outputHoldTimeRecord_[oldOwner];
184 holdRecord.totalCnt++;
185 holdRecord.totalCostUs += holdUs;
186
187 // now change owner
188 info.lastOwnerChangeTime = now;
189 info.owner = newOwner;
190
191 if (info.isInput && oldOwner == OWNED_BY_US && newOwner == OWNED_BY_OMX) {
192 if (inTotalCnt_ == 0) {
193 firstInTime_ = now;
194 }
195 inTotalCnt_++;
196 if (inTotalCnt_ % PRINT_PER_FRAME == 0) {
197 PrintStatistic(info.isInput, now);
198 firstInTime_ = now;
199 inputHoldTimeRecord_.fill({0, 0});
200 }
201 }
202 if (!info.isInput && oldOwner == OWNED_BY_OMX && newOwner == OWNED_BY_US) {
203 if (outRecord_.totalCnt == 0) {
204 firstOutTime_ = now;
205 }
206 outRecord_.totalCnt++;
207 if (outRecord_.totalCnt % PRINT_PER_FRAME == 0) {
208 PrintStatistic(info.isInput, now);
209 firstOutTime_ = now;
210 outputHoldTimeRecord_.fill({0, 0});
211 }
212 }
213 }
214
ChangeOwnerDebug(BufferInfo & info,BufferOwner newOwner)215 void HCodec::ChangeOwnerDebug(BufferInfo& info, BufferOwner newOwner)
216 {
217 std::array<int, OWNER_CNT>& arr = info.isInput ? inputOwner_ : outputOwner_;
218 const std::array<std::string, OWNER_CNT>& ownerStr = info.isInput ? inputOwnerStr_ : outputOwnerStr_;
219
220 BufferOwner oldOwner = info.owner;
221 auto now = chrono::steady_clock::now();
222 lastOwnerChangeTime_ = now;
223 circulateWarnPrintedTimes_ = 0;
224 arr[oldOwner]--;
225 arr[newOwner]++;
226 CountTrace(HITRACE_TAG_ZMEDIA, ownerStr[oldOwner], arr[oldOwner]);
227 CountTrace(HITRACE_TAG_ZMEDIA, ownerStr[newOwner], arr[newOwner]);
228
229 const char* oldOwnerStr = ToString(oldOwner);
230 const char* newOwnerStr = ToString(newOwner);
231 const char* idStr = info.isInput ? "inBufId" : "outBufId";
232
233 uint64_t holdUs = static_cast<uint64_t>
234 (chrono::duration_cast<chrono::microseconds>(now - info.lastOwnerChangeTime).count());
235 double holdMs = holdUs / US_TO_MS;
236 TotalCntAndCost& holdRecord = info.isInput ? inputHoldTimeRecord_[oldOwner] :
237 outputHoldTimeRecord_[oldOwner];
238 holdRecord.totalCnt++;
239 holdRecord.totalCostUs += holdUs;
240 double aveHoldMs = holdRecord.totalCostUs / US_TO_MS / holdRecord.totalCnt;
241
242 // now change owner
243 info.lastOwnerChangeTime = now;
244 info.owner = newOwner;
245 HLOGI("%s = %u, after hold %.1f ms (%.1f ms), %s -> %s, %d/%d/%d/%d",
246 idStr, info.bufferId, holdMs, aveHoldMs, oldOwnerStr, newOwnerStr,
247 arr[OWNED_BY_US], arr[OWNED_BY_USER], arr[OWNED_BY_OMX], arr[OWNED_BY_SURFACE]);
248
249 if (info.isInput && oldOwner == OWNED_BY_US && newOwner == OWNED_BY_OMX) {
250 UpdateInputRecord(info, now);
251 }
252 if (!info.isInput && oldOwner == OWNED_BY_OMX && newOwner == OWNED_BY_US) {
253 UpdateOutputRecord(info, now);
254 }
255 }
256
UpdateInputRecord(const BufferInfo & info,std::chrono::time_point<std::chrono::steady_clock> now)257 void HCodec::UpdateInputRecord(const BufferInfo& info, std::chrono::time_point<std::chrono::steady_clock> now)
258 {
259 if (!info.IsValidFrame()) {
260 return;
261 }
262 inTimeMap_[info.omxBuffer->pts] = now;
263 if (inTotalCnt_ == 0) {
264 firstInTime_ = now;
265 }
266 inTotalCnt_++;
267
268 uint64_t fromFirstInToNow = static_cast<uint64_t>
269 (chrono::duration_cast<chrono::microseconds>(now - firstInTime_).count());
270 if (fromFirstInToNow == 0) {
271 HLOGI("pts = %" PRId64 ", len = %u, flags = 0x%x",
272 info.omxBuffer->pts, info.omxBuffer->filledLen, info.omxBuffer->flag);
273 } else {
274 double inFps = inTotalCnt_ * US_TO_S / fromFirstInToNow;
275 HLOGI("pts = %" PRId64 ", len = %u, flags = 0x%x, in fps %.2f",
276 info.omxBuffer->pts, info.omxBuffer->filledLen, info.omxBuffer->flag, inFps);
277 }
278 }
279
UpdateOutputRecord(const BufferInfo & info,std::chrono::time_point<std::chrono::steady_clock> now)280 void HCodec::UpdateOutputRecord(const BufferInfo& info, std::chrono::time_point<std::chrono::steady_clock> now)
281 {
282 if (!info.IsValidFrame()) {
283 return;
284 }
285 auto it = inTimeMap_.find(info.omxBuffer->pts);
286 if (it == inTimeMap_.end()) {
287 return;
288 }
289 if (outRecord_.totalCnt == 0) {
290 firstOutTime_ = now;
291 }
292 outRecord_.totalCnt++;
293
294 uint64_t fromInToOut = static_cast<uint64_t>
295 (chrono::duration_cast<chrono::microseconds>(now - it->second).count());
296 inTimeMap_.erase(it);
297 outRecord_.totalCostUs += fromInToOut;
298 double oneFrameCostMs = fromInToOut / US_TO_MS;
299 double averageCostMs = outRecord_.totalCostUs / US_TO_MS / outRecord_.totalCnt;
300
301 uint64_t fromFirstOutToNow = static_cast<uint64_t>
302 (chrono::duration_cast<chrono::microseconds>(now - firstOutTime_).count());
303 if (fromFirstOutToNow == 0) {
304 HLOGI("pts = %" PRId64 ", len = %u, flags = 0x%x, "
305 "cost %.2f ms (%.2f ms)",
306 info.omxBuffer->pts, info.omxBuffer->filledLen, info.omxBuffer->flag,
307 oneFrameCostMs, averageCostMs);
308 } else {
309 double outFps = outRecord_.totalCnt * US_TO_S / fromFirstOutToNow;
310 HLOGI("pts = %" PRId64 ", len = %u, flags = 0x%x, "
311 "cost %.2f ms (%.2f ms), out fps %.2f",
312 info.omxBuffer->pts, info.omxBuffer->filledLen, info.omxBuffer->flag,
313 oneFrameCostMs, averageCostMs, outFps);
314 }
315 }
316
IsValidFrame() const317 bool HCodec::BufferInfo::IsValidFrame() const
318 {
319 if (omxBuffer->flag & OMX_BUFFERFLAG_EOS) {
320 return false;
321 }
322 if (omxBuffer->flag & OMX_BUFFERFLAG_CODECCONFIG) {
323 return false;
324 }
325 if (omxBuffer->filledLen == 0) {
326 return false;
327 }
328 return true;
329 }
330
331 #ifdef BUILD_ENG_VERSION
Dump(const string & prefix,uint64_t cnt,DumpMode dumpMode,bool isEncoder) const332 void HCodec::BufferInfo::Dump(const string& prefix, uint64_t cnt, DumpMode dumpMode, bool isEncoder) const
333 {
334 if (isInput) {
335 if (((dumpMode & DUMP_ENCODER_INPUT) && isEncoder) ||
336 ((dumpMode & DUMP_DECODER_INPUT) && !isEncoder)) {
337 Dump(prefix + "_Input", cnt);
338 }
339 } else {
340 if (((dumpMode & DUMP_ENCODER_OUTPUT) && isEncoder) ||
341 ((dumpMode & DUMP_DECODER_OUTPUT) && !isEncoder)) {
342 Dump(prefix + "_Output", cnt);
343 }
344 }
345 }
346
Dump(const string & prefix,uint64_t cnt) const347 void HCodec::BufferInfo::Dump(const string& prefix, uint64_t cnt) const
348 {
349 if (surfaceBuffer) {
350 DumpSurfaceBuffer(prefix, cnt);
351 } else {
352 DumpLinearBuffer(prefix);
353 }
354 }
355
DumpSurfaceBuffer(const std::string & prefix,uint64_t cnt) const356 void HCodec::BufferInfo::DumpSurfaceBuffer(const std::string& prefix, uint64_t cnt) const
357 {
358 if (omxBuffer->filledLen == 0) {
359 return;
360 }
361 const char* va = reinterpret_cast<const char*>(surfaceBuffer->GetVirAddr());
362 IF_TRUE_RETURN_VOID_WITH_MSG(va == nullptr, "null va");
363 int w = surfaceBuffer->GetWidth();
364 int h = surfaceBuffer->GetHeight();
365 int byteStride = surfaceBuffer->GetStride();
366 IF_TRUE_RETURN_VOID_WITH_MSG(byteStride == 0, "stride 0");
367 int alignedH = h;
368 uint32_t totalSize = surfaceBuffer->GetSize();
369 uint32_t seq = surfaceBuffer->GetSeqNum();
370 GraphicPixelFormat graphicFmt = static_cast<GraphicPixelFormat>(surfaceBuffer->GetFormat());
371 std::optional<PixelFmt> fmt = TypeConverter::GraphicFmtToFmt(graphicFmt);
372 IF_TRUE_RETURN_VOID_WITH_MSG(!fmt.has_value(), "unknown fmt %d", graphicFmt);
373
374 string suffix;
375 bool dumpAsVideo = true;
376 DecideDumpInfo(alignedH, totalSize, suffix, dumpAsVideo);
377
378 char name[128];
379 int ret = 0;
380 if (dumpAsVideo) {
381 ret = sprintf_s(name, sizeof(name), "%s/%s_%dx%d(%dx%d)_fmt%s.%s",
382 DUMP_PATH, prefix.c_str(), w, h, byteStride, alignedH,
383 fmt->strFmt.c_str(), suffix.c_str());
384 } else {
385 ret = sprintf_s(name, sizeof(name), "%s/%s_%" PRIu64 "_%dx%d(%d)_fmt%s_pts%" PRId64 "_seq%u.%s",
386 DUMP_PATH, prefix.c_str(), cnt, w, h, byteStride,
387 fmt->strFmt.c_str(), omxBuffer->pts, seq, suffix.c_str());
388 }
389 if (ret > 0) {
390 ofstream ofs(name, ios::binary | ios::app);
391 if (ofs.is_open()) {
392 ofs.write(va, totalSize);
393 } else {
394 LOGW("cannot open %s", name);
395 }
396 }
397 // if we unmap here, flush cache will fail
398 }
399
DecideDumpInfo(int & alignedH,uint32_t & totalSize,string & suffix,bool & dumpAsVideo) const400 void HCodec::BufferInfo::DecideDumpInfo(int& alignedH, uint32_t& totalSize, string& suffix, bool& dumpAsVideo) const
401 {
402 int h = surfaceBuffer->GetHeight();
403 int byteStride = surfaceBuffer->GetStride();
404 GraphicPixelFormat fmt = static_cast<GraphicPixelFormat>(surfaceBuffer->GetFormat());
405 switch (fmt) {
406 case GRAPHIC_PIXEL_FMT_YCBCR_420_P:
407 case GRAPHIC_PIXEL_FMT_YCRCB_420_SP:
408 case GRAPHIC_PIXEL_FMT_YCBCR_420_SP:
409 case GRAPHIC_PIXEL_FMT_YCBCR_P010:
410 case GRAPHIC_PIXEL_FMT_YCRCB_P010: {
411 OH_NativeBuffer_Planes *planes = nullptr;
412 GSError err = surfaceBuffer->GetPlanesInfo(reinterpret_cast<void**>(&planes));
413 if (err != GSERROR_OK || planes == nullptr) { // compressed
414 suffix = "bin";
415 dumpAsVideo = false;
416 return;
417 }
418 alignedH = static_cast<int32_t>(static_cast<int64_t>(planes->planes[1].offset) / byteStride);
419 totalSize = GetYuv420Size(byteStride, alignedH);
420 suffix = "yuv";
421 break;
422 }
423 case GRAPHIC_PIXEL_FMT_RGBA_8888: {
424 totalSize = static_cast<uint32_t>(byteStride * h);
425 suffix = "rgba";
426 break;
427 }
428 default: {
429 suffix = "bin";
430 dumpAsVideo = false;
431 break;
432 }
433 }
434 }
435
DumpLinearBuffer(const string & prefix) const436 void HCodec::BufferInfo::DumpLinearBuffer(const string& prefix) const
437 {
438 if (omxBuffer->filledLen == 0) {
439 return;
440 }
441 if (avBuffer == nullptr || avBuffer->memory_ == nullptr) {
442 LOGW("invalid avbuffer");
443 return;
444 }
445 const char* va = reinterpret_cast<const char*>(avBuffer->memory_->GetAddr());
446 if (va == nullptr) {
447 LOGW("null va");
448 return;
449 }
450
451 char name[128];
452 int ret = sprintf_s(name, sizeof(name), "%s/%s.bin", DUMP_PATH, prefix.c_str());
453 if (ret <= 0) {
454 LOGW("sprintf_s failed");
455 return;
456 }
457 ofstream ofs(name, ios::binary | ios::app);
458 if (ofs.is_open()) {
459 ofs.write(va, omxBuffer->filledLen);
460 } else {
461 LOGW("cannot open %s", name);
462 }
463 }
464 #endif // BUILD_ENG_VERSION
465 }