2012-01-06 14:49:06 -08:00
|
|
|
// Copyright 2011 Google Inc. All Rights Reserved.
|
2011-02-18 23:33:46 -08:00
|
|
|
//
|
|
|
|
// This code is licensed under the same terms as WebM:
|
|
|
|
// Software License Agreement: http://www.webmproject.org/license/software/
|
|
|
|
// Additional IP Rights Grant: http://www.webmproject.org/license/additional/
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// simple command line calling the WebPEncode function.
|
|
|
|
// Encodes a raw .YUV into WebP bitstream
|
|
|
|
//
|
|
|
|
// Author: Skal (pascal.massimino@gmail.com)
|
|
|
|
|
|
|
|
#include <stdio.h>
|
2011-03-26 09:27:45 -07:00
|
|
|
#include <stdlib.h>
|
2011-02-18 23:33:46 -08:00
|
|
|
#include <string.h>
|
|
|
|
|
2011-06-06 17:56:50 -07:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2011-02-18 23:33:46 -08:00
|
|
|
#include "webp/encode.h"
|
2012-12-11 11:02:39 +01:00
|
|
|
|
2012-12-03 18:20:00 -08:00
|
|
|
#include "./metadata.h"
|
2012-05-15 13:48:11 -07:00
|
|
|
#include "./stopwatch.h"
|
2012-12-11 11:02:39 +01:00
|
|
|
|
|
|
|
#include "./jpegdec.h"
|
|
|
|
#include "./pngdec.h"
|
2012-12-06 14:04:36 -08:00
|
|
|
#include "./tiffdec.h"
|
2013-01-22 18:28:52 -08:00
|
|
|
#include "./wicdec.h"
|
2012-12-11 11:02:39 +01:00
|
|
|
|
2011-07-15 14:53:03 -07:00
|
|
|
#ifndef WEBP_DLL
|
2011-11-04 15:20:08 -07:00
|
|
|
#if defined(__cplusplus) || defined(c_plusplus)
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2011-09-02 21:30:08 +00:00
|
|
|
extern void* VP8GetCPUInfo; // opaque forward declaration.
|
2011-11-04 15:20:08 -07:00
|
|
|
|
|
|
|
#if defined(__cplusplus) || defined(c_plusplus)
|
|
|
|
} // extern "C"
|
2011-07-15 14:53:03 -07:00
|
|
|
#endif
|
2011-11-04 15:20:08 -07:00
|
|
|
#endif // WEBP_DLL
|
2011-04-22 12:14:45 -07:00
|
|
|
|
2011-08-25 14:22:32 -07:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-18 23:33:46 -08:00
|
|
|
|
|
|
|
static int verbose = 0;
|
|
|
|
|
|
|
|
static int ReadYUV(FILE* in_file, WebPPicture* const pic) {
|
2012-07-18 21:58:53 +00:00
|
|
|
const int use_argb = pic->use_argb;
|
2011-02-18 23:33:46 -08:00
|
|
|
const int uv_width = (pic->width + 1) / 2;
|
|
|
|
const int uv_height = (pic->height + 1) / 2;
|
|
|
|
int y;
|
|
|
|
int ok = 0;
|
|
|
|
|
2012-07-18 21:58:53 +00:00
|
|
|
pic->use_argb = 0;
|
2011-02-18 23:33:46 -08:00
|
|
|
if (!WebPPictureAlloc(pic)) return ok;
|
|
|
|
|
|
|
|
for (y = 0; y < pic->height; ++y) {
|
|
|
|
if (fread(pic->y + y * pic->y_stride, pic->width, 1, in_file) != 1) {
|
|
|
|
goto End;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (y = 0; y < uv_height; ++y) {
|
|
|
|
if (fread(pic->u + y * pic->uv_stride, uv_width, 1, in_file) != 1)
|
|
|
|
goto End;
|
|
|
|
}
|
|
|
|
for (y = 0; y < uv_height; ++y) {
|
|
|
|
if (fread(pic->v + y * pic->uv_stride, uv_width, 1, in_file) != 1)
|
|
|
|
goto End;
|
|
|
|
}
|
|
|
|
ok = 1;
|
2012-07-18 21:58:53 +00:00
|
|
|
if (use_argb) ok = WebPPictureYUVAToARGB(pic);
|
2011-02-18 23:33:46 -08:00
|
|
|
|
|
|
|
End:
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2011-06-06 17:56:50 -07:00
|
|
|
#ifdef HAVE_WINCODEC_H
|
2011-02-18 23:33:46 -08:00
|
|
|
|
EXPERIMENTAL: add support for alpha channel
This is a (minor) bitstream change: if the 'color_space' bit is set to '1'
(which is normally an undefined/invalid behaviour), we add extra data at the
end of partition #0 (so-called 'extensions')
Namely, we add the size of the extension data as 3 bytes (little-endian),
followed by a set of bits telling which extensions we're incorporating.
The data then _preceeds_ this trailing tags.
This is all experimental, and you'll need to have
'#define WEBP_EXPERIMENTAL_FEATURES' in webp/types.h to enable this code
(at your own risk! :))
Still, this hack produces almost-valid WebP file for decoders that don't
check this color_space bit. In particular, previous 'dwebp' (and for instance
Chrome) will recognize this files and decode them, but without the alpha
of course. Other decoder will just see random extra stuff at the end of
partition #0.
To experiment with the alpha-channel, you need to compile on Unix platform
and use PNGs for input/output.
If 'alpha.png' is a source with alpha channel, then you can try (on Unix):
cwebp alpha.png -o alpha.webp
dwebp alpha.webp -o test.png
cwebp now has a '-noalpha' flag to ignore any alpha information from the
source, if present.
More hacking and experimenting welcome!
Change-Id: I3c7b1fd8411c9e7a9f77690e898479ad85c52f3e
2011-04-25 16:58:04 -07:00
|
|
|
static int ReadPicture(const char* const filename, WebPPicture* const pic,
|
2012-12-03 18:20:00 -08:00
|
|
|
int keep_alpha, Metadata* const metadata) {
|
2011-02-18 23:33:46 -08:00
|
|
|
int ok;
|
2012-12-03 18:20:00 -08:00
|
|
|
(void)metadata; // TODO(jzern): add metadata extraction using WIC
|
2011-02-18 23:33:46 -08:00
|
|
|
if (pic->width != 0 && pic->height != 0) {
|
|
|
|
// If image size is specified, infer it as YUV format.
|
|
|
|
FILE* in_file = fopen(filename, "rb");
|
|
|
|
if (in_file == NULL) {
|
|
|
|
fprintf(stderr, "Error! Cannot open input file '%s'\n", filename);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
ok = ReadYUV(in_file, pic);
|
|
|
|
fclose(in_file);
|
|
|
|
} else {
|
|
|
|
// If no size specified, try to decode it using WIC.
|
2013-01-22 18:28:52 -08:00
|
|
|
ok = ReadPictureWithWIC(filename, pic, keep_alpha);
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
|
|
|
if (!ok) {
|
|
|
|
fprintf(stderr, "Error! Could not process file %s\n", filename);
|
|
|
|
}
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2011-06-06 17:56:50 -07:00
|
|
|
#else // !HAVE_WINCODEC_H
|
2011-02-18 23:33:46 -08:00
|
|
|
|
|
|
|
typedef enum {
|
2012-07-02 11:24:23 -07:00
|
|
|
PNG_ = 0,
|
|
|
|
JPEG_,
|
2012-07-01 17:55:21 -07:00
|
|
|
TIFF_, // 'TIFF' clashes with libtiff
|
2012-01-27 17:39:47 -08:00
|
|
|
UNSUPPORTED
|
2011-02-18 23:33:46 -08:00
|
|
|
} InputFileFormat;
|
|
|
|
|
|
|
|
static InputFileFormat GetImageType(FILE* in_file) {
|
|
|
|
InputFileFormat format = UNSUPPORTED;
|
|
|
|
unsigned int magic;
|
|
|
|
unsigned char buf[4];
|
|
|
|
|
|
|
|
if ((fread(&buf[0], 4, 1, in_file) != 1) ||
|
|
|
|
(fseek(in_file, 0, SEEK_SET) != 0)) {
|
|
|
|
return format;
|
|
|
|
}
|
|
|
|
|
|
|
|
magic = (buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3];
|
|
|
|
if (magic == 0x89504E47U) {
|
2012-07-02 11:24:23 -07:00
|
|
|
format = PNG_;
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (magic >= 0xFFD8FF00U && magic <= 0xFFD8FFFFU) {
|
2012-07-02 11:24:23 -07:00
|
|
|
format = JPEG_;
|
2012-07-01 17:55:21 -07:00
|
|
|
} else if (magic == 0x49492A00 || magic == 0x4D4D002A) {
|
|
|
|
format = TIFF_;
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
|
|
|
return format;
|
|
|
|
}
|
|
|
|
|
EXPERIMENTAL: add support for alpha channel
This is a (minor) bitstream change: if the 'color_space' bit is set to '1'
(which is normally an undefined/invalid behaviour), we add extra data at the
end of partition #0 (so-called 'extensions')
Namely, we add the size of the extension data as 3 bytes (little-endian),
followed by a set of bits telling which extensions we're incorporating.
The data then _preceeds_ this trailing tags.
This is all experimental, and you'll need to have
'#define WEBP_EXPERIMENTAL_FEATURES' in webp/types.h to enable this code
(at your own risk! :))
Still, this hack produces almost-valid WebP file for decoders that don't
check this color_space bit. In particular, previous 'dwebp' (and for instance
Chrome) will recognize this files and decode them, but without the alpha
of course. Other decoder will just see random extra stuff at the end of
partition #0.
To experiment with the alpha-channel, you need to compile on Unix platform
and use PNGs for input/output.
If 'alpha.png' is a source with alpha channel, then you can try (on Unix):
cwebp alpha.png -o alpha.webp
dwebp alpha.webp -o test.png
cwebp now has a '-noalpha' flag to ignore any alpha information from the
source, if present.
More hacking and experimenting welcome!
Change-Id: I3c7b1fd8411c9e7a9f77690e898479ad85c52f3e
2011-04-25 16:58:04 -07:00
|
|
|
static int ReadPicture(const char* const filename, WebPPicture* const pic,
|
2012-12-03 18:20:00 -08:00
|
|
|
int keep_alpha, Metadata* const metadata) {
|
2011-02-18 23:33:46 -08:00
|
|
|
int ok = 0;
|
|
|
|
FILE* in_file = fopen(filename, "rb");
|
|
|
|
if (in_file == NULL) {
|
|
|
|
fprintf(stderr, "Error! Cannot open input file '%s'\n", filename);
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pic->width == 0 || pic->height == 0) {
|
|
|
|
// If no size specified, try to decode it as PNG/JPEG (as appropriate).
|
|
|
|
const InputFileFormat format = GetImageType(in_file);
|
2012-07-02 11:24:23 -07:00
|
|
|
if (format == PNG_) {
|
2012-12-04 19:00:30 -08:00
|
|
|
ok = ReadPNG(in_file, pic, keep_alpha, metadata);
|
2012-07-02 11:24:23 -07:00
|
|
|
} else if (format == JPEG_) {
|
2012-12-16 19:13:56 -08:00
|
|
|
ok = ReadJPEG(in_file, pic, metadata);
|
2012-07-01 17:55:21 -07:00
|
|
|
} else if (format == TIFF_) {
|
2012-12-15 19:41:03 -08:00
|
|
|
ok = ReadTIFF(filename, pic, keep_alpha, metadata);
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If image size is specified, infer it as YUV format.
|
|
|
|
ok = ReadYUV(in_file, pic);
|
|
|
|
}
|
|
|
|
if (!ok) {
|
|
|
|
fprintf(stderr, "Error! Could not process file %s\n", filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(in_file);
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2011-06-06 17:56:50 -07:00
|
|
|
#endif // !HAVE_WINCODEC_H
|
2011-02-18 23:33:46 -08:00
|
|
|
|
|
|
|
static void AllocExtraInfo(WebPPicture* const pic) {
|
|
|
|
const int mb_w = (pic->width + 15) / 16;
|
|
|
|
const int mb_h = (pic->height + 15) / 16;
|
|
|
|
pic->extra_info = (uint8_t*)malloc(mb_w * mb_h * sizeof(*pic->extra_info));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void PrintByteCount(const int bytes[4], int total_size,
|
|
|
|
int* const totals) {
|
|
|
|
int s;
|
|
|
|
int total = 0;
|
|
|
|
for (s = 0; s < 4; ++s) {
|
|
|
|
fprintf(stderr, "| %7d ", bytes[s]);
|
|
|
|
total += bytes[s];
|
|
|
|
if (totals) totals[s] += bytes[s];
|
|
|
|
}
|
2011-11-04 15:20:08 -07:00
|
|
|
fprintf(stderr, "| %7d (%.1f%%)\n", total, 100.f * total / total_size);
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void PrintPercents(const int counts[4], int total) {
|
|
|
|
int s;
|
|
|
|
for (s = 0; s < 4; ++s) {
|
|
|
|
fprintf(stderr, "| %2d%%", 100 * counts[s] / total);
|
|
|
|
}
|
2011-11-04 15:20:08 -07:00
|
|
|
fprintf(stderr, "| %7d\n", total);
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void PrintValues(const int values[4]) {
|
|
|
|
int s;
|
|
|
|
for (s = 0; s < 4; ++s) {
|
|
|
|
fprintf(stderr, "| %7d ", values[s]);
|
|
|
|
}
|
2011-11-04 15:20:08 -07:00
|
|
|
fprintf(stderr, "|\n");
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
|
|
|
|
2012-07-24 16:15:36 -07:00
|
|
|
static void PrintFullLosslessInfo(const WebPAuxStats* const stats,
|
|
|
|
const char* const description) {
|
|
|
|
fprintf(stderr, "Lossless-%s compressed size: %d bytes\n",
|
|
|
|
description, stats->lossless_size);
|
|
|
|
if (stats->lossless_features) {
|
|
|
|
fprintf(stderr, " * Lossless features used:");
|
|
|
|
if (stats->lossless_features & 1) fprintf(stderr, " PREDICTION");
|
|
|
|
if (stats->lossless_features & 2) fprintf(stderr, " CROSS-COLOR-TRANSFORM");
|
|
|
|
if (stats->lossless_features & 4) fprintf(stderr, " SUBTRACT-GREEN");
|
|
|
|
if (stats->lossless_features & 8) fprintf(stderr, " PALETTE");
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
fprintf(stderr, " * Precision Bits: histogram=%d transform=%d cache=%d\n",
|
|
|
|
stats->histogram_bits, stats->transform_bits, stats->cache_bits);
|
|
|
|
if (stats->palette_size > 0) {
|
|
|
|
fprintf(stderr, " * Palette size: %d\n", stats->palette_size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-09 11:27:46 +05:30
|
|
|
static void PrintExtraInfoLossless(const WebPPicture* const pic,
|
|
|
|
int short_output,
|
|
|
|
const char* const file_name) {
|
|
|
|
const WebPAuxStats* const stats = pic->stats;
|
|
|
|
if (short_output) {
|
|
|
|
fprintf(stderr, "%7d %2.2f\n", stats->coded_size, stats->PSNR[3]);
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "File: %s\n", file_name);
|
|
|
|
fprintf(stderr, "Dimension: %d x %d\n", pic->width, pic->height);
|
|
|
|
fprintf(stderr, "Output: %d bytes\n", stats->coded_size);
|
2012-07-24 16:15:36 -07:00
|
|
|
PrintFullLosslessInfo(stats, "ARGB");
|
2012-05-09 11:27:46 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void PrintExtraInfoLossy(const WebPPicture* const pic, int short_output,
|
|
|
|
const char* const file_name) {
|
2011-02-18 23:33:46 -08:00
|
|
|
const WebPAuxStats* const stats = pic->stats;
|
|
|
|
if (short_output) {
|
|
|
|
fprintf(stderr, "%7d %2.2f\n", stats->coded_size, stats->PSNR[3]);
|
2011-11-04 15:20:08 -07:00
|
|
|
} else {
|
2011-02-18 23:33:46 -08:00
|
|
|
const int num_i4 = stats->block_count[0];
|
|
|
|
const int num_i16 = stats->block_count[1];
|
|
|
|
const int num_skip = stats->block_count[2];
|
|
|
|
const int total = num_i4 + num_i16;
|
2012-01-20 07:20:56 -08:00
|
|
|
fprintf(stderr, "File: %s\n", file_name);
|
|
|
|
fprintf(stderr, "Dimension: %d x %d%s\n",
|
2012-06-07 17:32:29 -07:00
|
|
|
pic->width, pic->height,
|
|
|
|
stats->alpha_data_size ? " (with alpha)" : "");
|
2012-01-20 07:20:56 -08:00
|
|
|
fprintf(stderr, "Output: "
|
|
|
|
"%d bytes Y-U-V-All-PSNR %2.2f %2.2f %2.2f %2.2f dB\n",
|
2011-02-18 23:33:46 -08:00
|
|
|
stats->coded_size,
|
|
|
|
stats->PSNR[0], stats->PSNR[1], stats->PSNR[2], stats->PSNR[3]);
|
|
|
|
if (total > 0) {
|
|
|
|
int totals[4] = { 0, 0, 0, 0 };
|
|
|
|
fprintf(stderr, "block count: intra4: %d\n"
|
|
|
|
" intra16: %d (-> %.2f%%)\n",
|
|
|
|
num_i4, num_i16, 100.f * num_i16 / total);
|
|
|
|
fprintf(stderr, " skipped block: %d (%.2f%%)\n",
|
|
|
|
num_skip, 100.f * num_skip / total);
|
|
|
|
fprintf(stderr, "bytes used: header: %6d (%.1f%%)\n"
|
|
|
|
" mode-partition: %6d (%.1f%%)\n",
|
|
|
|
stats->header_bytes[0],
|
|
|
|
100.f * stats->header_bytes[0] / stats->coded_size,
|
|
|
|
stats->header_bytes[1],
|
|
|
|
100.f * stats->header_bytes[1] / stats->coded_size);
|
2012-07-24 16:15:36 -07:00
|
|
|
if (stats->alpha_data_size > 0) {
|
|
|
|
fprintf(stderr, " transparency: %6d (%.1f dB)\n",
|
|
|
|
stats->alpha_data_size, stats->PSNR[4]);
|
EXPERIMENTAL: add support for alpha channel
This is a (minor) bitstream change: if the 'color_space' bit is set to '1'
(which is normally an undefined/invalid behaviour), we add extra data at the
end of partition #0 (so-called 'extensions')
Namely, we add the size of the extension data as 3 bytes (little-endian),
followed by a set of bits telling which extensions we're incorporating.
The data then _preceeds_ this trailing tags.
This is all experimental, and you'll need to have
'#define WEBP_EXPERIMENTAL_FEATURES' in webp/types.h to enable this code
(at your own risk! :))
Still, this hack produces almost-valid WebP file for decoders that don't
check this color_space bit. In particular, previous 'dwebp' (and for instance
Chrome) will recognize this files and decode them, but without the alpha
of course. Other decoder will just see random extra stuff at the end of
partition #0.
To experiment with the alpha-channel, you need to compile on Unix platform
and use PNGs for input/output.
If 'alpha.png' is a source with alpha channel, then you can try (on Unix):
cwebp alpha.png -o alpha.webp
dwebp alpha.webp -o test.png
cwebp now has a '-noalpha' flag to ignore any alpha information from the
source, if present.
More hacking and experimenting welcome!
Change-Id: I3c7b1fd8411c9e7a9f77690e898479ad85c52f3e
2011-04-25 16:58:04 -07:00
|
|
|
}
|
2011-05-02 17:19:00 -07:00
|
|
|
if (stats->layer_data_size) {
|
|
|
|
fprintf(stderr, " enhancement: %6d\n",
|
|
|
|
stats->layer_data_size);
|
|
|
|
}
|
2011-02-18 23:33:46 -08:00
|
|
|
fprintf(stderr, " Residuals bytes "
|
|
|
|
"|segment 1|segment 2|segment 3"
|
|
|
|
"|segment 4| total\n");
|
|
|
|
fprintf(stderr, " intra4-coeffs: ");
|
|
|
|
PrintByteCount(stats->residual_bytes[0], stats->coded_size, totals);
|
|
|
|
fprintf(stderr, " intra16-coeffs: ");
|
|
|
|
PrintByteCount(stats->residual_bytes[1], stats->coded_size, totals);
|
|
|
|
fprintf(stderr, " chroma coeffs: ");
|
|
|
|
PrintByteCount(stats->residual_bytes[2], stats->coded_size, totals);
|
|
|
|
fprintf(stderr, " macroblocks: ");
|
|
|
|
PrintPercents(stats->segment_size, total);
|
|
|
|
fprintf(stderr, " quantizer: ");
|
|
|
|
PrintValues(stats->segment_quant);
|
|
|
|
fprintf(stderr, " filter level: ");
|
|
|
|
PrintValues(stats->segment_level);
|
|
|
|
fprintf(stderr, "------------------+---------");
|
|
|
|
fprintf(stderr, "+---------+---------+---------+-----------------\n");
|
|
|
|
fprintf(stderr, " segments total: ");
|
|
|
|
PrintByteCount(totals, stats->coded_size, NULL);
|
|
|
|
}
|
2012-07-24 16:15:36 -07:00
|
|
|
if (stats->lossless_size > 0) {
|
|
|
|
PrintFullLosslessInfo(stats, "alpha");
|
|
|
|
}
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
2012-07-24 16:15:36 -07:00
|
|
|
if (pic->extra_info != NULL) {
|
2011-02-18 23:33:46 -08:00
|
|
|
const int mb_w = (pic->width + 15) / 16;
|
|
|
|
const int mb_h = (pic->height + 15) / 16;
|
|
|
|
const int type = pic->extra_info_type;
|
|
|
|
int x, y;
|
|
|
|
for (y = 0; y < mb_h; ++y) {
|
|
|
|
for (x = 0; x < mb_w; ++x) {
|
|
|
|
const int c = pic->extra_info[x + y * mb_w];
|
|
|
|
if (type == 1) { // intra4/intra16
|
|
|
|
printf("%c", "+."[c]);
|
|
|
|
} else if (type == 2) { // segments
|
|
|
|
printf("%c", ".-*X"[c]);
|
|
|
|
} else if (type == 3) { // quantizers
|
|
|
|
printf("%.2d ", c);
|
|
|
|
} else if (type == 6 || type == 7) {
|
|
|
|
printf("%3d ", c);
|
|
|
|
} else {
|
|
|
|
printf("0x%.2x ", c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-25 14:22:32 -07:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-18 23:33:46 -08:00
|
|
|
|
|
|
|
static int MyWriter(const uint8_t* data, size_t data_size,
|
|
|
|
const WebPPicture* const pic) {
|
|
|
|
FILE* const out = (FILE*)pic->custom_ptr;
|
|
|
|
return data_size ? (fwrite(data, data_size, 1, out) == 1) : 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Dumps a picture as a PGM file using the IMC4 layout.
|
|
|
|
static int DumpPicture(const WebPPicture* const picture, const char* PGM_name) {
|
|
|
|
int y;
|
|
|
|
const int uv_width = (picture->width + 1) / 2;
|
|
|
|
const int uv_height = (picture->height + 1) / 2;
|
|
|
|
const int stride = (picture->width + 1) & ~1;
|
2012-06-04 15:50:05 -07:00
|
|
|
const int alpha_height =
|
|
|
|
WebPPictureHasTransparency(picture) ? picture->height : 0;
|
2011-07-07 16:08:15 -07:00
|
|
|
const int height = picture->height + uv_height + alpha_height;
|
2011-02-18 23:33:46 -08:00
|
|
|
FILE* const f = fopen(PGM_name, "wb");
|
2012-01-20 07:20:56 -08:00
|
|
|
if (f == NULL) return 0;
|
2011-02-18 23:33:46 -08:00
|
|
|
fprintf(f, "P5\n%d %d\n255\n", stride, height);
|
|
|
|
for (y = 0; y < picture->height; ++y) {
|
|
|
|
if (fwrite(picture->y + y * picture->y_stride, picture->width, 1, f) != 1)
|
|
|
|
return 0;
|
|
|
|
if (picture->width & 1) fputc(0, f); // pad
|
|
|
|
}
|
|
|
|
for (y = 0; y < uv_height; ++y) {
|
|
|
|
if (fwrite(picture->u + y * picture->uv_stride, uv_width, 1, f) != 1)
|
|
|
|
return 0;
|
|
|
|
if (fwrite(picture->v + y * picture->uv_stride, uv_width, 1, f) != 1)
|
|
|
|
return 0;
|
|
|
|
}
|
2011-07-07 16:08:15 -07:00
|
|
|
for (y = 0; y < alpha_height; ++y) {
|
|
|
|
if (fwrite(picture->a + y * picture->a_stride, picture->width, 1, f) != 1)
|
|
|
|
return 0;
|
|
|
|
if (picture->width & 1) fputc(0, f); // pad
|
|
|
|
}
|
2011-02-18 23:33:46 -08:00
|
|
|
fclose(f);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-01-11 12:25:36 -08:00
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// Metadata writing.
|
|
|
|
|
|
|
|
enum {
|
|
|
|
METADATA_EXIF = (1 << 0),
|
|
|
|
METADATA_ICCP = (1 << 1),
|
|
|
|
METADATA_XMP = (1 << 2),
|
|
|
|
METADATA_ALL = METADATA_EXIF | METADATA_ICCP | METADATA_XMP
|
|
|
|
};
|
|
|
|
|
2013-01-14 18:32:44 -08:00
|
|
|
static const int kChunkHeaderSize = 8;
|
|
|
|
static const int kTagSize = 4;
|
|
|
|
|
|
|
|
// Outputs, in little endian, 'num' bytes from 'val' to 'out'.
|
|
|
|
static int WriteLE(FILE* const out, uint32_t val, int num) {
|
|
|
|
uint8_t buf[4];
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < num; ++i) {
|
|
|
|
buf[i] = (uint8_t)(val & 0xff);
|
|
|
|
val >>= 8;
|
|
|
|
}
|
|
|
|
return (fwrite(buf, num, 1, out) == 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int WriteLE24(FILE* const out, uint32_t val) {
|
|
|
|
return WriteLE(out, val, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int WriteLE32(FILE* const out, uint32_t val) {
|
|
|
|
return WriteLE(out, val, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int WriteMetadataChunk(FILE* const out, const char fourcc[4],
|
|
|
|
const MetadataPayload* const payload) {
|
|
|
|
const uint8_t zero = 0;
|
|
|
|
const size_t need_padding = payload->size & 1;
|
|
|
|
int ok = (fwrite(fourcc, kTagSize, 1, out) == 1);
|
|
|
|
ok = ok && WriteLE32(out, (uint32_t)payload->size);
|
|
|
|
ok = ok && (fwrite(payload->bytes, payload->size, 1, out) == 1);
|
|
|
|
return ok && (fwrite(&zero, need_padding, need_padding, out) == need_padding);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sets 'flag' in 'vp8x_flags' and updates 'metadata_size' with the size of the
|
|
|
|
// chunk if there is metadata and 'keep' is true.
|
|
|
|
static int UpdateFlagsAndSize(const MetadataPayload* const payload,
|
|
|
|
int keep, int flag,
|
|
|
|
uint32_t* vp8x_flags, uint64_t* metadata_size) {
|
|
|
|
if (keep && payload->bytes != NULL && payload->size > 0) {
|
|
|
|
*vp8x_flags |= flag;
|
|
|
|
*metadata_size += kChunkHeaderSize + payload->size + (payload->size & 1);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Writes a WebP file using the image contained in 'memory_writer' and the
|
|
|
|
// metadata from 'metadata'. Metadata is controlled by 'keep_metadata' and the
|
|
|
|
// availability in 'metadata'. Returns true on success.
|
|
|
|
// For details see doc/webp-container-spec.txt#extended-file-format.
|
|
|
|
static int WriteWebPWithMetadata(FILE* const out,
|
|
|
|
const WebPPicture* const picture,
|
|
|
|
const WebPMemoryWriter* const memory_writer,
|
|
|
|
const Metadata* const metadata,
|
|
|
|
int keep_metadata) {
|
|
|
|
const char kVP8XHeader[] = "VP8X\x0a\x00\x00\x00";
|
|
|
|
const int kAlphaFlag = 0x10;
|
|
|
|
const int kEXIFFlag = 0x08;
|
|
|
|
const int kICCPFlag = 0x20;
|
|
|
|
const int kXMPFlag = 0x04;
|
|
|
|
const size_t kRiffHeaderSize = 12;
|
|
|
|
const size_t kMaxChunkPayload = ~0 - kChunkHeaderSize - 1;
|
|
|
|
const size_t kMinSize = kRiffHeaderSize + kChunkHeaderSize;
|
|
|
|
uint32_t flags = 0;
|
|
|
|
uint64_t metadata_size = 0;
|
|
|
|
const int write_exif = UpdateFlagsAndSize(&metadata->exif,
|
|
|
|
!!(keep_metadata & METADATA_EXIF),
|
|
|
|
kEXIFFlag, &flags, &metadata_size);
|
|
|
|
const int write_iccp = UpdateFlagsAndSize(&metadata->iccp,
|
|
|
|
!!(keep_metadata & METADATA_ICCP),
|
|
|
|
kICCPFlag, &flags, &metadata_size);
|
|
|
|
const int write_xmp = UpdateFlagsAndSize(&metadata->xmp,
|
|
|
|
!!(keep_metadata & METADATA_XMP),
|
|
|
|
kXMPFlag, &flags, &metadata_size);
|
|
|
|
uint8_t* webp = memory_writer->mem;
|
|
|
|
size_t webp_size = memory_writer->size;
|
|
|
|
if (webp_size < kMinSize) return 0;
|
|
|
|
if (webp_size - kChunkHeaderSize + metadata_size > kMaxChunkPayload) {
|
|
|
|
fprintf(stderr, "Error! Addition of metadata would exceed "
|
|
|
|
"container size limit.\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (metadata_size > 0) {
|
|
|
|
const int kVP8XChunkSize = 18;
|
|
|
|
const int has_vp8x = !memcmp(webp + kRiffHeaderSize, "VP8X", kTagSize);
|
|
|
|
const uint32_t riff_size = (uint32_t)(webp_size - kChunkHeaderSize +
|
|
|
|
(has_vp8x ? 0 : kVP8XChunkSize) +
|
|
|
|
metadata_size);
|
|
|
|
// RIFF
|
|
|
|
int ok = (fwrite(webp, kTagSize, 1, out) == 1);
|
|
|
|
// RIFF size (file header size is not recorded)
|
|
|
|
ok = ok && WriteLE32(out, riff_size);
|
|
|
|
webp += kChunkHeaderSize;
|
|
|
|
webp_size -= kChunkHeaderSize;
|
|
|
|
// WEBP
|
|
|
|
ok = ok && (fwrite(webp, kTagSize, 1, out) == 1);
|
|
|
|
webp += kTagSize;
|
|
|
|
webp_size -= kTagSize;
|
|
|
|
if (has_vp8x) { // update the existing VP8X flags
|
|
|
|
webp[kChunkHeaderSize] |= (uint8_t)(flags & 0xff);
|
|
|
|
ok = ok && (fwrite(webp, kVP8XChunkSize, 1, out) == 1);
|
|
|
|
webp_size -= kVP8XChunkSize;
|
|
|
|
} else {
|
|
|
|
const int is_lossless = !memcmp(webp, "VP8L", kTagSize);
|
|
|
|
// The alpha flag is forced with lossless images.
|
|
|
|
if (is_lossless) flags |= kAlphaFlag;
|
|
|
|
ok = ok && (fwrite(kVP8XHeader, kChunkHeaderSize, 1, out) == 1);
|
|
|
|
ok = ok && WriteLE32(out, flags);
|
|
|
|
ok = ok && WriteLE24(out, picture->width - 1);
|
|
|
|
ok = ok && WriteLE24(out, picture->height - 1);
|
|
|
|
}
|
|
|
|
if (write_iccp) ok = ok && WriteMetadataChunk(out, "ICCP", &metadata->iccp);
|
|
|
|
// Image
|
|
|
|
ok = ok && (fwrite(webp, webp_size, 1, out) == 1);
|
|
|
|
if (write_exif) ok = ok && WriteMetadataChunk(out, "EXIF", &metadata->exif);
|
|
|
|
if (write_xmp) ok = ok && WriteMetadataChunk(out, "XMP ", &metadata->xmp);
|
|
|
|
return ok;
|
|
|
|
} else {
|
|
|
|
// No metadata, just write the original image file.
|
|
|
|
return (fwrite(webp, webp_size, 1, out) == 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-25 14:22:32 -07:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-18 23:33:46 -08:00
|
|
|
|
2011-12-01 02:24:50 -08:00
|
|
|
static int ProgressReport(int percent, const WebPPicture* const picture) {
|
|
|
|
printf("[%s]: %3d %% \r",
|
2012-07-27 19:53:16 -07:00
|
|
|
(char*)picture->user_data, percent);
|
2011-12-01 02:24:50 -08:00
|
|
|
fflush(stdout);
|
|
|
|
return 1; // all ok
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
2011-03-25 15:04:11 -07:00
|
|
|
static void HelpShort(void) {
|
2011-02-18 23:33:46 -08:00
|
|
|
printf("Usage:\n\n");
|
|
|
|
printf(" cwebp [options] -q quality input.png -o output.webp\n\n");
|
|
|
|
printf("where quality is between 0 (poor) to 100 (very good).\n");
|
|
|
|
printf("Typical value is around 80.\n\n");
|
|
|
|
printf("Try -longhelp for an exhaustive list of advanced options.\n");
|
|
|
|
}
|
|
|
|
|
2011-03-25 15:04:11 -07:00
|
|
|
static void HelpLong(void) {
|
2011-02-18 23:33:46 -08:00
|
|
|
printf("Usage:\n");
|
|
|
|
printf(" cwebp [-preset <...>] [options] in_file [-o out_file]\n\n");
|
|
|
|
printf("If input size (-s) for an image is not specified, "
|
2012-07-13 14:55:39 -07:00
|
|
|
"it is assumed to be a PNG, JPEG or TIFF file.\n");
|
2011-06-06 17:56:50 -07:00
|
|
|
#ifdef HAVE_WINCODEC_H
|
2011-02-18 23:33:46 -08:00
|
|
|
printf("Windows builds can take as input any of the files handled by WIC\n");
|
|
|
|
#endif
|
|
|
|
printf("options:\n");
|
|
|
|
printf(" -h / -help ............ short help\n");
|
|
|
|
printf(" -H / -longhelp ........ long help\n");
|
|
|
|
printf(" -q <float> ............. quality factor (0:small..100:big)\n");
|
2011-12-01 15:11:34 +05:30
|
|
|
printf(" -alpha_q <int> ......... Transparency-compression quality "
|
|
|
|
"(0..100).\n");
|
2011-02-18 23:33:46 -08:00
|
|
|
printf(" -preset <string> ....... Preset setting, one of:\n");
|
|
|
|
printf(" default, photo, picture,\n");
|
|
|
|
printf(" drawing, icon, text\n");
|
|
|
|
printf(" -preset must come first, as it overwrites other parameters.");
|
|
|
|
printf("\n");
|
|
|
|
printf(" -m <int> ............... compression method (0=fast, 6=slowest)\n");
|
|
|
|
printf(" -segments <int> ........ number of segments to use (1..4)\n");
|
2011-05-04 22:59:51 -07:00
|
|
|
printf(" -size <int> ............ Target size (in bytes)\n");
|
|
|
|
printf(" -psnr <float> .......... Target PSNR (in dB. typically: 42)\n");
|
2011-02-18 23:33:46 -08:00
|
|
|
printf("\n");
|
|
|
|
printf(" -s <int> <int> ......... Input size (width x height) for YUV\n");
|
|
|
|
printf(" -sns <int> ............. Spatial Noise Shaping (0:off, 100:max)\n");
|
|
|
|
printf(" -f <int> ............... filter strength (0=off..100)\n");
|
|
|
|
printf(" -sharpness <int> ....... "
|
|
|
|
"filter sharpness (0:most .. 7:least sharp)\n");
|
|
|
|
printf(" -strong ................ use strong filter instead of simple.\n");
|
2011-08-23 15:58:22 -07:00
|
|
|
printf(" -partition_limit <int> . limit quality to fit the 512k limit on\n");
|
|
|
|
printf(" "
|
|
|
|
"the first partition (0=no degradation ... 100=full)\n");
|
2011-02-18 23:33:46 -08:00
|
|
|
printf(" -pass <int> ............ analysis pass number (1..10)\n");
|
|
|
|
printf(" -crop <x> <y> <w> <h> .. crop picture with the given rectangle\n");
|
2011-05-02 17:19:00 -07:00
|
|
|
printf(" -resize <w> <h> ........ resize picture (after any cropping)\n");
|
|
|
|
#ifdef WEBP_EXPERIMENTAL_FEATURES
|
|
|
|
printf(" -444 / -422 / -gray ..... Change colorspace\n");
|
|
|
|
#endif
|
2011-02-18 23:33:46 -08:00
|
|
|
printf(" -map <int> ............. print map of extra info.\n");
|
2012-01-20 07:20:56 -08:00
|
|
|
printf(" -print_psnr ............ prints averaged PSNR distortion.\n");
|
2012-10-18 08:26:40 -07:00
|
|
|
printf(" -print_ssim ............ prints averaged SSIM distortion.\n");
|
|
|
|
printf(" -print_lsim ............ prints local-similarity distortion.\n");
|
2011-02-18 23:33:46 -08:00
|
|
|
printf(" -d <file.pgm> .......... dump the compressed output (PGM file).\n");
|
2011-12-01 15:11:34 +05:30
|
|
|
printf(" -alpha_method <int> .... Transparency-compression method (0..1)\n");
|
2012-01-08 19:27:21 -08:00
|
|
|
printf(" -alpha_filter <string> . predictive filtering for alpha plane.\n");
|
|
|
|
printf(" One of: none, fast (default) or best.\n");
|
2012-01-17 08:18:22 +00:00
|
|
|
printf(" -alpha_cleanup ......... Clean RGB values in transparent area.\n");
|
2011-12-01 15:11:34 +05:30
|
|
|
printf(" -noalpha ............... discard any transparency information.\n");
|
2012-06-22 12:14:48 +05:30
|
|
|
printf(" -lossless .............. Encode image losslessly.\n");
|
|
|
|
printf(" -hint <string> ......... Specify image characteristics hint.\n");
|
2012-07-31 23:07:52 -07:00
|
|
|
printf(" One of: photo, picture or graph\n");
|
2011-05-02 17:19:00 -07:00
|
|
|
|
2013-01-11 12:25:36 -08:00
|
|
|
printf("\n");
|
|
|
|
printf(" -metadata <string> ..... comma separated list of metadata to\n");
|
|
|
|
printf(" ");
|
|
|
|
printf("copy from the input to the output if present.\n");
|
|
|
|
printf(" "
|
|
|
|
"Valid values: all, none (default), exif, iccp, xmp\n");
|
|
|
|
|
2011-02-18 23:33:46 -08:00
|
|
|
printf("\n");
|
|
|
|
printf(" -short ................. condense printed message\n");
|
|
|
|
printf(" -quiet ................. don't print anything.\n");
|
2011-03-24 16:17:10 -07:00
|
|
|
printf(" -version ............... print version number and exit.\n");
|
2011-07-15 14:53:03 -07:00
|
|
|
#ifndef WEBP_DLL
|
2011-04-22 12:14:45 -07:00
|
|
|
printf(" -noasm ................. disable all assembly optimizations.\n");
|
2011-07-15 14:53:03 -07:00
|
|
|
#endif
|
2011-02-18 23:33:46 -08:00
|
|
|
printf(" -v ..................... verbose, e.g. print encoding/decoding "
|
|
|
|
"times\n");
|
2011-12-01 02:24:50 -08:00
|
|
|
printf(" -progress .............. report encoding progress\n");
|
2011-02-18 23:33:46 -08:00
|
|
|
printf("\n");
|
|
|
|
printf("Experimental Options:\n");
|
|
|
|
printf(" -af .................... auto-adjust filter strength.\n");
|
|
|
|
printf(" -pre <int> ............. pre-processing filter\n");
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
2011-08-25 14:22:32 -07:00
|
|
|
//------------------------------------------------------------------------------
|
2011-06-02 06:55:03 -07:00
|
|
|
// Error messages
|
|
|
|
|
|
|
|
static const char* const kErrorMessages[] = {
|
|
|
|
"OK",
|
|
|
|
"OUT_OF_MEMORY: Out of memory allocating objects",
|
|
|
|
"BITSTREAM_OUT_OF_MEMORY: Out of memory re-allocating byte buffer",
|
|
|
|
"NULL_PARAMETER: NULL parameter passed to function",
|
|
|
|
"INVALID_CONFIGURATION: configuration is invalid",
|
2011-08-23 15:58:22 -07:00
|
|
|
"BAD_DIMENSION: Bad picture dimension. Maximum width and height "
|
|
|
|
"allowed is 16383 pixels.",
|
|
|
|
"PARTITION0_OVERFLOW: Partition #0 is too big to fit 512k.\n"
|
|
|
|
"To reduce the size of this partition, try using less segments "
|
|
|
|
"with the -segments option, and eventually reduce the number of "
|
|
|
|
"header bits using -partition_limit. More details are available "
|
|
|
|
"in the manual (`man cwebp`)",
|
|
|
|
"PARTITION_OVERFLOW: Partition is too big to fit 16M",
|
2011-12-01 03:34:22 -08:00
|
|
|
"BAD_WRITE: Picture writer returned an I/O error",
|
2011-12-01 02:24:50 -08:00
|
|
|
"FILE_TOO_BIG: File would be too big to fit in 4G",
|
|
|
|
"USER_ABORT: encoding abort requested by user"
|
2011-06-02 06:55:03 -07:00
|
|
|
};
|
|
|
|
|
2011-08-25 14:22:32 -07:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-18 23:33:46 -08:00
|
|
|
|
|
|
|
int main(int argc, const char *argv[]) {
|
2012-05-09 00:32:20 -07:00
|
|
|
int return_value = -1;
|
2011-02-18 23:33:46 -08:00
|
|
|
const char *in_file = NULL, *out_file = NULL, *dump_file = NULL;
|
|
|
|
FILE *out = NULL;
|
|
|
|
int c;
|
|
|
|
int short_output = 0;
|
|
|
|
int quiet = 0;
|
2011-12-01 15:11:34 +05:30
|
|
|
int keep_alpha = 1;
|
2011-02-18 23:33:46 -08:00
|
|
|
int crop = 0, crop_x = 0, crop_y = 0, crop_w = 0, crop_h = 0;
|
2011-05-02 17:19:00 -07:00
|
|
|
int resize_w = 0, resize_h = 0;
|
2011-12-01 02:24:50 -08:00
|
|
|
int show_progress = 0;
|
2013-01-11 12:25:36 -08:00
|
|
|
int keep_metadata = 0;
|
2011-02-18 23:33:46 -08:00
|
|
|
WebPPicture picture;
|
2012-10-18 08:26:40 -07:00
|
|
|
int print_distortion = -1; // -1=off, 0=PSNR, 1=SSIM, 2=LSIM
|
2012-01-20 07:20:56 -08:00
|
|
|
WebPPicture original_picture; // when PSNR or SSIM is requested
|
2011-02-18 23:33:46 -08:00
|
|
|
WebPConfig config;
|
|
|
|
WebPAuxStats stats;
|
2013-01-14 18:32:44 -08:00
|
|
|
WebPMemoryWriter memory_writer;
|
2012-12-03 18:20:00 -08:00
|
|
|
Metadata metadata;
|
2011-02-18 23:33:46 -08:00
|
|
|
Stopwatch stop_watch;
|
2011-05-02 17:19:00 -07:00
|
|
|
|
2012-12-03 18:20:00 -08:00
|
|
|
MetadataInit(&metadata);
|
2013-02-01 19:17:26 -08:00
|
|
|
WebPMemoryWriterInit(&memory_writer);
|
2012-01-20 07:20:56 -08:00
|
|
|
if (!WebPPictureInit(&picture) ||
|
|
|
|
!WebPPictureInit(&original_picture) ||
|
|
|
|
!WebPConfigInit(&config)) {
|
2011-02-18 23:33:46 -08:00
|
|
|
fprintf(stderr, "Error! Version mismatch!\n");
|
2012-07-27 18:56:55 -07:00
|
|
|
return -1;
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
HelpShort();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (c = 1; c < argc; ++c) {
|
|
|
|
if (!strcmp(argv[c], "-h") || !strcmp(argv[c], "-help")) {
|
|
|
|
HelpShort();
|
|
|
|
return 0;
|
|
|
|
} else if (!strcmp(argv[c], "-H") || !strcmp(argv[c], "-longhelp")) {
|
|
|
|
HelpLong();
|
|
|
|
return 0;
|
|
|
|
} else if (!strcmp(argv[c], "-o") && c < argc - 1) {
|
|
|
|
out_file = argv[++c];
|
|
|
|
} else if (!strcmp(argv[c], "-d") && c < argc - 1) {
|
|
|
|
dump_file = argv[++c];
|
|
|
|
config.show_compressed = 1;
|
2012-01-20 07:20:56 -08:00
|
|
|
} else if (!strcmp(argv[c], "-print_psnr")) {
|
2012-10-18 08:26:40 -07:00
|
|
|
config.show_compressed = 1;
|
|
|
|
print_distortion = 0;
|
|
|
|
} else if (!strcmp(argv[c], "-print_ssim")) {
|
2012-01-20 07:20:56 -08:00
|
|
|
config.show_compressed = 1;
|
|
|
|
print_distortion = 1;
|
2012-10-18 08:26:40 -07:00
|
|
|
} else if (!strcmp(argv[c], "-print_lsim")) {
|
|
|
|
config.show_compressed = 1;
|
|
|
|
print_distortion = 2;
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-short")) {
|
|
|
|
short_output++;
|
|
|
|
} else if (!strcmp(argv[c], "-s") && c < argc - 2) {
|
2011-03-26 09:27:45 -07:00
|
|
|
picture.width = strtol(argv[++c], NULL, 0);
|
|
|
|
picture.height = strtol(argv[++c], NULL, 0);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-m") && c < argc - 1) {
|
2011-03-26 09:27:45 -07:00
|
|
|
config.method = strtol(argv[++c], NULL, 0);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-q") && c < argc - 1) {
|
2011-06-28 16:02:56 +02:00
|
|
|
config.quality = (float)strtod(argv[++c], NULL);
|
2011-12-01 15:11:34 +05:30
|
|
|
} else if (!strcmp(argv[c], "-alpha_q") && c < argc - 1) {
|
|
|
|
config.alpha_quality = strtol(argv[++c], NULL, 0);
|
|
|
|
} else if (!strcmp(argv[c], "-alpha_method") && c < argc - 1) {
|
|
|
|
config.alpha_compression = strtol(argv[++c], NULL, 0);
|
2012-01-17 08:18:22 +00:00
|
|
|
} else if (!strcmp(argv[c], "-alpha_cleanup")) {
|
|
|
|
keep_alpha = keep_alpha ? 2 : 0;
|
2012-01-05 13:04:30 +05:30
|
|
|
} else if (!strcmp(argv[c], "-alpha_filter") && c < argc - 1) {
|
2012-01-08 19:27:21 -08:00
|
|
|
++c;
|
|
|
|
if (!strcmp(argv[c], "none")) {
|
|
|
|
config.alpha_filtering = 0;
|
|
|
|
} else if (!strcmp(argv[c], "fast")) {
|
|
|
|
config.alpha_filtering = 1;
|
|
|
|
} else if (!strcmp(argv[c], "best")) {
|
|
|
|
config.alpha_filtering = 2;
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "Error! Unrecognized alpha filter: %s\n", argv[c]);
|
|
|
|
goto Error;
|
|
|
|
}
|
2011-12-01 15:11:34 +05:30
|
|
|
} else if (!strcmp(argv[c], "-noalpha")) {
|
|
|
|
keep_alpha = 0;
|
2012-04-24 10:59:08 +00:00
|
|
|
} else if (!strcmp(argv[c], "-lossless")) {
|
|
|
|
config.lossless = 1;
|
2012-07-18 21:58:53 +00:00
|
|
|
picture.use_argb = 1;
|
2012-06-22 12:14:48 +05:30
|
|
|
} else if (!strcmp(argv[c], "-hint") && c < argc - 1) {
|
|
|
|
++c;
|
|
|
|
if (!strcmp(argv[c], "photo")) {
|
|
|
|
config.image_hint = WEBP_HINT_PHOTO;
|
|
|
|
} else if (!strcmp(argv[c], "picture")) {
|
|
|
|
config.image_hint = WEBP_HINT_PICTURE;
|
2012-07-31 23:07:52 -07:00
|
|
|
} else if (!strcmp(argv[c], "graph")) {
|
|
|
|
config.image_hint = WEBP_HINT_GRAPH;
|
2012-06-22 12:14:48 +05:30
|
|
|
} else {
|
|
|
|
fprintf(stderr, "Error! Unrecognized image hint: %s\n", argv[c]);
|
|
|
|
goto Error;
|
|
|
|
}
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-size") && c < argc - 1) {
|
2011-03-26 09:27:45 -07:00
|
|
|
config.target_size = strtol(argv[++c], NULL, 0);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-psnr") && c < argc - 1) {
|
2011-06-28 16:02:56 +02:00
|
|
|
config.target_PSNR = (float)strtod(argv[++c], NULL);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-sns") && c < argc - 1) {
|
2011-03-26 09:27:45 -07:00
|
|
|
config.sns_strength = strtol(argv[++c], NULL, 0);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-f") && c < argc - 1) {
|
2011-03-26 09:27:45 -07:00
|
|
|
config.filter_strength = strtol(argv[++c], NULL, 0);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-af")) {
|
|
|
|
config.autofilter = 1;
|
2011-05-05 18:10:08 -07:00
|
|
|
} else if (!strcmp(argv[c], "-strong")) {
|
2011-02-18 23:33:46 -08:00
|
|
|
config.filter_type = 1;
|
|
|
|
} else if (!strcmp(argv[c], "-sharpness") && c < argc - 1) {
|
2011-03-26 09:27:45 -07:00
|
|
|
config.filter_sharpness = strtol(argv[++c], NULL, 0);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-pass") && c < argc - 1) {
|
2011-03-26 09:27:45 -07:00
|
|
|
config.pass = strtol(argv[++c], NULL, 0);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-pre") && c < argc - 1) {
|
2011-03-26 09:27:45 -07:00
|
|
|
config.preprocessing = strtol(argv[++c], NULL, 0);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-segments") && c < argc - 1) {
|
2011-03-26 09:27:45 -07:00
|
|
|
config.segments = strtol(argv[++c], NULL, 0);
|
2011-08-23 15:58:22 -07:00
|
|
|
} else if (!strcmp(argv[c], "-partition_limit") && c < argc - 1) {
|
|
|
|
config.partition_limit = strtol(argv[++c], NULL, 0);
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-map") && c < argc - 1) {
|
2011-03-26 09:27:45 -07:00
|
|
|
picture.extra_info_type = strtol(argv[++c], NULL, 0);
|
2011-05-02 17:19:00 -07:00
|
|
|
#ifdef WEBP_EXPERIMENTAL_FEATURES
|
|
|
|
} else if (!strcmp(argv[c], "-444")) {
|
|
|
|
picture.colorspace = WEBP_YUV444;
|
|
|
|
} else if (!strcmp(argv[c], "-422")) {
|
|
|
|
picture.colorspace = WEBP_YUV422;
|
|
|
|
} else if (!strcmp(argv[c], "-gray")) {
|
|
|
|
picture.colorspace = WEBP_YUV400;
|
|
|
|
#endif
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-crop") && c < argc - 4) {
|
|
|
|
crop = 1;
|
2011-03-26 09:27:45 -07:00
|
|
|
crop_x = strtol(argv[++c], NULL, 0);
|
|
|
|
crop_y = strtol(argv[++c], NULL, 0);
|
|
|
|
crop_w = strtol(argv[++c], NULL, 0);
|
|
|
|
crop_h = strtol(argv[++c], NULL, 0);
|
2011-05-02 17:19:00 -07:00
|
|
|
} else if (!strcmp(argv[c], "-resize") && c < argc - 2) {
|
|
|
|
resize_w = strtol(argv[++c], NULL, 0);
|
|
|
|
resize_h = strtol(argv[++c], NULL, 0);
|
2011-07-15 14:53:03 -07:00
|
|
|
#ifndef WEBP_DLL
|
2011-04-22 12:14:45 -07:00
|
|
|
} else if (!strcmp(argv[c], "-noasm")) {
|
2011-09-02 21:30:08 +00:00
|
|
|
VP8GetCPUInfo = NULL;
|
2011-07-15 14:53:03 -07:00
|
|
|
#endif
|
2011-03-24 16:17:10 -07:00
|
|
|
} else if (!strcmp(argv[c], "-version")) {
|
|
|
|
const int version = WebPGetEncoderVersion();
|
|
|
|
printf("%d.%d.%d\n",
|
|
|
|
(version >> 16) & 0xff, (version >> 8) & 0xff, version & 0xff);
|
|
|
|
return 0;
|
2011-12-01 02:24:50 -08:00
|
|
|
} else if (!strcmp(argv[c], "-progress")) {
|
|
|
|
show_progress = 1;
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-quiet")) {
|
|
|
|
quiet = 1;
|
|
|
|
} else if (!strcmp(argv[c], "-preset") && c < argc - 1) {
|
|
|
|
WebPPreset preset;
|
|
|
|
++c;
|
|
|
|
if (!strcmp(argv[c], "default")) {
|
|
|
|
preset = WEBP_PRESET_DEFAULT;
|
|
|
|
} else if (!strcmp(argv[c], "photo")) {
|
|
|
|
preset = WEBP_PRESET_PHOTO;
|
|
|
|
} else if (!strcmp(argv[c], "picture")) {
|
|
|
|
preset = WEBP_PRESET_PICTURE;
|
|
|
|
} else if (!strcmp(argv[c], "drawing")) {
|
|
|
|
preset = WEBP_PRESET_DRAWING;
|
|
|
|
} else if (!strcmp(argv[c], "icon")) {
|
|
|
|
preset = WEBP_PRESET_ICON;
|
|
|
|
} else if (!strcmp(argv[c], "text")) {
|
|
|
|
preset = WEBP_PRESET_TEXT;
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "Error! Unrecognized preset: %s\n", argv[c]);
|
|
|
|
goto Error;
|
|
|
|
}
|
|
|
|
if (!WebPConfigPreset(&config, preset, config.quality)) {
|
2011-03-17 14:49:19 -07:00
|
|
|
fprintf(stderr, "Error! Could initialize configuration with preset.\n");
|
2011-02-18 23:33:46 -08:00
|
|
|
goto Error;
|
|
|
|
}
|
2013-01-11 12:25:36 -08:00
|
|
|
} else if (!strcmp(argv[c], "-metadata") && c < argc - 1) {
|
|
|
|
static const struct {
|
|
|
|
const char* option;
|
|
|
|
int flag;
|
|
|
|
} kTokens[] = {
|
|
|
|
{ "all", METADATA_ALL },
|
|
|
|
{ "none", 0 },
|
|
|
|
{ "exif", METADATA_EXIF },
|
|
|
|
{ "iccp", METADATA_ICCP },
|
|
|
|
{ "xmp", METADATA_XMP },
|
|
|
|
};
|
|
|
|
const size_t kNumTokens = sizeof(kTokens) / sizeof(kTokens[0]);
|
|
|
|
const char* start = argv[++c];
|
|
|
|
const char* const end = start + strlen(start);
|
|
|
|
|
|
|
|
while (start < end) {
|
|
|
|
size_t i;
|
|
|
|
const char* token = strchr(start, ',');
|
|
|
|
if (token == NULL) token = end;
|
|
|
|
|
|
|
|
for (i = 0; i < kNumTokens; ++i) {
|
|
|
|
if ((size_t)(token - start) == strlen(kTokens[i].option) &&
|
|
|
|
!strncmp(start, kTokens[i].option, strlen(kTokens[i].option))) {
|
|
|
|
if (kTokens[i].flag != 0) {
|
|
|
|
keep_metadata |= kTokens[i].flag;
|
|
|
|
} else {
|
|
|
|
keep_metadata = 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (i == kNumTokens) {
|
|
|
|
fprintf(stderr, "Error! Unknown metadata type '%.*s'\n",
|
|
|
|
(int)(token - start), start);
|
|
|
|
HelpLong();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
start = token + 1;
|
|
|
|
}
|
2013-01-14 18:32:44 -08:00
|
|
|
#ifdef HAVE_WINCODEC_H
|
2013-01-11 12:25:36 -08:00
|
|
|
if (keep_metadata != 0) {
|
|
|
|
// TODO(jzern): remove when -metadata is supported on all platforms.
|
|
|
|
fprintf(stderr, "Warning: -metadata is currently unsupported on this"
|
|
|
|
" platform. Ignoring this option!\n");
|
|
|
|
}
|
2013-01-14 18:32:44 -08:00
|
|
|
#endif
|
2011-02-18 23:33:46 -08:00
|
|
|
} else if (!strcmp(argv[c], "-v")) {
|
|
|
|
verbose = 1;
|
|
|
|
} else if (argv[c][0] == '-') {
|
|
|
|
fprintf(stderr, "Error! Unknown option '%s'\n", argv[c]);
|
|
|
|
HelpLong();
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
in_file = argv[c];
|
|
|
|
}
|
|
|
|
}
|
2012-05-09 00:32:20 -07:00
|
|
|
if (in_file == NULL) {
|
|
|
|
fprintf(stderr, "No input file specified!\n");
|
|
|
|
HelpShort();
|
|
|
|
goto Error;
|
|
|
|
}
|
2011-02-18 23:33:46 -08:00
|
|
|
|
2012-06-20 09:02:25 +05:30
|
|
|
// Check for unsupported command line options for lossless mode and log
|
|
|
|
// warning for such options.
|
2012-06-20 09:20:34 +00:00
|
|
|
if (!quiet && config.lossless == 1) {
|
2012-06-20 09:02:25 +05:30
|
|
|
if (config.target_size > 0 || config.target_PSNR > 0) {
|
|
|
|
fprintf(stderr, "Encoding for specified size or PSNR is not supported"
|
|
|
|
" for lossless encoding. Ignoring such option(s)!\n");
|
|
|
|
}
|
|
|
|
if (config.partition_limit > 0) {
|
|
|
|
fprintf(stderr, "Partition limit option is not required for lossless"
|
|
|
|
" encoding. Ignoring this option!\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-18 23:33:46 -08:00
|
|
|
if (!WebPValidateConfig(&config)) {
|
|
|
|
fprintf(stderr, "Error! Invalid configuration.\n");
|
|
|
|
goto Error;
|
|
|
|
}
|
|
|
|
|
2011-02-25 12:03:27 -08:00
|
|
|
// Read the input
|
2011-05-04 22:59:51 -07:00
|
|
|
if (verbose) {
|
2011-02-25 12:03:27 -08:00
|
|
|
StopwatchReadAndReset(&stop_watch);
|
2011-05-04 22:59:51 -07:00
|
|
|
}
|
2013-01-11 12:25:36 -08:00
|
|
|
if (!ReadPicture(in_file, &picture, keep_alpha,
|
|
|
|
(keep_metadata == 0) ? NULL : &metadata)) {
|
2012-01-20 07:20:56 -08:00
|
|
|
fprintf(stderr, "Error! Cannot read input picture file '%s'\n", in_file);
|
2011-03-17 14:49:19 -07:00
|
|
|
goto Error;
|
|
|
|
}
|
2011-12-01 02:24:50 -08:00
|
|
|
picture.progress_hook = (show_progress && !quiet) ? ProgressReport : NULL;
|
|
|
|
|
2011-02-25 12:03:27 -08:00
|
|
|
if (verbose) {
|
2013-01-25 00:44:16 -08:00
|
|
|
const double read_time = StopwatchReadAndReset(&stop_watch);
|
|
|
|
fprintf(stderr, "Time to read input: %.3fs\n", read_time);
|
2011-02-25 12:03:27 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Open the output
|
2011-02-18 23:33:46 -08:00
|
|
|
if (out_file) {
|
|
|
|
out = fopen(out_file, "wb");
|
2012-01-20 07:20:56 -08:00
|
|
|
if (out == NULL) {
|
2011-02-18 23:33:46 -08:00
|
|
|
fprintf(stderr, "Error! Cannot open output file '%s'\n", out_file);
|
|
|
|
goto Error;
|
|
|
|
} else {
|
|
|
|
if (!short_output && !quiet) {
|
|
|
|
fprintf(stderr, "Saving file '%s'\n", out_file);
|
|
|
|
}
|
|
|
|
}
|
2013-01-14 18:32:44 -08:00
|
|
|
if (keep_metadata == 0) {
|
|
|
|
picture.writer = MyWriter;
|
|
|
|
picture.custom_ptr = (void*)out;
|
|
|
|
} else {
|
|
|
|
picture.writer = WebPMemoryWrite;
|
|
|
|
picture.custom_ptr = (void*)&memory_writer;
|
|
|
|
}
|
2011-02-18 23:33:46 -08:00
|
|
|
} else {
|
|
|
|
out = NULL;
|
|
|
|
if (!quiet && !short_output) {
|
|
|
|
fprintf(stderr, "No output file specified (no -o flag). Encoding will\n");
|
|
|
|
fprintf(stderr, "be performed, but its results discarded.\n\n");
|
|
|
|
}
|
|
|
|
}
|
2012-07-24 16:15:36 -07:00
|
|
|
if (!quiet) {
|
|
|
|
picture.stats = &stats;
|
2012-07-27 19:53:16 -07:00
|
|
|
picture.user_data = (void*)in_file;
|
2012-07-24 16:15:36 -07:00
|
|
|
}
|
2011-02-18 23:33:46 -08:00
|
|
|
|
2011-02-25 12:03:27 -08:00
|
|
|
// Compress
|
2011-05-04 22:59:51 -07:00
|
|
|
if (verbose) {
|
2011-02-18 23:33:46 -08:00
|
|
|
StopwatchReadAndReset(&stop_watch);
|
2011-05-04 22:59:51 -07:00
|
|
|
}
|
2012-06-21 00:30:43 -07:00
|
|
|
if (crop != 0) {
|
|
|
|
// We use self-cropping using a view.
|
|
|
|
if (!WebPPictureView(&picture, crop_x, crop_y, crop_w, crop_h, &picture)) {
|
|
|
|
fprintf(stderr, "Error! Cannot crop picture\n");
|
|
|
|
goto Error;
|
|
|
|
}
|
2011-03-17 14:49:19 -07:00
|
|
|
}
|
2011-05-02 17:19:00 -07:00
|
|
|
if ((resize_w | resize_h) > 0) {
|
|
|
|
if (!WebPPictureRescale(&picture, resize_w, resize_h)) {
|
|
|
|
fprintf(stderr, "Error! Cannot resize picture\n");
|
|
|
|
goto Error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (picture.extra_info_type > 0) {
|
|
|
|
AllocExtraInfo(&picture);
|
|
|
|
}
|
2012-10-18 08:26:40 -07:00
|
|
|
if (print_distortion >= 0) { // Save original picture for later comparison
|
2012-01-20 07:20:56 -08:00
|
|
|
WebPPictureCopy(&picture, &original_picture);
|
|
|
|
}
|
2011-03-17 14:49:19 -07:00
|
|
|
if (!WebPEncode(&config, &picture)) {
|
|
|
|
fprintf(stderr, "Error! Cannot encode picture as WebP\n");
|
2011-06-02 06:55:03 -07:00
|
|
|
fprintf(stderr, "Error code: %d (%s)\n",
|
|
|
|
picture.error_code, kErrorMessages[picture.error_code]);
|
2011-03-17 14:49:19 -07:00
|
|
|
goto Error;
|
|
|
|
}
|
2011-02-18 23:33:46 -08:00
|
|
|
if (verbose) {
|
2013-01-25 00:44:16 -08:00
|
|
|
const double encode_time = StopwatchReadAndReset(&stop_watch);
|
|
|
|
fprintf(stderr, "Time to encode picture: %.3fs\n", encode_time);
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
2011-02-25 12:03:27 -08:00
|
|
|
|
|
|
|
// Write info
|
2011-05-04 22:59:51 -07:00
|
|
|
if (dump_file) {
|
2012-07-18 21:58:53 +00:00
|
|
|
if (picture.use_argb) {
|
2012-06-04 15:50:05 -07:00
|
|
|
fprintf(stderr, "Warning: can't dump file (-d option) in lossless mode.");
|
|
|
|
} else if (!DumpPicture(&picture, dump_file)) {
|
|
|
|
fprintf(stderr, "Warning, couldn't dump picture %s\n", dump_file);
|
|
|
|
}
|
2011-05-04 22:59:51 -07:00
|
|
|
}
|
2012-01-20 07:20:56 -08:00
|
|
|
|
2013-01-14 18:32:44 -08:00
|
|
|
if (keep_metadata != 0 && out != NULL) {
|
|
|
|
if (!WriteWebPWithMetadata(out, &picture, &memory_writer,
|
|
|
|
&metadata, keep_metadata)) {
|
|
|
|
fprintf(stderr, "Error writing WebP file with metadata!\n");
|
|
|
|
goto Error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-04 22:59:51 -07:00
|
|
|
if (!quiet) {
|
2012-05-09 11:27:46 +05:30
|
|
|
if (config.lossless) {
|
|
|
|
PrintExtraInfoLossless(&picture, short_output, in_file);
|
|
|
|
} else {
|
|
|
|
PrintExtraInfoLossy(&picture, short_output, in_file);
|
|
|
|
}
|
2012-01-20 07:20:56 -08:00
|
|
|
}
|
2012-10-18 08:26:40 -07:00
|
|
|
if (!quiet && !short_output && print_distortion >= 0) { // print distortion
|
|
|
|
static const char* distortion_names[] = { "PSNR", "SSIM", "LSIM" };
|
2012-01-20 07:20:56 -08:00
|
|
|
float values[5];
|
|
|
|
WebPPictureDistortion(&picture, &original_picture,
|
2012-10-18 08:26:40 -07:00
|
|
|
print_distortion, values);
|
2012-01-20 07:20:56 -08:00
|
|
|
fprintf(stderr, "%s: Y:%.2f U:%.2f V:%.2f A:%.2f Total:%.2f\n",
|
2012-10-18 08:26:40 -07:00
|
|
|
distortion_names[print_distortion],
|
2012-01-20 07:20:56 -08:00
|
|
|
values[0], values[1], values[2], values[3], values[4]);
|
2011-05-04 22:59:51 -07:00
|
|
|
}
|
2012-05-09 00:32:20 -07:00
|
|
|
return_value = 0;
|
2011-02-18 23:33:46 -08:00
|
|
|
|
|
|
|
Error:
|
2013-02-01 19:17:26 -08:00
|
|
|
free(memory_writer.mem);
|
2011-02-18 23:33:46 -08:00
|
|
|
free(picture.extra_info);
|
2012-12-03 18:20:00 -08:00
|
|
|
MetadataFree(&metadata);
|
2011-02-18 23:33:46 -08:00
|
|
|
WebPPictureFree(&picture);
|
2012-01-20 07:20:56 -08:00
|
|
|
WebPPictureFree(&original_picture);
|
2011-02-18 23:33:46 -08:00
|
|
|
if (out != NULL) {
|
|
|
|
fclose(out);
|
|
|
|
}
|
|
|
|
|
2012-05-09 00:32:20 -07:00
|
|
|
return return_value;
|
2011-02-18 23:33:46 -08:00
|
|
|
}
|
|
|
|
|
2011-08-25 14:22:32 -07:00
|
|
|
//------------------------------------------------------------------------------
|