xref: /aosp_15_r20/external/libvpx/examples/decode_to_md5.c (revision fb1b10ab9aebc7c7068eedab379b749d7e3900be)
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