1*fb1b10abSAndroid Build Coastguard Worker /*
2*fb1b10abSAndroid Build Coastguard Worker * Copyright (c) 2010 The WebM project authors. All Rights Reserved.
3*fb1b10abSAndroid Build Coastguard Worker *
4*fb1b10abSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license
5*fb1b10abSAndroid Build Coastguard Worker * that can be found in the LICENSE file in the root of the source
6*fb1b10abSAndroid Build Coastguard Worker * tree. An additional intellectual property rights grant can be found
7*fb1b10abSAndroid Build Coastguard Worker * in the file PATENTS. All contributing project authors may
8*fb1b10abSAndroid Build Coastguard Worker * be found in the AUTHORS file in the root of the source tree.
9*fb1b10abSAndroid Build Coastguard Worker */
10*fb1b10abSAndroid Build Coastguard Worker
11*fb1b10abSAndroid Build Coastguard Worker // Frame-by-frame MD5 Checksum
12*fb1b10abSAndroid Build Coastguard Worker // ===========================
13*fb1b10abSAndroid Build Coastguard Worker //
14*fb1b10abSAndroid Build Coastguard Worker // This example builds upon the simple decoder loop to show how checksums
15*fb1b10abSAndroid Build Coastguard Worker // of the decoded output can be generated. These are used for validating
16*fb1b10abSAndroid Build Coastguard Worker // decoder implementations against the reference implementation, for example.
17*fb1b10abSAndroid Build Coastguard Worker //
18*fb1b10abSAndroid Build Coastguard Worker // MD5 algorithm
19*fb1b10abSAndroid Build Coastguard Worker // -------------
20*fb1b10abSAndroid Build Coastguard Worker // The Message-Digest 5 (MD5) is a well known hash function. We have provided
21*fb1b10abSAndroid Build Coastguard Worker // an implementation derived from the RSA Data Security, Inc. MD5 Message-Digest
22*fb1b10abSAndroid Build Coastguard Worker // Algorithm for your use. Our implmentation only changes the interface of this
23*fb1b10abSAndroid Build Coastguard Worker // reference code. You must include the `md5_utils.h` header for access to these
24*fb1b10abSAndroid Build Coastguard Worker // functions.
25*fb1b10abSAndroid Build Coastguard Worker //
26*fb1b10abSAndroid Build Coastguard Worker // Processing The Decoded Data
27*fb1b10abSAndroid Build Coastguard Worker // ---------------------------
28*fb1b10abSAndroid Build Coastguard Worker // Each row of the image is passed to the MD5 accumulator. First the Y plane
29*fb1b10abSAndroid Build Coastguard Worker // is processed, then U, then V. It is important to honor the image's `stride`
30*fb1b10abSAndroid Build Coastguard Worker // values.
31*fb1b10abSAndroid Build Coastguard Worker
32*fb1b10abSAndroid Build Coastguard Worker #include <stdio.h>
33*fb1b10abSAndroid Build Coastguard Worker #include <stdlib.h>
34*fb1b10abSAndroid Build Coastguard Worker #include <string.h>
35*fb1b10abSAndroid Build Coastguard Worker
36*fb1b10abSAndroid Build Coastguard Worker #include "vpx/vp8dx.h"
37*fb1b10abSAndroid Build Coastguard Worker #include "vpx/vpx_decoder.h"
38*fb1b10abSAndroid Build Coastguard Worker
39*fb1b10abSAndroid Build Coastguard Worker #include "../md5_utils.h"
40*fb1b10abSAndroid Build Coastguard Worker #include "../tools_common.h"
41*fb1b10abSAndroid Build Coastguard Worker #include "../video_reader.h"
42*fb1b10abSAndroid Build Coastguard Worker #include "./vpx_config.h"
43*fb1b10abSAndroid Build Coastguard Worker
get_image_md5(const vpx_image_t * img,unsigned char digest[16])44*fb1b10abSAndroid Build Coastguard Worker static void get_image_md5(const vpx_image_t *img, unsigned char digest[16]) {
45*fb1b10abSAndroid Build Coastguard Worker int plane, y;
46*fb1b10abSAndroid Build Coastguard Worker MD5Context md5;
47*fb1b10abSAndroid Build Coastguard Worker
48*fb1b10abSAndroid Build Coastguard Worker MD5Init(&md5);
49*fb1b10abSAndroid Build Coastguard Worker
50*fb1b10abSAndroid Build Coastguard Worker for (plane = 0; plane < 3; ++plane) {
51*fb1b10abSAndroid Build Coastguard Worker const unsigned char *buf = img->planes[plane];
52*fb1b10abSAndroid Build Coastguard Worker const int stride = img->stride[plane];
53*fb1b10abSAndroid Build Coastguard Worker const int w = plane ? (img->d_w + 1) >> 1 : img->d_w;
54*fb1b10abSAndroid Build Coastguard Worker const int h = plane ? (img->d_h + 1) >> 1 : img->d_h;
55*fb1b10abSAndroid Build Coastguard Worker
56*fb1b10abSAndroid Build Coastguard Worker for (y = 0; y < h; ++y) {
57*fb1b10abSAndroid Build Coastguard Worker MD5Update(&md5, buf, w);
58*fb1b10abSAndroid Build Coastguard Worker buf += stride;
59*fb1b10abSAndroid Build Coastguard Worker }
60*fb1b10abSAndroid Build Coastguard Worker }
61*fb1b10abSAndroid Build Coastguard Worker
62*fb1b10abSAndroid Build Coastguard Worker MD5Final(digest, &md5);
63*fb1b10abSAndroid Build Coastguard Worker }
64*fb1b10abSAndroid Build Coastguard Worker
print_md5(FILE * stream,unsigned char digest[16])65*fb1b10abSAndroid Build Coastguard Worker static void print_md5(FILE *stream, unsigned char digest[16]) {
66*fb1b10abSAndroid Build Coastguard Worker int i;
67*fb1b10abSAndroid Build Coastguard Worker
68*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) fprintf(stream, "%02x", digest[i]);
69*fb1b10abSAndroid Build Coastguard Worker }
70*fb1b10abSAndroid Build Coastguard Worker
71*fb1b10abSAndroid Build Coastguard Worker static const char *exec_name;
72*fb1b10abSAndroid Build Coastguard Worker
usage_exit(void)73*fb1b10abSAndroid Build Coastguard Worker void usage_exit(void) {
74*fb1b10abSAndroid Build Coastguard Worker fprintf(stderr, "Usage: %s <infile> <outfile>\n", exec_name);
75*fb1b10abSAndroid Build Coastguard Worker exit(EXIT_FAILURE);
76*fb1b10abSAndroid Build Coastguard Worker }
77*fb1b10abSAndroid Build Coastguard Worker
main(int argc,char ** argv)78*fb1b10abSAndroid Build Coastguard Worker int main(int argc, char **argv) {
79*fb1b10abSAndroid Build Coastguard Worker int frame_cnt = 0;
80*fb1b10abSAndroid Build Coastguard Worker FILE *outfile = NULL;
81*fb1b10abSAndroid Build Coastguard Worker vpx_codec_ctx_t codec;
82*fb1b10abSAndroid Build Coastguard Worker VpxVideoReader *reader = NULL;
83*fb1b10abSAndroid Build Coastguard Worker const VpxVideoInfo *info = NULL;
84*fb1b10abSAndroid Build Coastguard Worker const VpxInterface *decoder = NULL;
85*fb1b10abSAndroid Build Coastguard Worker
86*fb1b10abSAndroid Build Coastguard Worker exec_name = argv[0];
87*fb1b10abSAndroid Build Coastguard Worker
88*fb1b10abSAndroid Build Coastguard Worker if (argc != 3) die("Invalid number of arguments.");
89*fb1b10abSAndroid Build Coastguard Worker
90*fb1b10abSAndroid Build Coastguard Worker reader = vpx_video_reader_open(argv[1]);
91*fb1b10abSAndroid Build Coastguard Worker if (!reader) die("Failed to open %s for reading.", argv[1]);
92*fb1b10abSAndroid Build Coastguard Worker
93*fb1b10abSAndroid Build Coastguard Worker if (!(outfile = fopen(argv[2], "wb")))
94*fb1b10abSAndroid Build Coastguard Worker die("Failed to open %s for writing.", argv[2]);
95*fb1b10abSAndroid Build Coastguard Worker
96*fb1b10abSAndroid Build Coastguard Worker info = vpx_video_reader_get_info(reader);
97*fb1b10abSAndroid Build Coastguard Worker
98*fb1b10abSAndroid Build Coastguard Worker decoder = get_vpx_decoder_by_fourcc(info->codec_fourcc);
99*fb1b10abSAndroid Build Coastguard Worker if (!decoder) die("Unknown input codec.");
100*fb1b10abSAndroid Build Coastguard Worker
101*fb1b10abSAndroid Build Coastguard Worker printf("Using %s\n", vpx_codec_iface_name(decoder->codec_interface()));
102*fb1b10abSAndroid Build Coastguard Worker
103*fb1b10abSAndroid Build Coastguard Worker if (vpx_codec_dec_init(&codec, decoder->codec_interface(), NULL, 0))
104*fb1b10abSAndroid Build Coastguard Worker die_codec(&codec, "Failed to initialize decoder");
105*fb1b10abSAndroid Build Coastguard Worker
106*fb1b10abSAndroid Build Coastguard Worker while (vpx_video_reader_read_frame(reader)) {
107*fb1b10abSAndroid Build Coastguard Worker vpx_codec_iter_t iter = NULL;
108*fb1b10abSAndroid Build Coastguard Worker vpx_image_t *img = NULL;
109*fb1b10abSAndroid Build Coastguard Worker size_t frame_size = 0;
110*fb1b10abSAndroid Build Coastguard Worker const unsigned char *frame =
111*fb1b10abSAndroid Build Coastguard Worker vpx_video_reader_get_frame(reader, &frame_size);
112*fb1b10abSAndroid Build Coastguard Worker if (vpx_codec_decode(&codec, frame, (unsigned int)frame_size, NULL, 0))
113*fb1b10abSAndroid Build Coastguard Worker die_codec(&codec, "Failed to decode frame");
114*fb1b10abSAndroid Build Coastguard Worker
115*fb1b10abSAndroid Build Coastguard Worker while ((img = vpx_codec_get_frame(&codec, &iter)) != NULL) {
116*fb1b10abSAndroid Build Coastguard Worker unsigned char digest[16];
117*fb1b10abSAndroid Build Coastguard Worker
118*fb1b10abSAndroid Build Coastguard Worker get_image_md5(img, digest);
119*fb1b10abSAndroid Build Coastguard Worker print_md5(outfile, digest);
120*fb1b10abSAndroid Build Coastguard Worker fprintf(outfile, " img-%dx%d-%04d.i420\n", img->d_w, img->d_h,
121*fb1b10abSAndroid Build Coastguard Worker ++frame_cnt);
122*fb1b10abSAndroid Build Coastguard Worker }
123*fb1b10abSAndroid Build Coastguard Worker }
124*fb1b10abSAndroid Build Coastguard Worker
125*fb1b10abSAndroid Build Coastguard Worker printf("Processed %d frames.\n", frame_cnt);
126*fb1b10abSAndroid Build Coastguard Worker if (vpx_codec_destroy(&codec)) die_codec(&codec, "Failed to destroy codec.");
127*fb1b10abSAndroid Build Coastguard Worker
128*fb1b10abSAndroid Build Coastguard Worker vpx_video_reader_close(reader);
129*fb1b10abSAndroid Build Coastguard Worker
130*fb1b10abSAndroid Build Coastguard Worker fclose(outfile);
131*fb1b10abSAndroid Build Coastguard Worker return EXIT_SUCCESS;
132*fb1b10abSAndroid Build Coastguard Worker }
133