/*
* soft_handler.cpp - soft image handler 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_handler.h"
#include "soft_video_buf_allocator.h"
#include "thread_pool.h"
#include "soft_worker.h"
#define DEFAULT_SOFT_BUF_COUNT 4
namespace XCam {
class SyncMeta
: public MetaBase
{
public:
SyncMeta ()
: _done (false)
, _error (XCAM_RETURN_NO_ERROR) {}
void signal_done (XCamReturn err);
void wakeup ();
XCamReturn signal_wait_ret ();
bool is_error () const;
private:
mutable Mutex _mutex;
Cond _cond;
bool _done;
XCamReturn _error;
};
void
SyncMeta::signal_done (XCamReturn err)
{
SmartLock locker (_mutex);
_done = true;
_error = err;
_cond.broadcast ();
}
void
SyncMeta::wakeup ()
{
SmartLock locker (_mutex);
_error = XCAM_RETURN_ERROR_UNKNOWN;
_cond.broadcast ();
}
XCamReturn
SyncMeta::signal_wait_ret ()
{
SmartLock locker (_mutex);
if (_done)
return _error;
_cond.wait (_mutex);
return _error;
}
bool
SyncMeta::is_error () const
{
SmartLock locker (_mutex);
return !xcam_ret_is_ok (_error);
}
SoftHandler::SoftHandler (const char* name)
: ImageHandler (name)
, _need_configure (true)
, _enable_allocator (true)
, _wip_buf_count (0)
{
}
SoftHandler::~SoftHandler ()
{
}
bool
SoftHandler::set_threads (const SmartPtr<ThreadPool> &pool)
{
_threads = pool;
return true;
}
bool
SoftHandler::set_out_video_info (const VideoBufferInfo &info)
{
XCAM_ASSERT (info.width && info.height && info.format);
_out_video_info = info;
return true;
}
bool
SoftHandler::enable_allocator (bool enable)
{
_enable_allocator = enable;
return true;
}
XCamReturn
SoftHandler::confirm_configured ()
{
XCamReturn ret = XCAM_RETURN_NO_ERROR;
XCAM_ASSERT (_need_configure);
if (_enable_allocator) {
XCAM_FAIL_RETURN (
ERROR, _out_video_info.is_valid (), XCAM_RETURN_ERROR_PARAM,
"soft_hander(%s) configure resource failed before reserver buffer since out video info was not set",
XCAM_STR (get_name ()));
set_allocator (new SoftVideoBufAllocator);
ret = reserve_buffers (_out_video_info, DEFAULT_SOFT_BUF_COUNT);
XCAM_FAIL_RETURN (
ERROR, ret == XCAM_RETURN_NO_ERROR, ret,
"soft_hander(%s) configure resource failed in reserving buffers", XCAM_STR (get_name ()));
}
if (_threads.ptr () && !_threads->is_running ()) {
ret = _threads->start ();
XCAM_FAIL_RETURN (
ERROR, ret == XCAM_RETURN_NO_ERROR, ret,
"soft_hander(%s) configure resource failed when starting threads", XCAM_STR (get_name ()));
}
_need_configure = false;
return ret;
}
XCamReturn
SoftHandler::execute_buffer (const SmartPtr<ImageHandler::Parameters> ¶m, bool sync)
{
XCamReturn ret = XCAM_RETURN_NO_ERROR;
SmartPtr<SyncMeta> sync_meta;
XCAM_FAIL_RETURN (
ERROR, param.ptr (), XCAM_RETURN_ERROR_PARAM,
"soft_hander(%s) execute buffer failed, params is null",
XCAM_STR (get_name ()));
if (_need_configure) {
ret = configure_resource (param);
XCAM_FAIL_RETURN (
WARNING, xcam_ret_is_ok (ret), ret,
"soft_hander(%s) configure resource failed", XCAM_STR (get_name ()));
ret = confirm_configured ();
XCAM_FAIL_RETURN (
WARNING, xcam_ret_is_ok (ret), ret,
"soft_hander(%s) confirm configure failed", XCAM_STR (get_name ()));
}
if (!param->out_buf.ptr () && _enable_allocator) {
param->out_buf = get_free_buf ();
XCAM_FAIL_RETURN (
ERROR, param->out_buf.ptr (), XCAM_RETURN_ERROR_PARAM,
"soft_hander:%s execute buffer failed, output buffer failed in allocation.",
XCAM_STR (get_name ()));
}
XCAM_ASSERT (!param->find_meta<SyncMeta> ().ptr ());
sync_meta = new SyncMeta ();
XCAM_ASSERT (sync_meta.ptr ());
param->add_meta (sync_meta);
#if 0
SmartPtr<SoftWorker> worker = get_first_worker ().dynamic_cast_ptr<SoftWorker> ();
XCAM_FAIL_RETURN (
WARNING, worker.ptr (), XCAM_RETURN_ERROR_PARAM,
"No worder set to soft_hander(%s)", XCAM_STR (get_name ()));
SmartPtr<Worker::Arguments> args = get_first_worker_args (worker, params);
XCAM_FAIL_RETURN (
WARNING, args.ptr (), XCAM_RETURN_ERROR_PARAM,
"soft_hander(%s) get first worker(%s) args failed",
XCAM_STR (get_name ()), XCAM_STR (worker->get_name ()));
_params.push (params);
ret = worker->work (args);
#else
_params.push (param);
ret = start_work (param);
#endif
if (!xcam_ret_is_ok (ret)) {
_params.erase (param);
XCAM_LOG_WARNING ("soft_hander(%s) execute buffer failed in starting workers", XCAM_STR (get_name ()));
return ret;
}
++_wip_buf_count;
_cur_sync = sync_meta;
if (sync) {
XCAM_ASSERT (sync_meta.ptr ());
ret = sync_meta->signal_wait_ret ();
_cur_sync.release ();
}
return ret;
}
XCamReturn
SoftHandler::finish ()
{
XCamReturn ret = XCAM_RETURN_NO_ERROR;
SmartPtr<SyncMeta> sync = _cur_sync;
if (sync.ptr ()) {
ret = sync->signal_wait_ret ();
}
XCAM_ASSERT (_params.is_empty ());
//wait for _wip_buf_count = 0
//if (ret == XCAM_RETURN_NO_ERROR)
// XCAM_ASSERT (_wip_buf_count == 0);
return ret;
}
XCamReturn
SoftHandler::terminate ()
{
SmartPtr<SyncMeta> sync = _cur_sync;
if (sync.ptr ()) {
sync->wakeup ();
sync.release ();
}
_params.clear ();
return ImageHandler::terminate ();
}
void
SoftHandler::work_well_done (const SmartPtr<ImageHandler::Parameters> ¶m, XCamReturn err)
{
XCAM_ASSERT (param.ptr ());
XCAM_ASSERT (xcam_ret_is_ok (err));
if (!xcam_ret_is_ok (err)) {
XCAM_LOG_WARNING ("soft_hander(%s) work_well_done but errno(%d) is not ok", XCAM_STR (get_name ()), (int)err);
//continue work
}
if (!_params.erase (param)) {
XCAM_LOG_ERROR(
"soft_hander(%s) last_work_done param already removed, who removed it?", XCAM_STR (get_name ()));
return;
}
XCAM_LOG_DEBUG ("soft_hander(%s) work well done", XCAM_STR (get_name ()));
param_ended (param, err);
}
void
SoftHandler::work_broken (const SmartPtr<ImageHandler::Parameters> ¶m, XCamReturn err)
{
XCAM_ASSERT (param.ptr ());
XCAM_ASSERT (!xcam_ret_is_ok (err));
if (xcam_ret_is_ok (err)) {
XCAM_LOG_WARNING ("soft_hander(%s) work_broken but the errno(%d) is ok", XCAM_STR (get_name ()), (int)err);
//continue work
}
if (!_params.erase (param)) {
//already removed by other handlers
return;
}
XCAM_LOG_WARNING ("soft_hander(%s) work broken", XCAM_STR (get_name ()));
param_ended (param, err);
}
void
SoftHandler::param_ended (SmartPtr<ImageHandler::Parameters> param, XCamReturn err)
{
XCAM_ASSERT (param.ptr ());
SmartPtr<SyncMeta> sync_meta = param->find_meta<SyncMeta> ();
XCAM_ASSERT (sync_meta.ptr ());
sync_meta->signal_done (err);
--_wip_buf_count;
execute_status_check (param, err);
}
bool
SoftHandler::check_work_continue (const SmartPtr<ImageHandler::Parameters> ¶m, XCamReturn err)
{
if (!xcam_ret_is_ok (err)) {
work_broken (param, err);
return false;
}
if (is_param_error (param)) {
XCAM_LOG_WARNING (
"soft_handler(%s) check_work_continue found param broken", XCAM_STR(get_name ()));
return false;
}
return true;
}
bool
SoftHandler::is_param_error (const SmartPtr<ImageHandler::Parameters> ¶m)
{
XCAM_ASSERT (param.ptr ());
SmartPtr<SyncMeta> meta = param->find_meta<SyncMeta> ();
if (!meta.ptr ()) { // return ok if param not set
XCAM_ASSERT (meta.ptr ());
return false;
}
return meta->is_error ();
}
}