uvg266/src/encmain.c

287 lines
8.4 KiB
C
Raw Normal View History

/*****************************************************************************
* This file is part of Kvazaar HEVC encoder.
2014-02-21 13:00:20 +00:00
*
* Copyright (C) 2013-2015 Tampere University of Technology and others (see
* COPYING file).
*
* Kvazaar is free software: you can redistribute it and/or modify it under
* the terms of the GNU Lesser General Public License as published by the
* Free Software Foundation; either version 2.1 of the License, or (at your
* option) any later version.
*
* Kvazaar is distributed in the hope that it will be useful, but WITHOUT ANY
* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along
* with Kvazaar. If not, see <http://www.gnu.org/licenses/>.
****************************************************************************/
/*
* \file
2013-09-18 14:29:30 +00:00
*
*/
#ifdef _WIN32
/* The following two defines must be located before the inclusion of any system header files. */
#define WINVER 0x0500
#define _WIN32_WINNT 0x0500
#include <io.h> /* _setmode() */
#include <fcntl.h> /* _O_BINARY */
#endif
#include "kvazaar_internal.h"
2013-09-19 09:48:25 +00:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
2014-04-11 09:42:37 +00:00
#include <time.h>
2014-06-11 05:48:36 +00:00
#include "checkpoint.h"
2013-09-19 09:48:25 +00:00
#include "global.h"
#include "config.h"
#include "threadqueue.h"
2013-09-19 09:48:25 +00:00
#include "encoder.h"
#include "encoderstate.h"
#include "cli.h"
2015-06-17 12:24:07 +00:00
#include "yuv_io.h"
2014-02-21 13:00:20 +00:00
/**
* \brief Open a file for reading.
*
* If the file is "-", stdin is used.
*
* \param filename name of the file to open or "-"
* \return the opened file or NULL if opening fails
*/
static FILE* open_input_file(const char* filename)
{
if (!strcmp(filename, "-")) return stdin;
return fopen(filename, "rb");
}
/**
* \brief Open a file for writing.
*
* If the file is "-", stdout is used.
*
* \param filename name of the file to open or "-"
* \return the opened file or NULL if opening fails
*/
static FILE* open_output_file(const char* filename)
{
if (!strcmp(filename, "-")) return stdout;
return fopen(filename, "wb");
}
2013-09-19 09:48:25 +00:00
/**
* \brief Program main function.
* \param argc Argument count from commandline
* \param argv Argument list
* \return Program exit state
*/
2013-09-19 09:48:25 +00:00
int main(int argc, char *argv[])
2014-02-21 13:00:20 +00:00
{
int retval = EXIT_SUCCESS;
2015-06-30 08:03:26 +00:00
kvz_config *cfg = NULL; //!< Global configuration
kvz_encoder* enc = NULL;
2013-09-19 09:48:25 +00:00
FILE *input = NULL; //!< input file (YUV)
FILE *output = NULL; //!< output file (HEVC NAL stream)
FILE *recout = NULL; //!< reconstructed YUV output, --debug
2014-04-11 09:42:37 +00:00
clock_t start_time = clock();
2014-06-16 08:18:11 +00:00
clock_t encoding_start_cpu_time;
CLOCK_T encoding_start_real_time;
clock_t encoding_end_cpu_time;
CLOCK_T encoding_end_real_time;
// Stdin and stdout need to be binary for input and output to work.
// Stderr needs to be text mode to convert \n to \r\n in Windows.
#ifdef _WIN32
_setmode( _fileno( stdin ), _O_BINARY );
_setmode( _fileno( stdout ), _O_BINARY );
_setmode( _fileno( stderr ), _O_TEXT );
#endif
2014-06-11 05:48:36 +00:00
CHECKPOINTS_INIT();
const kvz_api * const api = kvz_api_get(8);
2013-09-19 09:48:25 +00:00
// Handle configuration
cfg = api->config_alloc();
2014-02-21 13:00:20 +00:00
2013-09-19 09:48:25 +00:00
// If problem with configuration, print banner and shutdown
if (!cfg || !api->config_init(cfg) || !config_read(cfg,argc,argv)) {
print_version();
print_help();
2013-09-19 09:48:25 +00:00
goto exit_failure;
2013-09-19 09:48:25 +00:00
}
input = open_input_file(cfg->input);
2013-09-19 09:48:25 +00:00
if (input == NULL) {
fprintf(stderr, "Could not open input file, shutting down!\n");
goto exit_failure;
2013-09-19 09:48:25 +00:00
}
2014-02-21 13:00:20 +00:00
output = open_output_file(cfg->output);
if (output == NULL) {
fprintf(stderr, "Could not open output file, shutting down!\n");
goto exit_failure;
}
if (cfg->debug != NULL) {
recout = open_output_file(cfg->debug);
if (recout == NULL) {
fprintf(stderr, "Could not open reconstruction file (%s), shutting down!\n", cfg->debug);
goto exit_failure;
}
}
enc = api->encoder_open(cfg);
if (!enc) {
fprintf(stderr, "Failed to open encoder.\n");
goto exit_failure;
}
encoder_control_t *encoder = enc->control;
fprintf(stderr, "Input: %s, output: %s\n", cfg->input, cfg->output);
fprintf(stderr, " Video size: %dx%d (input=%dx%d)\n",
encoder->in.width, encoder->in.height,
encoder->in.real_width, encoder->in.real_height);
2015-06-09 07:20:21 +00:00
2015-06-17 12:24:07 +00:00
if (cfg->seek > 0 && !yuv_io_seek(input, cfg->seek, cfg->width, cfg->height)) {
2015-06-09 07:20:21 +00:00
fprintf(stderr, "Failed to seek %d frames.\n", cfg->seek);
goto exit_failure;
}
2014-06-05 07:09:25 +00:00
//Now, do the real stuff
{
2014-06-16 08:18:11 +00:00
GET_TIME(&encoding_start_real_time);
encoding_start_cpu_time = clock();
2014-06-05 07:09:25 +00:00
uint64_t bitstream_length = 0;
uint32_t frames_read = 0;
uint32_t frames_done = 0;
double psnr_sum[3] = { 0.0, 0.0, 0.0 };
for (;;) {
encoder_state_t *state = &enc->states[enc->cur_state_num];
2015-06-30 07:56:29 +00:00
kvz_picture *img_in = NULL;
if (!feof(input) && (cfg->frames == 0 || frames_read < cfg->frames)) {
// Try to read an input frame.
img_in = api->picture_alloc(encoder->in.width, encoder->in.height);
if (!img_in) {
fprintf(stderr, "Failed to allocate image.\n");
goto exit_failure;
}
if (yuv_io_read(input, cfg->width, cfg->height, img_in)) {
frames_read += 1;
} else {
// EOF or some error
api->picture_free(img_in);
img_in = NULL;
if (!feof(input)) {
fprintf(stderr, "Failed to read a frame %d\n", frames_read);
goto exit_failure;
}
}
}
2014-06-05 07:09:25 +00:00
kvz_data_chunk* chunks_out = NULL;
uint32_t len_out = 0;
if (!api->encoder_encode(enc, img_in, &chunks_out, &len_out, NULL)) {
fprintf(stderr, "Failed to encode image.\n");
api->picture_free(img_in);
goto exit_failure;
}
if (chunks_out == NULL && img_in == NULL) {
// We are done since there is no more input and output left.
break;
}
if (chunks_out != NULL) {
uint64_t written = 0;
// Write data into the output file.
for (kvz_data_chunk *chunk = chunks_out;
chunk != NULL;
chunk = chunk->next) {
assert(written + chunk->len <= len_out);
if (fwrite(chunk->data, sizeof(uint8_t), chunk->len, output) != chunk->len) {
fprintf(stderr, "Failed to write data to file.\n");
api->picture_free(img_in);
api->chunk_free(chunks_out);
goto exit_failure;
}
written += chunk->len;
}
fflush(output);
bitstream_length += len_out;
// Compute and print stats.
state = &enc->states[enc->cur_state_num];
double frame_psnr[3] = { 0.0, 0.0, 0.0 };
encoder_compute_stats(state, recout, frame_psnr);
frames_done += 1;
psnr_sum[0] += frame_psnr[0];
psnr_sum[1] += frame_psnr[1];
psnr_sum[2] += frame_psnr[2];
print_frame_info(state, frame_psnr);
}
api->picture_free(img_in);
api->chunk_free(chunks_out);
2014-06-05 07:09:25 +00:00
}
2014-06-16 08:18:11 +00:00
GET_TIME(&encoding_end_real_time);
encoding_end_cpu_time = clock();
threadqueue_flush(encoder->threadqueue);
2014-06-05 07:09:25 +00:00
// Coding finished
2013-09-19 09:48:25 +00:00
2014-06-05 07:09:25 +00:00
// Print statistics of the coding
fprintf(stderr, " Processed %d frames, %10llu bits AVG PSNR: %2.4f %2.4f %2.4f\n",
frames_done, (long long unsigned int)bitstream_length * 8,
psnr_sum[0] / frames_done, psnr_sum[1] / frames_done, psnr_sum[2] / frames_done);
2014-06-16 08:18:11 +00:00
fprintf(stderr, " Total CPU time: %.3f s.\n", ((float)(clock() - start_time)) / CLOCKS_PER_SEC);
{
2014-11-14 14:38:07 +00:00
double encoding_time = ( (double)(encoding_end_cpu_time - encoding_start_cpu_time) ) / (double) CLOCKS_PER_SEC;
2014-06-16 08:18:11 +00:00
double wall_time = CLOCK_T_AS_DOUBLE(encoding_end_real_time) - CLOCK_T_AS_DOUBLE(encoding_start_real_time);
2014-11-14 14:38:07 +00:00
fprintf(stderr, " Encoding time: %.3f s.\n", encoding_time);
fprintf(stderr, " Encoding wall time: %.3f s.\n", wall_time);
fprintf(stderr, " Encoding CPU usage: %.2f%%\n", encoding_time/wall_time*100.f);
fprintf(stderr, " FPS: %.2f\n", ((double)frames_done)/wall_time);
2014-06-16 08:18:11 +00:00
}
2014-06-05 07:09:25 +00:00
}
2013-09-19 09:48:25 +00:00
goto done;
exit_failure:
retval = EXIT_FAILURE;
2013-09-19 09:48:25 +00:00
done:
// deallocate structures
if (enc) api->encoder_close(enc);
if (cfg) api->config_destroy(cfg);
// close files
if (input) fclose(input);
if (output) fclose(output);
if (recout) fclose(recout);
2014-06-11 05:48:36 +00:00
CHECKPOINTS_FINALIZE();
return retval;
2013-09-19 09:48:25 +00:00
}