MythTV  0.27pre
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Groups Pages
SceneChangeDetector.cpp
Go to the documentation of this file.
1 // ANSI C headers
2 #include <cstdlib>
3 #include <cmath>
4 
5 // MythTV headers
6 #include "mythcorecontext.h" /* gContext */
7 #include "mythplayer.h"
8 #include "mythlogging.h"
9 
10 // Commercial Flagging headers
11 #include "CommDetector2.h"
12 #include "FrameAnalyzer.h"
13 #include "quickselect.h"
14 #include "HistogramAnalyzer.h"
15 #include "SceneChangeDetector.h"
16 
17 using namespace commDetector2;
18 using namespace frameAnalyzer;
19 
20 namespace {
21 
22 int
23 scenechange_data_sort_desc_frequency(const void *aa, const void *bb)
24 {
25  /* Descending by frequency, then ascending by color. */
26  const struct SceneChangeDetector::scenechange_data *sc1 =
27  (const struct SceneChangeDetector::scenechange_data*)aa;
28  const struct SceneChangeDetector::scenechange_data *sc2 =
29  (const struct SceneChangeDetector::scenechange_data*)bb;
30  int freqdiff = sc2->frequency - sc1->frequency;
31  return freqdiff ? freqdiff : sc1->color - sc2->color;
32 }
33 
34 void
35 scenechange_data_init(SceneChangeDetector::SceneChangeData *scdata,
37 {
38  unsigned int ncolors = sizeof(*hh)/sizeof((*hh)[0]);
39 
40  for (unsigned int ii = 0; ii < ncolors; ii++)
41  {
42  (*scdata)[ii].color = ii;
43  (*scdata)[ii].frequency = (*hh)[ii];
44  }
45  qsort(*scdata, sizeof(*scdata)/sizeof((*scdata)[0]), sizeof((*scdata)[0]),
46  scenechange_data_sort_desc_frequency);
47 }
48 
49 unsigned short
50 scenechange_data_diff(const SceneChangeDetector::SceneChangeData *sc1,
52 {
53  /*
54  * Compute a notion of "difference" that takes into account the difference
55  * in relative frequencies of the dominant colors.
56  */
57  unsigned short diff = 0;
58  for (unsigned int ii = 0; ii < sizeof(*sc1)/sizeof((*sc1)[0]); ii++)
59  diff += abs((*sc1)[ii].frequency - (*sc2)[ii].frequency) +
60  abs((*sc1)[ii].color - (*sc2)[ii].color);
61  return diff;
62 }
63 
64 bool
65 writeData(QString filename, const unsigned short *scdiff, long long nframes)
66 {
67  FILE *fp;
68  long long frameno;
69 
70  QByteArray fname = filename.toLocal8Bit();
71  if (!(fp = fopen(fname.constData(), "w")))
72  return false;
73  for (frameno = 0; frameno < nframes; frameno++)
74  (void)fprintf(fp, "%5u\n", scdiff[frameno]);
75  if (fclose(fp))
76  LOG(VB_COMMFLAG, LOG_ERR, QString("Error closing %1: %2")
77  .arg(filename).arg(strerror(errno)));
78  return true;
79 }
80 
81 void
82 computeChangeMap(FrameAnalyzer::FrameMap *changeMap, long long nframes,
83  const unsigned short *scdiff, unsigned short mindiff)
84 {
85  /*
86  * Look for sudden changes in histogram.
87  */
88  long long frameno;
89 
90  changeMap->clear();
91  for (frameno = 0; frameno < nframes; frameno++)
92  {
93  if (scdiff[frameno] > mindiff)
94  changeMap->insert(frameno, 0);
95  }
96 }
97 
98 }; /* namespace */
99 
101  QString debugdir)
102  : FrameAnalyzer()
103  , histogramAnalyzer(ha)
104  , fps(0.0f)
105  , scdata(NULL)
106  , scdiff(NULL)
107  , debugLevel(0)
108  , debugdata(debugdir + "/SceneChangeDetector.txt")
111 {
112  LOG(VB_COMMFLAG, LOG_INFO, "SceneChangeDetector");
113 
114  /*
115  * debugLevel:
116  * 0: no debugging
117  * 2: extra verbosity [O(nframes)]
118  */
119  debugLevel = gCoreContext->GetNumSetting("SceneChangeDetectorDebugLevel", 0);
120 
121  if (debugLevel >= 1)
122  {
123  createDebugDirectory(debugdir,
124  QString("SceneChangeDetector debugLevel %1").arg(debugLevel));
125  debug_scenechange = true;
126  }
127 }
128 
130 {
131  if (scdata)
132  delete []scdata;
133  if (scdiff)
134  delete []scdiff;
135 }
136 
139  long long nframes)
140 {
142  histogramAnalyzer->MythPlayerInited(player, nframes);
143 
144  fps = player->GetFrameRate();
145 
146  scdata = new SceneChangeData[nframes];
147  memset(scdata, 0, nframes * sizeof(*scdata));
148 
149  scdiff = new unsigned short[nframes];
150  memset(scdiff, 0, nframes * sizeof(*scdiff));
151 
152  QSize video_disp_dim = player->GetVideoSize();
153 
154  LOG(VB_COMMFLAG, LOG_INFO,
155  QString("SceneChangeDetector::MythPlayerInited %1x%2")
156  .arg(video_disp_dim.width())
157  .arg(video_disp_dim.height()));
158 
159  return ares;
160 }
161 
163 SceneChangeDetector::analyzeFrame(const VideoFrame *frame, long long frameno,
164  long long *pNextFrame)
165 {
166  *pNextFrame = NEXTFRAME;
167 
168  if (histogramAnalyzer->analyzeFrame(frame, frameno) ==
170  return ANALYZE_OK;
171 
172  LOG(VB_COMMFLAG, LOG_ERR,
173  QString("SceneChangeDetector::analyzeFrame error at frame %1")
174  .arg(frameno));
175  return ANALYZE_ERROR;
176 }
177 
178 int
179 SceneChangeDetector::finished(long long nframes, bool final)
180 {
181  if (histogramAnalyzer->finished(nframes, final))
182  return -1;
183 
184  LOG(VB_COMMFLAG, LOG_INFO, QString("SceneChangeDetector::finished(%1)")
185  .arg(nframes));
186 
187  const HistogramAnalyzer::Histogram *histogram =
189  for (unsigned int frameno = 0; frameno < nframes; frameno++)
190  (void)scenechange_data_init(&scdata[frameno], &histogram[frameno]);
191  scdiff[0] = 0;
192  for (unsigned int frameno = 1; frameno < nframes; frameno++)
193  scdiff[frameno] = scenechange_data_diff(&scdata[frameno - 1],
194  &scdata[frameno]);
195 
197  {
198  if (final && writeData(debugdata, scdiff, nframes))
199  {
200  LOG(VB_COMMFLAG, LOG_INFO,
201  QString("SceneChangeDetector::finished wrote %1")
202  .arg(debugdata));
203  scenechange_done = true;
204  }
205  }
206 
207  /* Identify all scene-change frames (changeMap). */
208  unsigned short *scdiffsort = new unsigned short[nframes];
209  memcpy(scdiffsort, scdiff, nframes * sizeof(*scdiff));
210  unsigned short mindiff = quick_select_ushort(scdiffsort, nframes,
211  (int)(0.979472 * nframes));
212  LOG(VB_COMMFLAG, LOG_INFO,
213  QString("SceneChangeDetector::finished applying threshold value %1")
214  .arg(mindiff));
215  computeChangeMap(&changeMap, nframes, scdiff, mindiff);
216  delete []scdiffsort;
217  if (debugLevel >= 2)
218  frameAnalyzerReportMapms(&changeMap, fps, "SC frame");
219 
220  return 0;
221 }
222 
223 int
225 {
226  return histogramAnalyzer->reportTime();
227 }
228 
229 /* vim: set expandtab tabstop=4 shiftwidth=4: */