/*
* soft_geo_mapper.cpp - soft geometry mapper implementation
*
* Copyright (c) 2017 Intel Corporation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* Author: Wind Yuan <feng.yuan@intel.com>
*/
#include "soft_geo_mapper.h"
#include "soft_geo_tasks_priv.h"
#define XCAM_GEO_MAP_ALIGNMENT_X 8
#define XCAM_GEO_MAP_ALIGNMENT_Y 2
namespace XCam {
DECLARE_WORK_CALLBACK (CbGeoMapTask, SoftGeoMapper, remap_task_done);
SoftGeoMapper::SoftGeoMapper (const char *name)
: SoftHandler (name)
{
}
SoftGeoMapper::~SoftGeoMapper ()
{
}
bool
SoftGeoMapper::set_lookup_table (const PointFloat2 *data, uint32_t width, uint32_t height)
{
XCAM_FAIL_RETURN(
ERROR, width > 1 && height > 1 && data, false,
"SoftGeoMapper(%s) set loop up table need w>1 and h>1, but width:%d, height:%d",
XCAM_STR (get_name ()), width, height);
_lookup_table = new Float2Image (width, height);
XCAM_FAIL_RETURN(
ERROR, _lookup_table.ptr () && _lookup_table->is_valid (), false,
"SoftGeoMapper(%s) set loop up table failed in data allocation",
XCAM_STR (get_name ()));
for (uint32_t i = 0; i < height; ++i) {
Float2 *ret = _lookup_table->get_buf_ptr (0, i);
const PointFloat2 *line = &data[i * width];
for (uint32_t j = 0; j < width; ++j) {
ret[j].x = line [j].x;
ret[j].y = line [j].y;
}
}
return true;
}
XCamReturn
SoftGeoMapper::remap (
const SmartPtr<VideoBuffer> &in,
SmartPtr<VideoBuffer> &out_buf)
{
SmartPtr<ImageHandler::Parameters> param = new ImageHandler::Parameters (in, out_buf);
XCamReturn ret = execute_buffer (param, true);
if (xcam_ret_is_ok (ret) && !out_buf.ptr ()) {
out_buf = param->out_buf;
}
return ret;
}
XCamReturn
SoftGeoMapper::configure_resource (const SmartPtr<Parameters> ¶m)
{
XCAM_FAIL_RETURN(
ERROR, _lookup_table.ptr () && _lookup_table->is_valid (), XCAM_RETURN_ERROR_PARAM,
"SoftGeoMapper(%s) configure failed, look_up_table was not set correctly",
XCAM_STR (get_name ()));
const VideoBufferInfo &in_info = param->in_buf->get_video_info ();
XCAM_FAIL_RETURN (
ERROR, in_info.format == V4L2_PIX_FMT_NV12, XCAM_RETURN_ERROR_PARAM,
"SoftGeoMapper(:%s) only support format(NV12) but input format is %s",
XCAM_STR(get_name ()), xcam_fourcc_to_string (in_info.format));
Float2 factors;
get_factors (factors.x, factors.y);
if (XCAM_DOUBLE_EQUAL_AROUND (factors.x, 0.0f) ||
XCAM_DOUBLE_EQUAL_AROUND (factors.y, 0.0f)) {
auto_calculate_factors (_lookup_table->get_width (), _lookup_table->get_height ());
}
uint32_t width, height;
get_output_size (width, height);
VideoBufferInfo out_info;
out_info.init (
in_info.format, width, height,
XCAM_ALIGN_UP (width, XCAM_GEO_MAP_ALIGNMENT_X),
XCAM_ALIGN_UP (height, XCAM_GEO_MAP_ALIGNMENT_Y));
set_out_video_info (out_info);
XCAM_ASSERT (!_map_task.ptr ());
_map_task = new XCamSoftTasks::GeoMapTask (new CbGeoMapTask(this));
XCAM_ASSERT (_map_task.ptr ());
return XCAM_RETURN_NO_ERROR;
}
XCamReturn
SoftGeoMapper::start_remap_task (const SmartPtr<ImageHandler::Parameters> ¶m)
{
XCAM_ASSERT (_map_task.ptr ());
XCAM_ASSERT (_lookup_table.ptr ());
Float2 factors;
get_factors (factors.x, factors.y);
SmartPtr<VideoBuffer> in_buf = param->in_buf, out_buf = param->out_buf;
SmartPtr<XCamSoftTasks::GeoMapTask::Args> args = new XCamSoftTasks::GeoMapTask::Args (param);
args->in_luma = new UcharImage (in_buf, 0);
args->in_uv = new Uchar2Image (in_buf, 1);
args->out_luma = new UcharImage (out_buf, 0);
args->out_uv = new Uchar2Image (out_buf, 1);
args->lookup_table = _lookup_table;
args->factors = factors;
uint32_t thread_x = 2, thread_y = 2;
WorkSize work_unit = _map_task->get_work_uint ();
WorkSize global_size (
xcam_ceil (args->out_luma->get_width (), work_unit.value[0]) / work_unit.value[0],
xcam_ceil (args->out_luma->get_height (), work_unit.value[1]) / work_unit.value[1]);
WorkSize local_size (
xcam_ceil(global_size.value[0], thread_x) / thread_x ,
xcam_ceil(global_size.value[1], thread_y) / thread_y);
_map_task->set_local_size (local_size);
_map_task->set_global_size (global_size);
param->in_buf.release ();
return _map_task->work (args);
}
XCamReturn
SoftGeoMapper::start_work (const SmartPtr<ImageHandler::Parameters> ¶m)
{
XCamReturn ret = XCAM_RETURN_NO_ERROR;
XCAM_ASSERT (param->out_buf.ptr ());
ret = start_remap_task (param);
XCAM_FAIL_RETURN (
ERROR, xcam_ret_is_ok (ret), ret,
"geo_mapper:%s start_work failed on idx0", XCAM_STR (get_name ()));
param->in_buf.release ();
return ret;
};
XCamReturn
SoftGeoMapper::terminate ()
{
if (_map_task.ptr ()) {
_map_task->stop ();
_map_task.release ();
}
return SoftHandler::terminate ();
}
void
SoftGeoMapper::remap_task_done (
const SmartPtr<Worker> &worker, const SmartPtr<Worker::Arguments> &base, const XCamReturn error)
{
XCAM_UNUSED (worker);
XCAM_ASSERT (worker.ptr () == _map_task.ptr ());
SmartPtr<XCamSoftTasks::GeoMapTask::Args> args = base.dynamic_cast_ptr<XCamSoftTasks::GeoMapTask::Args> ();
XCAM_ASSERT (args.ptr ());
const SmartPtr<ImageHandler::Parameters> param = args->get_param ();
if (!check_work_continue (param, error))
return;
work_well_done (param, error);
}
SmartPtr<SoftHandler> create_soft_geo_mapper ()
{
SmartPtr<SoftHandler> mapper = new SoftGeoMapper ();
XCAM_ASSERT (mapper.ptr ());
return mapper;
}
SmartPtr<GeoMapper>
GeoMapper::create_soft_geo_mapper ()
{
SmartPtr<SoftHandler> handler = XCam::create_soft_geo_mapper ();
return handler.dynamic_cast_ptr<GeoMapper> ();
}
}