/* * x3a_image_process_center.cpp - 3a process center * * Copyright (c) 2014-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 <feng.yuan@intel.com> */ #include "x3a_image_process_center.h" namespace XCam { X3aImageProcessCenter::X3aImageProcessCenter() : _callback (NULL) { XCAM_LOG_DEBUG ("X3aImageProcessCenter construction"); } X3aImageProcessCenter::~X3aImageProcessCenter() { stop (); XCAM_LOG_DEBUG ("~X3aImageProcessCenter destruction"); } bool X3aImageProcessCenter::set_image_callback (ImageProcessCallback *callback) { XCAM_ASSERT (!_callback); _callback = callback; return true; } bool X3aImageProcessCenter::insert_processor (SmartPtr<ImageProcessor> &processor) { _image_processors.push_back (processor); XCAM_LOG_INFO ("Add processor(%s) into image processor center", XCAM_STR (processor->get_name())); return true; } bool X3aImageProcessCenter::has_processors () { return !_image_processors.empty(); } XCamReturn X3aImageProcessCenter::start () { XCamReturn ret = XCAM_RETURN_NO_ERROR; if (_image_processors.empty()) { XCAM_LOG_ERROR ("process center start failed, no processor found"); return XCAM_RETURN_ERROR_PARAM; } for (ImageProcessorList::iterator i_pro = _image_processors.begin (); i_pro != _image_processors.end(); ++i_pro) { SmartPtr<ImageProcessor> &processor = *i_pro; XCAM_ASSERT (processor.ptr()); processor->set_callback (this); ret = processor->start (); if (ret != XCAM_RETURN_NO_ERROR) { XCAM_LOG_ERROR ("processor(%s) start failed", XCAM_STR(processor->get_name())); break; } } if (ret != XCAM_RETURN_NO_ERROR) stop(); else { XCAM_LOG_INFO ("3a process center started"); } return ret; } XCamReturn X3aImageProcessCenter::stop () { for (ImageProcessorList::iterator i_pro = _image_processors.begin (); i_pro != _image_processors.end(); ++i_pro) { SmartPtr<ImageProcessor> &processor = *i_pro; XCAM_ASSERT (processor.ptr()); processor->stop (); } XCAM_LOG_INFO ("3a process center stopped"); _image_processors.clear(); return XCAM_RETURN_NO_ERROR; } bool X3aImageProcessCenter::put_buffer (SmartPtr<VideoBuffer> &buf) { XCAM_ASSERT (!_image_processors.empty()); if (_image_processors.empty()) return false; ImageProcessorList::iterator i_pro = _image_processors.begin (); SmartPtr<ImageProcessor> &processor = *i_pro; if (processor->push_buffer (buf) != XCAM_RETURN_NO_ERROR) return false; return true; } XCamReturn X3aImageProcessCenter::put_3a_results (X3aResultList &results) { XCamReturn ret = XCAM_RETURN_NO_ERROR; XCAM_FAIL_RETURN (ERROR, !results.empty(), XCAM_RETURN_ERROR_PARAM, "results empty"); for (ImageProcessorIter i_pro = _image_processors.begin(); i_pro != _image_processors.end(); i_pro++) { SmartPtr<ImageProcessor> &processor = *i_pro; XCAM_ASSERT (processor.ptr()); ret = processor->push_3a_results (results); if (ret != XCAM_RETURN_NO_ERROR && ret != XCAM_RETURN_BYPASS) { XCAM_LOG_WARNING ("processor(%s) gailed on results", XCAM_STR(processor->get_name())); break; } if (results.empty ()) { XCAM_LOG_DEBUG ("results done"); return XCAM_RETURN_NO_ERROR; } } if (!results.empty()) { XCAM_LOG_DEBUG ("process center: results left without being processed"); return XCAM_RETURN_BYPASS; } return XCAM_RETURN_NO_ERROR; } XCamReturn X3aImageProcessCenter::put_3a_result (SmartPtr<X3aResult> &result) { XCamReturn ret = XCAM_RETURN_NO_ERROR; XCAM_FAIL_RETURN (ERROR, !result.ptr(), XCAM_RETURN_ERROR_PARAM, "result empty"); for (ImageProcessorIter i_pro = _image_processors.begin(); i_pro != _image_processors.end(); i_pro++) { SmartPtr<ImageProcessor> &processor = *i_pro; XCAM_ASSERT (processor.ptr()); ret = processor->push_3a_result (result); if (ret == XCAM_RETURN_BYPASS) continue; if (ret == XCAM_RETURN_NO_ERROR) return XCAM_RETURN_NO_ERROR; XCAM_LOG_WARNING ("processor(%s) failed on result", XCAM_STR(processor->get_name())); return ret; } if (ret == XCAM_RETURN_BYPASS) { XCAM_LOG_WARNING ("processor center: no processor can handle result()"); } return ret; } void X3aImageProcessCenter::process_buffer_done (ImageProcessor *processor, const SmartPtr<VideoBuffer> &buf) { ImageProcessorIter i_pro = _image_processors.begin(); for (; i_pro != _image_processors.end(); ++i_pro) { SmartPtr<ImageProcessor> &cur_pro = *i_pro; XCAM_ASSERT (cur_pro.ptr()); if (cur_pro.ptr() == processor) break; } XCAM_ASSERT (i_pro != _image_processors.end()); if (i_pro == _image_processors.end()) { XCAM_LOG_ERROR ("processor doesn't found from list of image center"); return; } if (++i_pro != _image_processors.end()) { SmartPtr<ImageProcessor> &next_processor = *i_pro; SmartPtr<VideoBuffer> cur_buf = buf; XCAM_ASSERT (next_processor.ptr()); XCamReturn ret = next_processor->push_buffer (cur_buf); if (ret != XCAM_RETURN_NO_ERROR) { XCAM_LOG_ERROR ("processor(%s) failed in push_buffer", next_processor->get_name()); } return; } //all processor done if (_callback) _callback->process_buffer_done (processor, buf); else ImageProcessCallback::process_buffer_done (processor, buf); } void X3aImageProcessCenter::process_buffer_failed (ImageProcessor *processor, const SmartPtr<VideoBuffer> &buf) { if (_callback) _callback->process_buffer_failed(processor, buf); else ImageProcessCallback::process_buffer_failed (processor, buf); } void X3aImageProcessCenter::process_image_result_done (ImageProcessor *processor, const SmartPtr<X3aResult> &result) { if (_callback) _callback->process_image_result_done(processor, result); else ImageProcessCallback::process_image_result_done (processor, result); } };