| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright (c) 2010 The WebM project authors. All Rights Reserved. | 2 * Copyright (c) 2010 The WebM project authors. All Rights Reserved. |
| 3 * | 3 * |
| 4 * Use of this source code is governed by a BSD-style license | 4 * Use of this source code is governed by a BSD-style license |
| 5 * that can be found in the LICENSE file in the root of the source | 5 * that can be found in the LICENSE file in the root of the source |
| 6 * tree. An additional intellectual property rights grant can be found | 6 * tree. An additional intellectual property rights grant can be found |
| 7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
| 8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
| 9 */ | 9 */ |
| 10 | 10 |
| (...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 59 #include "./video_writer.h" | 59 #include "./video_writer.h" |
| 60 | 60 |
| 61 static const char *exec_name; | 61 static const char *exec_name; |
| 62 | 62 |
| 63 void usage_exit() { | 63 void usage_exit() { |
| 64 fprintf(stderr, "Usage: %s <codec> <width> <height> <infile> <outfile>\n", | 64 fprintf(stderr, "Usage: %s <codec> <width> <height> <infile> <outfile>\n", |
| 65 exec_name); | 65 exec_name); |
| 66 exit(EXIT_FAILURE); | 66 exit(EXIT_FAILURE); |
| 67 } | 67 } |
| 68 | 68 |
| 69 static void get_frame_stats(vpx_codec_ctx_t *ctx, | 69 static int get_frame_stats(vpx_codec_ctx_t *ctx, |
| 70 const vpx_image_t *img, | 70 const vpx_image_t *img, |
| 71 vpx_codec_pts_t pts, | 71 vpx_codec_pts_t pts, |
| 72 unsigned int duration, | 72 unsigned int duration, |
| 73 vpx_enc_frame_flags_t flags, | 73 vpx_enc_frame_flags_t flags, |
| 74 unsigned int deadline, | 74 unsigned int deadline, |
| 75 vpx_fixed_buf_t *stats) { | 75 vpx_fixed_buf_t *stats) { |
| 76 int got_pkts = 0; |
| 76 vpx_codec_iter_t iter = NULL; | 77 vpx_codec_iter_t iter = NULL; |
| 77 const vpx_codec_cx_pkt_t *pkt = NULL; | 78 const vpx_codec_cx_pkt_t *pkt = NULL; |
| 78 const vpx_codec_err_t res = vpx_codec_encode(ctx, img, pts, duration, flags, | 79 const vpx_codec_err_t res = vpx_codec_encode(ctx, img, pts, duration, flags, |
| 79 deadline); | 80 deadline); |
| 80 if (res != VPX_CODEC_OK) | 81 if (res != VPX_CODEC_OK) |
| 81 die_codec(ctx, "Failed to get frame stats."); | 82 die_codec(ctx, "Failed to get frame stats."); |
| 82 | 83 |
| 83 while ((pkt = vpx_codec_get_cx_data(ctx, &iter)) != NULL) { | 84 while ((pkt = vpx_codec_get_cx_data(ctx, &iter)) != NULL) { |
| 85 got_pkts = 1; |
| 86 |
| 84 if (pkt->kind == VPX_CODEC_STATS_PKT) { | 87 if (pkt->kind == VPX_CODEC_STATS_PKT) { |
| 85 const uint8_t *const pkt_buf = pkt->data.twopass_stats.buf; | 88 const uint8_t *const pkt_buf = pkt->data.twopass_stats.buf; |
| 86 const size_t pkt_size = pkt->data.twopass_stats.sz; | 89 const size_t pkt_size = pkt->data.twopass_stats.sz; |
| 87 stats->buf = realloc(stats->buf, stats->sz + pkt_size); | 90 stats->buf = realloc(stats->buf, stats->sz + pkt_size); |
| 88 memcpy((uint8_t *)stats->buf + stats->sz, pkt_buf, pkt_size); | 91 memcpy((uint8_t *)stats->buf + stats->sz, pkt_buf, pkt_size); |
| 89 stats->sz += pkt_size; | 92 stats->sz += pkt_size; |
| 90 } | 93 } |
| 91 } | 94 } |
| 95 |
| 96 return got_pkts; |
| 92 } | 97 } |
| 93 | 98 |
| 94 static void encode_frame(vpx_codec_ctx_t *ctx, | 99 static int encode_frame(vpx_codec_ctx_t *ctx, |
| 95 const vpx_image_t *img, | 100 const vpx_image_t *img, |
| 96 vpx_codec_pts_t pts, | 101 vpx_codec_pts_t pts, |
| 97 unsigned int duration, | 102 unsigned int duration, |
| 98 vpx_enc_frame_flags_t flags, | 103 vpx_enc_frame_flags_t flags, |
| 99 unsigned int deadline, | 104 unsigned int deadline, |
| 100 VpxVideoWriter *writer) { | 105 VpxVideoWriter *writer) { |
| 106 int got_pkts = 0; |
| 101 vpx_codec_iter_t iter = NULL; | 107 vpx_codec_iter_t iter = NULL; |
| 102 const vpx_codec_cx_pkt_t *pkt = NULL; | 108 const vpx_codec_cx_pkt_t *pkt = NULL; |
| 103 const vpx_codec_err_t res = vpx_codec_encode(ctx, img, pts, duration, flags, | 109 const vpx_codec_err_t res = vpx_codec_encode(ctx, img, pts, duration, flags, |
| 104 deadline); | 110 deadline); |
| 105 if (res != VPX_CODEC_OK) | 111 if (res != VPX_CODEC_OK) |
| 106 die_codec(ctx, "Failed to encode frame."); | 112 die_codec(ctx, "Failed to encode frame."); |
| 107 | 113 |
| 108 while ((pkt = vpx_codec_get_cx_data(ctx, &iter)) != NULL) { | 114 while ((pkt = vpx_codec_get_cx_data(ctx, &iter)) != NULL) { |
| 115 got_pkts = 1; |
| 109 if (pkt->kind == VPX_CODEC_CX_FRAME_PKT) { | 116 if (pkt->kind == VPX_CODEC_CX_FRAME_PKT) { |
| 110 const int keyframe = (pkt->data.frame.flags & VPX_FRAME_IS_KEY) != 0; | 117 const int keyframe = (pkt->data.frame.flags & VPX_FRAME_IS_KEY) != 0; |
| 111 | 118 |
| 112 if (!vpx_video_writer_write_frame(writer, pkt->data.frame.buf, | 119 if (!vpx_video_writer_write_frame(writer, pkt->data.frame.buf, |
| 113 pkt->data.frame.sz, | 120 pkt->data.frame.sz, |
| 114 pkt->data.frame.pts)) | 121 pkt->data.frame.pts)) |
| 115 die_codec(ctx, "Failed to write compressed frame."); | 122 die_codec(ctx, "Failed to write compressed frame."); |
| 116 printf(keyframe ? "K" : "."); | 123 printf(keyframe ? "K" : "."); |
| 117 fflush(stdout); | 124 fflush(stdout); |
| 118 } | 125 } |
| 119 } | 126 } |
| 127 |
| 128 return got_pkts; |
| 129 } |
| 130 |
| 131 static vpx_fixed_buf_t pass0(vpx_image_t *raw, |
| 132 FILE *infile, |
| 133 const VpxInterface *encoder, |
| 134 const vpx_codec_enc_cfg_t *cfg) { |
| 135 vpx_codec_ctx_t codec; |
| 136 int frame_count = 0; |
| 137 vpx_fixed_buf_t stats = {NULL, 0}; |
| 138 |
| 139 if (vpx_codec_enc_init(&codec, encoder->codec_interface(), cfg, 0)) |
| 140 die_codec(&codec, "Failed to initialize encoder"); |
| 141 |
| 142 // Calculate frame statistics. |
| 143 while (vpx_img_read(raw, infile)) { |
| 144 ++frame_count; |
| 145 get_frame_stats(&codec, raw, frame_count, 1, 0, VPX_DL_BEST_QUALITY, |
| 146 &stats); |
| 147 } |
| 148 |
| 149 // Flush encoder. |
| 150 while (get_frame_stats(&codec, NULL, frame_count, 1, 0, |
| 151 VPX_DL_BEST_QUALITY, &stats)) {} |
| 152 |
| 153 printf("Pass 0 complete. Processed %d frames.\n", frame_count); |
| 154 if (vpx_codec_destroy(&codec)) |
| 155 die_codec(&codec, "Failed to destroy codec."); |
| 156 |
| 157 return stats; |
| 158 } |
| 159 |
| 160 static void pass1(vpx_image_t *raw, |
| 161 FILE *infile, |
| 162 const char *outfile_name, |
| 163 const VpxInterface *encoder, |
| 164 const vpx_codec_enc_cfg_t *cfg) { |
| 165 VpxVideoInfo info = { |
| 166 encoder->fourcc, |
| 167 cfg->g_w, |
| 168 cfg->g_h, |
| 169 {cfg->g_timebase.num, cfg->g_timebase.den} |
| 170 }; |
| 171 VpxVideoWriter *writer = NULL; |
| 172 vpx_codec_ctx_t codec; |
| 173 int frame_count = 0; |
| 174 |
| 175 writer = vpx_video_writer_open(outfile_name, kContainerIVF, &info); |
| 176 if (!writer) |
| 177 die("Failed to open %s for writing", outfile_name); |
| 178 |
| 179 if (vpx_codec_enc_init(&codec, encoder->codec_interface(), cfg, 0)) |
| 180 die_codec(&codec, "Failed to initialize encoder"); |
| 181 |
| 182 // Encode frames. |
| 183 while (vpx_img_read(raw, infile)) { |
| 184 ++frame_count; |
| 185 encode_frame(&codec, raw, frame_count, 1, 0, VPX_DL_BEST_QUALITY, writer); |
| 186 } |
| 187 |
| 188 // Flush encoder. |
| 189 while (encode_frame(&codec, NULL, -1, 1, 0, VPX_DL_BEST_QUALITY, writer)) {} |
| 190 |
| 191 printf("\n"); |
| 192 |
| 193 if (vpx_codec_destroy(&codec)) |
| 194 die_codec(&codec, "Failed to destroy codec."); |
| 195 |
| 196 vpx_video_writer_close(writer); |
| 197 |
| 198 printf("Pass 1 complete. Processed %d frames.\n", frame_count); |
| 120 } | 199 } |
| 121 | 200 |
| 122 int main(int argc, char **argv) { | 201 int main(int argc, char **argv) { |
| 123 FILE *infile = NULL; | 202 FILE *infile = NULL; |
| 124 VpxVideoWriter *writer = NULL; | 203 int w, h; |
| 125 vpx_codec_ctx_t codec; | 204 vpx_codec_ctx_t codec; |
| 126 vpx_codec_enc_cfg_t cfg; | 205 vpx_codec_enc_cfg_t cfg; |
| 127 vpx_image_t raw; | 206 vpx_image_t raw; |
| 128 vpx_codec_err_t res; | 207 vpx_codec_err_t res; |
| 129 vpx_fixed_buf_t stats = {0}; | 208 vpx_fixed_buf_t stats; |
| 130 VpxVideoInfo info = {0}; | 209 |
| 131 const VpxInterface *encoder = NULL; | 210 const VpxInterface *encoder = NULL; |
| 132 int pass; | |
| 133 const int fps = 30; // TODO(dkovalev) add command line argument | 211 const int fps = 30; // TODO(dkovalev) add command line argument |
| 134 const int bitrate = 200; // kbit/s TODO(dkovalev) add command line argument | 212 const int bitrate = 200; // kbit/s TODO(dkovalev) add command line argument |
| 135 const char *const codec_arg = argv[1]; | 213 const char *const codec_arg = argv[1]; |
| 136 const char *const width_arg = argv[2]; | 214 const char *const width_arg = argv[2]; |
| 137 const char *const height_arg = argv[3]; | 215 const char *const height_arg = argv[3]; |
| 138 const char *const infile_arg = argv[4]; | 216 const char *const infile_arg = argv[4]; |
| 139 const char *const outfile_arg = argv[5]; | 217 const char *const outfile_arg = argv[5]; |
| 140 exec_name = argv[0]; | 218 exec_name = argv[0]; |
| 141 | 219 |
| 142 if (argc != 6) | 220 if (argc != 6) |
| 143 die("Invalid number of arguments."); | 221 die("Invalid number of arguments."); |
| 144 | 222 |
| 145 encoder = get_vpx_encoder_by_name(codec_arg); | 223 encoder = get_vpx_encoder_by_name(codec_arg); |
| 146 if (!encoder) | 224 if (!encoder) |
| 147 die("Unsupported codec."); | 225 die("Unsupported codec."); |
| 148 | 226 |
| 149 info.codec_fourcc = encoder->fourcc; | 227 w = strtol(width_arg, NULL, 0); |
| 150 info.time_base.numerator = 1; | 228 h = strtol(height_arg, NULL, 0); |
| 151 info.time_base.denominator = fps; | |
| 152 info.frame_width = strtol(width_arg, NULL, 0); | |
| 153 info.frame_height = strtol(height_arg, NULL, 0); | |
| 154 | 229 |
| 155 if (info.frame_width <= 0 || | 230 if (w <= 0 || h <= 0 || (w % 2) != 0 || (h % 2) != 0) |
| 156 info.frame_height <= 0 || | 231 die("Invalid frame size: %dx%d", w, h); |
| 157 (info.frame_width % 2) != 0 || | |
| 158 (info.frame_height % 2) != 0) { | |
| 159 die("Invalid frame size: %dx%d", info.frame_width, info.frame_height); | |
| 160 } | |
| 161 | 232 |
| 162 if (!vpx_img_alloc(&raw, VPX_IMG_FMT_I420, info.frame_width, | 233 if (!vpx_img_alloc(&raw, VPX_IMG_FMT_I420, w, h, 1)) |
| 163 info.frame_height, 1)) { | 234 die("Failed to allocate image", w, h); |
| 164 die("Failed to allocate image", info.frame_width, info.frame_height); | |
| 165 } | |
| 166 | |
| 167 writer = vpx_video_writer_open(outfile_arg, kContainerIVF, &info); | |
| 168 if (!writer) | |
| 169 die("Failed to open %s for writing", outfile_arg); | |
| 170 | 235 |
| 171 printf("Using %s\n", vpx_codec_iface_name(encoder->codec_interface())); | 236 printf("Using %s\n", vpx_codec_iface_name(encoder->codec_interface())); |
| 172 | 237 |
| 238 // Configuration |
| 173 res = vpx_codec_enc_config_default(encoder->codec_interface(), &cfg, 0); | 239 res = vpx_codec_enc_config_default(encoder->codec_interface(), &cfg, 0); |
| 174 if (res) | 240 if (res) |
| 175 die_codec(&codec, "Failed to get default codec config."); | 241 die_codec(&codec, "Failed to get default codec config."); |
| 176 | 242 |
| 177 cfg.g_w = info.frame_width; | 243 cfg.g_w = w; |
| 178 cfg.g_h = info.frame_height; | 244 cfg.g_h = h; |
| 179 cfg.g_timebase.num = info.time_base.numerator; | 245 cfg.g_timebase.num = 1; |
| 180 cfg.g_timebase.den = info.time_base.denominator; | 246 cfg.g_timebase.den = fps; |
| 181 cfg.rc_target_bitrate = bitrate; | 247 cfg.rc_target_bitrate = bitrate; |
| 182 | 248 |
| 183 for (pass = 0; pass < 2; ++pass) { | 249 if (!(infile = fopen(infile_arg, "rb"))) |
| 184 int frame_count = 0; | 250 die("Failed to open %s for reading", infile_arg); |
| 185 | 251 |
| 186 if (pass == 0) { | 252 // Pass 0 |
| 187 cfg.g_pass = VPX_RC_FIRST_PASS; | 253 cfg.g_pass = VPX_RC_FIRST_PASS; |
| 188 } else { | 254 stats = pass0(&raw, infile, encoder, &cfg); |
| 189 cfg.g_pass = VPX_RC_LAST_PASS; | |
| 190 cfg.rc_twopass_stats_in = stats; | |
| 191 } | |
| 192 | 255 |
| 193 if (!(infile = fopen(infile_arg, "rb"))) | 256 // Pass 1 |
| 194 die("Failed to open %s for reading", infile_arg); | 257 rewind(infile); |
| 195 | 258 cfg.g_pass = VPX_RC_LAST_PASS; |
| 196 if (vpx_codec_enc_init(&codec, encoder->codec_interface(), &cfg, 0)) | 259 cfg.rc_twopass_stats_in = stats; |
| 197 die_codec(&codec, "Failed to initialize encoder"); | 260 pass1(&raw, infile, outfile_arg, encoder, &cfg); |
| 198 | 261 free(stats.buf); |
| 199 while (vpx_img_read(&raw, infile)) { | |
| 200 ++frame_count; | |
| 201 | |
| 202 if (pass == 0) { | |
| 203 get_frame_stats(&codec, &raw, frame_count, 1, 0, VPX_DL_BEST_QUALITY, | |
| 204 &stats); | |
| 205 } else { | |
| 206 encode_frame(&codec, &raw, frame_count, 1, 0, VPX_DL_BEST_QUALITY, | |
| 207 writer); | |
| 208 } | |
| 209 } | |
| 210 | |
| 211 if (pass == 0) { | |
| 212 get_frame_stats(&codec, NULL, frame_count, 1, 0, VPX_DL_BEST_QUALITY, | |
| 213 &stats); | |
| 214 } else { | |
| 215 printf("\n"); | |
| 216 } | |
| 217 | |
| 218 fclose(infile); | |
| 219 printf("Pass %d complete. Processed %d frames.\n", pass + 1, frame_count); | |
| 220 if (vpx_codec_destroy(&codec)) | |
| 221 die_codec(&codec, "Failed to destroy codec."); | |
| 222 } | |
| 223 | 262 |
| 224 vpx_img_free(&raw); | 263 vpx_img_free(&raw); |
| 225 free(stats.buf); | 264 fclose(infile); |
| 226 | |
| 227 vpx_video_writer_close(writer); | |
| 228 | 265 |
| 229 return EXIT_SUCCESS; | 266 return EXIT_SUCCESS; |
| 230 } | 267 } |
| OLD | NEW |