blob: 6c2a35f68709441f4e3bb0eb6aa87779afaed81a [file] [log] [blame]
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -04001/*
2 * Copyright (c) 2017, Alliance for Open Media. All rights reserved
3 *
4 * This source code is subject to the terms of the BSD 2 Clause License and
5 * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
6 * was not distributed with this source code in the LICENSE file, you can
7 * obtain it at www.aomedia.org/license/software. If the Alliance for Open
8 * Media Patent License 1.0 was not distributed with this source code in the
9 * PATENTS file, you can obtain it at www.aomedia.org/license/patent.
10 */
11#include <wx/wx.h>
12#include <wx/aboutdlg.h>
13#include <wx/cmdline.h>
14#include <wx/dcbuffer.h>
15#include "./tools_common.h"
16#include "./video_reader.h"
17#include "aom/aom_decoder.h"
18#include "aom/aomdx.h"
19#include "av1/common/accounting.h"
20#include "av1/common/onyxc_int.h"
Nathan E. Egge2693ca52017-02-22 16:23:02 -050021#include "av1/decoder/inspection.h"
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -040022
23#define OD_SIGNMASK(a) (-((a) < 0))
24#define OD_FLIPSIGNI(a, b) (((a) + OD_SIGNMASK(b)) ^ OD_SIGNMASK(b))
25#define OD_DIV_ROUND(x, y) (((x) + OD_FLIPSIGNI((y) >> 1, x)) / (y))
26
Nathan E. Egge625bc262017-02-22 10:42:54 -050027enum {
28 OD_LUMA_MASK = 1 << 0,
29 OD_CB_MASK = 1 << 1,
30 OD_CR_MASK = 1 << 2,
31 OD_ALL_MASK = OD_LUMA_MASK | OD_CB_MASK | OD_CR_MASK
32};
33
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -040034class AV1Decoder {
35 private:
36 FILE *input;
37 wxString path;
38
39 AvxVideoReader *reader;
40 const AvxVideoInfo *info;
41 const AvxInterface *decoder;
42
Nathan E. Egge2693ca52017-02-22 16:23:02 -050043 insp_frame_data frame_data;
44
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -040045 aom_codec_ctx_t codec;
46
47 public:
48 aom_image_t *image;
49 int frame;
50
Nathan E. Egge625bc262017-02-22 10:42:54 -050051 int plane_mask;
52
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -040053 AV1Decoder();
54 ~AV1Decoder();
55
56 bool open(const wxString &path);
57 void close();
58 bool step();
59
60 int getWidth() const;
61 int getHeight() const;
62
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -040063 bool getAccountingStruct(Accounting **acct);
Nathan E. Egge2693ca52017-02-22 16:23:02 -050064 bool setInspectionCallback();
65
66 static void inspect(void *decoder, void *data);
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -040067};
68
69AV1Decoder::AV1Decoder()
70 : reader(NULL), info(NULL), decoder(NULL), image(NULL), frame(0) {}
71
72AV1Decoder::~AV1Decoder() {}
73
74bool AV1Decoder::open(const wxString &path) {
75 reader = aom_video_reader_open(path.mb_str());
76 if (!reader) {
77 fprintf(stderr, "Failed to open %s for reading.", path.mb_str().data());
78 return false;
79 }
80 this->path = path;
81 info = aom_video_reader_get_info(reader);
82 decoder = get_aom_decoder_by_fourcc(info->codec_fourcc);
83 if (!decoder) {
84 fprintf(stderr, "Unknown input codec.");
85 return false;
86 }
87 printf("Using %s\n", aom_codec_iface_name(decoder->codec_interface()));
88 if (aom_codec_dec_init(&codec, decoder->codec_interface(), NULL, 0)) {
89 fprintf(stderr, "Failed to initialize decoder.");
90 return false;
91 }
Nathan E. Egge2693ca52017-02-22 16:23:02 -050092 ifd_init(&frame_data, info->frame_width, info->frame_height);
93 setInspectionCallback();
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -040094 return true;
95}
96
97void AV1Decoder::close() {}
98
99bool AV1Decoder::step() {
100 if (aom_video_reader_read_frame(reader)) {
101 size_t frame_size;
102 const unsigned char *frame_data;
103 frame_data = aom_video_reader_get_frame(reader, &frame_size);
104 if (aom_codec_decode(&codec, frame_data, frame_size, NULL, 0)) {
105 fprintf(stderr, "Failed to decode frame.");
106 return false;
107 } else {
108 aom_codec_iter_t iter = NULL;
109 image = aom_codec_get_frame(&codec, &iter);
110 if (image != NULL) {
111 frame++;
112 return true;
113 }
114 return false;
115 }
116 }
117 return false;
118}
119
120int AV1Decoder::getWidth() const { return info->frame_width; }
121
122int AV1Decoder::getHeight() const { return info->frame_height; }
123
124bool AV1Decoder::getAccountingStruct(Accounting **accounting) {
125 return aom_codec_control(&codec, AV1_GET_ACCOUNTING, accounting) ==
126 AOM_CODEC_OK;
127}
128
Nathan E. Egge2693ca52017-02-22 16:23:02 -0500129bool AV1Decoder::setInspectionCallback() {
130 aom_inspect_init ii;
131 ii.inspect_cb = AV1Decoder::inspect;
132 ii.inspect_ctx = (void *)this;
133 return aom_codec_control(&codec, AV1_SET_INSPECTION_CALLBACK, &ii) ==
134 AOM_CODEC_OK;
135}
136
137void AV1Decoder::inspect(void *pbi, void *data) {
138 AV1Decoder *decoder = (AV1Decoder *)data;
139 ifd_inspect(&decoder->frame_data, pbi);
140}
141
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400142#define MIN_ZOOM (1)
143#define MAX_ZOOM (4)
144
145class AnalyzerPanel : public wxPanel {
146 DECLARE_EVENT_TABLE()
147
148 private:
149 AV1Decoder decoder;
150 const wxString path;
151
152 int zoom;
153 unsigned char *pixels;
154
155 const bool bit_accounting;
156 double *bpp_q3;
157
Nathan E. Egge625bc262017-02-22 10:42:54 -0500158 int plane_mask;
159
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400160 // The display size is the decode size, scaled by the zoom.
161 int getDisplayWidth() const;
162 int getDisplayHeight() const;
163
164 bool updateDisplaySize();
165
166 void computeBitsPerPixel();
167
168 public:
169 AnalyzerPanel(wxWindow *parent, const wxString &path,
170 const bool bit_accounting);
171 ~AnalyzerPanel();
172
173 bool open(const wxString &path);
174 void close();
175 void render();
176 bool nextFrame();
177 void refresh();
178
179 int getZoom() const;
180 bool setZoom(int zoom);
181
Nathan E. Egge625bc262017-02-22 10:42:54 -0500182 void setShowPlane(bool show_plane, int mask);
183
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400184 void onPaint(wxPaintEvent &event); // NOLINT
185};
186
187BEGIN_EVENT_TABLE(AnalyzerPanel, wxPanel)
188EVT_PAINT(AnalyzerPanel::onPaint)
189END_EVENT_TABLE()
190
191AnalyzerPanel::AnalyzerPanel(wxWindow *parent, const wxString &path,
192 const bool bit_accounting)
193 : wxPanel(parent), path(path), zoom(0), pixels(NULL),
Nathan E. Egge625bc262017-02-22 10:42:54 -0500194 bit_accounting(bit_accounting), bpp_q3(NULL), plane_mask(OD_ALL_MASK) {}
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400195
196AnalyzerPanel::~AnalyzerPanel() { close(); }
197
Nathan E. Egge625bc262017-02-22 10:42:54 -0500198void AnalyzerPanel::setShowPlane(bool show_plane, int mask) {
199 if (show_plane) {
200 plane_mask |= mask;
201 } else {
202 plane_mask &= ~mask;
203 }
204}
205
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400206void AnalyzerPanel::render() {
207 aom_image_t *img = decoder.image;
208 int y_stride = img->stride[0];
209 int cb_stride = img->stride[1];
210 int cr_stride = img->stride[2];
211 int p_stride = 3 * getDisplayWidth();
212 unsigned char *y_row = img->planes[0];
213 unsigned char *cb_row = img->planes[1];
214 unsigned char *cr_row = img->planes[2];
215 unsigned char *p_row = pixels;
216 for (int j = 0; j < decoder.getHeight(); j++) {
217 unsigned char *y = y_row;
218 unsigned char *cb = cb_row;
219 unsigned char *cr = cr_row;
220 unsigned char *p = p_row;
221 for (int i = 0; i < decoder.getWidth(); i++) {
222 int64_t yval;
223 int64_t cbval;
224 int64_t crval;
Nathan E. Egge625bc262017-02-22 10:42:54 -0500225 int pmask;
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400226 unsigned rval;
227 unsigned gval;
228 unsigned bval;
229 yval = *y;
230 cbval = *cb;
231 crval = *cr;
Nathan E. Egge625bc262017-02-22 10:42:54 -0500232 pmask = plane_mask;
233 if (pmask & OD_LUMA_MASK) {
234 yval -= 16;
235 } else {
236 yval = 128;
237 }
238 cbval = ((pmask & OD_CB_MASK) >> 1) * (cbval - 128);
239 crval = ((pmask & OD_CR_MASK) >> 2) * (crval - 128);
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400240 /*This is intentionally slow and very accurate.*/
241 rval = OD_CLAMPI(0, (int32_t)OD_DIV_ROUND(
242 2916394880000LL * yval + 4490222169144LL * crval,
243 9745792000LL),
244 65535);
245 gval = OD_CLAMPI(0, (int32_t)OD_DIV_ROUND(2916394880000LL * yval -
246 534117096223LL * cbval -
247 1334761232047LL * crval,
248 9745792000LL),
249 65535);
250 bval = OD_CLAMPI(0, (int32_t)OD_DIV_ROUND(
251 2916394880000LL * yval + 5290866304968LL * cbval,
252 9745792000LL),
253 65535);
254 unsigned char *px_row = p;
255 for (int v = 0; v < zoom; v++) {
256 unsigned char *px = px_row;
257 for (int u = 0; u < zoom; u++) {
258 *(px + 0) = (unsigned char)(rval >> 8);
259 *(px + 1) = (unsigned char)(gval >> 8);
260 *(px + 2) = (unsigned char)(bval >> 8);
261 px += 3;
262 }
263 px_row += p_stride;
264 }
265 int dc = ((y - y_row) & 1) | (1 - img->x_chroma_shift);
266 y++;
267 cb += dc;
268 cr += dc;
269 p += zoom * 3;
270 }
271 int dc = -((j & 1) | (1 - img->y_chroma_shift));
272 y_row += y_stride;
273 cb_row += dc & cb_stride;
274 cr_row += dc & cr_stride;
275 p_row += zoom * p_stride;
276 }
277}
278
279void AnalyzerPanel::computeBitsPerPixel() {
280 Accounting *acct;
281 double bpp_total;
282 int totals_q3[MAX_SYMBOL_TYPES] = { 0 };
283 int sym_count[MAX_SYMBOL_TYPES] = { 0 };
284 decoder.getAccountingStruct(&acct);
285 for (int j = 0; j < decoder.getHeight(); j++) {
286 for (int i = 0; i < decoder.getWidth(); i++) {
287 bpp_q3[j * decoder.getWidth() + i] = 0.0;
288 }
289 }
290 bpp_total = 0;
291 for (int i = 0; i < acct->syms.num_syms; i++) {
292 AccountingSymbol *s;
293 s = &acct->syms.syms[i];
294 totals_q3[s->id] += s->bits;
295 sym_count[s->id] += s->samples;
296 }
297 printf("=== Frame: %-3i ===\n", decoder.frame - 1);
298 for (int i = 0; i < acct->syms.dictionary.num_strs; i++) {
299 if (totals_q3[i]) {
300 printf("%30s = %10.3f (%f bit/symbol)\n", acct->syms.dictionary.strs[i],
301 (float)totals_q3[i] / 8, (float)totals_q3[i] / 8 / sym_count[i]);
302 }
303 }
304 printf("\n");
305}
306
307bool AnalyzerPanel::nextFrame() {
308 if (decoder.step()) {
309 refresh();
310 return true;
311 }
312 return false;
313}
314
315void AnalyzerPanel::refresh() {
316 if (bit_accounting) {
317 computeBitsPerPixel();
318 }
319 render();
320}
321
322int AnalyzerPanel::getDisplayWidth() const { return zoom * decoder.getWidth(); }
323
324int AnalyzerPanel::getDisplayHeight() const {
325 return zoom * decoder.getHeight();
326}
327
328bool AnalyzerPanel::updateDisplaySize() {
329 unsigned char *p = (unsigned char *)malloc(
330 sizeof(*p) * 3 * getDisplayWidth() * getDisplayHeight());
331 if (p == NULL) {
332 return false;
333 }
334 free(pixels);
335 pixels = p;
336 SetSize(getDisplayWidth(), getDisplayHeight());
337 return true;
338}
339
340bool AnalyzerPanel::open(const wxString &path) {
341 if (!decoder.open(path)) {
342 return false;
343 }
344 if (!setZoom(MIN_ZOOM)) {
345 return false;
346 }
347 if (bit_accounting) {
348 bpp_q3 = (double *)malloc(sizeof(*bpp_q3) * decoder.getWidth() *
349 decoder.getHeight());
350 if (bpp_q3 == NULL) {
351 fprintf(stderr, "Could not allocate memory for bit accounting\n");
352 close();
353 return false;
354 }
355 }
356 if (!nextFrame()) {
357 close();
358 return false;
359 }
360 SetFocus();
361 return true;
362}
363
364void AnalyzerPanel::close() {
365 decoder.close();
366 free(pixels);
367 pixels = NULL;
368 free(bpp_q3);
369 bpp_q3 = NULL;
370}
371
372int AnalyzerPanel::getZoom() const { return zoom; }
373
374bool AnalyzerPanel::setZoom(int z) {
375 if (z <= MAX_ZOOM && z >= MIN_ZOOM && zoom != z) {
376 int old_zoom = zoom;
377 zoom = z;
378 if (!updateDisplaySize()) {
379 zoom = old_zoom;
380 return false;
381 }
382 return true;
383 }
384 return false;
385}
386
387void AnalyzerPanel::onPaint(wxPaintEvent &) {
388 wxBitmap bmp(wxImage(getDisplayWidth(), getDisplayHeight(), pixels, true));
389 wxBufferedPaintDC dc(this, bmp);
390}
391
392class AnalyzerFrame : public wxFrame {
393 DECLARE_EVENT_TABLE()
394
395 private:
396 AnalyzerPanel *panel;
397 const bool bit_accounting;
398
399 wxMenu *fileMenu;
400 wxMenu *viewMenu;
401 wxMenu *playbackMenu;
402
403 public:
404 AnalyzerFrame(const bool bit_accounting); // NOLINT
405
406 void onOpen(wxCommandEvent &event); // NOLINT
407 void onClose(wxCommandEvent &event); // NOLINT
408 void onQuit(wxCommandEvent &event); // NOLINT
409
410 void onZoomIn(wxCommandEvent &event); // NOLINT
411 void onZoomOut(wxCommandEvent &event); // NOLINT
412 void onActualSize(wxCommandEvent &event); // NOLINT
413
Nathan E. Egge625bc262017-02-22 10:42:54 -0500414 void onToggleViewMenuCheckBox(wxCommandEvent &event); // NOLINT
415 void onResetAndToggleViewMenuCheckBox(wxCommandEvent &event); // NOLINT
416
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400417 void onNextFrame(wxCommandEvent &event); // NOLINT
418 void onGotoFrame(wxCommandEvent &event); // NOLINT
419 void onRestart(wxCommandEvent &event); // NOLINT
420
421 void onAbout(wxCommandEvent &event); // NOLINT
422
423 bool open(const wxString &path);
424 bool setZoom(int zoom);
Nathan E. Egge625bc262017-02-22 10:42:54 -0500425 void updateViewMenu();
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400426};
427
428enum {
429 wxID_NEXT_FRAME = 6000,
Nathan E. Egge625bc262017-02-22 10:42:54 -0500430 wxID_SHOW_Y,
431 wxID_SHOW_U,
432 wxID_SHOW_V,
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400433 wxID_GOTO_FRAME,
434 wxID_RESTART,
435 wxID_ACTUAL_SIZE
436};
437
438BEGIN_EVENT_TABLE(AnalyzerFrame, wxFrame)
439EVT_MENU(wxID_OPEN, AnalyzerFrame::onOpen)
440EVT_MENU(wxID_CLOSE, AnalyzerFrame::onClose)
441EVT_MENU(wxID_EXIT, AnalyzerFrame::onQuit)
442EVT_MENU(wxID_ZOOM_IN, AnalyzerFrame::onZoomIn)
443EVT_MENU(wxID_ZOOM_OUT, AnalyzerFrame::onZoomOut)
444EVT_MENU(wxID_ACTUAL_SIZE, AnalyzerFrame::onActualSize)
Nathan E. Egge625bc262017-02-22 10:42:54 -0500445EVT_MENU(wxID_SHOW_Y, AnalyzerFrame::onResetAndToggleViewMenuCheckBox)
446EVT_MENU(wxID_SHOW_U, AnalyzerFrame::onResetAndToggleViewMenuCheckBox)
447EVT_MENU(wxID_SHOW_V, AnalyzerFrame::onResetAndToggleViewMenuCheckBox)
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400448EVT_MENU(wxID_NEXT_FRAME, AnalyzerFrame::onNextFrame)
449EVT_MENU(wxID_GOTO_FRAME, AnalyzerFrame::onGotoFrame)
450EVT_MENU(wxID_RESTART, AnalyzerFrame::onRestart)
451EVT_MENU(wxID_ABOUT, AnalyzerFrame::onAbout)
452END_EVENT_TABLE()
453
454AnalyzerFrame::AnalyzerFrame(const bool bit_accounting)
455 : wxFrame(NULL, wxID_ANY, _("AV1 Stream Analyzer"), wxDefaultPosition,
456 wxDefaultSize, wxDEFAULT_FRAME_STYLE),
457 panel(NULL), bit_accounting(bit_accounting) {
458 wxMenuBar *mb = new wxMenuBar();
459
460 fileMenu = new wxMenu();
461 fileMenu->Append(wxID_OPEN, _("&Open...\tCtrl-O"), _("Open daala file"));
462 fileMenu->Append(wxID_CLOSE, _("&Close\tCtrl-W"), _("Close daala file"));
463 fileMenu->Enable(wxID_CLOSE, false);
464 fileMenu->Append(wxID_EXIT, _("E&xit\tCtrl-Q"), _("Quit this program"));
465 mb->Append(fileMenu, _("&File"));
466
467 wxAcceleratorEntry entries[2];
468 entries[0].Set(wxACCEL_CTRL, (int)'=', wxID_ZOOM_IN);
469 entries[1].Set(wxACCEL_CTRL | wxACCEL_SHIFT, (int)'-', wxID_ZOOM_OUT);
470 wxAcceleratorTable accel(2, entries);
471 this->SetAcceleratorTable(accel);
472
473 viewMenu = new wxMenu();
474 viewMenu->Append(wxID_ZOOM_IN, _("Zoom-In\tCtrl-+"), _("Double image size"));
475 viewMenu->Append(wxID_ZOOM_OUT, _("Zoom-Out\tCtrl--"), _("Half image size"));
476 viewMenu->Append(wxID_ACTUAL_SIZE, _("Actual size\tCtrl-0"),
477 _("Actual size of the frame"));
Nathan E. Egge625bc262017-02-22 10:42:54 -0500478 viewMenu->AppendSeparator();
479 viewMenu->AppendCheckItem(wxID_SHOW_Y, _("&Y plane\tCtrl-Y"),
480 _("Show Y plane"));
481 viewMenu->AppendCheckItem(wxID_SHOW_U, _("&U plane\tCtrl-U"),
482 _("Show U plane"));
483 viewMenu->AppendCheckItem(wxID_SHOW_V, _("&V plane\tCtrl-V"),
484 _("Show V plane"));
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400485 mb->Append(viewMenu, _("&View"));
486
487 playbackMenu = new wxMenu();
488 playbackMenu->Append(wxID_NEXT_FRAME, _("Next frame\tCtrl-."),
489 _("Go to next frame"));
490 /*playbackMenu->Append(wxID_RESTART, _("&Restart\tCtrl-R"),
491 _("Set video to frame 0"));
492 playbackMenu->Append(wxID_GOTO_FRAME, _("Jump to Frame\tCtrl-J"),
493 _("Go to frame number"));*/
494 mb->Append(playbackMenu, _("&Playback"));
495
496 wxMenu *helpMenu = new wxMenu();
497 helpMenu->Append(wxID_ABOUT, _("&About...\tF1"), _("Show about dialog"));
498 mb->Append(helpMenu, _("&Help"));
499
500 SetMenuBar(mb);
501
502 CreateStatusBar(1);
503}
504
505void AnalyzerFrame::onOpen(wxCommandEvent &WXUNUSED(event)) {
506 wxFileDialog openFileDialog(this, _("Open file"), wxEmptyString,
507 wxEmptyString, _("AV1 files (*.ivf)|*.ivf"),
508 wxFD_OPEN | wxFD_FILE_MUST_EXIST);
509 if (openFileDialog.ShowModal() != wxID_CANCEL) {
510 open(openFileDialog.GetPath());
511 }
512}
513
514void AnalyzerFrame::onClose(wxCommandEvent &WXUNUSED(event)) {}
515
516void AnalyzerFrame::onQuit(wxCommandEvent &WXUNUSED(event)) { Close(true); }
517
518void AnalyzerFrame::onZoomIn(wxCommandEvent &WXUNUSED(event)) {
519 setZoom(panel->getZoom() + 1);
520}
521
522void AnalyzerFrame::onZoomOut(wxCommandEvent &WXUNUSED(event)) {
523 setZoom(panel->getZoom() - 1);
524}
525
526void AnalyzerFrame::onActualSize(wxCommandEvent &WXUNUSED(event)) {
527 setZoom(MIN_ZOOM);
528}
529
Nathan E. Egge625bc262017-02-22 10:42:54 -0500530void AnalyzerFrame::onToggleViewMenuCheckBox(wxCommandEvent &event) { // NOLINT
531 GetMenuBar()->Check(event.GetId(), event.IsChecked());
532 updateViewMenu();
533}
534
535void AnalyzerFrame::onResetAndToggleViewMenuCheckBox(
536 wxCommandEvent &event) { // NOLINT
537 int id = event.GetId();
538 if (id != wxID_SHOW_Y && id != wxID_SHOW_U && id != wxID_SHOW_V) {
539 GetMenuBar()->Check(wxID_SHOW_Y, true);
540 GetMenuBar()->Check(wxID_SHOW_U, true);
541 GetMenuBar()->Check(wxID_SHOW_V, true);
542 }
543 onToggleViewMenuCheckBox(event);
544}
545
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400546void AnalyzerFrame::onNextFrame(wxCommandEvent &WXUNUSED(event)) {
547 panel->nextFrame();
548 panel->Refresh(false);
549}
550
551void AnalyzerFrame::onGotoFrame(wxCommandEvent &WXUNUSED(event)) {}
552
553void AnalyzerFrame::onRestart(wxCommandEvent &WXUNUSED(event)) {}
554
555void AnalyzerFrame::onAbout(wxCommandEvent &WXUNUSED(event)) {
556 wxAboutDialogInfo info;
557 info.SetName(_("AV1 Bitstream Analyzer"));
558 info.SetVersion(_("0.1-beta"));
559 info.SetDescription(
560 _("This program implements a bitstream analyzer for AV1"));
561 info.SetCopyright(
562 wxT("(C) 2017 Alliance for Open Media <negge@mozilla.com>"));
563 wxAboutBox(info);
564}
565
566bool AnalyzerFrame::open(const wxString &path) {
567 panel = new AnalyzerPanel(this, path, bit_accounting);
568 if (panel->open(path)) {
569 SetClientSize(panel->GetSize());
570 return true;
571 } else {
572 delete panel;
573 return false;
574 }
575}
576
577bool AnalyzerFrame::setZoom(int zoom) {
578 if (panel->setZoom(zoom)) {
579 GetMenuBar()->Enable(wxID_ACTUAL_SIZE, zoom != MIN_ZOOM);
580 GetMenuBar()->Enable(wxID_ZOOM_IN, zoom != MAX_ZOOM);
581 GetMenuBar()->Enable(wxID_ZOOM_OUT, zoom != MIN_ZOOM);
582 SetClientSize(panel->GetSize());
583 panel->render();
584 panel->Refresh();
585 return true;
586 }
587 return false;
588}
589
Nathan E. Egge625bc262017-02-22 10:42:54 -0500590void AnalyzerFrame::updateViewMenu() {
591 panel->setShowPlane(GetMenuBar()->IsChecked(wxID_SHOW_Y), OD_LUMA_MASK);
592 panel->setShowPlane(GetMenuBar()->IsChecked(wxID_SHOW_U), OD_CB_MASK);
593 panel->setShowPlane(GetMenuBar()->IsChecked(wxID_SHOW_V), OD_CR_MASK);
594 SetClientSize(panel->GetSize());
595 panel->render();
596 panel->Refresh(false);
597}
598
Nathan E. Eggef4fa01e2016-10-11 14:20:20 -0400599class Analyzer : public wxApp {
600 private:
601 AnalyzerFrame *frame;
602
603 public:
604 void OnInitCmdLine(wxCmdLineParser &parser); // NOLINT
605 bool OnCmdLineParsed(wxCmdLineParser &parser); // NOLINT
606};
607
608static const wxCmdLineEntryDesc CMD_LINE_DESC[] = {
609 { wxCMD_LINE_SWITCH, _("h"), _("help"), _("Display this help and exit."),
610 wxCMD_LINE_VAL_NONE, wxCMD_LINE_OPTION_HELP },
611 { wxCMD_LINE_SWITCH, _("a"), _("bit-accounting"), _("Enable bit accounting"),
612 wxCMD_LINE_VAL_NONE, wxCMD_LINE_PARAM_OPTIONAL },
613 { wxCMD_LINE_PARAM, NULL, NULL, _("input.ivf"), wxCMD_LINE_VAL_STRING,
614 wxCMD_LINE_PARAM_OPTIONAL },
615 { wxCMD_LINE_NONE }
616};
617
618void Analyzer::OnInitCmdLine(wxCmdLineParser &parser) { // NOLINT
619 parser.SetDesc(CMD_LINE_DESC);
620 parser.SetSwitchChars(_("-"));
621}
622
623bool Analyzer::OnCmdLineParsed(wxCmdLineParser &parser) { // NOLINT
624 bool bit_accounting = parser.Found(_("a"));
625 if (bit_accounting && !CONFIG_ACCOUNTING) {
626 fprintf(stderr,
627 "Bit accounting support not found. "
628 "Recompile with:\n./configure --enable-accounting\n");
629 return false;
630 }
631 frame = new AnalyzerFrame(parser.Found(_("a")));
632 frame->Show();
633 if (parser.GetParamCount() > 0) {
634 return frame->open(parser.GetParam(0));
635 }
636 return true;
637}
638
639void usage_exit(void) {
640 fprintf(stderr, "uhh\n");
641 exit(EXIT_FAILURE);
642}
643
644IMPLEMENT_APP(Analyzer)