| /* |
| * device_manager.h - device manager |
| * |
| * 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 <[email protected]> |
| */ |
| |
| #include "device_manager.h" |
| #include "poll_thread.h" |
| #include "xcam_thread.h" |
| #include "x3a_image_process_center.h" |
| #include "x3a_analyzer_manager.h" |
| |
| #define XCAM_FAILED_STOP(exp, msg, ...) \ |
| if ((exp) != XCAM_RETURN_NO_ERROR) { \ |
| XCAM_LOG_ERROR (msg, ## __VA_ARGS__); \ |
| stop (); \ |
| return ret; \ |
| } |
| |
| namespace XCam { |
| |
| class MessageThread |
| : public Thread |
| { |
| public: |
| explicit MessageThread (DeviceManager *dev_manager) |
| : Thread ("MessageThread") |
| , _manager (dev_manager) |
| {} |
| |
| protected: |
| virtual bool loop (); |
| |
| DeviceManager *_manager; |
| }; |
| |
| bool |
| MessageThread::loop() |
| { |
| XCamReturn ret = _manager->message_loop(); |
| if (ret == XCAM_RETURN_NO_ERROR || ret == XCAM_RETURN_ERROR_TIMEOUT) |
| return true; |
| |
| return false; |
| } |
| |
| XCamMessage::XCamMessage (XCamMessageType type, int64_t timestamp, const char *message) |
| : timestamp (timestamp) |
| , msg_id (type) |
| , msg (NULL) |
| { |
| if (message) |
| this->msg = strndup (message, XCAM_MAX_STR_SIZE); |
| } |
| |
| XCamMessage::~XCamMessage () |
| { |
| if (msg) |
| xcam_free (msg); |
| } |
| |
| DeviceManager::DeviceManager() |
| : _has_3a (true) |
| , _is_running (false) |
| { |
| _3a_process_center = new X3aImageProcessCenter; |
| XCAM_LOG_DEBUG ("~DeviceManager construction"); |
| } |
| |
| DeviceManager::~DeviceManager() |
| { |
| XCAM_LOG_DEBUG ("~DeviceManager destruction"); |
| } |
| |
| bool |
| DeviceManager::set_capture_device (SmartPtr<V4l2Device> device) |
| { |
| if (is_running()) |
| return false; |
| |
| XCAM_ASSERT (device.ptr () && !_device.ptr ()); |
| _device = device; |
| return true; |
| } |
| |
| bool |
| DeviceManager::set_event_device (SmartPtr<V4l2SubDevice> device) |
| { |
| if (is_running()) |
| return false; |
| |
| XCAM_ASSERT (device.ptr () && !_subdevice.ptr ()); |
| _subdevice = device; |
| return true; |
| } |
| |
| bool |
| DeviceManager::set_3a_analyzer (SmartPtr<X3aAnalyzer> analyzer) |
| { |
| if (is_running()) |
| return false; |
| |
| XCAM_ASSERT (analyzer.ptr () && !_3a_analyzer.ptr ()); |
| _3a_analyzer = analyzer; |
| |
| return true; |
| } |
| |
| bool |
| DeviceManager::set_smart_analyzer (SmartPtr<SmartAnalyzer> analyzer) |
| { |
| if (is_running()) |
| return false; |
| |
| XCAM_ASSERT (analyzer.ptr () && !_smart_analyzer.ptr ()); |
| _smart_analyzer = analyzer; |
| |
| return true; |
| } |
| |
| bool |
| DeviceManager::add_image_processor (SmartPtr<ImageProcessor> processor) |
| { |
| if (is_running()) |
| return false; |
| |
| XCAM_ASSERT (processor.ptr ()); |
| return _3a_process_center->insert_processor (processor); |
| } |
| |
| bool |
| DeviceManager::set_poll_thread (SmartPtr<PollThread> thread) |
| { |
| if (is_running ()) |
| return false; |
| |
| XCAM_ASSERT (thread.ptr () && !_poll_thread.ptr ()); |
| _poll_thread = thread; |
| return true; |
| } |
| |
| XCamReturn |
| DeviceManager::start () |
| { |
| XCamReturn ret = XCAM_RETURN_NO_ERROR; |
| |
| // start device |
| XCAM_ASSERT (_device->is_opened()); |
| if (!_device.ptr() || !_device->is_opened()) { |
| XCAM_FAILED_STOP (ret = XCAM_RETURN_ERROR_FILE, "capture device not ready"); |
| } |
| XCAM_FAILED_STOP (ret = _device->start(), "capture device start failed"); |
| |
| //start subdevice |
| //XCAM_ASSERT (_subdevice->is_opened()); |
| if (_subdevice.ptr()) { |
| if (!_subdevice->is_opened()) |
| XCAM_FAILED_STOP (ret = XCAM_RETURN_ERROR_FILE, "event device not ready"); |
| XCAM_FAILED_STOP (ret = _subdevice->start(), "start event device failed"); |
| } |
| |
| if (_has_3a) { |
| // Initialize and start analyzer |
| uint32_t width = 0, height = 0; |
| uint32_t fps_n = 0, fps_d = 0; |
| double framerate = 30.0; |
| |
| if (!_3a_analyzer.ptr()) { |
| _3a_analyzer = X3aAnalyzerManager::instance()->create_analyzer(); |
| if (!_3a_analyzer.ptr()) { |
| XCAM_FAILED_STOP (ret = XCAM_RETURN_ERROR_PARAM, "create analyzer failed"); |
| } |
| } |
| if (_3a_analyzer->prepare_handlers () != XCAM_RETURN_NO_ERROR) { |
| XCAM_FAILED_STOP (ret = XCAM_RETURN_ERROR_PARAM, "prepare analyzer handler failed"); |
| } |
| _3a_analyzer->set_results_callback (this); |
| |
| _device->get_size (width, height); |
| _device->get_framerate (fps_n, fps_d); |
| if (fps_d) |
| framerate = (double)fps_n / (double)fps_d; |
| XCAM_FAILED_STOP ( |
| ret = _3a_analyzer->init (width, height, framerate), |
| "initialize analyzer failed"); |
| |
| XCAM_FAILED_STOP (ret = _3a_analyzer->start (), "start analyzer failed"); |
| |
| if (_smart_analyzer.ptr()) { |
| if (_smart_analyzer->prepare_handlers () != XCAM_RETURN_NO_ERROR) { |
| XCAM_LOG_INFO ("prepare smart analyzer handler failed"); |
| } |
| _smart_analyzer->set_results_callback (this); |
| if (_smart_analyzer->init (width, height, framerate) != XCAM_RETURN_NO_ERROR) { |
| XCAM_LOG_INFO ("initialize smart analyzer failed"); |
| } |
| if (_smart_analyzer->start () != XCAM_RETURN_NO_ERROR) { |
| XCAM_LOG_INFO ("start smart analyzer failed"); |
| } |
| } |
| |
| if (!_3a_process_center->has_processors ()) { |
| XCAM_LOG_ERROR ("image processors empty"); |
| } |
| |
| _3a_process_center->set_image_callback(this); |
| XCAM_FAILED_STOP (ret = _3a_process_center->start (), "3A process center start failed"); |
| |
| } |
| |
| //Initialize and start poll thread |
| XCAM_ASSERT (_poll_thread.ptr ()); |
| _poll_thread->set_capture_device (_device); |
| if (_subdevice.ptr ()) |
| _poll_thread->set_event_device (_subdevice); |
| _poll_thread->set_poll_callback (this); |
| _poll_thread->set_stats_callback (this); |
| |
| XCAM_FAILED_STOP (ret = _poll_thread->start(), "start poll failed"); |
| |
| _is_running = true; |
| |
| XCAM_LOG_DEBUG ("Device manager started"); |
| return XCAM_RETURN_NO_ERROR; |
| } |
| |
| XCamReturn |
| DeviceManager::stop () |
| { |
| _is_running = false; |
| |
| if (_poll_thread.ptr()) |
| _poll_thread->stop (); |
| |
| if (_3a_analyzer.ptr()) { |
| _3a_analyzer->stop (); |
| _3a_analyzer->deinit (); |
| } |
| if (_smart_analyzer.ptr()) { |
| _smart_analyzer->stop (); |
| _smart_analyzer->deinit (); |
| } |
| |
| if (_3a_process_center.ptr()) |
| _3a_process_center->stop (); |
| |
| if (_subdevice.ptr ()) |
| _subdevice->stop (); |
| |
| _device->stop (); |
| |
| _poll_thread.release (); |
| |
| XCAM_LOG_DEBUG ("Device manager stopped"); |
| return XCAM_RETURN_NO_ERROR; |
| } |
| |
| XCamReturn |
| DeviceManager::x3a_stats_ready (const SmartPtr<X3aStats> &stats) |
| { |
| XCamReturn ret = XCAM_RETURN_NO_ERROR; |
| X3aResultList results; |
| XCAM_ASSERT (_3a_analyzer.ptr()); |
| |
| ret = _3a_analyzer->push_3a_stats (stats); |
| XCAM_FAIL_RETURN (ERROR, |
| ret == XCAM_RETURN_NO_ERROR, |
| ret, |
| "analyze 3a statistics failed"); |
| |
| return XCAM_RETURN_NO_ERROR; |
| } |
| |
| XCamReturn |
| DeviceManager::dvs_stats_ready () |
| { |
| XCAM_ASSERT (false); |
| // TODO |
| return XCAM_RETURN_NO_ERROR; |
| } |
| |
| XCamReturn |
| DeviceManager::scaled_image_ready (const SmartPtr<VideoBuffer> &buffer) |
| { |
| XCamReturn ret = XCAM_RETURN_NO_ERROR; |
| if (!_smart_analyzer.ptr()) { |
| return XCAM_RETURN_NO_ERROR; |
| } |
| |
| ret = _smart_analyzer->push_buffer (buffer); |
| XCAM_FAIL_RETURN ( |
| ERROR, ret == XCAM_RETURN_NO_ERROR, ret, |
| "push frame buffer failed"); |
| |
| return XCAM_RETURN_NO_ERROR; |
| } |
| |
| |
| XCamReturn |
| DeviceManager::poll_buffer_ready (SmartPtr<VideoBuffer> &buf) |
| { |
| if (_has_3a) { |
| if (_3a_process_center->put_buffer (buf) == false) |
| return XCAM_RETURN_ERROR_UNKNOWN; |
| } |
| return XCAM_RETURN_NO_ERROR; |
| } |
| |
| XCamReturn |
| DeviceManager::poll_buffer_failed (int64_t timestamp, const char *msg) |
| { |
| post_message (XCAM_MESSAGE_BUF_ERROR, timestamp, msg); |
| return XCAM_RETURN_NO_ERROR; |
| } |
| |
| void |
| DeviceManager::x3a_calculation_done (XAnalyzer *analyzer, X3aResultList &results) |
| { |
| XCamReturn ret = _3a_process_center->put_3a_results (results); |
| if (ret != XCAM_RETURN_NO_ERROR && ret != XCAM_RETURN_BYPASS) { |
| XCAM_LOG_WARNING ("apply 3a results failed"); |
| return; |
| } |
| AnalyzerCallback::x3a_calculation_done (analyzer, results); |
| } |
| |
| void |
| DeviceManager::x3a_calculation_failed (XAnalyzer *analyzer, int64_t timestamp, const char *msg) |
| { |
| AnalyzerCallback::x3a_calculation_failed (analyzer, timestamp, msg); |
| } |
| |
| void |
| DeviceManager::process_buffer_done (ImageProcessor *processor, const SmartPtr<VideoBuffer> &buf) |
| { |
| ImageProcessCallback::process_buffer_done (processor, buf); |
| handle_buffer (buf); |
| } |
| |
| void |
| DeviceManager::process_buffer_failed (ImageProcessor *processor, const SmartPtr<VideoBuffer> &buf) |
| { |
| ImageProcessCallback::process_buffer_failed (processor, buf); |
| } |
| |
| void |
| DeviceManager::process_image_result_done (ImageProcessor *processor, const SmartPtr<X3aResult> &result) |
| { |
| ImageProcessCallback::process_image_result_done (processor, result); |
| } |
| |
| void |
| DeviceManager::post_message (XCamMessageType type, int64_t timestamp, const char *msg) |
| { |
| SmartPtr<XCamMessage> new_msg = new XCamMessage (type, timestamp, msg); |
| _msg_queue.push (new_msg); |
| } |
| |
| XCamReturn |
| DeviceManager::message_loop () |
| { |
| const static int32_t msg_time_out = -1; //wait until wakeup |
| SmartPtr<XCamMessage> msg = _msg_queue.pop (msg_time_out); |
| if (!msg.ptr ()) |
| return XCAM_RETURN_ERROR_THREAD; |
| handle_message (msg); |
| return XCAM_RETURN_NO_ERROR; |
| } |
| |
| }; |