blob: ba67d2ea6e76783e2f710294f6ca1319951a9541 [file] [log] [blame]
Joe Dragob2f46c12019-04-24 20:14:36 -07001// Copyright 2019 Joe Drago. All rights reserved.
2// SPDX-License-Identifier: BSD-2-Clause
3
4#include "avif/avif.h"
5
Joe Dragoddd23452020-04-15 13:23:08 -07006#include "avifjpeg.h"
Joe Dragoc647acc2020-03-10 18:06:12 -07007#include "avifpng.h"
Joe Dragob2f46c12019-04-24 20:14:36 -07008#include "avifutil.h"
9#include "y4m.h"
10
Joe Dragoa0bc0532020-06-26 13:08:59 -070011#include <inttypes.h>
Joe Dragob2f46c12019-04-24 20:14:36 -070012#include <stdio.h>
Joe Dragoc647acc2020-03-10 18:06:12 -070013#include <stdlib.h>
Joe Drago7f989852019-07-24 10:35:17 -070014#include <string.h>
Joe Dragob2f46c12019-04-24 20:14:36 -070015
Joe Dragoddd23452020-04-15 13:23:08 -070016#define DEFAULT_JPEG_QUALITY 90
17
Joe Dragoc8fc62f2020-02-28 09:39:40 -080018#define NEXTARG() \
19 if (((argIndex + 1) == argc) || (argv[argIndex + 1][0] == '-')) { \
20 fprintf(stderr, "%s requires an argument.", arg); \
21 return 1; \
22 } \
23 arg = argv[++argIndex]
24
wantehchang37cf0c62020-03-08 17:25:18 -070025static void syntax(void)
Joe Dragob2f46c12019-04-24 20:14:36 -070026{
Wan-Teh Changa0133252020-04-23 16:02:49 -070027 printf("Syntax: avifdec [options] input.avif output.[jpg|jpeg|png|y4m]\n");
Joe Dragoa0bc0532020-06-26 13:08:59 -070028 printf(" avifdec --info input.avif\n");
Joe Dragob2f46c12019-04-24 20:14:36 -070029 printf("Options:\n");
Joe Dragoc8fc62f2020-02-28 09:39:40 -080030 printf(" -h,--help : Show syntax help\n");
Shun Sakai69814262020-11-06 18:38:10 +090031 printf(" -V,--version : Show the version number\n");
Joe Drago5a987902021-06-07 15:04:25 -070032 printf(" -j,--jobs J : Number of jobs (worker threads, default: 1. Use \"all\" to use all available cores)\n");
Joe Dragoc8fc62f2020-02-28 09:39:40 -080033 printf(" -c,--codec C : AV1 codec to use (choose from versions list below)\n");
Joe Dragoddd23452020-04-15 13:23:08 -070034 printf(" -d,--depth D : Output depth [8,16]. (PNG only; For y4m, depth is retained, and JPEG is always 8bpc)\n");
Wan-Teh Changa0133252020-04-23 16:02:49 -070035 printf(" -q,--quality Q : Output quality [0-100]. (JPEG only, default: %d)\n", DEFAULT_JPEG_QUALITY);
Joe Dragoc7301f32021-07-14 14:35:36 -070036 printf(" --png-compress L : Set PNG compression level (PNG only; 0-9, 0=none, 9=max). Defaults to libpng's builtin default.\n");
Joe Dragob7be24a2020-11-06 17:28:42 -080037 printf(" -u,--upsampling U : Chroma upsampling (for 420/422). automatic (default), fastest, best, nearest, or bilinear\n");
Wan-Teh Changd37ef742021-04-07 12:35:18 -070038 printf(" -r,--raw-color : Output raw RGB values instead of multiplying by alpha when saving to opaque formats\n");
39 printf(" (JPEG only; not applicable to y4m)\n");
Shun Sakaie3c279d2022-04-12 00:50:09 +090040 printf(" --index I : When decoding an image sequence or progressive image, specify which frame index to decode (Default: 0)\n");
Joe Drago8df04be2021-06-14 14:22:18 -070041 printf(" --progressive : Enable progressive AVIF processing. If a progressive image is encountered and --progressive is passed,\n");
42 printf(" avifdec will use --index to choose which layer to decode (in progressive order).\n");
Joe Dragof5b98a62021-05-10 12:25:29 -070043 printf(" --no-strict : Disable strict decoding, which disables strict validation checks and errors\n");
Joe Dragoa0bc0532020-06-26 13:08:59 -070044 printf(" -i,--info : Decode all frames and display all image information instead of saving to disk\n");
Joe Drago2fadd1d2020-09-11 12:31:14 -070045 printf(" --ignore-icc : If the input file contains an embedded ICC profile, ignore it (no-op if absent)\n");
Wan-Teh Chang980d5852021-08-03 20:02:38 -070046 printf(" --size-limit C : Specifies the image size limit (in total pixels) that should be tolerated.\n");
47 printf(" Default: %u, set to a smaller value to further restrict.\n", AVIF_DEFAULT_IMAGE_SIZE_LIMIT);
Joe Dragob2f46c12019-04-24 20:14:36 -070048 printf("\n");
Joe Drago14968d82019-11-11 12:07:51 -080049 avifPrintVersions();
Joe Dragob2f46c12019-04-24 20:14:36 -070050}
51
52int main(int argc, char * argv[])
53{
54 const char * inputFilename = NULL;
55 const char * outputFilename = NULL;
Joe Dragoc647acc2020-03-10 18:06:12 -070056 int requestedDepth = 0;
Joe Dragoede5c202020-11-11 09:42:57 -080057 int jobs = 1;
Joe Dragoddd23452020-04-15 13:23:08 -070058 int jpegQuality = DEFAULT_JPEG_QUALITY;
Joe Dragoc7301f32021-07-14 14:35:36 -070059 int pngCompressionLevel = -1; // -1 is a sentinel to avifPNGWrite() to skip calling png_set_compression_level()
Joe Dragoc8fc62f2020-02-28 09:39:40 -080060 avifCodecChoice codecChoice = AVIF_CODEC_CHOICE_AUTO;
Joe Dragoa0bc0532020-06-26 13:08:59 -070061 avifBool infoOnly = AVIF_FALSE;
Joe Drago0a0547b2020-10-30 12:28:33 -070062 avifChromaUpsampling chromaUpsampling = AVIF_CHROMA_UPSAMPLING_AUTOMATIC;
Joe Drago2fadd1d2020-09-11 12:31:14 -070063 avifBool ignoreICC = AVIF_FALSE;
Yuan Tongd4f317b2021-03-01 22:18:22 +080064 avifBool rawColor = AVIF_FALSE;
Joe Dragobffba3b2021-05-26 15:46:10 -070065 avifBool allowProgressive = AVIF_FALSE;
Joe Dragof5b98a62021-05-10 12:25:29 -070066 avifStrictFlags strictFlags = AVIF_STRICT_ENABLED;
Joe Dragobffba3b2021-05-26 15:46:10 -070067 uint32_t frameIndex = 0;
Wan-Teh Chang980d5852021-08-03 20:02:38 -070068 uint32_t imageSizeLimit = AVIF_DEFAULT_IMAGE_SIZE_LIMIT;
Joe Dragob2f46c12019-04-24 20:14:36 -070069
70 if (argc < 2) {
wantehchang37cf0c62020-03-08 17:25:18 -070071 syntax();
72 return 1;
Joe Dragob2f46c12019-04-24 20:14:36 -070073 }
74
Joe Dragob2f46c12019-04-24 20:14:36 -070075 int argIndex = 1;
Joe Dragob2f46c12019-04-24 20:14:36 -070076 while (argIndex < argc) {
77 const char * arg = argv[argIndex];
78
79 if (!strcmp(arg, "-h") || !strcmp(arg, "--help")) {
wantehchang37cf0c62020-03-08 17:25:18 -070080 syntax();
81 return 0;
Shun Sakai69814262020-11-06 18:38:10 +090082 } else if (!strcmp(arg, "-V") || !strcmp(arg, "--version")) {
83 avifPrintVersions();
84 return 0;
Joe Dragoede5c202020-11-11 09:42:57 -080085 } else if (!strcmp(arg, "-j") || !strcmp(arg, "--jobs")) {
86 NEXTARG();
Joe Drago5a987902021-06-07 15:04:25 -070087 if (!strcmp(arg, "all")) {
Joe Drago4eadcc52021-06-16 11:56:34 -070088 jobs = avifQueryCPUCount();
Joe Drago5a987902021-06-07 15:04:25 -070089 } else {
90 jobs = atoi(arg);
91 if (jobs < 1) {
92 jobs = 1;
93 }
Joe Dragoede5c202020-11-11 09:42:57 -080094 }
Joe Dragoc8fc62f2020-02-28 09:39:40 -080095 } else if (!strcmp(arg, "-c") || !strcmp(arg, "--codec")) {
96 NEXTARG();
97 codecChoice = avifCodecChoiceFromName(arg);
98 if (codecChoice == AVIF_CODEC_CHOICE_AUTO) {
99 fprintf(stderr, "ERROR: Unrecognized codec: %s\n", arg);
100 return 1;
101 } else {
102 const char * codecName = avifCodecName(codecChoice, AVIF_CODEC_FLAG_CAN_DECODE);
103 if (codecName == NULL) {
104 fprintf(stderr, "ERROR: AV1 Codec cannot decode: %s\n", arg);
105 return 1;
106 }
107 }
Joe Dragoc647acc2020-03-10 18:06:12 -0700108 } else if (!strcmp(arg, "-d") || !strcmp(arg, "--depth")) {
109 NEXTARG();
110 requestedDepth = atoi(arg);
111 if ((requestedDepth != 8) && (requestedDepth != 16)) {
112 fprintf(stderr, "ERROR: invalid depth: %s\n", arg);
113 return 1;
114 }
Joe Dragoddd23452020-04-15 13:23:08 -0700115 } else if (!strcmp(arg, "-q") || !strcmp(arg, "--quality")) {
116 NEXTARG();
117 jpegQuality = atoi(arg);
Wan-Teh Changa0133252020-04-23 16:02:49 -0700118 if (jpegQuality < 0) {
119 jpegQuality = 0;
Joe Dragoddd23452020-04-15 13:23:08 -0700120 } else if (jpegQuality > 100) {
121 jpegQuality = 100;
122 }
Joe Dragoc7301f32021-07-14 14:35:36 -0700123 } else if (!strcmp(arg, "--png-compress")) {
124 NEXTARG();
125 pngCompressionLevel = atoi(arg);
126 if (pngCompressionLevel < 0) {
127 pngCompressionLevel = 0;
128 } else if (pngCompressionLevel > 9) {
129 pngCompressionLevel = 9;
130 }
Joe Drago17fee3f2020-07-01 17:08:50 -0700131 } else if (!strcmp(arg, "-u") || !strcmp(arg, "--upsampling")) {
132 NEXTARG();
Joe Drago0a0547b2020-10-30 12:28:33 -0700133 if (!strcmp(arg, "automatic")) {
134 chromaUpsampling = AVIF_CHROMA_UPSAMPLING_AUTOMATIC;
Joe Dragob7be24a2020-11-06 17:28:42 -0800135 } else if (!strcmp(arg, "fastest")) {
136 chromaUpsampling = AVIF_CHROMA_UPSAMPLING_FASTEST;
137 } else if (!strcmp(arg, "best")) {
138 chromaUpsampling = AVIF_CHROMA_UPSAMPLING_BEST_QUALITY;
Joe Drago17fee3f2020-07-01 17:08:50 -0700139 } else if (!strcmp(arg, "nearest")) {
140 chromaUpsampling = AVIF_CHROMA_UPSAMPLING_NEAREST;
Joe Drago0a0547b2020-10-30 12:28:33 -0700141 } else if (!strcmp(arg, "bilinear")) {
142 chromaUpsampling = AVIF_CHROMA_UPSAMPLING_BILINEAR;
Joe Drago17fee3f2020-07-01 17:08:50 -0700143 } else {
144 fprintf(stderr, "ERROR: invalid upsampling: %s\n", arg);
145 return 1;
146 }
Yuan Tongd4f317b2021-03-01 22:18:22 +0800147 } else if (!strcmp(arg, "-r") || !strcmp(arg, "--raw-color")) {
148 rawColor = AVIF_TRUE;
Joe Dragobffba3b2021-05-26 15:46:10 -0700149 } else if (!strcmp(arg, "--progressive")) {
150 allowProgressive = AVIF_TRUE;
151 } else if (!strcmp(arg, "--index")) {
152 NEXTARG();
153 frameIndex = (uint32_t)atoi(arg);
Joe Dragof5b98a62021-05-10 12:25:29 -0700154 } else if (!strcmp(arg, "--no-strict")) {
155 strictFlags = AVIF_STRICT_DISABLED;
Joe Dragoa0bc0532020-06-26 13:08:59 -0700156 } else if (!strcmp(arg, "-i") || !strcmp(arg, "--info")) {
157 infoOnly = AVIF_TRUE;
Joe Drago2fadd1d2020-09-11 12:31:14 -0700158 } else if (!strcmp(arg, "--ignore-icc")) {
159 ignoreICC = AVIF_TRUE;
Wan-Teh Chang980d5852021-08-03 20:02:38 -0700160 } else if (!strcmp(arg, "--size-limit")) {
161 NEXTARG();
162 imageSizeLimit = strtoul(arg, NULL, 10);
163 if ((imageSizeLimit > AVIF_DEFAULT_IMAGE_SIZE_LIMIT) || (imageSizeLimit == 0)) {
164 fprintf(stderr, "ERROR: invalid image size limit: %s\n", arg);
165 return 1;
166 }
Joe Dragob2f46c12019-04-24 20:14:36 -0700167 } else {
168 // Positional argument
169 if (!inputFilename) {
170 inputFilename = arg;
171 } else if (!outputFilename) {
172 outputFilename = arg;
173 } else {
174 fprintf(stderr, "Too many positional arguments: %s\n", arg);
wantehchang37cf0c62020-03-08 17:25:18 -0700175 syntax();
Joe Dragob2f46c12019-04-24 20:14:36 -0700176 return 1;
177 }
178 }
179
180 ++argIndex;
181 }
182
Joe Dragoa0bc0532020-06-26 13:08:59 -0700183 if (!inputFilename) {
wantehchang37cf0c62020-03-08 17:25:18 -0700184 syntax();
185 return 1;
Joe Dragob2f46c12019-04-24 20:14:36 -0700186 }
187
Joe Dragoa0bc0532020-06-26 13:08:59 -0700188 if (infoOnly) {
189 if (!inputFilename || outputFilename) {
190 syntax();
191 return 1;
192 }
Joe Drago9b942c22021-05-06 12:33:12 -0700193
194 avifDecoder * decoder = avifDecoderCreate();
195 decoder->maxThreads = jobs;
196 decoder->codecChoice = codecChoice;
Wan-Teh Chang980d5852021-08-03 20:02:38 -0700197 decoder->imageSizeLimit = imageSizeLimit;
Joe Drago9b942c22021-05-06 12:33:12 -0700198 decoder->strictFlags = strictFlags;
Joe Dragobffba3b2021-05-26 15:46:10 -0700199 decoder->allowProgressive = allowProgressive;
Joe Drago9b942c22021-05-06 12:33:12 -0700200 avifResult result = avifDecoderSetIOFile(decoder, inputFilename);
201 if (result != AVIF_RESULT_OK) {
202 fprintf(stderr, "Cannot open file for read: %s\n", inputFilename);
203 avifDecoderDestroy(decoder);
204 return 1;
205 }
206 result = avifDecoderParse(decoder);
207 if (result == AVIF_RESULT_OK) {
208 printf("Image decoded: %s\n", inputFilename);
209 avifContainerDump(decoder);
210
211 printf(" * %" PRIu64 " timescales per second, %2.2f seconds (%" PRIu64 " timescales), %d frame%s\n",
212 decoder->timescale,
213 decoder->duration,
214 decoder->durationInTimescales,
215 decoder->imageCount,
216 (decoder->imageCount == 1) ? "" : "s");
Joe Dragobffba3b2021-05-26 15:46:10 -0700217 if (decoder->imageCount > 1) {
218 printf(" * %s Frames: (%u expected frames)\n",
219 (decoder->progressiveState != AVIF_PROGRESSIVE_STATE_UNAVAILABLE) ? "Progressive Image" : "Image Sequence",
220 decoder->imageCount);
221 } else {
222 printf(" * Frame:\n");
223 }
Joe Drago9b942c22021-05-06 12:33:12 -0700224
Joe Dragobffba3b2021-05-26 15:46:10 -0700225 int currIndex = 0;
Wan-Teh Changc3c8afb2021-07-13 16:26:51 -0700226 avifResult nextImageResult;
227 while ((nextImageResult = avifDecoderNextImage(decoder)) == AVIF_RESULT_OK) {
Joe Dragobffba3b2021-05-26 15:46:10 -0700228 printf(" * Decoded frame [%d] [pts %2.2f (%" PRIu64 " timescales)] [duration %2.2f (%" PRIu64 " timescales)] [%ux%u]\n",
229 currIndex,
Joe Drago9b942c22021-05-06 12:33:12 -0700230 decoder->imageTiming.pts,
231 decoder->imageTiming.ptsInTimescales,
232 decoder->imageTiming.duration,
Joe Dragobffba3b2021-05-26 15:46:10 -0700233 decoder->imageTiming.durationInTimescales,
234 decoder->image->width,
235 decoder->image->height);
236 ++currIndex;
Joe Dragobffba3b2021-05-26 15:46:10 -0700237 }
238 if (nextImageResult != AVIF_RESULT_NO_IMAGES_REMAINING) {
239 printf("ERROR: Failed to decode frame: %s\n", avifResultToString(nextImageResult));
240 avifDumpDiagnostics(&decoder->diag);
Joe Drago9b942c22021-05-06 12:33:12 -0700241 }
242 } else {
243 printf("ERROR: Failed to decode image: %s\n", avifResultToString(result));
244 avifDumpDiagnostics(&decoder->diag);
245 }
246
247 avifDecoderDestroy(decoder);
248 return 0;
Joe Dragoa0bc0532020-06-26 13:08:59 -0700249 } else {
250 if (!inputFilename || !outputFilename) {
251 syntax();
252 return 1;
253 }
254 }
255
Joe Dragoede5c202020-11-11 09:42:57 -0800256 printf("Decoding with AV1 codec '%s' (%d worker thread%s), please wait...\n",
257 avifCodecName(codecChoice, AVIF_CODEC_FLAG_CAN_DECODE),
258 jobs,
259 (jobs == 1) ? "" : "s");
Joe Dragoc8fc62f2020-02-28 09:39:40 -0800260
wantehchangddaa5c42020-03-08 17:23:58 -0700261 int returnCode = 0;
Joe Drago0b05eee2019-06-12 13:24:39 -0700262 avifDecoder * decoder = avifDecoderCreate();
Joe Dragoede5c202020-11-11 09:42:57 -0800263 decoder->maxThreads = jobs;
Joe Drago1bf61a12020-09-11 15:21:23 -0700264 decoder->codecChoice = codecChoice;
Wan-Teh Chang980d5852021-08-03 20:02:38 -0700265 decoder->imageSizeLimit = imageSizeLimit;
Joe Drago9b942c22021-05-06 12:33:12 -0700266 decoder->strictFlags = strictFlags;
Joe Dragobffba3b2021-05-26 15:46:10 -0700267 decoder->allowProgressive = allowProgressive;
Joe Dragoc647acc2020-03-10 18:06:12 -0700268
Joe Dragobffba3b2021-05-26 15:46:10 -0700269 avifResult result = avifDecoderSetIOFile(decoder, inputFilename);
270 if (result != AVIF_RESULT_OK) {
271 fprintf(stderr, "Cannot open file for read: %s\n", inputFilename);
272 returnCode = 1;
273 goto cleanup;
274 }
Joe Drago2fadd1d2020-09-11 12:31:14 -0700275
Joe Dragobffba3b2021-05-26 15:46:10 -0700276 result = avifDecoderParse(decoder);
277 if (result != AVIF_RESULT_OK) {
278 fprintf(stderr, "ERROR: Failed to parse image: %s\n", avifResultToString(result));
279 returnCode = 1;
280 goto cleanup;
281 }
282
283 result = avifDecoderNthImage(decoder, frameIndex);
284 if (result != AVIF_RESULT_OK) {
285 fprintf(stderr, "ERROR: Failed to decode image: %s\n", avifResultToString(result));
286 returnCode = 1;
287 goto cleanup;
288 }
289
290 printf("Image decoded: %s\n", inputFilename);
291 printf("Image details:\n");
292 avifImageDump(decoder->image, 0, 0, decoder->progressiveState);
293
294 if (ignoreICC && (decoder->image->icc.size > 0)) {
295 printf("[--ignore-icc] Discarding ICC profile.\n");
296 avifImageSetProfileICC(decoder->image, NULL, 0);
297 }
298
299 avifAppFileFormat outputFormat = avifGuessFileFormat(outputFilename);
300 if (outputFormat == AVIF_APP_FILE_FORMAT_UNKNOWN) {
301 fprintf(stderr, "Cannot determine output file extension: %s\n", outputFilename);
302 returnCode = 1;
303 } else if (outputFormat == AVIF_APP_FILE_FORMAT_Y4M) {
304 if (!y4mWrite(outputFilename, decoder->image)) {
Wan-Teh Chang01884e42020-04-27 17:33:04 -0700305 returnCode = 1;
Joe Dragobffba3b2021-05-26 15:46:10 -0700306 }
307 } else if (outputFormat == AVIF_APP_FILE_FORMAT_JPEG) {
308 // Bypass alpha multiply step during conversion
309 if (rawColor) {
310 decoder->image->alphaPremultiplied = AVIF_TRUE;
311 }
312 if (!avifJPEGWrite(outputFilename, decoder->image, jpegQuality, chromaUpsampling)) {
313 returnCode = 1;
314 }
315 } else if (outputFormat == AVIF_APP_FILE_FORMAT_PNG) {
Joe Dragoc7301f32021-07-14 14:35:36 -0700316 if (!avifPNGWrite(outputFilename, decoder->image, requestedDepth, chromaUpsampling, pngCompressionLevel)) {
Wan-Teh Changa0133252020-04-23 16:02:49 -0700317 returnCode = 1;
wantehchangddaa5c42020-03-08 17:23:58 -0700318 }
Joe Dragob2f46c12019-04-24 20:14:36 -0700319 } else {
Wan-Teh Chang4f107042021-07-13 16:34:24 -0700320 fprintf(stderr, "Unsupported output file extension: %s\n", outputFilename);
wantehchangddaa5c42020-03-08 17:23:58 -0700321 returnCode = 1;
Joe Dragob2f46c12019-04-24 20:14:36 -0700322 }
Joe Dragobffba3b2021-05-26 15:46:10 -0700323
324cleanup:
Wan-Teh Change4b118f2021-07-13 16:41:55 -0700325 if (returnCode != 0) {
326 avifDumpDiagnostics(&decoder->diag);
Joe Dragobffba3b2021-05-26 15:46:10 -0700327 }
Wan-Teh Change4b118f2021-07-13 16:41:55 -0700328 avifDecoderDestroy(decoder);
wantehchangddaa5c42020-03-08 17:23:58 -0700329 return returnCode;
Joe Dragob2f46c12019-04-24 20:14:36 -0700330}