blob: c0cbd98e22379a6ae2e9f6d232d6c80f52a63b6f [file] [log] [blame]
/*
* cl_image_processor.cpp - CL image processor
*
* Copyright (c) 2015 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 <[email protected]>
*/
#include "cl_image_processor.h"
#include "cl_context.h"
#include "cl_device.h"
#include "cl_image_handler.h"
#include "cl_demo_handler.h"
#include "xcam_thread.h"
namespace XCam {
class CLHandlerThread
: public Thread
{
public:
CLHandlerThread (CLImageProcessor *processor)
: Thread ("CLHandlerThread")
, _processor (processor)
{}
~CLHandlerThread () {}
virtual bool loop ();
private:
CLImageProcessor *_processor;
};
bool CLHandlerThread::loop ()
{
XCAM_ASSERT (_processor);
XCamReturn ret = _processor->process_cl_buffer_queue ();
if (ret != XCAM_RETURN_NO_ERROR && ret != XCAM_RETURN_BYPASS)
return false;
return true;
}
class CLBufferNotifyThread
: public Thread
{
public:
CLBufferNotifyThread (CLImageProcessor *processor)
: Thread ("CLBufNtfThrd")
, _processor (processor)
{}
~CLBufferNotifyThread () {}
virtual bool loop ();
private:
CLImageProcessor *_processor;
};
bool CLBufferNotifyThread::loop ()
{
XCAM_ASSERT (_processor);
XCamReturn ret = _processor->process_done_buffer ();
if (ret < XCAM_RETURN_NO_ERROR)
return false;
return true;
}
CLImageProcessor::CLImageProcessor (const char* name)
: ImageProcessor (name ? name : "CLImageProcessor")
, _seq_num (0)
, _keep_attached_buffer (false)
{
_context = CLDevice::instance ()->get_context ();
XCAM_ASSERT (_context.ptr());
_handler_thread = new CLHandlerThread (this);
XCAM_ASSERT (_handler_thread.ptr ());
_done_buf_thread = new CLBufferNotifyThread (this);
XCAM_ASSERT (_done_buf_thread.ptr ());
XCAM_LOG_DEBUG ("CLImageProcessor constructed");
XCAM_OBJ_PROFILING_INIT;
}
CLImageProcessor::~CLImageProcessor ()
{
XCAM_LOG_DEBUG ("CLImageProcessor destructed");
}
void
CLImageProcessor::keep_attached_buf(bool flag)
{
_keep_attached_buffer = flag;
}
bool
CLImageProcessor::add_handler (SmartPtr<CLImageHandler> &handler)
{
XCAM_ASSERT (handler.ptr ());
_handlers.push_back (handler);
return true;
}
CLImageProcessor::ImageHandlerList::iterator
CLImageProcessor::handlers_begin ()
{
return _handlers.begin ();
}
CLImageProcessor::ImageHandlerList::iterator
CLImageProcessor::handlers_end ()
{
return _handlers.end ();
}
SmartPtr<CLContext>
CLImageProcessor::get_cl_context ()
{
return _context;
}
bool
CLImageProcessor::can_process_result (SmartPtr<X3aResult> &result)
{
XCAM_UNUSED (result);
return false;
}
XCamReturn
CLImageProcessor::apply_3a_results (X3aResultList &results)
{
XCAM_UNUSED (results);
return XCAM_RETURN_NO_ERROR;
}
XCamReturn
CLImageProcessor::apply_3a_result (SmartPtr<X3aResult> &result)
{
XCAM_UNUSED (result);
return XCAM_RETURN_NO_ERROR;
}
XCamReturn
CLImageProcessor::process_buffer (SmartPtr<VideoBuffer> &input, SmartPtr<VideoBuffer> &output)
{
XCamReturn ret = XCAM_RETURN_NO_ERROR;
XCAM_ASSERT (input.ptr ());
// Always set to NULL, output buf should be handled in CLBufferNotifyThread
output = NULL;
STREAM_LOCK;
if (_handlers.empty()) {
ret = create_handlers ();
}
XCAM_FAIL_RETURN (
WARNING,
!_handlers.empty () && ret == XCAM_RETURN_NO_ERROR,
XCAM_RETURN_ERROR_CL,
"CL image processor create handlers failed");
SmartPtr<PriorityBuffer> p_buf = new PriorityBuffer;
p_buf->set_seq_num (_seq_num++);
p_buf->data = input;
p_buf->handler = *(_handlers.begin ());
XCAM_FAIL_RETURN (
WARNING,
_process_buffer_queue.push_priority_buf (p_buf),
XCAM_RETURN_ERROR_UNKNOWN,
"CLImageProcessor push priority buffer failed");
return XCAM_RETURN_BYPASS;
}
XCamReturn
CLImageProcessor::process_done_buffer ()
{
SmartPtr<VideoBuffer> done_buf = _done_buffer_queue.pop (-1);
if (!done_buf.ptr ())
return XCAM_RETURN_ERROR_THREAD;
//notify buffer done, only in this thread
notify_process_buffer_done (done_buf);
return XCAM_RETURN_NO_ERROR;
}
uint32_t
CLImageProcessor::check_ready_buffers ()
{
uint32_t ready_count = 0;
bool is_ready_or_disabled = false;
UnsafePriorityBufferList::iterator i = _not_ready_buffers.begin ();
while (i != _not_ready_buffers.end()) {
SmartPtr<PriorityBuffer> buf = *i;
XCAM_ASSERT (buf.ptr () && buf->handler.ptr ());
{
is_ready_or_disabled = (!buf->handler->is_handler_enabled () || buf->handler->is_ready ());
}
if (is_ready_or_disabled) {
ready_count ++;
_process_buffer_queue.push_priority_buf (buf);
_not_ready_buffers.erase (i++);
} else
++i;
}
return ready_count;
}
XCamReturn
CLImageProcessor::process_cl_buffer_queue ()
{
XCamReturn ret = XCAM_RETURN_NO_ERROR;
SmartPtr<PriorityBuffer> p_buf;
const int32_t timeout = 5000; // 5ms
uint32_t ready_count = 0;
{
STREAM_LOCK; // make sure handler APIs are protected
check_ready_buffers ();
}
p_buf = _process_buffer_queue.pop (timeout);
if (!p_buf.ptr ()) {
//XCAM_LOG_DEBUG ("cl buffer queue stopped");
return XCAM_RETURN_BYPASS;
}
SmartPtr<VideoBuffer> data = p_buf->data;
SmartPtr<CLImageHandler> handler = p_buf->handler;
SmartPtr <VideoBuffer> out_data;
XCAM_ASSERT (data.ptr () && handler.ptr ());
XCAM_LOG_DEBUG ("buf:%d, rank:%d\n", p_buf->seq_num, p_buf->rank);
{
STREAM_LOCK;
if (handler->is_handler_enabled () && !handler->is_ready ()) {
_not_ready_buffers.push_back (p_buf);
return XCAM_RETURN_NO_ERROR;
}
ready_count = check_ready_buffers ();
if (ready_count) {
_process_buffer_queue.push_priority_buf (p_buf);
return XCAM_RETURN_BYPASS;
}
ret = handler->execute (data, out_data);
XCAM_FAIL_RETURN (
WARNING,
(ret == XCAM_RETURN_NO_ERROR || ret == XCAM_RETURN_BYPASS),
ret,
"CLImageProcessor execute image handler failed");
XCAM_ASSERT (out_data.ptr ());
if (ret == XCAM_RETURN_BYPASS)
return ret;
// for loop in handler, find next handler
ImageHandlerList::iterator i_handler = _handlers.begin ();
while (i_handler != _handlers.end ())
{
if (handler.ptr () == (*i_handler).ptr ()) {
++i_handler;
break;
}
++i_handler;
}
//skip all disabled handlers
while (i_handler != _handlers.end () && !(*i_handler)->is_handler_enabled ())
++i_handler;
if (i_handler != _handlers.end ())
p_buf->handler = *i_handler;
else
p_buf->handler = NULL;
}
// buffer processed by all handlers, done
if (!p_buf->handler.ptr ()) {
if (!_keep_attached_buffer && out_data.ptr ())
out_data->clear_attached_buffers ();
XCAM_OBJ_PROFILING_START;
CLDevice::instance()->get_context ()->finish ();
XCAM_OBJ_PROFILING_END (get_name (), XCAM_OBJ_DUR_FRAME_NUM);
// buffer done, push back
_done_buffer_queue.push (out_data);
return XCAM_RETURN_NO_ERROR;
}
p_buf->data = out_data;
p_buf->down_rank ();
XCAM_FAIL_RETURN (
WARNING,
_process_buffer_queue.push_priority_buf (p_buf),
XCAM_RETURN_ERROR_UNKNOWN,
"CLImageProcessor push priority buffer failed");
return ret;
}
XCamReturn
CLImageProcessor::emit_start ()
{
_done_buffer_queue.resume_pop ();
_process_buffer_queue.resume_pop ();
if (!_done_buf_thread->start ())
return XCAM_RETURN_ERROR_THREAD;
if (!_handler_thread->start ())
return XCAM_RETURN_ERROR_THREAD;
return XCAM_RETURN_NO_ERROR;
}
void
CLImageProcessor::emit_stop ()
{
_process_buffer_queue.pause_pop();
_done_buffer_queue.pause_pop ();
for (ImageHandlerList::iterator i_handler = _handlers.begin ();
i_handler != _handlers.end (); ++i_handler) {
(*i_handler)->emit_stop ();
}
_handler_thread->stop ();
_done_buf_thread->stop ();
_not_ready_buffers.clear ();
_process_buffer_queue.clear ();
_done_buffer_queue.clear ();
}
XCamReturn
CLImageProcessor::create_handlers ()
{
SmartPtr<CLImageHandler> demo_handler;
demo_handler = create_cl_demo_image_handler (_context);
// demo_handler = create_cl_binary_demo_image_handler (_context);
XCAM_FAIL_RETURN (
WARNING,
demo_handler.ptr (),
XCAM_RETURN_ERROR_CL,
"CLImageProcessor create demo handler failed");
add_handler (demo_handler);
return XCAM_RETURN_NO_ERROR;
}
};