2014-10-30 12:59:37 +00:00
|
|
|
/* Copyright 2014 Google Inc. All Rights Reserved.
|
|
|
|
|
2015-12-11 10:11:51 +00:00
|
|
|
Distributed under MIT license.
|
2015-11-27 10:27:11 +00:00
|
|
|
See file LICENSE for detail or copy at https://opensource.org/licenses/MIT
|
2014-10-30 12:59:37 +00:00
|
|
|
*/
|
|
|
|
|
2015-11-27 10:27:11 +00:00
|
|
|
/* Example main() function for Brotli library. */
|
|
|
|
|
2014-10-30 12:59:37 +00:00
|
|
|
#include <fcntl.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
2016-04-19 14:29:10 +00:00
|
|
|
#include <cstdlib>
|
|
|
|
#include <cstring>
|
2015-11-25 22:07:48 +00:00
|
|
|
#include <ctime>
|
2015-10-02 11:12:20 +00:00
|
|
|
#include <string>
|
2016-06-13 09:01:04 +00:00
|
|
|
#include <vector>
|
2015-10-02 11:12:20 +00:00
|
|
|
|
2014-10-30 12:59:37 +00:00
|
|
|
#include "../dec/decode.h"
|
2016-06-13 09:01:04 +00:00
|
|
|
#include "../enc/encode.h"
|
2016-04-19 14:29:10 +00:00
|
|
|
|
|
|
|
#if !defined(_WIN32)
|
|
|
|
#include <unistd.h>
|
|
|
|
#else
|
|
|
|
#include <io.h>
|
|
|
|
|
|
|
|
#define STDIN_FILENO _fileno(stdin)
|
|
|
|
#define STDOUT_FILENO _fileno(stdout)
|
|
|
|
#define S_IRUSR S_IREAD
|
|
|
|
#define S_IWUSR S_IWRITE
|
|
|
|
#define fdopen _fdopen
|
|
|
|
#define unlink _unlink
|
|
|
|
|
|
|
|
#define fopen ms_fopen
|
|
|
|
#define open ms_open
|
|
|
|
|
2016-04-29 13:37:52 +00:00
|
|
|
#if defined(_MSC_VER) && (_MSC_VER >= 1400)
|
|
|
|
#define fseek _fseeki64
|
|
|
|
#define ftell _ftelli64
|
|
|
|
#endif
|
|
|
|
|
2016-04-19 14:29:10 +00:00
|
|
|
static inline FILE* ms_fopen(const char *filename, const char *mode) {
|
|
|
|
FILE* result = 0;
|
|
|
|
fopen_s(&result, filename, mode);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int ms_open(const char *filename, int oflag, int pmode) {
|
|
|
|
int result = -1;
|
|
|
|
_sopen_s(&result, filename, oflag | O_BINARY, _SH_DENYNO, pmode);
|
|
|
|
return result;
|
|
|
|
}
|
2016-04-29 13:37:52 +00:00
|
|
|
#endif /* WIN32 */
|
2014-10-30 12:59:37 +00:00
|
|
|
|
2015-05-11 12:14:05 +00:00
|
|
|
static bool ParseQuality(const char* s, int* quality) {
|
|
|
|
if (s[0] >= '0' && s[0] <= '9') {
|
|
|
|
*quality = s[0] - '0';
|
|
|
|
if (s[1] >= '0' && s[1] <= '9') {
|
|
|
|
*quality = *quality * 10 + s[1] - '0';
|
|
|
|
return s[2] == 0;
|
|
|
|
}
|
|
|
|
return s[1] == 0;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-10-30 12:59:37 +00:00
|
|
|
static void ParseArgv(int argc, char **argv,
|
|
|
|
char **input_path,
|
|
|
|
char **output_path,
|
2016-06-13 09:01:04 +00:00
|
|
|
char **dictionary_path,
|
2014-10-30 12:59:37 +00:00
|
|
|
int *force,
|
2015-05-11 12:14:05 +00:00
|
|
|
int *quality,
|
2015-09-01 11:44:17 +00:00
|
|
|
int *decompress,
|
|
|
|
int *repeat,
|
2015-10-02 11:12:20 +00:00
|
|
|
int *verbose,
|
|
|
|
int *lgwin) {
|
2014-10-30 12:59:37 +00:00
|
|
|
*force = 0;
|
|
|
|
*input_path = 0;
|
|
|
|
*output_path = 0;
|
2015-09-01 11:44:17 +00:00
|
|
|
*repeat = 1;
|
|
|
|
*verbose = 0;
|
2015-10-02 11:12:20 +00:00
|
|
|
*lgwin = 22;
|
2014-10-30 12:59:37 +00:00
|
|
|
{
|
|
|
|
size_t argv0_len = strlen(argv[0]);
|
|
|
|
*decompress =
|
|
|
|
argv0_len >= 5 && strcmp(&argv[0][argv0_len - 5], "unbro") == 0;
|
|
|
|
}
|
|
|
|
for (int k = 1; k < argc; ++k) {
|
|
|
|
if (!strcmp("--force", argv[k]) ||
|
|
|
|
!strcmp("-f", argv[k])) {
|
|
|
|
if (*force != 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
*force = 1;
|
|
|
|
continue;
|
|
|
|
} else if (!strcmp("--decompress", argv[k]) ||
|
|
|
|
!strcmp("--uncompress", argv[k]) ||
|
|
|
|
!strcmp("-d", argv[k])) {
|
|
|
|
*decompress = 1;
|
|
|
|
continue;
|
2015-09-01 11:44:17 +00:00
|
|
|
} else if (!strcmp("--verbose", argv[k]) ||
|
|
|
|
!strcmp("-v", argv[k])) {
|
|
|
|
if (*verbose != 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
*verbose = 1;
|
|
|
|
continue;
|
2014-10-30 12:59:37 +00:00
|
|
|
}
|
|
|
|
if (k < argc - 1) {
|
|
|
|
if (!strcmp("--input", argv[k]) ||
|
2014-11-17 14:31:00 +00:00
|
|
|
!strcmp("--in", argv[k]) ||
|
2014-10-30 12:59:37 +00:00
|
|
|
!strcmp("-i", argv[k])) {
|
|
|
|
if (*input_path != 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
*input_path = argv[k + 1];
|
|
|
|
++k;
|
|
|
|
continue;
|
|
|
|
} else if (!strcmp("--output", argv[k]) ||
|
2014-11-17 14:31:00 +00:00
|
|
|
!strcmp("--out", argv[k]) ||
|
2014-10-30 12:59:37 +00:00
|
|
|
!strcmp("-o", argv[k])) {
|
|
|
|
if (*output_path != 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
*output_path = argv[k + 1];
|
|
|
|
++k;
|
|
|
|
continue;
|
2016-06-13 09:01:04 +00:00
|
|
|
} else if (!strcmp("--custom-dictionary", argv[k])) {
|
|
|
|
if (*dictionary_path != 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
*dictionary_path = argv[k + 1];
|
|
|
|
++k;
|
|
|
|
continue;
|
2015-05-11 12:14:05 +00:00
|
|
|
} else if (!strcmp("--quality", argv[k]) ||
|
|
|
|
!strcmp("-q", argv[k])) {
|
|
|
|
if (!ParseQuality(argv[k + 1], quality)) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
++k;
|
|
|
|
continue;
|
2015-09-01 11:44:17 +00:00
|
|
|
} else if (!strcmp("--repeat", argv[k]) ||
|
|
|
|
!strcmp("-r", argv[k])) {
|
|
|
|
if (!ParseQuality(argv[k + 1], repeat)) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
++k;
|
|
|
|
continue;
|
2015-10-02 11:12:20 +00:00
|
|
|
} else if (!strcmp("--window", argv[k]) ||
|
|
|
|
!strcmp("-w", argv[k])) {
|
|
|
|
if (!ParseQuality(argv[k + 1], lgwin)) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
if (*lgwin < 10 || *lgwin >= 25) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
++k;
|
|
|
|
continue;
|
2014-10-30 12:59:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
error:
|
|
|
|
fprintf(stderr,
|
2015-05-11 12:14:05 +00:00
|
|
|
"Usage: %s [--force] [--quality n] [--decompress]"
|
2015-09-01 11:44:17 +00:00
|
|
|
" [--input filename] [--output filename] [--repeat iters]"
|
2016-06-13 09:01:04 +00:00
|
|
|
" [--verbose] [--window n] [--custom-dictionary filename]\n",
|
2014-10-30 12:59:37 +00:00
|
|
|
argv[0]);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2014-11-17 14:31:00 +00:00
|
|
|
static FILE* OpenInputFile(const char* input_path) {
|
|
|
|
if (input_path == 0) {
|
|
|
|
return fdopen(STDIN_FILENO, "rb");
|
|
|
|
}
|
|
|
|
FILE* f = fopen(input_path, "rb");
|
|
|
|
if (f == 0) {
|
|
|
|
perror("fopen");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
2014-10-30 12:59:37 +00:00
|
|
|
static FILE *OpenOutputFile(const char *output_path, const int force) {
|
2014-11-17 14:31:00 +00:00
|
|
|
if (output_path == 0) {
|
|
|
|
return fdopen(STDOUT_FILENO, "wb");
|
|
|
|
}
|
2015-10-02 11:12:20 +00:00
|
|
|
int excl = force ? 0 : O_EXCL;
|
|
|
|
int fd = open(output_path, O_CREAT | excl | O_WRONLY | O_TRUNC,
|
2014-10-30 12:59:37 +00:00
|
|
|
S_IRUSR | S_IWUSR);
|
|
|
|
if (fd < 0) {
|
2015-10-02 11:12:20 +00:00
|
|
|
if (!force) {
|
|
|
|
struct stat statbuf;
|
|
|
|
if (stat(output_path, &statbuf) == 0) {
|
|
|
|
fprintf(stderr, "output file exists\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
2014-10-30 12:59:37 +00:00
|
|
|
perror("open");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
return fdopen(fd, "wb");
|
|
|
|
}
|
|
|
|
|
2016-06-13 09:01:04 +00:00
|
|
|
static int64_t FileSize(const char *path) {
|
2015-09-01 11:44:17 +00:00
|
|
|
FILE *f = fopen(path, "rb");
|
|
|
|
if (f == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
2015-10-06 09:23:44 +00:00
|
|
|
if (fseek(f, 0L, SEEK_END) != 0) {
|
|
|
|
fclose(f);
|
|
|
|
return -1;
|
|
|
|
}
|
2015-09-01 11:44:17 +00:00
|
|
|
int64_t retval = ftell(f);
|
2015-10-23 10:05:43 +00:00
|
|
|
if (fclose(f) != 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
2015-09-01 11:44:17 +00:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2016-06-13 09:01:04 +00:00
|
|
|
static std::vector<uint8_t> ReadDictionary(const char* path) {
|
|
|
|
FILE *f = fopen(path, "rb");
|
|
|
|
if (f == NULL) {
|
|
|
|
perror("fopen");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t file_size = FileSize(path);
|
|
|
|
if (file_size == -1) {
|
|
|
|
fprintf(stderr, "could not get size of dictionary file");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const int kMaxDictionarySize = (1 << 24) - 16;
|
|
|
|
if (file_size > kMaxDictionarySize) {
|
|
|
|
fprintf(stderr, "dictionary is larger than maximum allowed: %d\n",
|
|
|
|
kMaxDictionarySize);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<uint8_t> buffer;
|
|
|
|
buffer.resize(static_cast<size_t>(file_size));
|
|
|
|
size_t bytes_read = fread(buffer.data(), sizeof(uint8_t), buffer.size(), f);
|
|
|
|
if (bytes_read != buffer.size()) {
|
|
|
|
fprintf(stderr, "could not read dictionary\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
2015-11-25 22:07:48 +00:00
|
|
|
static const size_t kFileBufferSize = 65536;
|
|
|
|
|
2016-06-13 09:01:04 +00:00
|
|
|
static int Decompress(FILE* fin, FILE* fout, const char* dictionary_path) {
|
|
|
|
/* Dictionary should be kept during first rounds of decompression. */
|
|
|
|
std::vector<uint8_t> dictionary;
|
2016-04-19 14:29:10 +00:00
|
|
|
BrotliState* s = BrotliCreateState(NULL, NULL, NULL);
|
|
|
|
if (!s) {
|
|
|
|
fprintf(stderr, "out of memory\n");
|
2016-06-13 09:01:04 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (dictionary_path != NULL) {
|
|
|
|
dictionary = ReadDictionary(dictionary_path);
|
|
|
|
BrotliSetCustomDictionary(dictionary.size(), dictionary.data(), s);
|
2016-04-19 14:29:10 +00:00
|
|
|
}
|
2015-11-25 22:07:48 +00:00
|
|
|
uint8_t* input = new uint8_t[kFileBufferSize];
|
|
|
|
uint8_t* output = new uint8_t[kFileBufferSize];
|
|
|
|
size_t total_out;
|
|
|
|
size_t available_in;
|
|
|
|
const uint8_t* next_in;
|
|
|
|
size_t available_out = kFileBufferSize;
|
|
|
|
uint8_t* next_out = output;
|
|
|
|
BrotliResult result = BROTLI_RESULT_NEEDS_MORE_INPUT;
|
|
|
|
while (1) {
|
|
|
|
if (result == BROTLI_RESULT_NEEDS_MORE_INPUT) {
|
|
|
|
if (feof(fin)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
available_in = fread(input, 1, kFileBufferSize, fin);
|
|
|
|
next_in = input;
|
|
|
|
if (ferror(fin)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (result == BROTLI_RESULT_NEEDS_MORE_OUTPUT) {
|
|
|
|
fwrite(output, 1, kFileBufferSize, fout);
|
|
|
|
if (ferror(fout)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
available_out = kFileBufferSize;
|
|
|
|
next_out = output;
|
|
|
|
} else {
|
2016-01-05 21:31:36 +00:00
|
|
|
break; /* Error or success. */
|
2015-11-25 22:07:48 +00:00
|
|
|
}
|
|
|
|
result = BrotliDecompressStream(&available_in, &next_in,
|
2016-04-19 14:29:10 +00:00
|
|
|
&available_out, &next_out, &total_out, s);
|
2015-11-25 22:07:48 +00:00
|
|
|
}
|
|
|
|
if (next_out != output) {
|
2016-04-19 14:29:10 +00:00
|
|
|
fwrite(output, 1, static_cast<size_t>(next_out - output), fout);
|
2015-11-25 22:07:48 +00:00
|
|
|
}
|
|
|
|
delete[] input;
|
|
|
|
delete[] output;
|
2016-04-19 14:29:10 +00:00
|
|
|
BrotliDestroyState(s);
|
2015-11-25 22:07:48 +00:00
|
|
|
if ((result == BROTLI_RESULT_NEEDS_MORE_OUTPUT) || ferror(fout)) {
|
|
|
|
fprintf(stderr, "failed to write output\n");
|
2016-06-13 09:01:04 +00:00
|
|
|
return 0;
|
2015-11-25 22:07:48 +00:00
|
|
|
} else if (result != BROTLI_RESULT_SUCCESS) { /* Error or needs more input. */
|
|
|
|
fprintf(stderr, "corrupt input\n");
|
2016-06-13 09:01:04 +00:00
|
|
|
return 0;
|
2015-11-25 22:07:48 +00:00
|
|
|
}
|
2016-06-13 09:01:04 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int Compress(int quality, int lgwin, FILE* fin, FILE* fout,
|
|
|
|
const char *dictionary_path) {
|
|
|
|
BrotliEncoderState* s = BrotliEncoderCreateInstance(0, 0, 0);
|
|
|
|
uint8_t* buffer = reinterpret_cast<uint8_t*>(malloc(kFileBufferSize << 1));
|
|
|
|
uint8_t* input = buffer;
|
|
|
|
uint8_t* output = buffer + kFileBufferSize;
|
|
|
|
size_t available_in = 0;
|
|
|
|
const uint8_t* next_in = NULL;
|
|
|
|
size_t available_out = kFileBufferSize;
|
|
|
|
uint8_t* next_out = output;
|
|
|
|
int is_eof = 0;
|
|
|
|
int is_ok = 1;
|
|
|
|
|
|
|
|
if (!s || !buffer) {
|
|
|
|
is_ok = 0;
|
|
|
|
goto finish;
|
|
|
|
}
|
|
|
|
|
|
|
|
BrotliEncoderSetParameter(s, BROTLI_PARAM_QUALITY, (uint32_t)quality);
|
|
|
|
BrotliEncoderSetParameter(s, BROTLI_PARAM_LGWIN, (uint32_t)lgwin);
|
|
|
|
if (dictionary_path != NULL) {
|
|
|
|
std::vector<uint8_t> dictionary = ReadDictionary(dictionary_path);
|
|
|
|
BrotliEncoderSetCustomDictionary(s, dictionary.size(),
|
|
|
|
reinterpret_cast<const uint8_t*>(dictionary.data()));
|
|
|
|
}
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
if (available_in == 0 && !is_eof) {
|
|
|
|
available_in = fread(input, 1, kFileBufferSize, fin);
|
|
|
|
next_in = input;
|
|
|
|
if (ferror(fin)) break;
|
|
|
|
is_eof = feof(fin);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!BrotliEncoderCompressStream(s,
|
|
|
|
is_eof ? BROTLI_OPERATION_FINISH : BROTLI_OPERATION_PROCESS,
|
|
|
|
&available_in, &next_in, &available_out, &next_out, NULL)) {
|
|
|
|
is_ok = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (available_out != kFileBufferSize) {
|
|
|
|
size_t out_size = kFileBufferSize - available_out;
|
|
|
|
fwrite(output, 1, out_size, fout);
|
|
|
|
if (ferror(fout)) break;
|
|
|
|
available_out = kFileBufferSize;
|
|
|
|
next_out = output;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (BrotliEncoderIsFinished(s)) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
finish:
|
|
|
|
free(buffer);
|
|
|
|
BrotliEncoderDestroyInstance(s);
|
|
|
|
|
|
|
|
if (!is_ok) {
|
|
|
|
/* Should detect OOM? */
|
|
|
|
fprintf(stderr, "failed to compress data\n");
|
|
|
|
return 0;
|
|
|
|
} else if (ferror(fout)) {
|
|
|
|
fprintf(stderr, "failed to write output\n");
|
|
|
|
return 0;
|
|
|
|
} else if (ferror(fin)) {
|
|
|
|
fprintf(stderr, "failed to read input\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
2015-11-25 22:07:48 +00:00
|
|
|
}
|
|
|
|
|
2014-10-30 12:59:37 +00:00
|
|
|
int main(int argc, char** argv) {
|
|
|
|
char *input_path = 0;
|
|
|
|
char *output_path = 0;
|
2016-06-13 09:01:04 +00:00
|
|
|
char *dictionary_path = 0;
|
2014-10-30 12:59:37 +00:00
|
|
|
int force = 0;
|
2015-05-11 12:14:05 +00:00
|
|
|
int quality = 11;
|
2014-10-30 12:59:37 +00:00
|
|
|
int decompress = 0;
|
2015-09-01 11:44:17 +00:00
|
|
|
int repeat = 1;
|
|
|
|
int verbose = 0;
|
2015-10-02 11:12:20 +00:00
|
|
|
int lgwin = 0;
|
2016-06-13 09:01:04 +00:00
|
|
|
ParseArgv(argc, argv, &input_path, &output_path, &dictionary_path, &force,
|
2015-10-02 11:12:20 +00:00
|
|
|
&quality, &decompress, &repeat, &verbose, &lgwin);
|
2015-09-01 11:44:17 +00:00
|
|
|
const clock_t clock_start = clock();
|
|
|
|
for (int i = 0; i < repeat; ++i) {
|
|
|
|
FILE* fin = OpenInputFile(input_path);
|
|
|
|
FILE* fout = OpenOutputFile(output_path, force);
|
2016-06-13 09:01:04 +00:00
|
|
|
int is_ok = false;
|
2015-09-01 11:44:17 +00:00
|
|
|
if (decompress) {
|
2016-06-13 09:01:04 +00:00
|
|
|
is_ok = Decompress(fin, fout, dictionary_path);
|
2015-09-01 11:44:17 +00:00
|
|
|
} else {
|
2016-06-13 09:01:04 +00:00
|
|
|
is_ok = Compress(quality, lgwin, fin, fout, dictionary_path);
|
|
|
|
}
|
|
|
|
if (!is_ok) {
|
|
|
|
unlink(output_path);
|
|
|
|
exit(1);
|
2015-09-01 11:44:17 +00:00
|
|
|
}
|
|
|
|
if (fclose(fin) != 0) {
|
|
|
|
perror("fclose");
|
2014-10-30 12:59:37 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
2015-09-01 11:44:17 +00:00
|
|
|
if (fclose(fout) != 0) {
|
|
|
|
perror("fclose");
|
2015-04-23 13:26:08 +00:00
|
|
|
exit(1);
|
2014-10-30 12:59:37 +00:00
|
|
|
}
|
|
|
|
}
|
2015-09-01 11:44:17 +00:00
|
|
|
if (verbose) {
|
|
|
|
const clock_t clock_end = clock();
|
|
|
|
double duration =
|
|
|
|
static_cast<double>(clock_end - clock_start) / CLOCKS_PER_SEC;
|
|
|
|
if (duration < 1e-9) {
|
|
|
|
duration = 1e-9;
|
|
|
|
}
|
2015-10-23 10:05:43 +00:00
|
|
|
int64_t uncompressed_size = FileSize(decompress ? output_path : input_path);
|
|
|
|
if (uncompressed_size == -1) {
|
|
|
|
fprintf(stderr, "failed to determine uncompressed file size\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
double uncompressed_bytes_in_MB =
|
2016-04-19 14:29:10 +00:00
|
|
|
static_cast<double>(repeat * uncompressed_size) / (1024.0 * 1024.0);
|
2015-09-01 11:44:17 +00:00
|
|
|
if (decompress) {
|
|
|
|
printf("Brotli decompression speed: ");
|
|
|
|
} else {
|
|
|
|
printf("Brotli compression speed: ");
|
|
|
|
}
|
|
|
|
printf("%g MB/s\n", uncompressed_bytes_in_MB / duration);
|
2014-10-30 12:59:37 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|