Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 1 | /* |
Yaowu Xu | 9c01aa1 | 2016-09-01 14:32:49 -0700 | [diff] [blame] | 2 | * Copyright (c) 2016, Alliance for Open Media. All rights reserved |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 3 | * |
Yaowu Xu | 9c01aa1 | 2016-09-01 14:32:49 -0700 | [diff] [blame] | 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. |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 10 | */ |
| 11 | |
Yaowu Xu | 9c01aa1 | 2016-09-01 14:32:49 -0700 | [diff] [blame] | 12 | // AOM Set Active and ROI Maps |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 13 | // =========================== |
| 14 | // |
Yaowu Xu | 9c01aa1 | 2016-09-01 14:32:49 -0700 | [diff] [blame] | 15 | // This is an example demonstrating how to control the AOM encoder's |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 16 | // ROI and Active maps. |
| 17 | // |
| 18 | // ROI (Reigon of Interest) maps are a way for the application to assign |
| 19 | // each macroblock in the image to a region, and then set quantizer and |
| 20 | // filtering parameters on that image. |
| 21 | // |
| 22 | // Active maps are a way for the application to specify on a |
| 23 | // macroblock-by-macroblock basis whether there is any activity in that |
| 24 | // macroblock. |
| 25 | // |
| 26 | // |
| 27 | // Configuration |
| 28 | // ------------- |
| 29 | // An ROI map is set on frame 22. If the width of the image in macroblocks |
| 30 | // is evenly divisble by 4, then the output will appear to have distinct |
| 31 | // columns, where the quantizer, loopfilter, and static threshold differ |
| 32 | // from column to column. |
| 33 | // |
| 34 | // An active map is set on frame 33. If the width of the image in macroblocks |
| 35 | // is evenly divisble by 4, then the output will appear to have distinct |
| 36 | // columns, where one column will have motion and the next will not. |
| 37 | // |
| 38 | // The active map is cleared on frame 44. |
| 39 | // |
| 40 | // Observing The Effects |
| 41 | // --------------------- |
| 42 | // Use the `simple_decoder` example to decode this sample, and observe |
| 43 | // the change in the image at frames 22, 33, and 44. |
| 44 | |
Jim Bankoski | 812506b | 2014-08-19 14:56:09 -0700 | [diff] [blame] | 45 | #include <assert.h> |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 46 | #include <stdio.h> |
| 47 | #include <stdlib.h> |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 48 | #include <string.h> |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 49 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 50 | #include "aom/aomcx.h" |
| 51 | #include "aom/aom_encoder.h" |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 52 | |
Tom Finegan | 9e96bdc | 2015-02-04 16:11:57 -0800 | [diff] [blame] | 53 | #include "../tools_common.h" |
| 54 | #include "../video_writer.h" |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 55 | |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 56 | static const char *exec_name; |
| 57 | |
James Zern | 59e7a47 | 2015-05-09 10:33:26 -0700 | [diff] [blame] | 58 | void usage_exit(void) { |
Alex Converse | 61ecd7f | 2014-03-10 16:13:49 -0700 | [diff] [blame] | 59 | fprintf(stderr, "Usage: %s <codec> <width> <height> <infile> <outfile>\n", |
| 60 | exec_name); |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 61 | exit(EXIT_FAILURE); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 62 | } |
| 63 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 64 | static void set_active_map(const aom_codec_enc_cfg_t *cfg, |
| 65 | aom_codec_ctx_t *codec) { |
Tom Finegan | 560119c | 2014-02-19 16:13:15 -0800 | [diff] [blame] | 66 | unsigned int i; |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 67 | aom_active_map_t map = { 0, 0, 0 }; |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 68 | |
Alex Converse | b528d49 | 2014-03-10 16:08:58 -0700 | [diff] [blame] | 69 | map.rows = (cfg->g_h + 15) / 16; |
| 70 | map.cols = (cfg->g_w + 15) / 16; |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 71 | |
| 72 | map.active_map = (uint8_t *)malloc(map.rows * map.cols); |
clang-format | 397d964 | 2016-08-08 18:51:16 -0700 | [diff] [blame] | 73 | for (i = 0; i < map.rows * map.cols; ++i) map.active_map[i] = i % 2; |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 74 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 75 | if (aom_codec_control(codec, AOME_SET_ACTIVEMAP, &map)) |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 76 | die_codec(codec, "Failed to set active map"); |
| 77 | |
| 78 | free(map.active_map); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 79 | } |
| 80 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 81 | static void unset_active_map(const aom_codec_enc_cfg_t *cfg, |
| 82 | aom_codec_ctx_t *codec) { |
| 83 | aom_active_map_t map = { 0, 0, 0 }; |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 84 | |
Alex Converse | b528d49 | 2014-03-10 16:08:58 -0700 | [diff] [blame] | 85 | map.rows = (cfg->g_h + 15) / 16; |
| 86 | map.cols = (cfg->g_w + 15) / 16; |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 87 | map.active_map = NULL; |
| 88 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 89 | if (aom_codec_control(codec, AOME_SET_ACTIVEMAP, &map)) |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 90 | die_codec(codec, "Failed to set active map"); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 91 | } |
| 92 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 93 | static int encode_frame(aom_codec_ctx_t *codec, aom_image_t *img, |
| 94 | int frame_index, AvxVideoWriter *writer) { |
Dmitry Kovalev | a8e674d | 2014-08-15 17:15:17 -0700 | [diff] [blame] | 95 | int got_pkts = 0; |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 96 | aom_codec_iter_t iter = NULL; |
| 97 | const aom_codec_cx_pkt_t *pkt = NULL; |
| 98 | const aom_codec_err_t res = |
| 99 | aom_codec_encode(codec, img, frame_index, 1, 0, AOM_DL_GOOD_QUALITY); |
| 100 | if (res != AOM_CODEC_OK) die_codec(codec, "Failed to encode frame"); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 101 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 102 | while ((pkt = aom_codec_get_cx_data(codec, &iter)) != NULL) { |
Dmitry Kovalev | a8e674d | 2014-08-15 17:15:17 -0700 | [diff] [blame] | 103 | got_pkts = 1; |
| 104 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 105 | if (pkt->kind == AOM_CODEC_CX_FRAME_PKT) { |
| 106 | const int keyframe = (pkt->data.frame.flags & AOM_FRAME_IS_KEY) != 0; |
| 107 | if (!aom_video_writer_write_frame(writer, pkt->data.frame.buf, |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 108 | pkt->data.frame.sz, |
| 109 | pkt->data.frame.pts)) { |
| 110 | die_codec(codec, "Failed to write compressed frame"); |
| 111 | } |
| 112 | |
| 113 | printf(keyframe ? "K" : "."); |
| 114 | fflush(stdout); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 115 | } |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 116 | } |
Dmitry Kovalev | a8e674d | 2014-08-15 17:15:17 -0700 | [diff] [blame] | 117 | |
| 118 | return got_pkts; |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 119 | } |
| 120 | |
| 121 | int main(int argc, char **argv) { |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 122 | FILE *infile = NULL; |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 123 | aom_codec_ctx_t codec; |
| 124 | aom_codec_enc_cfg_t cfg; |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 125 | int frame_count = 0; |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 126 | aom_image_t raw; |
| 127 | aom_codec_err_t res; |
| 128 | AvxVideoInfo info; |
| 129 | AvxVideoWriter *writer = NULL; |
| 130 | const AvxInterface *encoder = NULL; |
clang-format | 397d964 | 2016-08-08 18:51:16 -0700 | [diff] [blame] | 131 | const int fps = 2; // TODO(dkovalev) add command line argument |
Alex Converse | b528d49 | 2014-03-10 16:08:58 -0700 | [diff] [blame] | 132 | const double bits_per_pixel_per_frame = 0.067; |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 133 | |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 134 | exec_name = argv[0]; |
clang-format | 397d964 | 2016-08-08 18:51:16 -0700 | [diff] [blame] | 135 | if (argc != 6) die("Invalid number of arguments"); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 136 | |
Yaowu Xu | 7083560 | 2014-05-13 09:32:18 -0700 | [diff] [blame] | 137 | memset(&info, 0, sizeof(info)); |
| 138 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 139 | encoder = get_aom_encoder_by_name(argv[1]); |
Jim Bankoski | 812506b | 2014-08-19 14:56:09 -0700 | [diff] [blame] | 140 | if (encoder == NULL) { |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 141 | die("Unsupported codec."); |
Jim Bankoski | 812506b | 2014-08-19 14:56:09 -0700 | [diff] [blame] | 142 | } |
| 143 | assert(encoder != NULL); |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 144 | info.codec_fourcc = encoder->fourcc; |
Alex Converse | 61ecd7f | 2014-03-10 16:13:49 -0700 | [diff] [blame] | 145 | info.frame_width = strtol(argv[2], NULL, 0); |
| 146 | info.frame_height = strtol(argv[3], NULL, 0); |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 147 | info.time_base.numerator = 1; |
| 148 | info.time_base.denominator = fps; |
| 149 | |
clang-format | 397d964 | 2016-08-08 18:51:16 -0700 | [diff] [blame] | 150 | if (info.frame_width <= 0 || info.frame_height <= 0 || |
| 151 | (info.frame_width % 2) != 0 || (info.frame_height % 2) != 0) { |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 152 | die("Invalid frame size: %dx%d", info.frame_width, info.frame_height); |
| 153 | } |
| 154 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 155 | if (!aom_img_alloc(&raw, AOM_IMG_FMT_I420, info.frame_width, |
clang-format | 397d964 | 2016-08-08 18:51:16 -0700 | [diff] [blame] | 156 | info.frame_height, 1)) { |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 157 | die("Failed to allocate image."); |
| 158 | } |
| 159 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 160 | printf("Using %s\n", aom_codec_iface_name(encoder->codec_interface())); |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 161 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 162 | res = aom_codec_enc_config_default(encoder->codec_interface(), &cfg, 0); |
clang-format | 397d964 | 2016-08-08 18:51:16 -0700 | [diff] [blame] | 163 | if (res) die_codec(&codec, "Failed to get default codec config."); |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 164 | |
| 165 | cfg.g_w = info.frame_width; |
| 166 | cfg.g_h = info.frame_height; |
| 167 | cfg.g_timebase.num = info.time_base.numerator; |
| 168 | cfg.g_timebase.den = info.time_base.denominator; |
clang-format | 397d964 | 2016-08-08 18:51:16 -0700 | [diff] [blame] | 169 | cfg.rc_target_bitrate = |
| 170 | (unsigned int)(bits_per_pixel_per_frame * cfg.g_w * cfg.g_h * fps / 1000); |
Alex Converse | 61ecd7f | 2014-03-10 16:13:49 -0700 | [diff] [blame] | 171 | cfg.g_lag_in_frames = 0; |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 172 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 173 | writer = aom_video_writer_open(argv[5], kContainerIVF, &info); |
clang-format | 397d964 | 2016-08-08 18:51:16 -0700 | [diff] [blame] | 174 | if (!writer) die("Failed to open %s for writing.", argv[5]); |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 175 | |
Alex Converse | 61ecd7f | 2014-03-10 16:13:49 -0700 | [diff] [blame] | 176 | if (!(infile = fopen(argv[4], "rb"))) |
| 177 | die("Failed to open %s for reading.", argv[4]); |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 178 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 179 | if (aom_codec_enc_init(&codec, encoder->codec_interface(), &cfg, 0)) |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 180 | die_codec(&codec, "Failed to initialize encoder"); |
| 181 | |
Dmitry Kovalev | a8e674d | 2014-08-15 17:15:17 -0700 | [diff] [blame] | 182 | // Encode frames. |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 183 | while (aom_img_read(&raw, infile)) { |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 184 | ++frame_count; |
| 185 | |
Jim Bankoski | a7a3909 | 2016-05-20 06:25:40 -0700 | [diff] [blame] | 186 | if (frame_count == 33) { |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 187 | set_active_map(&cfg, &codec); |
| 188 | } else if (frame_count == 44) { |
| 189 | unset_active_map(&cfg, &codec); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 190 | } |
| 191 | |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 192 | encode_frame(&codec, &raw, frame_count, writer); |
| 193 | } |
Dmitry Kovalev | a8e674d | 2014-08-15 17:15:17 -0700 | [diff] [blame] | 194 | |
| 195 | // Flush encoder. |
clang-format | 397d964 | 2016-08-08 18:51:16 -0700 | [diff] [blame] | 196 | while (encode_frame(&codec, NULL, -1, writer)) { |
| 197 | } |
Dmitry Kovalev | a8e674d | 2014-08-15 17:15:17 -0700 | [diff] [blame] | 198 | |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 199 | printf("\n"); |
| 200 | fclose(infile); |
| 201 | printf("Processed %d frames.\n", frame_count); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 202 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 203 | aom_img_free(&raw); |
| 204 | if (aom_codec_destroy(&codec)) die_codec(&codec, "Failed to destroy codec."); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 205 | |
Yaowu Xu | f883b42 | 2016-08-30 14:01:10 -0700 | [diff] [blame] | 206 | aom_video_writer_close(writer); |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 207 | |
Dmitry Kovalev | 8837b02 | 2014-02-13 14:18:05 -0800 | [diff] [blame] | 208 | return EXIT_SUCCESS; |
Dmitry Kovalev | 50fa585 | 2014-01-07 15:15:25 -0800 | [diff] [blame] | 209 | } |