MythTV  master
TemplateMatcher.cpp
Go to the documentation of this file.
1 // C++ headers
2 #include <algorithm>
3 #include <cmath>
4 #include <cstdlib>
5 
6 // Qt headers
7 #include <QFile>
8 #include <QFileInfo>
9 #include <utility>
10 
11 // MythTV headers
14 #include "libmythtv/mythplayer.h"
15 
16 // Commercial Flagging headers
17 #include "BlankFrameDetector.h"
18 #include "CommDetector2.h"
19 #include "EdgeDetector.h"
20 #include "FrameAnalyzer.h"
21 #include "PGMConverter.h"
22 #include "TemplateFinder.h"
23 #include "TemplateMatcher.h"
24 #include "pgm.h"
25 
26 extern "C" {
27 #include "libavutil/imgutils.h"
28 }
29 
30 using namespace commDetector2;
31 using namespace frameAnalyzer;
32 
33 namespace {
34 
35 int pgm_set(const AVFrame *pict, int height)
36 {
37  const int width = pict->linesize[0];
38  const int size = height * width;
39 
40  int score = 0;
41  for (int ii = 0; ii < size; ii++)
42  if (pict->data[0][ii])
43  score++;
44  return score;
45 }
46 
47 int pgm_match(const AVFrame *tmpl, const AVFrame *test, int height,
48  int radius, unsigned short *pscore)
49 {
50  /* Return the number of matching "edge" and non-edge pixels. */
51  const int width = tmpl->linesize[0];
52 
53  if (width != test->linesize[0])
54  {
55  LOG(VB_COMMFLAG, LOG_ERR,
56  QString("pgm_match widths don't match: %1 != %2")
57  .arg(width).arg(test->linesize[0]));
58  return -1;
59  }
60 
61  int score = 0;
62  for (int rr = 0; rr < height; rr++)
63  {
64  for (int cc = 0; cc < width; cc++)
65  {
66  if (!tmpl->data[0][(rr * width) + cc])
67  continue;
68 
69  int r2min = std::max(0, rr - radius);
70  int r2max = std::min(height, rr + radius);
71 
72  int c2min = std::max(0, cc - radius);
73  int c2max = std::min(width, cc + radius);
74 
75  for (int r2 = r2min; r2 <= r2max; r2++)
76  {
77  for (int c2 = c2min; c2 <= c2max; c2++)
78  {
79  if (test->data[0][(r2 * width) + c2])
80  {
81  score++;
82  goto next_pixel;
83  }
84  }
85  }
86 next_pixel:
87  ;
88  }
89  }
90 
91  *pscore = score;
92  return 0;
93 }
94 
95 bool readMatches(const QString& filename, unsigned short *matches, long long nframes)
96 {
97  QByteArray fname = filename.toLocal8Bit();
98  FILE *fp = fopen(fname.constData(), "r");
99  if (fp == nullptr)
100  return false;
101 
102  for (long long frameno = 0; frameno < nframes; frameno++)
103  {
104  int nitems = fscanf(fp, "%20hu", &matches[frameno]);
105  if (nitems != 1)
106  {
107  LOG(VB_COMMFLAG, LOG_ERR,
108  QString("Not enough data in %1: frame %2")
109  .arg(filename).arg(frameno));
110  goto error;
111  }
112  }
113 
114  if (fclose(fp))
115  LOG(VB_COMMFLAG, LOG_ERR, QString("Error closing %1: %2")
116  .arg(filename, strerror(errno)));
117  return true;
118 
119 error:
120  if (fclose(fp))
121  LOG(VB_COMMFLAG, LOG_ERR, QString("Error closing %1: %2")
122  .arg(filename, strerror(errno)));
123  return false;
124 }
125 
126 bool writeMatches(const QString& filename, unsigned short *matches, long long nframes)
127 {
128  QByteArray fname = filename.toLocal8Bit();
129  FILE *fp = fopen(fname.constData(), "w");
130  if (fp == nullptr)
131  return false;
132 
133  for (long long frameno = 0; frameno < nframes; frameno++)
134  (void)fprintf(fp, "%hu\n", matches[frameno]);
135 
136  if (fclose(fp))
137  LOG(VB_COMMFLAG, LOG_ERR, QString("Error closing %1: %2")
138  .arg(filename, strerror(errno)));
139  return true;
140 }
141 
142 int finishedDebug(long long nframes, const unsigned short *matches,
143  const unsigned char *match)
144 {
145  ushort score = matches[0];
146  ushort low = score;
147  ushort high = score;
148  long long startframe = 0;
149 
150  for (long long frameno = 1; frameno < nframes; frameno++)
151  {
152  score = matches[frameno];
153  if (match[frameno - 1] == match[frameno])
154  {
155  low = std::min(score, low);
156  high = std::max(score, high);
157  continue;
158  }
159 
160  LOG(VB_COMMFLAG, LOG_INFO, QString("Frame %1-%2: %3 L-H: %4-%5 (%6)")
161  .arg(startframe, 6).arg(frameno - 1, 6)
162  .arg(match[frameno - 1] ? "logo " : " no-logo")
163  .arg(low, 4).arg(high, 4).arg(frameno - startframe, 5));
164 
165  low = score;
166  high = score;
167  startframe = frameno;
168  }
169 
170  return 0;
171 }
172 
173 int sort_ascending(const void *aa, const void *bb)
174 {
175  return *(unsigned short*)aa - *(unsigned short*)bb;
176 }
177 
178 long long matchspn(long long nframes, const unsigned char *match, long long frameno,
179  unsigned char acceptval)
180 {
181  /*
182  * strspn(3)-style interface: return the first frame number that does not
183  * match "acceptval".
184  */
185  while (frameno < nframes && match[frameno] == acceptval)
186  frameno++;
187  return frameno;
188 }
189 
190 unsigned int range_area(const unsigned short *freq, unsigned short start,
191  unsigned short end)
192 {
193  /* Return the integrated area under the curve of the plotted histogram. */
194  const unsigned short width = end - start;
195 
196  uint sum = 0;
197  uint nsamples = 0;
198  for (ushort matchcnt = start; matchcnt < end; matchcnt++)
199  {
200  if (freq[matchcnt])
201  {
202  sum += freq[matchcnt];
203  nsamples++;
204  }
205  }
206  if (!nsamples)
207  return 0;
208  return width * sum / nsamples;
209 }
210 
211 unsigned short pick_mintmpledges(const unsigned short *matches,
212  long long nframes)
213 {
214  /*
215  * Most frames either match the template very well, or don't match
216  * very well at all. This allows us to assume a bimodal
217  * distribution of the values in a frequency histogram of the
218  * "matches" array.
219  *
220  * Return a local minima between the two modes representing the
221  * threshold value that decides whether or not a frame matches the
222  * template.
223  *
224  * See "mythcommflag-analyze" and its output.
225  */
226 
227  /*
228  * TUNABLE:
229  *
230  * Given a point to test, require the integrated area to the left
231  * of the point to be greater than some (larger) area to the right
232  * of the point.
233  */
234  static constexpr float kLeftWidth = 0.04;
235  static constexpr float kMiddleWidth = 0.04;
236  static constexpr float kRightWidth = 0.04;
237 
238  static constexpr float kMatchStart = 0.20;
239  static constexpr float kMatchEnd = 0.80;
240 
241  auto *sorted = new unsigned short[nframes];
242  memcpy(sorted, matches, nframes * sizeof(*matches));
243  qsort(sorted, nframes, sizeof(*sorted), sort_ascending);
244  ushort minmatch = sorted[0];
245  ushort maxmatch = sorted[nframes - 1];
246  ushort matchrange = maxmatch - minmatch;
247  /* degenerate minmatch==maxmatch case is gracefully handled */
248 
249  auto leftwidth = (unsigned short)(kLeftWidth * matchrange);
250  auto middlewidth = (unsigned short)(kMiddleWidth * matchrange);
251  auto rightwidth = (unsigned short)(kRightWidth * matchrange);
252 
253  int nfreq = maxmatch + 1;
254  auto *freq = new unsigned short[nfreq];
255  memset(freq, 0, nfreq * sizeof(*freq));
256  for (long long frameno = 0; frameno < nframes; frameno++)
257  freq[matches[frameno]]++; /* freq[<matchcnt>] = <framecnt> */
258 
259  ushort matchstart = minmatch + (unsigned short)(kMatchStart * matchrange);
260  ushort matchend = minmatch + (unsigned short)(kMatchEnd * matchrange);
261 
262  int local_minimum = matchstart;
263  uint maxdelta = 0;
264  for (int matchcnt = matchstart + leftwidth + (middlewidth / 2);
265  matchcnt < matchend - rightwidth - middlewidth / 2;
266  matchcnt++)
267  {
268  ushort p0 = matchcnt - leftwidth - (middlewidth / 2);
269  ushort p1 = p0 + leftwidth;
270  ushort p2 = p1 + middlewidth;
271  ushort p3 = p2 + rightwidth;
272 
273  uint leftscore = range_area(freq, p0, p1);
274  uint middlescore = range_area(freq, p1, p2);
275  uint rightscore = range_area(freq, p2, p3);
276  if (middlescore < leftscore && middlescore < rightscore)
277  {
278  unsigned int delta = (leftscore - middlescore) +
279  (rightscore - middlescore);
280  if (delta > maxdelta)
281  {
282  local_minimum = matchcnt;
283  maxdelta = delta;
284  }
285  }
286  }
287 
288  LOG(VB_COMMFLAG, LOG_INFO,
289  QString("pick_mintmpledges minmatch=%1 maxmatch=%2"
290  " matchstart=%3 matchend=%4 widths=%5,%6,%7 local_minimum=%8")
291  .arg(minmatch).arg(maxmatch).arg(matchstart).arg(matchend)
292  .arg(leftwidth).arg(middlewidth).arg(rightwidth)
293  .arg(local_minimum));
294 
295  delete []freq;
296  delete []sorted;
297  return local_minimum;
298 }
299 
300 }; /* namespace */
301 
302 TemplateMatcher::TemplateMatcher(std::shared_ptr<PGMConverter> pgmc,
303  std::shared_ptr<EdgeDetector> ed,
304  TemplateFinder *tf, const QString& debugdir) :
305  m_pgmConverter(std::move(pgmc)),
306  m_edgeDetector(std::move(ed)), m_templateFinder(tf),
307  m_debugDir(debugdir),
309  m_debugData(debugdir + "/TemplateMatcher-pgm.txt")
310 #else /* !PGM_CONVERT_GREYSCALE */
311  m_debugData(debugdir + "/TemplateMatcher-yuv.txt")
312 #endif /* !PGM_CONVERT_GREYSCALE */
313 {
314  /*
315  * debugLevel:
316  * 0: no debugging
317  * 1: cache frame edge counts into debugdir [1 file]
318  * 2: extra verbosity [O(nframes)]
319  */
320  m_debugLevel = gCoreContext->GetNumSetting("TemplateMatcherDebugLevel", 0);
321 
322  if (m_debugLevel >= 1)
323  {
325  QString("TemplateMatcher debugLevel %1").arg(m_debugLevel));
326  m_debugMatches = true;
327  if (m_debugLevel >= 2)
328  m_debugRemoveRunts = true;
329  }
330 }
331 
333 {
334  delete []m_matches;
335  delete []m_match;
336  av_freep(reinterpret_cast<void*>(&m_cropped.data[0]));
337 }
338 
341  long long nframes)
342 {
343  m_player = _player;
345 
348  if (m_tmpl == nullptr)
349  {
350  LOG(VB_COMMFLAG, LOG_ERR,
351  QString("TemplateMatcher::MythPlayerInited: no template"));
352  return ANALYZE_FATAL;
353  }
354 
355  if (av_image_alloc(m_cropped.data, m_cropped.linesize,
356  m_tmplWidth, m_tmplHeight, AV_PIX_FMT_GRAY8, IMAGE_ALIGN))
357  {
358  LOG(VB_COMMFLAG, LOG_ERR,
359  QString("TemplateMatcher::MythPlayerInited "
360  "av_image_alloc cropped (%1x%2) failed")
361  .arg(m_tmplWidth).arg(m_tmplHeight));
362  return ANALYZE_FATAL;
363  }
364 
365  if (m_pgmConverter->MythPlayerInited(m_player))
366  goto free_cropped;
367 
368  m_matches = new unsigned short[nframes];
369  memset(m_matches, 0, nframes * sizeof(*m_matches));
370 
371  m_match = new unsigned char[nframes];
372 
373  if (m_debugMatches)
374  {
375  if (readMatches(m_debugData, m_matches, nframes))
376  {
377  LOG(VB_COMMFLAG, LOG_INFO,
378  QString("TemplateMatcher::MythPlayerInited read %1")
379  .arg(m_debugData));
380  m_matchesDone = true;
381  }
382  }
383 
384  if (m_matchesDone)
385  return ANALYZE_FINISHED;
386 
387  return ANALYZE_OK;
388 
389 free_cropped:
390  av_freep(reinterpret_cast<void*>(&m_cropped.data[0]));
391  return ANALYZE_FATAL;
392 }
393 
395 TemplateMatcher::analyzeFrame(const MythVideoFrame *frame, long long frameno,
396  long long *pNextFrame)
397 {
398  /*
399  * TUNABLE:
400  *
401  * The matching area should be a lot smaller than the area used by
402  * TemplateFinder, so use a smaller percentile than the TemplateFinder
403  * (intensity requirements to be considered an "edge" are lower, because
404  * there should be less pollution from non-template edges).
405  *
406  * Higher values mean fewer edge pixels in the candidate template area;
407  * occluded or faint templates might be missed.
408  *
409  * Lower values mean more edge pixels in the candidate template area;
410  * non-template edges can be picked up and cause false identification of
411  * matches.
412  */
413  const int FRAMESGMPCTILE = 70;
414 
415  /*
416  * TUNABLE:
417  *
418  * The per-pixel search radius for matching the template. Meant to deal
419  * with template edges that might minimally slide around (such as for
420  * animated lighting effects).
421  *
422  * Higher values will pick up more pixels as matching the template
423  * (possibly false matches).
424  *
425  * Lower values will require more exact template matches, possibly missing
426  * true matches.
427  *
428  * The TemplateMatcher accumulates all its state in the "matches" array to
429  * be processed later by TemplateMatcher::finished.
430  */
431  const int JITTER_RADIUS = 0;
432 
433  const AVFrame *edges = nullptr;
434  int pgmwidth = 0;
435  int pgmheight = 0;
436  std::chrono::microseconds start {0us};
437  std::chrono::microseconds end {0us};
438 
439  *pNextFrame = kNextFrame;
440 
441  const AVFrame *pgm = m_pgmConverter->getImage(frame, frameno, &pgmwidth, &pgmheight);
442  if (pgm == nullptr)
443  goto error;
444 
445  start = nowAsDuration<std::chrono::microseconds>();
446 
447  if (pgm_crop(&m_cropped, pgm, pgmheight, m_tmplRow, m_tmplCol,
449  goto error;
450 
451  edges = m_edgeDetector->detectEdges(&m_cropped, m_tmplHeight, FRAMESGMPCTILE);
452  if (edges == nullptr)
453  goto error;
454 
455  if (pgm_match(m_tmpl, edges, m_tmplHeight, JITTER_RADIUS, &m_matches[frameno]))
456  goto error;
457 
458  end = nowAsDuration<std::chrono::microseconds>();
459  m_analyzeTime += (end - start);
460 
461  return ANALYZE_OK;
462 
463 error:
464  LOG(VB_COMMFLAG, LOG_ERR,
465  QString("TemplateMatcher::analyzeFrame error at frame %1 of %2")
466  .arg(frameno));
467  return ANALYZE_ERROR;
468 }
469 
470 int
471 TemplateMatcher::finished(long long nframes, bool final)
472 {
473  /*
474  * TUNABLE:
475  *
476  * Eliminate false negatives and false positives by eliminating breaks and
477  * segments shorter than these periods, subject to maximum bounds.
478  *
479  * Higher values could eliminate real breaks or segments entirely.
480  * Lower values can yield more false "short" breaks or segments.
481  */
482  const int MINBREAKLEN = (int)roundf(45 * m_fps); /* frames */
483  const int MINSEGLEN = (int)roundf(105 * m_fps); /* frames */
484 
485  int minbreaklen = 1;
486  int minseglen = 1;
487  long long brkb = 0;
488 
490  {
491  if (final && writeMatches(m_debugData, m_matches, nframes))
492  {
493  LOG(VB_COMMFLAG, LOG_INFO,
494  QString("TemplateMatcher::finished wrote %1") .arg(m_debugData));
495  m_matchesDone = true;
496  }
497  }
498 
499  int tmpledges = pgm_set(m_tmpl, m_tmplHeight);
500  int mintmpledges = pick_mintmpledges(m_matches, nframes);
501 
502  LOG(VB_COMMFLAG, LOG_INFO,
503  QString("TemplateMatcher::finished %1x%2@(%3,%4),"
504  " %5 edge pixels, want %6")
505  .arg(m_tmplWidth).arg(m_tmplHeight).arg(m_tmplCol).arg(m_tmplRow)
506  .arg(tmpledges).arg(mintmpledges));
507 
508  for (long long ii = 0; ii < nframes; ii++)
509  m_match[ii] = m_matches[ii] >= mintmpledges ? 1 : 0;
510 
511  if (m_debugLevel >= 2)
512  {
513  if (final && finishedDebug(nframes, m_matches, m_match))
514  goto error;
515  }
516 
517  /*
518  * Construct breaks; find the first logo break.
519  */
520  m_breakMap.clear();
521  brkb = m_match[0] ? matchspn(nframes, m_match, 0, m_match[0]) : 0;
522  while (brkb < nframes)
523  {
524  /* Skip over logo-less frames; find the next logo frame (brke). */
525  long long brke = matchspn(nframes, m_match, brkb, m_match[brkb]);
526  long long brklen = brke - brkb;
527  m_breakMap.insert(brkb, brklen);
528 
529  /* Find the next logo break. */
530  brkb = matchspn(nframes, m_match, brke, m_match[brke]);
531  }
532 
533  /* Clean up the "noise". */
534  minbreaklen = 1;
535  minseglen = 1;
536  for (;;)
537  {
538  bool f1 = false;
539  bool f2 = false;
540  if (minbreaklen <= MINBREAKLEN)
541  {
542  f1 = removeShortBreaks(&m_breakMap, m_fps, minbreaklen,
544  minbreaklen++;
545  }
546  if (minseglen <= MINSEGLEN)
547  {
548  f2 = removeShortSegments(&m_breakMap, nframes, m_fps, minseglen,
550  minseglen++;
551  }
552  if (minbreaklen > MINBREAKLEN && minseglen > MINSEGLEN)
553  break;
554  if (m_debugRemoveRunts && (f1 || f2))
555  frameAnalyzerReportMap(&m_breakMap, m_fps, "** TM Break");
556  }
557 
558  /*
559  * Report breaks.
560  */
561  frameAnalyzerReportMap(&m_breakMap, m_fps, "TM Break");
562 
563  return 0;
564 
565 error:
566  return -1;
567 }
568 
569 int
571 {
572  if (m_pgmConverter->reportTime())
573  return -1;
574 
575  LOG(VB_COMMFLAG, LOG_INFO, QString("TM Time: analyze=%1s")
576  .arg(strftimeval(m_analyzeTime)));
577  return 0;
578 }
579 
580 int
581 TemplateMatcher::templateCoverage(long long nframes, bool final) const
582 {
583  /*
584  * Return <0 for too little logo coverage (some content has no logo), 0 for
585  * "correct" coverage, and >0 for too much coverage (some commercials have
586  * logos).
587  */
588 
589  /*
590  * TUNABLE:
591  *
592  * Expect 20%-45% of total length to be commercials.
593  */
594  static const int64_t MINBREAKS { nframes * 20 / 100 };
595  static const int64_t MAXBREAKS { nframes * 45 / 100 };
596 
597  const int64_t brklen = frameAnalyzerMapSum(&m_breakMap);
598  const bool good = brklen >= MINBREAKS && brklen <= MAXBREAKS;
599 
600  if (m_debugLevel >= 1)
601  {
602  if (!m_tmpl)
603  {
604  LOG(VB_COMMFLAG, LOG_ERR,
605  QString("TemplateMatcher: no template (wanted %2-%3%)")
606  .arg(100 * MINBREAKS / nframes)
607  .arg(100 * MAXBREAKS / nframes));
608  }
609  else if (!final)
610  {
611  LOG(VB_COMMFLAG, LOG_ERR,
612  QString("TemplateMatcher has %1% breaks (real-time flagging)")
613  .arg(100 * brklen / nframes));
614  }
615  else if (good)
616  {
617  LOG(VB_COMMFLAG, LOG_INFO, QString("TemplateMatcher has %1% breaks")
618  .arg(100 * brklen / nframes));
619  }
620  else
621  {
622  LOG(VB_COMMFLAG, LOG_INFO,
623  QString("TemplateMatcher has %1% breaks (wanted %2-%3%)")
624  .arg(100 * brklen / nframes)
625  .arg(100 * MINBREAKS / nframes)
626  .arg(100 * MAXBREAKS / nframes));
627  }
628  }
629 
630  if (!final)
631  return 0; /* real-time flagging */
632 
633  if (brklen < MINBREAKS)
634  return 1;
635  if (brklen <= MAXBREAKS)
636  return 0;
637  return -1;
638 }
639 
640 int
642  long long nframes)
643 {
644  const FrameAnalyzer::FrameMap *blankMap = blankFrameDetector->getBlanks();
645 
646  /*
647  * TUNABLE:
648  *
649  * Tune the search for blank frames near appearances and disappearances of
650  * the template.
651  *
652  * TEMPLATE_DISAPPEARS_EARLY: Handle the scenario where the template can
653  * disappear "early" before switching to commercial. Delay the beginning of
654  * the commercial break by searching forwards to find a blank frame.
655  *
656  * Setting this value too low can yield false positives. If template
657  * does indeed disappear "early" (common in US broadcast), the end of
658  * the broadcast segment can be misidentified as part of the beginning
659  * of the commercial break.
660  *
661  * Setting this value too high can yield or worsen false negatives. If
662  * the template presence extends at all into the commercial break,
663  * immediate cuts to commercial without any intervening blank frames
664  * can cause the broadcast to "continue" even further into the
665  * commercial break.
666  *
667  * TEMPLATE_DISAPPEARS_LATE: Handle the scenario where the template
668  * disappears "late" after having already switched to commercial (template
669  * presence extends into commercial break). Accelerate the beginning of the
670  * commercial break by searching backwards to find a blank frame.
671  *
672  * Setting this value too low can yield false negatives. If the
673  * template does extend deep into the commercial break, the first part
674  * of the commercial break can be misidentifed as part of the
675  * broadcast.
676  *
677  * Setting this value too high can yield or worsen false positives. If
678  * the template disappears extremely early (too early for
679  * TEMPLATE_DISAPPEARS_EARLY), blank frames before the template
680  * disappears can cause even more of the end of the broadcast segment
681  * to be misidentified as the first part of the commercial break.
682  *
683  * TEMPLATE_REAPPEARS_LATE: Handle the scenario where the template
684  * reappears "late" after having already returned to the broadcast.
685  * Accelerate the beginning of the broadcast by searching backwards to find
686  * a blank frame.
687  *
688  * Setting this value too low can yield false positives. If the
689  * template does indeed reappear "late" (common in US broadcast), the
690  * beginning of the broadcast segment can be misidentified as the end
691  * of the commercial break.
692  *
693  * Setting this value too high can yield or worsen false negatives. If
694  * the template actually reappears early, blank frames before the
695  * template reappears can cause even more of the end of the commercial
696  * break to be misidentified as the first part of the broadcast break.
697  *
698  * TEMPLATE_REAPPEARS_EARLY: Handle the scenario where the template
699  * reappears "early" before resuming the broadcast. Delay the beginning of
700  * the broadcast by searching forwards to find a blank frame.
701  *
702  * Setting this value too low can yield false negatives. If the
703  * template does reappear "early", the the last part of the commercial
704  * break can be misidentified as part of the beginning of the
705  * following broadcast segment.
706  *
707  * Setting this value too high can yield or worsen false positives. If
708  * the template reappears extremely late (too late for
709  * TEMPLATE_REAPPEARS_LATE), blank frames after the template reappears
710  * can cause even more of the broadcast segment can be misidentified
711  * as part of the end of the commercial break.
712  */
713  const int BLANK_NEARBY = (int)roundf(0.5F * m_fps);
714  const int TEMPLATE_DISAPPEARS_EARLY = (int)roundf(25 * m_fps);
715  const int TEMPLATE_DISAPPEARS_LATE = (int)roundf(0 * m_fps);
716  const int TEMPLATE_REAPPEARS_LATE = (int)roundf(35 * m_fps);
717  const int TEMPLATE_REAPPEARS_EARLY = (int)roundf(1.5F * m_fps);
718 
719  LOG(VB_COMMFLAG, LOG_INFO, QString("TemplateMatcher adjusting for blanks"));
720 
721  FrameAnalyzer::FrameMap::Iterator ii = m_breakMap.begin();
722  long long prevbrke = 0;
723  while (ii != m_breakMap.end())
724  {
725  FrameAnalyzer::FrameMap::Iterator iinext = ii;
726  ++iinext;
727 
728  /*
729  * Where the template disappears, look for nearby blank frames. Only
730  * adjust beginning-of-breaks that are not at the very beginning. Do
731  * not search into adjacent breaks.
732  */
733  const long long brkb = ii.key();
734  const long long brke = brkb + *ii;
735  FrameAnalyzer::FrameMap::const_iterator jj = blankMap->constEnd();
736  if (brkb > 0)
737  {
738  jj = frameMapSearchForwards(blankMap,
739  std::max(prevbrke,
740  brkb - std::max(BLANK_NEARBY, TEMPLATE_DISAPPEARS_LATE)),
741  std::min(brke,
742  brkb + std::max(BLANK_NEARBY, TEMPLATE_DISAPPEARS_EARLY)));
743  }
744  long long newbrkb = brkb;
745  if (jj != blankMap->constEnd())
746  {
747  newbrkb = jj.key();
748  long long adj = *jj / 2;
749  adj = std::min<long long>(adj, MAX_BLANK_FRAMES);
750  newbrkb += adj;
751  }
752 
753  /*
754  * Where the template reappears, look for nearby blank frames. Do not
755  * search into adjacent breaks.
756  */
757  FrameAnalyzer::FrameMap::const_iterator kk = frameMapSearchBackwards(
758  blankMap,
759  std::max(newbrkb,
760  brke - std::max(BLANK_NEARBY, TEMPLATE_REAPPEARS_LATE)),
761  std::min(iinext == m_breakMap.end() ? nframes : iinext.key(),
762  brke + std::max(BLANK_NEARBY, TEMPLATE_REAPPEARS_EARLY)));
763  long long newbrke = brke;
764  if (kk != blankMap->constEnd())
765  {
766  newbrke = kk.key();
767  long long adj = *kk;
768  newbrke += adj;
769  adj /= 2;
770  adj = std::min<long long>(adj, MAX_BLANK_FRAMES);
771  newbrke -= adj;
772  }
773 
774  /*
775  * Adjust breakMap for blank frames.
776  */
777  long long newbrklen = newbrke - newbrkb;
778  if (newbrkb != brkb)
779  {
780  m_breakMap.erase(ii);
781  if (newbrkb < nframes && newbrklen)
782  m_breakMap.insert(newbrkb, newbrklen);
783  }
784  else if (newbrke != brke)
785  {
786  if (newbrklen)
787  {
788  m_breakMap.remove(newbrkb);
789  m_breakMap.insert(newbrkb, newbrklen);
790  }
791  else
792  {
793  m_breakMap.erase(ii);
794  }
795  }
796 
797  prevbrke = newbrke;
798  ii = iinext;
799  }
800 
801  /*
802  * Report breaks.
803  */
804  frameAnalyzerReportMap(&m_breakMap, m_fps, "TM Break");
805  return 0;
806 }
807 
808 int
810 {
811  breaks->clear();
812  for (auto bb = m_breakMap.begin();
813  bb != m_breakMap.end();
814  ++bb)
815  {
816  breaks->insert(bb.key(), *bb);
817  }
818  return 0;
819 }
820 
821 /* vim: set expandtab tabstop=4 shiftwidth=4: */
TemplateMatcher::m_tmplCol
int m_tmplCol
Definition: TemplateMatcher.h:64
TemplateMatcher.h
frameAnalyzer::removeShortBreaks
bool removeShortBreaks(FrameAnalyzer::FrameMap *breakMap, float fps, int minbreaklen, bool verbose)
Definition: FrameAnalyzer.cpp:96
FrameAnalyzer::ANALYZE_ERROR
@ ANALYZE_ERROR
Definition: FrameAnalyzer.h:38
error
static void error(const char *str,...)
Definition: vbi.cpp:37
TemplateMatcher::m_player
MythPlayer * m_player
Definition: TemplateMatcher.h:80
TemplateMatcher::m_fps
float m_fps
Definition: TemplateMatcher.h:72
CommDetector2.h
freq
static const std::array< const uint32_t, 4 > freq
Definition: element.cpp:45
TemplateMatcher::m_edgeDetector
std::shared_ptr< EdgeDetector > m_edgeDetector
Definition: TemplateMatcher.h:60
cc
Definition: cc.h:9
FrameAnalyzer::FrameMap
QMap< long long, long long > FrameMap
Definition: FrameAnalyzer.h:45
anonymous_namespace{TemplateMatcher.cpp}::pgm_set
int pgm_set(const AVFrame *pict, int height)
Definition: TemplateMatcher.cpp:35
TemplateMatcher::m_tmplRow
int m_tmplRow
Definition: TemplateMatcher.h:63
FrameAnalyzer::ANALYZE_OK
@ ANALYZE_OK
Definition: FrameAnalyzer.h:37
TemplateMatcher::m_debugLevel
int m_debugLevel
Definition: TemplateMatcher.h:77
MythPlayer::GetFrameRate
float GetFrameRate(void) const
Definition: mythplayer.h:134
LOG
#define LOG(_MASK_, _LEVEL_, _QSTRING_)
Definition: mythlogging.h:39
mythplayer.h
MythPlayer
Definition: mythplayer.h:83
commDetector2::createDebugDirectory
void createDebugDirectory(const QString &dirname, const QString &comment)
Definition: CommDetector2.cpp:225
mythburn.FILE
int FILE
Definition: mythburn.py:139
TemplateMatcher::m_breakMap
FrameAnalyzer::FrameMap m_breakMap
Definition: TemplateMatcher.h:74
anonymous_namespace{TemplateMatcher.cpp}::readMatches
bool readMatches(const QString &filename, unsigned short *matches, long long nframes)
Definition: TemplateMatcher.cpp:95
frameAnalyzer::removeShortSegments
bool removeShortSegments(FrameAnalyzer::FrameMap *breakMap, long long nframes, float fps, int minseglen, bool verbose)
Definition: FrameAnalyzer.cpp:143
frameAnalyzer::frameMapSearchBackwards
FrameAnalyzer::FrameMap::const_iterator frameMapSearchBackwards(const FrameAnalyzer::FrameMap *frameMap, long long markbegin, long long mark)
Definition: FrameAnalyzer.cpp:280
AVFrame
struct AVFrame AVFrame
Definition: BorderDetector.h:15
anonymous_namespace{TemplateMatcher.cpp}::sort_ascending
int sort_ascending(const void *aa, const void *bb)
Definition: TemplateMatcher.cpp:173
TemplateFinder
Definition: TemplateFinder.h:30
mythlogging.h
pgm.h
PGM_CONVERT_GREYSCALE
#define PGM_CONVERT_GREYSCALE
Definition: PGMConverter.h:28
PGMConverter.h
TemplateMatcher::m_templateFinder
TemplateFinder * m_templateFinder
Definition: TemplateMatcher.h:61
anonymous_namespace{TemplateMatcher.cpp}::writeMatches
bool writeMatches(const QString &filename, unsigned short *matches, long long nframes)
Definition: TemplateMatcher.cpp:126
TemplateMatcher::m_matchesDone
bool m_matchesDone
Definition: TemplateMatcher.h:83
anonymous_namespace{TemplateMatcher.cpp}::pick_mintmpledges
unsigned short pick_mintmpledges(const unsigned short *matches, long long nframes)
Definition: TemplateMatcher.cpp:211
BlankFrameDetector.h
p2
static guint32 * p2
Definition: goom_core.cpp:26
TemplateMatcher::m_debugMatches
bool m_debugMatches
Definition: TemplateMatcher.h:81
TemplateMatcher::templateCoverage
int templateCoverage(long long nframes, bool final) const
Definition: TemplateMatcher.cpp:581
pgm_crop
int pgm_crop(AVFrame *dst, const AVFrame *src, [[maybe_unused]] int srcheight, int srcrow, int srccol, int cropwidth, int cropheight)
Definition: pgm.cpp:168
TemplateMatcher::TemplateMatcher
TemplateMatcher(std::shared_ptr< PGMConverter > pgmc, std::shared_ptr< EdgeDetector > ed, TemplateFinder *tf, const QString &debugdir)
Definition: TemplateMatcher.cpp:302
anonymous_namespace{TemplateMatcher.cpp}::finishedDebug
int finishedDebug(long long nframes, const unsigned short *matches, const unsigned char *match)
Definition: TemplateMatcher.cpp:142
TemplateMatcher::m_debugDir
QString m_debugDir
Definition: TemplateMatcher.h:78
FrameAnalyzer::analyzeFrameResult
analyzeFrameResult
Definition: FrameAnalyzer.h:36
FrameAnalyzer::kNextFrame
static const long long kNextFrame
Definition: FrameAnalyzer.h:59
FrameAnalyzer.h
TemplateMatcher::m_pgmConverter
std::shared_ptr< PGMConverter > m_pgmConverter
Definition: TemplateMatcher.h:59
FrameAnalyzer::ANALYZE_FATAL
@ ANALYZE_FATAL
Definition: FrameAnalyzer.h:40
TemplateMatcher::m_cropped
AVFrame m_cropped
Definition: TemplateMatcher.h:73
gCoreContext
MythCoreContext * gCoreContext
This global variable contains the MythCoreContext instance for the app.
Definition: mythcorecontext.cpp:55
TemplateMatcher::m_debugData
QString m_debugData
Definition: TemplateMatcher.h:79
commDetector2::strftimeval
QString strftimeval(std::chrono::microseconds usecs)
Definition: CommDetector2.cpp:263
EdgeDetector.h
p1
static guint32 * p1
Definition: goom_core.cpp:26
MythCoreContext::GetNumSetting
int GetNumSetting(const QString &key, int defaultval=0)
Definition: mythcorecontext.cpp:916
TemplateMatcher::m_matches
unsigned short * m_matches
Definition: TemplateMatcher.h:69
BlankFrameDetector
Definition: BlankFrameDetector.h:15
frameAnalyzer::frameAnalyzerMapSum
long long frameAnalyzerMapSum(const FrameAnalyzer::FrameMap *frameMap)
Definition: FrameAnalyzer.cpp:89
BlankFrameDetector::getBlanks
const FrameAnalyzer::FrameMap * getBlanks(void) const
Definition: BlankFrameDetector.h:34
frameAnalyzer::frameAnalyzerReportMap
void frameAnalyzerReportMap(const FrameAnalyzer::FrameMap *frameMap, float fps, const char *comment)
Definition: FrameAnalyzer.cpp:17
mythcorecontext.h
frameAnalyzer::frameMapSearchForwards
FrameAnalyzer::FrameMap::const_iterator frameMapSearchForwards(const FrameAnalyzer::FrameMap *frameMap, long long mark, long long markend)
Definition: FrameAnalyzer.cpp:253
anonymous_namespace{TemplateMatcher.cpp}::matchspn
long long matchspn(long long nframes, const unsigned char *match, long long frameno, unsigned char acceptval)
Definition: TemplateMatcher.cpp:178
TemplateMatcher::m_debugRemoveRunts
bool m_debugRemoveRunts
Definition: TemplateMatcher.h:82
TemplateMatcher::analyzeFrame
enum analyzeFrameResult analyzeFrame(const MythVideoFrame *frame, long long frameno, long long *pNextFrame) override
Definition: TemplateMatcher.cpp:395
TemplateMatcher::finished
int finished(long long nframes, bool final) override
Definition: TemplateMatcher.cpp:471
commDetector2
Definition: CommDetector2.cpp:189
TemplateMatcher::m_tmpl
const struct AVFrame * m_tmpl
Definition: TemplateMatcher.h:62
TemplateMatcher::~TemplateMatcher
~TemplateMatcher(void) override
Definition: TemplateMatcher.cpp:332
TemplateMatcher::computeBreaks
int computeBreaks(FrameMap *breaks)
Definition: TemplateMatcher.cpp:809
frameAnalyzer
Definition: FrameAnalyzer.cpp:7
MAX_BLANK_FRAMES
static constexpr int64_t MAX_BLANK_FRAMES
Definition: CommDetectorBase.h:11
TemplateMatcher::reportTime
int reportTime(void) const override
Definition: TemplateMatcher.cpp:570
TemplateFinder.h
TemplateMatcher::m_tmplHeight
int m_tmplHeight
Definition: TemplateMatcher.h:66
MythVideoFrame
Definition: mythframe.h:87
TemplateMatcher::MythPlayerInited
enum analyzeFrameResult MythPlayerInited(MythPlayer *player, long long nframes) override
Definition: TemplateMatcher.cpp:340
TemplateMatcher::m_analyzeTime
std::chrono::microseconds m_analyzeTime
Definition: TemplateMatcher.h:84
FrameAnalyzer::ANALYZE_FINISHED
@ ANALYZE_FINISHED
Definition: FrameAnalyzer.h:39
TemplateMatcher::adjustForBlanks
int adjustForBlanks(const BlankFrameDetector *blankFrameDetector, long long nframes)
Definition: TemplateMatcher.cpp:641
anonymous_namespace{TemplateMatcher.cpp}::range_area
unsigned int range_area(const unsigned short *freq, unsigned short start, unsigned short end)
Definition: TemplateMatcher.cpp:190
build_compdb.filename
filename
Definition: build_compdb.py:21
TemplateFinder::getTemplate
const struct AVFrame * getTemplate(int *prow, int *pcol, int *pwidth, int *pheight) const
Definition: TemplateFinder.cpp:1043
TemplateMatcher::m_tmplWidth
int m_tmplWidth
Definition: TemplateMatcher.h:65
TemplateMatcher::m_match
unsigned char * m_match
Definition: TemplateMatcher.h:70
uint
unsigned int uint
Definition: freesurround.h:24
anonymous_namespace{TemplateMatcher.cpp}::pgm_match
int pgm_match(const AVFrame *tmpl, const AVFrame *test, int height, int radius, unsigned short *pscore)
Definition: TemplateMatcher.cpp:47