/* Copyright (c) 2013-2016, The Linux Foundation. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials provided
* with the distribution.
* * Neither the name of The Linux Foundation nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
* IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
// System dependencies
#include <pthread.h>
#include <stdlib.h>
#define TIME_H <SYSTEM_HEADER_PREFIX/time.h>
#include TIME_H
// JPEG dependencies
#include "mm_jpeg_interface.h"
#include "mm_jpeg_ionbuf.h"
// Camera dependencies
#include "mm_camera_dbg.h"
#define MIN(a,b) (((a) < (b)) ? (a) : (b))
#define MAX(a,b) (((a) > (b)) ? (a) : (b))
#define CLAMP(x, min, max) MIN(MAX((x), (min)), (max))
#define TIME_IN_US(r) ((uint64_t)r.tv_sec * 1000000LL + (uint64_t)r.tv_usec)
struct timeval dtime[2];
/** DUMP_TO_FILE:
* @filename: file name
* @p_addr: address of the buffer
* @len: buffer length
*
* dump the image to the file
**/
#define DUMP_TO_FILE(filename, p_addr, len) ({ \
size_t rc = 0; \
FILE *fp = fopen(filename, "w+"); \
if (fp) { \
rc = fwrite(p_addr, 1, len, fp); \
fclose(fp); \
} else { \
LOGE("cannot dump image"); \
} \
})
static int g_count = 1, g_i;
typedef struct {
char *filename;
int width;
int height;
char *out_filename;
int format;
} jpeg_test_input_t;
typedef struct {
char *filename;
int width;
int height;
char *out_filename;
pthread_mutex_t lock;
pthread_cond_t cond;
buffer_t input;
buffer_t output;
int use_ion;
uint32_t handle;
mm_jpegdec_ops_t ops;
uint32_t job_id[5];
mm_jpeg_decode_params_t params;
mm_jpeg_job_t job;
uint32_t session_id;
} mm_jpegdec_intf_test_t;
typedef struct {
char *format_str;
int eColorFormat;
} mm_jpegdec_col_fmt_t;
#define ARR_SZ(a) (sizeof(a)/sizeof(a[0]))
static const mm_jpegdec_col_fmt_t col_formats[] =
{
{ "YCRCBLP_H2V2", (int)MM_JPEG_COLOR_FORMAT_YCRCBLP_H2V2 },
{ "YCBCRLP_H2V2", (int)MM_JPEG_COLOR_FORMAT_YCBCRLP_H2V2 },
{ "YCRCBLP_H2V1", (int)MM_JPEG_COLOR_FORMAT_YCRCBLP_H2V1 },
{ "YCBCRLP_H2V1", (int)MM_JPEG_COLOR_FORMAT_YCBCRLP_H2V1 },
{ "YCRCBLP_H1V2", (int)MM_JPEG_COLOR_FORMAT_YCRCBLP_H1V2 },
{ "YCBCRLP_H1V2", (int)MM_JPEG_COLOR_FORMAT_YCBCRLP_H1V2 },
{ "YCRCBLP_H1V1", (int)MM_JPEG_COLOR_FORMAT_YCRCBLP_H1V1 },
{ "YCBCRLP_H1V1", (int)MM_JPEG_COLOR_FORMAT_YCBCRLP_H1V1 }
};
static void mm_jpegdec_decode_callback(jpeg_job_status_t status,
uint32_t client_hdl,
uint32_t jobId,
mm_jpeg_output_t *p_output,
void *userData)
{
mm_jpegdec_intf_test_t *p_obj = (mm_jpegdec_intf_test_t *)userData;
if (status == JPEG_JOB_STATUS_ERROR) {
LOGE("Decode error");
} else {
gettimeofday(&dtime[1], NULL);
LOGE("Decode time %llu ms",
((TIME_IN_US(dtime[1]) - TIME_IN_US(dtime[0]))/1000));
LOGE("Decode success file%s addr %p len %zu",
p_obj->out_filename,
p_output->buf_vaddr, p_output->buf_filled_len);
DUMP_TO_FILE(p_obj->out_filename, p_output->buf_vaddr, p_output->buf_filled_len);
}
g_i++;
if (g_i >= g_count) {
LOGE("Signal the thread");
pthread_cond_signal(&p_obj->cond);
}
}
int mm_jpegdec_test_alloc(buffer_t *p_buffer, int use_pmem)
{
int ret = 0;
/*Allocate buffers*/
if (use_pmem) {
p_buffer->addr = (uint8_t *)buffer_allocate(p_buffer, 0);
if (NULL == p_buffer->addr) {
LOGE("Error");
return -1;
}
} else {
/* Allocate heap memory */
p_buffer->addr = (uint8_t *)malloc(p_buffer->size);
if (NULL == p_buffer->addr) {
LOGE("Error");
return -1;
}
}
return ret;
}
void mm_jpegdec_test_free(buffer_t *p_buffer)
{
if (p_buffer->addr == NULL)
return;
if (p_buffer->p_pmem_fd >= 0)
buffer_deallocate(p_buffer);
else
free(p_buffer->addr);
memset(p_buffer, 0x0, sizeof(buffer_t));
}
int mm_jpegdec_test_read(mm_jpegdec_intf_test_t *p_obj)
{
int rc = 0;
FILE *fp = NULL;
size_t file_size = 0;
fp = fopen(p_obj->filename, "rb");
if (!fp) {
LOGE("error");
return -1;
}
fseek(fp, 0, SEEK_END);
file_size = (size_t)ftell(fp);
fseek(fp, 0, SEEK_SET);
LOGE("input file size is %zu",
file_size);
p_obj->input.size = file_size;
/* allocate buffers */
rc = mm_jpegdec_test_alloc(&p_obj->input, p_obj->use_ion);
if (rc) {
LOGE("Error");
return -1;
}
fread(p_obj->input.addr, 1, p_obj->input.size, fp);
fclose(fp);
return 0;
}
void chromaScale(mm_jpeg_color_format format, double *cScale)
{
double scale;
switch(format) {
case MM_JPEG_COLOR_FORMAT_YCRCBLP_H2V2:
case MM_JPEG_COLOR_FORMAT_YCBCRLP_H2V2:
scale = 1.5;
break;
case MM_JPEG_COLOR_FORMAT_YCRCBLP_H2V1:
case MM_JPEG_COLOR_FORMAT_YCBCRLP_H2V1:
case MM_JPEG_COLOR_FORMAT_YCRCBLP_H1V2:
case MM_JPEG_COLOR_FORMAT_YCBCRLP_H1V2:
scale = 2.0;
break;
case MM_JPEG_COLOR_FORMAT_YCRCBLP_H1V1:
case MM_JPEG_COLOR_FORMAT_YCBCRLP_H1V1:
scale = 3.0;
break;
case MM_JPEG_COLOR_FORMAT_MONOCHROME:
scale = 1.0;
break;
default:
scale = 0;
LOGE("color format Error");
}
*cScale = scale;
}
static int decode_init(jpeg_test_input_t *p_input, mm_jpegdec_intf_test_t *p_obj)
{
int rc = -1;
size_t size = (size_t)(CEILING16(p_input->width) * CEILING16(p_input->height));
double cScale;
mm_jpeg_decode_params_t *p_params = &p_obj->params;
mm_jpeg_decode_job_t *p_job_params = &p_obj->job.decode_job;
p_obj->filename = p_input->filename;
p_obj->width = p_input->width;
p_obj->height = p_input->height;
p_obj->out_filename = p_input->out_filename;
p_obj->use_ion = 1;
pthread_mutex_init(&p_obj->lock, NULL);
pthread_cond_init(&p_obj->cond, NULL);
chromaScale(p_input->format, &cScale);
p_obj->output.size = (size_t)((double)size * cScale);
rc = mm_jpegdec_test_alloc(&p_obj->output, p_obj->use_ion);
if (rc) {
LOGE("Error");
return -1;
}
rc = mm_jpegdec_test_read(p_obj);
if (rc) {
LOGE("Error");
return -1;
}
/* set encode parameters */
p_params->jpeg_cb = mm_jpegdec_decode_callback;
p_params->userdata = p_obj;
p_params->color_format = p_input->format;
/* dest buffer config */
p_params->dest_buf[0].buf_size = p_obj->output.size;
p_params->dest_buf[0].buf_vaddr = p_obj->output.addr;
p_params->dest_buf[0].fd = p_obj->output.p_pmem_fd;
p_params->dest_buf[0].format = MM_JPEG_FMT_YUV;
p_params->dest_buf[0].offset.mp[0].len = (uint32_t)size;
p_params->dest_buf[0].offset.mp[1].len =
(uint32_t)((double)size * (cScale - 1.0));
p_params->dest_buf[0].offset.mp[0].stride = CEILING16(p_input->width);
p_params->dest_buf[0].offset.mp[0].scanline = CEILING16(p_input->height);
p_params->dest_buf[0].offset.mp[1].stride = CEILING16(p_input->width);
p_params->dest_buf[0].offset.mp[1].scanline = CEILING16(p_input->height);
p_params->dest_buf[0].index = 0;
p_params->num_dst_bufs = 1;
/* src buffer config*/
p_params->src_main_buf[0].buf_size = p_obj->input.size;
p_params->src_main_buf[0].buf_vaddr = p_obj->input.addr;
p_params->src_main_buf[0].fd = p_obj->input.p_pmem_fd;
p_params->src_main_buf[0].index = 0;
p_params->src_main_buf[0].format = MM_JPEG_FMT_BITSTREAM;
/*
p_params->src_main_buf[0].offset.mp[0].len = size;
p_params->src_main_buf[0].offset.mp[1].len = size >> 1;
*/
p_params->num_src_bufs = 1;
p_job_params->dst_index = 0;
p_job_params->src_index = 0;
p_job_params->rotation = 0;
/* main dimension */
p_job_params->main_dim.src_dim.width = p_obj->width;
p_job_params->main_dim.src_dim.height = p_obj->height;
p_job_params->main_dim.dst_dim.width = p_obj->width;
p_job_params->main_dim.dst_dim.height = p_obj->height;
p_job_params->main_dim.crop.top = 0;
p_job_params->main_dim.crop.left = 0;
p_job_params->main_dim.crop.width = p_obj->width;
p_job_params->main_dim.crop.height = p_obj->height;
return 0;
}
void omx_test_dec_print_usage()
{
fprintf(stderr, "Usage: program_name [options]\n");
fprintf(stderr, "Mandatory options:\n");
fprintf(stderr, " -I FILE\t\tPath to the input file.\n");
fprintf(stderr, " -O FILE\t\tPath for the output file.\n");
fprintf(stderr, " -W WIDTH\t\tOutput image width\n");
fprintf(stderr, " -H HEIGHT\t\tOutput image height\n");
fprintf(stderr, "Optional:\n");
fprintf(stderr, " -F FORMAT\t\tDefault image format:\n");
fprintf(stderr, "\t\t\t\t%s (0), %s (1), %s (2) %s (3)\n"
"%s (4), %s (5), %s (6) %s (7)\n",
col_formats[0].format_str, col_formats[1].format_str,
col_formats[2].format_str, col_formats[3].format_str,
col_formats[4].format_str, col_formats[5].format_str,
col_formats[6].format_str, col_formats[7].format_str
);
fprintf(stderr, "\n");
}
static int mm_jpegdec_test_get_input(int argc, char *argv[],
jpeg_test_input_t *p_test)
{
int c;
while ((c = getopt(argc, argv, "I:O:W:H:F:")) != -1) {
switch (c) {
case 'O':
p_test->out_filename = optarg;
fprintf(stderr, "%-25s%s\n", "Output image path",
p_test->out_filename);
break;
case 'I':
p_test->filename = optarg;
fprintf(stderr, "%-25s%s\n", "Input image path", p_test->filename);
break;
case 'W':
p_test->width = atoi(optarg);
fprintf(stderr, "%-25s%d\n", "Default width", p_test->width);
break;
case 'H':
p_test->height = atoi(optarg);
fprintf(stderr, "%-25s%d\n", "Default height", p_test->height);
break;
case 'F': {
int format = 0;
format = atoi(optarg);
int num_formats = ARR_SZ(col_formats);
format = CLAMP(format, 0, num_formats);
p_test->format = col_formats[format].eColorFormat;
fprintf(stderr, "%-25s%s\n", "Default image format",
col_formats[format].format_str);
break;
}
default:;
}
}
if (!p_test->filename || !p_test->filename || !p_test->width ||
!p_test->height) {
fprintf(stderr, "Missing required arguments.\n");
omx_test_dec_print_usage();
return -1;
}
return 0;
}
static int decode_test(jpeg_test_input_t *p_input)
{
int rc = 0;
mm_jpegdec_intf_test_t jpeg_obj;
int i = 0;
memset(&jpeg_obj, 0x0, sizeof(jpeg_obj));
rc = decode_init(p_input, &jpeg_obj);
if (rc) {
LOGE("Error");
return -1;
}
jpeg_obj.handle = jpegdec_open(&jpeg_obj.ops);
if (jpeg_obj.handle == 0) {
LOGE("Error");
goto end;
}
rc = jpeg_obj.ops.create_session(jpeg_obj.handle, &jpeg_obj.params,
&jpeg_obj.job.decode_job.session_id);
if (jpeg_obj.job.decode_job.session_id == 0) {
LOGE("Error");
goto end;
}
for (i = 0; i < g_count; i++) {
jpeg_obj.job.job_type = JPEG_JOB_TYPE_DECODE;
LOGE("Starting decode job");
gettimeofday(&dtime[0], NULL);
fprintf(stderr, "Starting decode of %s into %s outw %d outh %d\n\n",
p_input->filename, p_input->out_filename,
p_input->width, p_input->height);
rc = jpeg_obj.ops.start_job(&jpeg_obj.job, &jpeg_obj.job_id[i]);
if (rc) {
LOGE("Error");
goto end;
}
}
/*
usleep(5);
jpeg_obj.ops.abort_job(jpeg_obj.job_id[0]);
*/
pthread_mutex_lock(&jpeg_obj.lock);
pthread_cond_wait(&jpeg_obj.cond, &jpeg_obj.lock);
pthread_mutex_unlock(&jpeg_obj.lock);
fprintf(stderr, "Decode time %llu ms\n",
((TIME_IN_US(dtime[1]) - TIME_IN_US(dtime[0]))/1000));
jpeg_obj.ops.destroy_session(jpeg_obj.job.decode_job.session_id);
jpeg_obj.ops.close(jpeg_obj.handle);
end:
mm_jpegdec_test_free(&jpeg_obj.input);
mm_jpegdec_test_free(&jpeg_obj.output);
return 0;
}
/** main:
*
* Arguments:
* @argc
* @argv
*
* Return:
* 0 or -ve values
*
* Description:
* main function
*
**/
int main(int argc, char* argv[])
{
jpeg_test_input_t dec_test_input;
int ret;
memset(&dec_test_input, 0, sizeof(dec_test_input));
ret = mm_jpegdec_test_get_input(argc, argv, &dec_test_input);
if (ret) {
return -1;
}
return decode_test(&dec_test_input);
}