blob: f299d46944fb8f0d7ef967de7096a12f112c206f [file] [log] [blame]
Wind Yuan75564b12015-01-15 06:51:15 -05001/*
2 * main.cpp - test
3 *
4 * Copyright (c) 2014 Intel Corporation
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 *
18 * Author: Wind Yuan <feng.yuan@intel.com>
Wind Yuan136aa4f2016-09-08 02:56:31 -040019 * Author: Yinhang Liu <yinhangx.liu@intel.com>
20 * Author: Wei Zong <wei.zong@intel.com>
Wind Yuan75564b12015-01-15 06:51:15 -050021 */
22
23#include "device_manager.h"
Sameer Kibey7b429292015-08-07 10:55:38 -070024#include "uvc_device.h"
Jia Meng0a532932015-10-15 08:44:09 -040025#include "fake_v4l2_device.h"
Wind Yuan75564b12015-01-15 06:51:15 -050026#include "x3a_analyzer_simple.h"
Yinhang Liue37a8722016-06-27 18:14:35 +080027#include "analyzer_loader.h"
Yinhang Liu4afa3642016-05-13 19:13:07 +080028#include "smart_analyzer_loader.h"
Wind Yuan75564b12015-01-15 06:51:15 -050029#if HAVE_IA_AIQ
Yinhang Liu7110f892017-06-30 13:11:10 +080030#include "isp/atomisp_device.h"
31#include "isp/isp_controller.h"
32#include "isp/isp_image_processor.h"
33#include "isp/isp_poll_thread.h"
34#include "isp/x3a_analyzer_aiq.h"
Yinhang Liu6c367352016-03-31 20:23:22 +080035#include "x3a_analyze_tuner.h"
Yinhang Liu454a3892016-11-07 16:30:35 +080036#include "dynamic_analyzer_loader.h"
Yinhang Liu7110f892017-06-30 13:11:10 +080037#include "isp/hybrid_analyzer_loader.h"
Wind Yuan75564b12015-01-15 06:51:15 -050038#endif
Wind Yuand4427312015-02-11 16:09:00 +080039#if HAVE_LIBCL
Yinhang Liu7110f892017-06-30 13:11:10 +080040#include "ocl/cl_3a_image_processor.h"
41#include "ocl/cl_post_image_processor.h"
42#include "ocl/cl_csc_image_processor.h"
43#include "ocl/cl_tnr_handler.h"
Wind Yuane4ba0c92015-03-26 16:59:06 +080044#endif
45#if HAVE_LIBDRM
46#include "drm_display.h"
Wind Yuand4427312015-02-11 16:09:00 +080047#endif
Jia Meng0a532932015-10-15 08:44:09 -040048#include "fake_poll_thread.h"
Wind Yuan136aa4f2016-09-08 02:56:31 -040049#include "image_file_handle.h"
Jia Meng66efecf2015-06-17 11:11:10 +000050#include <base/xcam_3a_types.h>
Wind Yuan75564b12015-01-15 06:51:15 -050051#include <unistd.h>
52#include <signal.h>
53#include <stdlib.h>
ShincyTue521cf92015-03-27 15:13:51 +080054#include <string>
Yinhang Liu81c44e02015-06-03 15:47:48 +080055#include <getopt.h>
John Yeae9da732015-01-22 13:36:36 +080056#include "test_common.h"
57
Wind Yuan75564b12015-01-15 06:51:15 -050058using namespace XCam;
59
wujunkai16682950fb2015-10-19 18:09:05 +080060#define IMX185_WDR_CPF "/etc/atomisp/imx185_wdr.cpf"
61
ShincyTue521cf92015-03-27 15:13:51 +080062static Mutex g_mutex;
63static Cond g_cond;
64static bool g_stop = false;
65
Wind Yuan75564b12015-01-15 06:51:15 -050066class MainDeviceManager
67 : public DeviceManager
68{
69public:
70 MainDeviceManager ()
Wind Yuan136aa4f2016-09-08 02:56:31 -040071 : _save_file (false)
Wind Yuan75564b12015-01-15 06:51:15 -050072 , _interval (1)
Yinhang Liud9346492015-12-17 11:14:25 +080073 , _frame_width (0)
74 , _frame_height (0)
Wind Yuan75564b12015-01-15 06:51:15 -050075 , _frame_count (0)
ShincyTue521cf92015-03-27 15:13:51 +080076 , _frame_save (0)
Wind Yuan683f8662015-03-27 18:52:38 +080077 , _enable_display (false)
Wind Yuane4ba0c92015-03-26 16:59:06 +080078 {
79#if HAVE_LIBDRM
Yinhang Liu5855e6f2016-11-14 21:04:22 +080080 _display = DrmDisplay::instance ();
Wind Yuane4ba0c92015-03-26 16:59:06 +080081#endif
Wind Yuan91625802015-06-24 15:36:01 +080082 XCAM_OBJ_PROFILING_INIT;
Wind Yuane4ba0c92015-03-26 16:59:06 +080083 }
Wind Yuan75564b12015-01-15 06:51:15 -050084
85 ~MainDeviceManager () {
Wind Yuan136aa4f2016-09-08 02:56:31 -040086 _file_handle.close ();
Wind Yuan75564b12015-01-15 06:51:15 -050087 }
88
89 void enable_save_file (bool enable) {
90 _save_file = enable;
91 }
Yinhang Liud9346492015-12-17 11:14:25 +080092
Wind Yuan75564b12015-01-15 06:51:15 -050093 void set_interval (uint32_t inteval) {
94 _interval = inteval;
95 }
Yinhang Liud9346492015-12-17 11:14:25 +080096
97 void set_frame_width (uint32_t frame_width) {
98 _frame_width = frame_width;
99 }
100
101 void set_frame_height (uint32_t frame_height) {
102 _frame_height = frame_height;
103 }
104
ShincyTue521cf92015-03-27 15:13:51 +0800105 void set_frame_save (uint32_t frame_save) {
106 _frame_save = frame_save;
107 }
Wind Yuan75564b12015-01-15 06:51:15 -0500108
Wind Yuan683f8662015-03-27 18:52:38 +0800109 void enable_display(bool value) {
110 _enable_display = value;
111 }
112
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800113#if HAVE_LIBDRM
wangfeicf4c4e72015-05-25 17:11:41 +0800114 void set_display_mode(DrmDisplayMode mode) {
115 _display->set_display_mode (mode);
116 }
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800117#endif
wangfeicf4c4e72015-05-25 17:11:41 +0800118
Wind Yuan75564b12015-01-15 06:51:15 -0500119protected:
Wind Yuan73af3932015-07-02 17:52:43 +0800120 virtual void handle_message (const SmartPtr<XCamMessage> &msg);
121 virtual void handle_buffer (const SmartPtr<VideoBuffer> &buf);
Wind Yuan75564b12015-01-15 06:51:15 -0500122
Wind Yuan73af3932015-07-02 17:52:43 +0800123 int display_buf (const SmartPtr<VideoBuffer> &buf);
Wind Yuane4ba0c92015-03-26 16:59:06 +0800124
Wind Yuan75564b12015-01-15 06:51:15 -0500125private:
126 void open_file ();
Wind Yuan75564b12015-01-15 06:51:15 -0500127
Wind Yuan136aa4f2016-09-08 02:56:31 -0400128private:
Wind Yuan75564b12015-01-15 06:51:15 -0500129 bool _save_file;
130 uint32_t _interval;
Yinhang Liud9346492015-12-17 11:14:25 +0800131 uint32_t _frame_width;
132 uint32_t _frame_height;
Wind Yuan75564b12015-01-15 06:51:15 -0500133 uint32_t _frame_count;
ShincyTue521cf92015-03-27 15:13:51 +0800134 uint32_t _frame_save;
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800135 bool _enable_display;
Wind Yuan136aa4f2016-09-08 02:56:31 -0400136 ImageFileHandle _file_handle;
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800137#if HAVE_LIBDRM
Wind Yuane4ba0c92015-03-26 16:59:06 +0800138 SmartPtr<DrmDisplay> _display;
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800139#endif
Wind Yuan91625802015-06-24 15:36:01 +0800140 XCAM_OBJ_PROFILING_DEFINES;
Wind Yuan75564b12015-01-15 06:51:15 -0500141};
142
143void
Wind Yuan73af3932015-07-02 17:52:43 +0800144MainDeviceManager::handle_message (const SmartPtr<XCamMessage> &msg)
Wind Yuan75564b12015-01-15 06:51:15 -0500145{
146 XCAM_UNUSED (msg);
147}
148
149void
Wind Yuan73af3932015-07-02 17:52:43 +0800150MainDeviceManager::handle_buffer (const SmartPtr<VideoBuffer> &buf)
Wind Yuan75564b12015-01-15 06:51:15 -0500151{
Wind Yuan6d144ce2017-08-16 18:29:29 +0800152 if (!buf.ptr ()) {
153 XCAM_LOG_WARNING ("video buffer is null, handle buffer failed.");
154 return;
155 }
156
Wind Yuan683f8662015-03-27 18:52:38 +0800157 FPS_CALCULATION (fps_buf, 30);
Wind Yuan91625802015-06-24 15:36:01 +0800158 XCAM_OBJ_PROFILING_START;
159
Wind Yuan683f8662015-03-27 18:52:38 +0800160 if (_enable_display)
161 display_buf (buf);
Wind Yuane4ba0c92015-03-26 16:59:06 +0800162
Yinhang Liu070c7592017-04-13 19:22:17 +0800163 XCAM_OBJ_PROFILING_END("main_dev_manager_display", XCAM_OBJ_DUR_FRAME_NUM);
Wind Yuan91625802015-06-24 15:36:01 +0800164
Wind Yuan75564b12015-01-15 06:51:15 -0500165 if (!_save_file)
166 return ;
167
168 if ((_frame_count++ % _interval) != 0)
169 return;
170
ShincyTue521cf92015-03-27 15:13:51 +0800171 if ((_frame_save != 0) && (_frame_count > _frame_save)) {
172 SmartLock locker (g_mutex);
173 g_stop = true;
174 g_cond.broadcast ();
175 return;
176 }
177
Wind Yuan75564b12015-01-15 06:51:15 -0500178 open_file ();
Jia Meng6c9241d2015-06-12 11:08:45 +0000179
Wind Yuan136aa4f2016-09-08 02:56:31 -0400180 if (!_file_handle.is_valid ()) {
Jia Meng6c9241d2015-06-12 11:08:45 +0000181 XCAM_LOG_ERROR ("open file failed");
Jia Meng54a2dcb2015-08-20 15:23:47 +0800182 return;
Jia Meng6c9241d2015-06-12 11:08:45 +0000183 }
Wind Yuan6d144ce2017-08-16 18:29:29 +0800184 _file_handle.write_buf (buf);
Wind Yuan75564b12015-01-15 06:51:15 -0500185}
186
Wind Yuane4ba0c92015-03-26 16:59:06 +0800187int
Wind Yuan73af3932015-07-02 17:52:43 +0800188MainDeviceManager::display_buf (const SmartPtr<VideoBuffer> &data)
Wind Yuane4ba0c92015-03-26 16:59:06 +0800189{
190#if HAVE_LIBDRM
191 XCamReturn ret = XCAM_RETURN_NO_ERROR;
Wind Yuan73af3932015-07-02 17:52:43 +0800192 SmartPtr<VideoBuffer> buf = data;
Wind Yuane4ba0c92015-03-26 16:59:06 +0800193 const VideoBufferInfo & frame_info = buf->get_video_info ();
Yinhang Liuaadb8ba2016-11-18 18:43:29 +0800194 struct v4l2_rect rect = { 0, 0, frame_info.width, frame_info.height};
Wind Yuane4ba0c92015-03-26 16:59:06 +0800195
196 if (!_display->is_render_inited ()) {
Yinhang Liud9346492015-12-17 11:14:25 +0800197 ret = _display->render_init (0, 0, this->_frame_width, this->_frame_height,
198 frame_info.format, &rect);
Wind Yuane4ba0c92015-03-26 16:59:06 +0800199 CHECK (ret, "display failed on render_init");
200 }
201 ret = _display->render_setup_frame_buffer (buf);
202 CHECK (ret, "display failed on framebuf set");
203 ret = _display->render_buffer (buf);
204 CHECK (ret, "display failed on rendering");
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800205#else
206 XCAM_UNUSED (data);
Wind Yuane4ba0c92015-03-26 16:59:06 +0800207#endif
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800208
Wind Yuane4ba0c92015-03-26 16:59:06 +0800209 return 0;
210}
211
212
Wind Yuan75564b12015-01-15 06:51:15 -0500213void
214MainDeviceManager::open_file ()
215{
Wind Yuan136aa4f2016-09-08 02:56:31 -0400216 if (_file_handle.is_valid () && (_frame_save == 0))
Wind Yuan75564b12015-01-15 06:51:15 -0500217 return;
ShincyTue521cf92015-03-27 15:13:51 +0800218
219 std::string file_name = DEFAULT_SAVE_FILE_NAME;
220
221 if (_frame_save != 0) {
222 file_name += std::to_string(_frame_count);
223 }
224 file_name += ".raw";
225
Wind Yuan136aa4f2016-09-08 02:56:31 -0400226 if (_file_handle.open (file_name.c_str (), "wb") != XCAM_RETURN_NO_ERROR) {
227 XCAM_LOG_WARNING ("create file(%s) failed", file_name.c_str ());
Wind Yuan71111322016-02-24 22:56:27 +0800228 }
Wind Yuan71111322016-02-24 22:56:27 +0800229}
230
Wind Yuan75564b12015-01-15 06:51:15 -0500231#define V4L2_CAPTURE_MODE_STILL 0x2000
232#define V4L2_CAPTURE_MODE_VIDEO 0x4000
233#define V4L2_CAPTURE_MODE_PREVIEW 0x8000
234
Wind Yuan26e9e212015-04-16 15:55:45 +0800235typedef enum {
236 AnalyzerTypeSimple = 0,
Yinhang Liu6c367352016-03-31 20:23:22 +0800237 AnalyzerTypeAiqTuner,
Wind Yuane25ce3f2015-05-04 18:07:29 +0800238 AnalyzerTypeDynamic,
Jia Meng9724cfe2015-08-12 15:04:45 +0800239 AnalyzerTypeHybrid,
Wind Yuan26e9e212015-04-16 15:55:45 +0800240} AnalyzerType;
241
Wind Yuan75564b12015-01-15 06:51:15 -0500242void dev_stop_handler(int sig)
243{
244 XCAM_UNUSED (sig);
245
246 SmartLock locker (g_mutex);
247 g_stop = true;
248 g_cond.broadcast ();
249
250 //exit(0);
251}
252
253void print_help (const char *bin_name)
254{
255 printf ("Usage: %s [-a analyzer]\n"
Yinhang Liu81c44e02015-06-03 15:47:48 +0800256 "Configurations:\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800257 "\t -a analyzer specify a analyzer\n"
258 "\t select from [simple"
259#if HAVE_IA_AIQ
zongwave213b5d62016-11-25 18:46:53 +0800260 ", aiq"
Yinhang Liuc2f19082015-08-28 12:27:48 +0800261#if HAVE_LIBCL
zongwave213b5d62016-11-25 18:46:53 +0800262 ", dynamic, hybrid"
Yinhang Liuc2f19082015-08-28 12:27:48 +0800263#endif
Yinhang Liu454a3892016-11-07 16:30:35 +0800264#endif
zongwave213b5d62016-11-25 18:46:53 +0800265 "], default is [simple]\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800266 "\t -m mem_type specify video memory type\n"
267 "\t mem_type select from [dma, mmap], default is [mmap]\n"
268 "\t -s save file to %s\n"
269 "\t -n interval save file on every [interval] frame\n"
270 "\t -f pixel_fmt specify output pixel format\n"
271 "\t pixel_fmt select from [NV12, YUYV, BA10, BA12], default is [NV12]\n"
Yinhang Liu2bdc67a2017-02-28 11:27:49 +0800272 "\t -W image_width specify image width, default is [1920]\n"
273 "\t -H image_height specify image height, default is [1080]\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800274 "\t -d cap_mode specify capture mode\n"
275 "\t cap_mode select from [video, still], default is [video]\n"
276 "\t -i frame_save specify the frame count to save, default is 0 which means endless\n"
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800277 "\t -p preview on enable local display, need root privilege\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800278 "\t --usb specify node for usb camera device, enables capture path through USB camera \n"
279 "\t specify [/dev/video4, /dev/video5] depending on which node USB camera is attached\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800280 "\t -e display_mode preview mode\n"
281 "\t select from [primary, overlay], default is [primary]\n"
282 "\t --sync set analyzer in sync mode\n"
283 "\t -r raw_input specify the path of raw image as fake source instead of live camera\n"
284 "\t -h help\n"
Yinhang Liuc2f19082015-08-28 12:27:48 +0800285#if HAVE_LIBCL
Yinhang Liu81c44e02015-06-03 15:47:48 +0800286 "CL features:\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800287 "\t -c process image with cl kernel\n"
288#if HAVE_IA_AIQ
289 "\t -b brightness specify brightness level\n"
290 "\t brightness level select from [0, 256], default is [128]\n"
291#endif
292 "\t --capture specify the capture stage of image\n"
293 "\t capture_stage select from [bayer, tonemapping], default is [tonemapping]\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800294 "\t --tnr specify temporal noise reduction type, default is tnr off\n"
Wind Yuan9d5e6a32017-01-26 03:09:39 -0500295 "\t only support [yuv]\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800296 "\t --tnr-level specify tnr level\n"
297 "\t --wdr-mode specify wdr mode. select from [gaussian, haleq]\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800298 "\t --enable-bnr enable bayer noise reduction\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800299 "\t --defog-mode specify defog mode\n"
300 "\t select from [disabled, retinex, dcp], default is [disabled]\n"
301 "\t --wavelet-mode specify wavelet denoise mode, default is off\n"
302 "\t select from [0:disable, 1:Hat Y, 2:Hat UV, 3:Haar Y, 4:Haar UV, 5:Haar YUV, 6:Haar Bayes Shrink]\n"
303 "\t --3d-denoise specify 3D Denoise mode\n"
304 "\t select from [disabled, yuv, uv], default is [disabled]\n"
Yinhang Liu4ea70612016-05-26 18:26:29 +0800305 "\t --enable-wireframe enable wire frame\n"
Yinhang Liu454a3892016-11-07 16:30:35 +0800306 "\t --pipeline specify pipe mode\n"
307 "\t select from [basic, advance, extreme], default is [basic]\n"
308 "\t --disable-post disable cl post image processor\n"
Yinhang Liuc2f19082015-08-28 12:27:48 +0800309#endif
Wind Yuan75564b12015-01-15 06:51:15 -0500310 , bin_name
311 , DEFAULT_SAVE_FILE_NAME);
312}
313
314int main (int argc, char *argv[])
315{
316 XCamReturn ret = XCAM_RETURN_NO_ERROR;
Wind Yuan75564b12015-01-15 06:51:15 -0500317 SmartPtr<V4l2Device> device;
Yinhang Liu454a3892016-11-07 16:30:35 +0800318#if HAVE_IA_AIQ
Yinhang Liu5c1381c2016-11-19 16:35:38 +0800319 SmartPtr<V4l2SubDevice> event_device;
Wind Yuan75564b12015-01-15 06:51:15 -0500320 SmartPtr<IspController> isp_controller;
Wind Yuane4ba0c92015-03-26 16:59:06 +0800321 SmartPtr<ImageProcessor> isp_processor;
Yinhang Liu454a3892016-11-07 16:30:35 +0800322#endif
323 SmartPtr<X3aAnalyzer> analyzer;
324 SmartPtr<AnalyzerLoader> loader;
Wind Yuan26e9e212015-04-16 15:55:45 +0800325 AnalyzerType analyzer_type = AnalyzerTypeSimple;
Wind Yuane4ba0c92015-03-26 16:59:06 +0800326
Wind Yuand4427312015-02-11 16:09:00 +0800327#if HAVE_LIBCL
Yinhang Liu454a3892016-11-07 16:30:35 +0800328 bool have_cl_processor = false;
329 SmartPtr<SmartAnalyzer> smart_analyzer;
Yinhang Liuc64ed592016-07-15 18:53:34 +0800330 bool have_cl_post_processor = true;
Wind Yuane4ba0c92015-03-26 16:59:06 +0800331 SmartPtr<CL3aImageProcessor> cl_processor;
Yinhang Liu2ab1f992016-03-15 10:41:17 +0800332 SmartPtr<CLPostImageProcessor> cl_post_processor;
Yinhang Liuc2f19082015-08-28 12:27:48 +0800333 uint32_t tnr_type = CL_TNR_DISABLE;
334 uint32_t denoise_type = 0;
335 uint8_t tnr_level = 0;
Wangfei1487b9b2015-08-28 15:10:39 +0800336 CL3aImageProcessor::PipelineProfile pipeline_mode = CL3aImageProcessor::BasicPipelineProfile;
wujunkai166a84ba052015-09-08 15:54:12 +0800337 CL3aImageProcessor::CaptureStage capture_stage = CL3aImageProcessor::TonemappingStage;
wujunkai16606f77622016-03-18 14:41:10 +0800338 CL3aImageProcessor::CLTonemappingMode wdr_mode = CL3aImageProcessor::WDRdisabled;
Yinhang Liuc64ed592016-07-15 18:53:34 +0800339
Yinhang Liu454a3892016-11-07 16:30:35 +0800340#if HAVE_IA_AIQ
341 int32_t brightness_level = 128;
342#endif
Wind Yuanf4e17ab2016-05-05 02:54:01 +0800343 uint32_t defog_type = 0;
zongwaveb7140442016-03-25 18:15:12 +0800344 CLWaveletBasis wavelet_mode = CL_WAVELET_DISABLED;
zongwave98d23202016-06-27 13:46:27 +0800345 uint32_t wavelet_channel = CL_IMAGE_CHANNEL_UV;
346 bool wavelet_bayes_shrink = false;
Yinhang Liud341dbe2016-07-25 12:00:17 +0800347 uint32_t denoise_3d_mode = 0;
348 uint8_t denoise_3d_ref_count = 3;
Yinhang Liu4ea70612016-05-26 18:26:29 +0800349 bool wireframe_type = false;
zongwave1b10d6f2016-10-25 17:47:51 +0800350 bool image_warp_type = false;
Yinhang Liuc64ed592016-07-15 18:53:34 +0800351#endif
zongwaveb7140442016-03-25 18:15:12 +0800352
Yinhang Liu454a3892016-11-07 16:30:35 +0800353 bool need_display = false;
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800354#if HAVE_LIBDRM
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800355 DrmDisplayMode display_mode = DRM_DISPLAY_MODE_PRIMARY;
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800356#endif
Yinhang Liu454a3892016-11-07 16:30:35 +0800357 enum v4l2_memory v4l2_mem_type = V4L2_MEMORY_MMAP;
358 const char *bin_name = argv[0];
359 uint32_t capture_mode = V4L2_CAPTURE_MODE_VIDEO;
360 uint32_t pixel_format = V4L2_PIX_FMT_NV12;
361
Sameer Kibey7b429292015-08-07 10:55:38 -0700362 bool have_usbcam = 0;
Wind Yuan28935912017-08-24 19:18:34 +0800363 std::string usb_device_name;
Jia Meng8f94a102015-08-11 16:24:19 +0800364 bool sync_mode = false;
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800365 bool save_file = false;
366 uint32_t interval_frames = 1;
367 uint32_t save_frames = 0;
368 uint32_t frame_rate;
369 uint32_t frame_width = 1920;
370 uint32_t frame_height = 1080;
Wind Yuan28935912017-08-24 19:18:34 +0800371 std::string path_to_fake;
Wind Yuan75564b12015-01-15 06:51:15 -0500372
Yinhang Liu454a3892016-11-07 16:30:35 +0800373 int opt;
Yinhang Liu2bdc67a2017-02-28 11:27:49 +0800374 const char *short_opts = "sca:n:m:f:W:H:d:b:pi:e:r:h";
Yinhang Liu81c44e02015-06-03 15:47:48 +0800375 const struct option long_opts[] = {
Yinhang Liu81c44e02015-06-03 15:47:48 +0800376 {"tnr", required_argument, NULL, 'T'},
377 {"tnr-level", required_argument, NULL, 'L'},
Yinhang Liu2bdc67a2017-02-28 11:27:49 +0800378 {"wdr-mode", required_argument, NULL, 'w'},
ShincyTu63e2c262015-06-26 10:52:51 +0800379 {"enable-bnr", no_argument, NULL, 'B'},
Wind Yuanf4e17ab2016-05-05 02:54:01 +0800380 {"defog-mode", required_argument, NULL, 'X'},
zongwave2ba7f642016-03-15 19:24:12 +0800381 {"wavelet-mode", required_argument, NULL, 'V'},
zongwavebec89e12016-07-23 12:34:20 +0800382 {"3d-denoise", required_argument, NULL, 'N'},
Yinhang Liu4ea70612016-05-26 18:26:29 +0800383 {"enable-wireframe", no_argument, NULL, 'F'},
zongwave1b10d6f2016-10-25 17:47:51 +0800384 {"enable-warp", no_argument, NULL, 'A'},
Sameer Kibey7b429292015-08-07 10:55:38 -0700385 {"usb", required_argument, NULL, 'U'},
Jia Meng8f94a102015-08-11 16:24:19 +0800386 {"sync", no_argument, NULL, 'Y'},
wujunkai166a84ba052015-09-08 15:54:12 +0800387 {"capture", required_argument, NULL, 'C'},
Wangfei1487b9b2015-08-28 15:10:39 +0800388 {"pipeline", required_argument, NULL, 'P'},
Yinhang Liub0ad64b2016-03-18 16:03:51 +0800389 {"disable-post", no_argument, NULL, 'O'},
Yinhang Liu81c44e02015-06-03 15:47:48 +0800390 {0, 0, 0, 0},
391 };
392
393 while ((opt = getopt_long(argc, argv, short_opts, long_opts, NULL)) != -1) {
Wind Yuan75564b12015-01-15 06:51:15 -0500394 switch (opt) {
395 case 'a': {
Wind Yuan6b437392016-02-02 00:31:30 +0800396 XCAM_ASSERT (optarg);
Yinhang Liu454a3892016-11-07 16:30:35 +0800397 if (!strcmp (optarg, "simple"))
Wind Yuan26e9e212015-04-16 15:55:45 +0800398 analyzer_type = AnalyzerTypeSimple;
Wind Yuan75564b12015-01-15 06:51:15 -0500399#if HAVE_IA_AIQ
400 else if (!strcmp (optarg, "aiq"))
Yinhang Liu6c367352016-03-31 20:23:22 +0800401 analyzer_type = AnalyzerTypeAiqTuner;
Yinhang Liu454a3892016-11-07 16:30:35 +0800402#if HAVE_LIBCL
403 else if (!strcmp (optarg, "dynamic"))
404 analyzer_type = AnalyzerTypeDynamic;
Jia Meng9724cfe2015-08-12 15:04:45 +0800405 else if (!strcmp (optarg, "hybrid"))
406 analyzer_type = AnalyzerTypeHybrid;
Wind Yuan75564b12015-01-15 06:51:15 -0500407#endif
Yinhang Liu454a3892016-11-07 16:30:35 +0800408#endif
Wind Yuan75564b12015-01-15 06:51:15 -0500409 else {
410 print_help (bin_name);
411 return -1;
412 }
413 break;
414 }
415
Wind Yuand4427312015-02-11 16:09:00 +0800416 case 'm': {
Wind Yuan6b437392016-02-02 00:31:30 +0800417 XCAM_ASSERT (optarg);
Wind Yuand4427312015-02-11 16:09:00 +0800418 if (!strcmp (optarg, "dma"))
419 v4l2_mem_type = V4L2_MEMORY_DMABUF;
420 else if (!strcmp (optarg, "mmap"))
421 v4l2_mem_type = V4L2_MEMORY_MMAP;
422 else
423 print_help (bin_name);
424 break;
425 }
426
Wind Yuan75564b12015-01-15 06:51:15 -0500427 case 's':
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800428 save_file = true;
Wind Yuan75564b12015-01-15 06:51:15 -0500429 break;
430 case 'n':
Yinhang Liu1aa97602016-05-27 14:58:21 +0800431 XCAM_ASSERT (optarg);
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800432 interval_frames = atoi(optarg);
433 break;
434 case 'i':
435 XCAM_ASSERT (optarg);
436 save_frames = atoi(optarg);
Wind Yuan75564b12015-01-15 06:51:15 -0500437 break;
ShincyTu2808b592015-03-13 17:17:25 +0800438 case 'f':
Yinhang Liu1aa97602016-05-27 14:58:21 +0800439 XCAM_ASSERT (optarg);
ShincyTuc8466402015-03-31 11:03:39 +0800440 CHECK_EXP ((strlen(optarg) == 4), "invalid pixel format\n");
441 pixel_format = v4l2_fourcc ((unsigned)optarg[0],
442 (unsigned)optarg[1],
443 (unsigned)optarg[2],
444 (unsigned)optarg[3]);
ShincyTu2808b592015-03-13 17:17:25 +0800445 break;
446 case 'd':
Wind Yuan6b437392016-02-02 00:31:30 +0800447 XCAM_ASSERT (optarg);
ShincyTu2808b592015-03-13 17:17:25 +0800448 if (!strcmp (optarg, "still"))
449 capture_mode = V4L2_CAPTURE_MODE_STILL;
450 else if (!strcmp (optarg, "video"))
451 capture_mode = V4L2_CAPTURE_MODE_VIDEO;
452 else {
453 print_help (bin_name);
454 return -1;
455 }
456 break;
Sameer Kibey7b429292015-08-07 10:55:38 -0700457 case 'U':
Yinhang Liu1aa97602016-05-27 14:58:21 +0800458 XCAM_ASSERT (optarg);
Sameer Kibey7b429292015-08-07 10:55:38 -0700459 have_usbcam = true;
Wind Yuan28935912017-08-24 19:18:34 +0800460 usb_device_name = optarg;
461 XCAM_LOG_DEBUG("using USB camera plugged in at node: %s", XCAM_STR(usb_device_name.c_str()));
Sameer Kibey7b429292015-08-07 10:55:38 -0700462 break;
Yinhang Liu2bdc67a2017-02-28 11:27:49 +0800463 case 'W':
Wind Yuan6b437392016-02-02 00:31:30 +0800464 XCAM_ASSERT (optarg);
Yinhang Liu2bdc67a2017-02-28 11:27:49 +0800465 frame_width = atoi(optarg);
466 break;
467 case 'H':
468 XCAM_ASSERT (optarg);
469 frame_height = atoi(optarg);
Yinhang Liud9346492015-12-17 11:14:25 +0800470 break;
wangfeicf4c4e72015-05-25 17:11:41 +0800471 case 'e': {
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800472#if HAVE_LIBDRM
Wind Yuan6b437392016-02-02 00:31:30 +0800473 XCAM_ASSERT (optarg);
wangfeicf4c4e72015-05-25 17:11:41 +0800474 if (!strcmp (optarg, "primary"))
475 display_mode = DRM_DISPLAY_MODE_PRIMARY;
476 else if (!strcmp (optarg, "overlay"))
477 display_mode = DRM_DISPLAY_MODE_OVERLAY;
478 else {
479 print_help (bin_name);
480 return -1;
481 }
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800482#else
483 XCAM_LOG_WARNING ("preview is not supported");
484#endif
wangfeicf4c4e72015-05-25 17:11:41 +0800485 break;
486 }
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800487
Jia Meng8f94a102015-08-11 16:24:19 +0800488 case 'Y':
489 sync_mode = true;
490 break;
Yinhang Liuc2f19082015-08-28 12:27:48 +0800491#if HAVE_LIBCL
Yinhang Liu454a3892016-11-07 16:30:35 +0800492 case 'c':
493 have_cl_processor = true;
494 break;
495#if HAVE_IA_AIQ
496 case 'b':
497 XCAM_ASSERT (optarg);
498 brightness_level = atoi(optarg);
499 if(brightness_level < 0 || brightness_level > 256) {
500 print_help (bin_name);
501 return -1;
502 }
503 break;
504#endif
Wind Yuan9d5e6a32017-01-26 03:09:39 -0500505
ShincyTu63e2c262015-06-26 10:52:51 +0800506 case 'B': {
507 denoise_type |= XCAM_DENOISE_TYPE_BNR;
508 break;
509 }
Wangfei8e5e3e42016-02-18 19:41:54 +0800510 case 'X': {
Wind Yuanf4e17ab2016-05-05 02:54:01 +0800511 XCAM_ASSERT (optarg);
512 defog_type = true;
513 if (!strcmp (optarg, "disabled"))
514 defog_type = CLPostImageProcessor::DefogDisabled;
515 else if (!strcmp (optarg, "retinex"))
516 defog_type = CLPostImageProcessor::DefogRetinex;
517 else if (!strcmp (optarg, "dcp"))
518 defog_type = CLPostImageProcessor::DefogDarkChannelPrior;
519 else {
520 print_help (bin_name);
521 return -1;
522 }
Wangfei8e5e3e42016-02-18 19:41:54 +0800523 break;
524 }
zongwave80adcf52016-02-29 12:38:06 +0800525 case 'V': {
zongwave2ba7f642016-03-15 19:24:12 +0800526 XCAM_ASSERT (optarg);
527 if (atoi(optarg) < 0 || atoi(optarg) > 255) {
528 print_help (bin_name);
529 return -1;
530 }
zongwave28a747b2016-03-17 12:49:08 +0800531 if (atoi(optarg) == 1) {
zongwaveb7140442016-03-25 18:15:12 +0800532 wavelet_mode = CL_WAVELET_HAT;
zongwave98d23202016-06-27 13:46:27 +0800533 wavelet_channel = CL_IMAGE_CHANNEL_Y;
zongwave28a747b2016-03-17 12:49:08 +0800534 } else if (atoi(optarg) == 2) {
zongwaveb7140442016-03-25 18:15:12 +0800535 wavelet_mode = CL_WAVELET_HAT;
zongwave98d23202016-06-27 13:46:27 +0800536 wavelet_channel = CL_IMAGE_CHANNEL_UV;
zongwaveb7140442016-03-25 18:15:12 +0800537 } else if (atoi(optarg) == 3) {
538 wavelet_mode = CL_WAVELET_HAAR;
zongwave98d23202016-06-27 13:46:27 +0800539 wavelet_channel = CL_IMAGE_CHANNEL_Y;
zongwaveb7140442016-03-25 18:15:12 +0800540 } else if (atoi(optarg) == 4) {
541 wavelet_mode = CL_WAVELET_HAAR;
zongwave98d23202016-06-27 13:46:27 +0800542 wavelet_channel = CL_IMAGE_CHANNEL_UV;
zongwaveb7140442016-03-25 18:15:12 +0800543 } else if (atoi(optarg) == 5) {
544 wavelet_mode = CL_WAVELET_HAAR;
zongwave98d23202016-06-27 13:46:27 +0800545 wavelet_channel = CL_IMAGE_CHANNEL_UV | CL_IMAGE_CHANNEL_Y;
546 } else if (atoi(optarg) == 6) {
547 wavelet_mode = CL_WAVELET_HAAR;
548 wavelet_channel = CL_IMAGE_CHANNEL_UV | CL_IMAGE_CHANNEL_Y;
549 wavelet_bayes_shrink = true;
zongwave28a747b2016-03-17 12:49:08 +0800550 } else {
zongwaveb7140442016-03-25 18:15:12 +0800551 wavelet_mode = CL_WAVELET_DISABLED;
zongwave28a747b2016-03-17 12:49:08 +0800552 }
zongwave80adcf52016-02-29 12:38:06 +0800553 break;
554 }
zongwavebec89e12016-07-23 12:34:20 +0800555 case 'N': {
556 XCAM_ASSERT (optarg);
557 denoise_3d_mode = true;
558 if (!strcmp (optarg, "disabled"))
559 denoise_3d_mode = CLPostImageProcessor::Denoise3DDisabled;
560 else if (!strcmp (optarg, "yuv"))
561 denoise_3d_mode = CLPostImageProcessor::Denoise3DYuv;
562 else if (!strcmp (optarg, "uv"))
563 denoise_3d_mode = CLPostImageProcessor::Denoise3DUV;
564 else {
565 print_help (bin_name);
566 return -1;
567 }
568 break;
569 }
Yinhang Liu4ea70612016-05-26 18:26:29 +0800570 case 'F': {
571 wireframe_type = true;
572 break;
573 }
zongwave1b10d6f2016-10-25 17:47:51 +0800574 case 'A': {
575 image_warp_type = true;
576 break;
577 }
Yinhang Liu81c44e02015-06-03 15:47:48 +0800578 case 'T': {
Wind Yuan6b437392016-02-02 00:31:30 +0800579 XCAM_ASSERT (optarg);
Yinhang Liu81c44e02015-06-03 15:47:48 +0800580 if (!strcasecmp (optarg, "yuv"))
581 tnr_type = CL_TNR_TYPE_YUV;
Yinhang Liu81c44e02015-06-03 15:47:48 +0800582 else {
Wind Yuan9d5e6a32017-01-26 03:09:39 -0500583 printf ("--tnr only support <yuv>, <%s> is not supported\n", optarg);
Yinhang Liu81c44e02015-06-03 15:47:48 +0800584 print_help (bin_name);
585 return -1;
586 }
587 break;
588 }
589 case 'L': {
Wind Yuan6b437392016-02-02 00:31:30 +0800590 XCAM_ASSERT (optarg);
Yinhang Liu81c44e02015-06-03 15:47:48 +0800591 if (atoi(optarg) < 0 || atoi(optarg) > 255) {
592 print_help (bin_name);
593 return -1;
594 }
595 tnr_level = atoi(optarg);
596 break;
597 }
Yinhang Liu2bdc67a2017-02-28 11:27:49 +0800598 case 'w': {
wujunkai16606f77622016-03-18 14:41:10 +0800599 XCAM_ASSERT (optarg);
600 if (!strcasecmp (optarg, "gaussian"))
601 wdr_mode = CL3aImageProcessor::Gaussian;
602 else if (!strcasecmp (optarg, "haleq"))
603 wdr_mode = CL3aImageProcessor::Haleq;
wujunkai16682950fb2015-10-19 18:09:05 +0800604
wujunkai166b0bd5512016-02-18 18:25:42 +0800605 pixel_format = V4L2_PIX_FMT_SGRBG12;
wujunkai166b0bd5512016-02-18 18:25:42 +0800606 setenv ("AIQ_CPF_PATH", IMX185_WDR_CPF, 1);
607 break;
608 }
Wangfei1487b9b2015-08-28 15:10:39 +0800609 case 'P': {
Wind Yuan6b437392016-02-02 00:31:30 +0800610 XCAM_ASSERT (optarg);
Wangfei1487b9b2015-08-28 15:10:39 +0800611 if (!strcasecmp (optarg, "basic"))
612 pipeline_mode = CL3aImageProcessor::BasicPipelineProfile;
613 else if (!strcasecmp (optarg, "advance"))
614 pipeline_mode = CL3aImageProcessor::AdvancedPipelineProfile;
Wangfei16b65db2015-09-11 14:44:28 +0800615 else if (!strcasecmp (optarg, "extreme"))
616 pipeline_mode = CL3aImageProcessor::ExtremePipelineProfile;
Wangfei1487b9b2015-08-28 15:10:39 +0800617 else {
618 print_help (bin_name);
619 return -1;
620 }
621 break;
622 }
wujunkai166a84ba052015-09-08 15:54:12 +0800623 case 'C': {
Wind Yuan6b437392016-02-02 00:31:30 +0800624 XCAM_ASSERT (optarg);
wujunkai166a84ba052015-09-08 15:54:12 +0800625 if (!strcmp (optarg, "bayer"))
626 capture_stage = CL3aImageProcessor::BasicbayerStage;
627 break;
628 }
Yinhang Liub0ad64b2016-03-18 16:03:51 +0800629 case 'O': {
630 have_cl_post_processor = false;
631 break;
632 }
Yinhang Liuc2f19082015-08-28 12:27:48 +0800633#endif
Jia Meng0a532932015-10-15 08:44:09 -0400634 case 'r': {
Yinhang Liu1aa97602016-05-27 14:58:21 +0800635 XCAM_ASSERT (optarg);
636 XCAM_LOG_INFO ("use raw image %s as input source", optarg);
Wind Yuan28935912017-08-24 19:18:34 +0800637 path_to_fake = optarg;
Jia Meng0a532932015-10-15 08:44:09 -0400638 break;
639 }
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800640 case 'p': {
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800641#if HAVE_LIBDRM
Yinhang Liu454a3892016-11-07 16:30:35 +0800642 need_display = true;
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800643#else
644 XCAM_LOG_WARNING ("preview is not supported, disable preview now");
645 need_display = false;
646#endif
Yinhang Liu454a3892016-11-07 16:30:35 +0800647 break;
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800648 }
Wind Yuan75564b12015-01-15 06:51:15 -0500649 case 'h':
650 print_help (bin_name);
651 return 0;
652
653 default:
654 print_help (bin_name);
655 return -1;
656 }
657 }
658
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800659 SmartPtr<MainDeviceManager> device_manager = new MainDeviceManager ();
660 device_manager->enable_save_file (save_file);
661 device_manager->set_interval (interval_frames);
662 device_manager->set_frame_save (save_frames);
663 device_manager->set_frame_width (frame_width);
664 device_manager->set_frame_height (frame_height);
Yinhang Liu5855e6f2016-11-14 21:04:22 +0800665
ShincyTu2808b592015-03-13 17:17:25 +0800666 if (!device.ptr ()) {
Wind Yuan28935912017-08-24 19:18:34 +0800667 if (path_to_fake.c_str ()) {
Jia Meng0a532932015-10-15 08:44:09 -0400668 device = new FakeV4l2Device ();
669 } else if (have_usbcam) {
Wind Yuan28935912017-08-24 19:18:34 +0800670 device = new UVCDevice (usb_device_name.c_str ());
Yinhang Liu454a3892016-11-07 16:30:35 +0800671 }
672#if HAVE_IA_AIQ
673 else {
Sameer Kibey7b429292015-08-07 10:55:38 -0700674 if (capture_mode == V4L2_CAPTURE_MODE_STILL)
675 device = new AtomispDevice (CAPTURE_DEVICE_STILL);
676 else if (capture_mode == V4L2_CAPTURE_MODE_VIDEO)
677 device = new AtomispDevice (CAPTURE_DEVICE_VIDEO);
678 else
679 device = new AtomispDevice (DEFAULT_CAPTURE_DEVICE);
680 }
Yinhang Liu454a3892016-11-07 16:30:35 +0800681#endif
ShincyTu2808b592015-03-13 17:17:25 +0800682 }
Yinhang Liu5c1381c2016-11-19 16:35:38 +0800683
Yinhang Liu454a3892016-11-07 16:30:35 +0800684#if HAVE_IA_AIQ
Wind Yuan75564b12015-01-15 06:51:15 -0500685 if (!isp_controller.ptr ())
686 isp_controller = new IspController (device);
Yinhang Liu454a3892016-11-07 16:30:35 +0800687#endif
Wind Yuan26e9e212015-04-16 15:55:45 +0800688
689 switch (analyzer_type) {
Wind Yuan3d1d15e2015-04-16 17:40:47 +0800690 case AnalyzerTypeSimple:
691 analyzer = new X3aAnalyzerSimple ();
692 break;
Wind Yuan26e9e212015-04-16 15:55:45 +0800693#if HAVE_IA_AIQ
Yinhang Liu6c367352016-03-31 20:23:22 +0800694 case AnalyzerTypeAiqTuner: {
695 SmartPtr<X3aAnalyzer> aiq_analyzer = new X3aAnalyzerAiq (isp_controller, DEFAULT_CPF_FILE);
696 SmartPtr<X3aAnalyzeTuner> tuner_analyzer = new X3aAnalyzeTuner ();
697 XCAM_ASSERT (aiq_analyzer.ptr () && tuner_analyzer.ptr ());
698 tuner_analyzer->set_analyzer (aiq_analyzer);
699 analyzer = tuner_analyzer;
Wind Yuan3d1d15e2015-04-16 17:40:47 +0800700 break;
Yinhang Liu6c367352016-03-31 20:23:22 +0800701 }
Yinhang Liu454a3892016-11-07 16:30:35 +0800702#if HAVE_LIBCL
703 case AnalyzerTypeDynamic: {
704 const char *path_of_3a = DEFAULT_DYNAMIC_3A_LIB;
705 SmartPtr<DynamicAnalyzerLoader> dynamic_loader = new DynamicAnalyzerLoader (path_of_3a);
706 loader = dynamic_loader.dynamic_cast_ptr<AnalyzerLoader> ();
707 analyzer = dynamic_loader->load_analyzer (loader);
708 CHECK_EXP (analyzer.ptr (), "load dynamic 3a lib(%s) failed", path_of_3a);
709 break;
710 }
Jia Meng9724cfe2015-08-12 15:04:45 +0800711 case AnalyzerTypeHybrid: {
Yinhang Liu454a3892016-11-07 16:30:35 +0800712 const char *path_of_3a = DEFAULT_HYBRID_3A_LIB;
Yinhang Liue37a8722016-06-27 18:14:35 +0800713 SmartPtr<HybridAnalyzerLoader> hybrid_loader = new HybridAnalyzerLoader (path_of_3a);
714 hybrid_loader->set_cpf_path (DEFAULT_CPF_FILE);
715 hybrid_loader->set_isp_controller (isp_controller);
716 loader = hybrid_loader.dynamic_cast_ptr<AnalyzerLoader> ();
717 analyzer = hybrid_loader->load_analyzer (loader);
Jia Meng9724cfe2015-08-12 15:04:45 +0800718 CHECK_EXP (analyzer.ptr (), "load hybrid 3a lib(%s) failed", path_of_3a);
719 break;
720 }
Wind Yuan26e9e212015-04-16 15:55:45 +0800721#endif
Yinhang Liu454a3892016-11-07 16:30:35 +0800722#endif
Wind Yuan3d1d15e2015-04-16 17:40:47 +0800723 default:
724 print_help (bin_name);
725 return -1;
Wind Yuan26e9e212015-04-16 15:55:45 +0800726 }
Jia Meng8f94a102015-08-11 16:24:19 +0800727 XCAM_ASSERT (analyzer.ptr ());
728 analyzer->set_sync_mode (sync_mode);
Wind Yuan75564b12015-01-15 06:51:15 -0500729
Yinhang Liu3fef5d72016-07-25 14:43:20 +0800730#if HAVE_LIBCL
731 SmartHandlerList smart_handlers = SmartAnalyzerLoader::load_smart_handlers (DEFAULT_SMART_ANALYSIS_LIB_DIR);
732 if (!smart_handlers.empty ()) {
733 smart_analyzer = new SmartAnalyzer ();
734 if (smart_analyzer.ptr ()) {
735 SmartHandlerList::iterator i_handler = smart_handlers.begin ();
Wind Yuan136aa4f2016-09-08 02:56:31 -0400736 for (; i_handler != smart_handlers.end (); ++i_handler)
Yinhang Liu3fef5d72016-07-25 14:43:20 +0800737 {
738 XCAM_ASSERT ((*i_handler).ptr ());
739 smart_analyzer->add_handler (*i_handler);
740 }
741 } else {
742 XCAM_LOG_WARNING ("load smart analyzer(%s) failed, please check.", DEFAULT_SMART_ANALYSIS_LIB_DIR);
743 }
744 }
Yinhang Liu454a3892016-11-07 16:30:35 +0800745
746 if (smart_analyzer.ptr ()) {
747 if (smart_analyzer->prepare_handlers () != XCAM_RETURN_NO_ERROR) {
748 XCAM_LOG_WARNING ("analyzer(%s) prepare handlers failed", smart_analyzer->get_name ());
749 }
750 device_manager->set_smart_analyzer (smart_analyzer);
751 }
Yinhang Liu3fef5d72016-07-25 14:43:20 +0800752#endif
753
Wind Yuan75564b12015-01-15 06:51:15 -0500754 signal(SIGINT, dev_stop_handler);
755
756 device->set_sensor_id (0);
ShincyTu2808b592015-03-13 17:17:25 +0800757 device->set_capture_mode (capture_mode);
Wind Yuan75564b12015-01-15 06:51:15 -0500758 //device->set_mem_type (V4L2_MEMORY_DMABUF);
Wind Yuand4427312015-02-11 16:09:00 +0800759 device->set_mem_type (v4l2_mem_type);
Wind Yuan75564b12015-01-15 06:51:15 -0500760 device->set_buffer_count (8);
Jia Meng757f3c92015-09-09 11:18:36 +0800761 if (pixel_format == V4L2_PIX_FMT_SGRBG12) {
762 frame_rate = 30;
763 device->set_framerate (frame_rate, 1);
764 }
Yinhang Liuc64ed592016-07-15 18:53:34 +0800765#if HAVE_LIBCL
Jia Meng757f3c92015-09-09 11:18:36 +0800766 else {
767 frame_rate = 25;
768 device->set_framerate (frame_rate, 1);
Yinhang Liuefe4f0f2016-12-29 13:17:22 +0800769 if(wdr_mode != CL3aImageProcessor::WDRdisabled) {
wujunkai166a9a55f22015-10-10 15:53:32 +0800770 XCAM_LOG_WARNING("Tonemapping is only applicable under BA12 format. Disable tonemapping automatically.");
Yinhang Liuefe4f0f2016-12-29 13:17:22 +0800771 wdr_mode = CL3aImageProcessor::WDRdisabled;
wujunkai166a9a55f22015-10-10 15:53:32 +0800772 }
Jia Meng757f3c92015-09-09 11:18:36 +0800773 }
Yinhang Liuc64ed592016-07-15 18:53:34 +0800774#endif
Wind Yuan75564b12015-01-15 06:51:15 -0500775 ret = device->open ();
776 CHECK (ret, "device(%s) open failed", device->get_device_name());
Yinhang Liud9346492015-12-17 11:14:25 +0800777 ret = device->set_format (frame_width, frame_height, pixel_format, V4L2_FIELD_NONE, frame_width * 2);
Wind Yuan75564b12015-01-15 06:51:15 -0500778 CHECK (ret, "device(%s) set format failed", device->get_device_name());
779
Yinhang Liu5c1381c2016-11-19 16:35:38 +0800780#if HAVE_IA_AIQ
781 if (!event_device.ptr ())
782 event_device = new V4l2SubDevice (DEFAULT_EVENT_DEVICE);
Wind Yuan75564b12015-01-15 06:51:15 -0500783 ret = event_device->open ();
Wind Yuanb67045c2015-08-27 13:50:53 +0800784 if (ret == XCAM_RETURN_NO_ERROR) {
785 CHECK (ret, "event device(%s) open failed", event_device->get_device_name());
786 int event = V4L2_EVENT_ATOMISP_3A_STATS_READY;
787 ret = event_device->subscribe_event (event);
788 CHECK_CONTINUE (
789 ret,
790 "device(%s) subscribe event(%d) failed",
791 event_device->get_device_name(), event);
792 event = V4L2_EVENT_FRAME_SYNC;
793 ret = event_device->subscribe_event (event);
794 CHECK_CONTINUE (
795 ret,
796 "device(%s) subscribe event(%d) failed",
797 event_device->get_device_name(), event);
798
799 device_manager->set_event_device (event_device);
800 }
Yinhang Liu5c1381c2016-11-19 16:35:38 +0800801#endif
Wind Yuan75564b12015-01-15 06:51:15 -0500802
803 device_manager->set_capture_device (device);
Wind Yuan75564b12015-01-15 06:51:15 -0500804 if (analyzer.ptr())
zongwave03954a32015-09-22 15:40:44 +0800805 device_manager->set_3a_analyzer (analyzer);
Wind Yuan3d1d15e2015-04-16 17:40:47 +0800806
Yinhang Liu454a3892016-11-07 16:30:35 +0800807#if HAVE_IA_AIQ
808#if HAVE_LIBCL
Wind Yuan3d1d15e2015-04-16 17:40:47 +0800809 if (have_cl_processor)
810 isp_processor = new IspExposureImageProcessor (isp_controller);
811 else
Yinhang Liu454a3892016-11-07 16:30:35 +0800812#endif
Wind Yuan3d1d15e2015-04-16 17:40:47 +0800813 isp_processor = new IspImageProcessor (isp_controller);
814
815 XCAM_ASSERT (isp_processor.ptr ());
Wind Yuane4ba0c92015-03-26 16:59:06 +0800816 device_manager->add_image_processor (isp_processor);
Yinhang Liu454a3892016-11-07 16:30:35 +0800817#endif
Yinhang Liuc2f19082015-08-28 12:27:48 +0800818#if HAVE_LIBCL
Wind Yuand4427312015-02-11 16:09:00 +0800819 if (have_cl_processor) {
Wind Yuane4ba0c92015-03-26 16:59:06 +0800820 cl_processor = new CL3aImageProcessor ();
Wind Yuan7ddf2602015-04-14 18:49:45 +0800821 cl_processor->set_stats_callback(device_manager);
Jia Meng66efecf2015-06-17 11:11:10 +0000822 cl_processor->set_denoise (denoise_type);
wujunkai166a84ba052015-09-08 15:54:12 +0800823 cl_processor->set_capture_stage (capture_stage);
wujunkai166c8f3b442016-02-29 17:12:09 +0800824
Yinhang Liuefe4f0f2016-12-29 13:17:22 +0800825 cl_processor->set_tonemapping (wdr_mode);
826 if (wdr_mode != CL3aImageProcessor::WDRdisabled) {
827 cl_processor->set_gamma (false);
828 cl_processor->set_3a_stats_bits (12);
wujunkai166c8f3b442016-02-29 17:12:09 +0800829 }
Yinhang Liuefe4f0f2016-12-29 13:17:22 +0800830
Yinhang Liu81c44e02015-06-03 15:47:48 +0800831 cl_processor->set_tnr (tnr_type, tnr_level);
Wangfei1487b9b2015-08-28 15:10:39 +0800832 cl_processor->set_profile (pipeline_mode);
Yinhang Liu454a3892016-11-07 16:30:35 +0800833#if HAVE_IA_AIQ
yaowang1a5f12122015-07-27 14:53:38 +0800834 analyzer->set_parameter_brightness((brightness_level - 128) / 128.0);
Yinhang Liu454a3892016-11-07 16:30:35 +0800835#endif
Wind Yuand4427312015-02-11 16:09:00 +0800836 device_manager->add_image_processor (cl_processor);
837 }
Yinhang Liu3c17fea2016-03-15 10:47:03 +0800838
Yinhang Liub0ad64b2016-03-18 16:03:51 +0800839 if (have_cl_post_processor) {
840 cl_post_processor = new CLPostImageProcessor ();
Yinhang Liu3c721c12016-03-15 15:03:51 +0800841
Yinhang Liud341dbe2016-07-25 12:00:17 +0800842 cl_post_processor->set_stats_callback (device_manager);
Wind Yuanf4e17ab2016-05-05 02:54:01 +0800843 cl_post_processor->set_defog_mode ((CLPostImageProcessor::CLDefogMode)defog_type);
Yinhang Liud341dbe2016-07-25 12:00:17 +0800844 cl_post_processor->set_wavelet (wavelet_mode, wavelet_channel, wavelet_bayes_shrink);
845 cl_post_processor->set_3ddenoise_mode ((CLPostImageProcessor::CL3DDenoiseMode) denoise_3d_mode, denoise_3d_ref_count);
Yinhang Liu3c721c12016-03-15 15:03:51 +0800846
Yinhang Liud341dbe2016-07-25 12:00:17 +0800847 cl_post_processor->set_wireframe (wireframe_type);
zongwave1b10d6f2016-10-25 17:47:51 +0800848 cl_post_processor->set_image_warp (image_warp_type);
849 if (smart_analyzer.ptr () && (wireframe_type || image_warp_type)) {
Yinhang Liud341dbe2016-07-25 12:00:17 +0800850 cl_post_processor->set_scaler (true);
851 cl_post_processor->set_scaler_factor (640.0 / frame_width);
852 }
zongwavebec89e12016-07-23 12:34:20 +0800853
Yinhang Liud6d824a2016-03-22 20:43:00 +0800854 if (need_display) {
Yinhang Liucff638b2017-09-25 21:16:28 +0800855 need_display = false;
856 XCAM_LOG_WARNING ("CLVideoBuffer doesn't support local preview, disable local preview now");
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800857 }
Yinhang Liucff638b2017-09-25 21:16:28 +0800858
Yinhang Liu6987f2b2017-09-30 10:39:58 +0800859 if (need_display) {
860#if HAVE_LIBDRM
861 if (DrmDisplay::set_preview (need_display)) {
862 device_manager->set_display_mode (display_mode);
863 cl_post_processor->set_output_format (V4L2_PIX_FMT_XBGR32);
864 } else {
865 need_display = false;
866 XCAM_LOG_WARNING ("set preview failed, disable local preview now");
867 }
868#else
869 XCAM_LOG_WARNING ("preview is not supported, disable preview now");
870 need_display = false;
871#endif
872 }
Yinhang Liucff638b2017-09-25 21:16:28 +0800873 device_manager->enable_display (need_display);
874
Yinhang Liub0ad64b2016-03-18 16:03:51 +0800875 device_manager->add_image_processor (cl_post_processor);
Yinhang Liu3c17fea2016-03-15 10:47:03 +0800876 }
Wind Yuand4427312015-02-11 16:09:00 +0800877#endif
Wind Yuane4ba0c92015-03-26 16:59:06 +0800878
Jia Meng0a532932015-10-15 08:44:09 -0400879 SmartPtr<PollThread> poll_thread;
Yinhang Liu454a3892016-11-07 16:30:35 +0800880 if (have_usbcam) {
881 poll_thread = new PollThread ();
Wind Yuan28935912017-08-24 19:18:34 +0800882 } else if (path_to_fake.c_str ()) {
883 poll_thread = new FakePollThread (path_to_fake.c_str ());
Yinhang Liu454a3892016-11-07 16:30:35 +0800884 }
885#if HAVE_IA_AIQ
Yinhang Liue37a8722016-06-27 18:14:35 +0800886 else {
887 SmartPtr<IspPollThread> isp_poll_thread = new IspPollThread ();
888 isp_poll_thread->set_isp_controller (isp_controller);
889 poll_thread = isp_poll_thread;
890 }
Yinhang Liu454a3892016-11-07 16:30:35 +0800891#endif
Jia Meng0a532932015-10-15 08:44:09 -0400892 device_manager->set_poll_thread (poll_thread);
893
Wind Yuan75564b12015-01-15 06:51:15 -0500894 ret = device_manager->start ();
895 CHECK (ret, "device manager start failed");
896
Yinhang Liuc64ed592016-07-15 18:53:34 +0800897#if HAVE_LIBCL
Jia Meng757f3c92015-09-09 11:18:36 +0800898 // hard code exposure range and max gain for imx185 WDR
Yinhang Liuefe4f0f2016-12-29 13:17:22 +0800899 if (wdr_mode != CL3aImageProcessor::WDRdisabled) {
Jia Meng757f3c92015-09-09 11:18:36 +0800900 if (frame_rate == 30)
901 analyzer->set_ae_exposure_time_range (80 * 1110 * 1000 / 37125, 1120 * 1110 * 1000 / 37125);
902 else
Jia Menga6580232015-10-08 16:51:28 +0800903 analyzer->set_ae_exposure_time_range (80 * 1320 * 1000 / 37125, 1120 * 1320 * 1000 / 37125);
Jia Meng757f3c92015-09-09 11:18:36 +0800904 analyzer->set_ae_max_analog_gain (3.98); // 12dB
905 }
Yinhang Liuc64ed592016-07-15 18:53:34 +0800906#endif
Jia Meng757f3c92015-09-09 11:18:36 +0800907
Wind Yuan75564b12015-01-15 06:51:15 -0500908 // wait for interruption
909 {
910 SmartLock locker (g_mutex);
911 while (!g_stop)
912 g_cond.wait (g_mutex);
913 }
914
915 ret = device_manager->stop();
916 CHECK_CONTINUE (ret, "device manager stop failed");
917 device->close ();
Yinhang Liu5c1381c2016-11-19 16:35:38 +0800918#if HAVE_IA_AIQ
Wind Yuan75564b12015-01-15 06:51:15 -0500919 event_device->close ();
Yinhang Liu5c1381c2016-11-19 16:35:38 +0800920#endif
Wind Yuan75564b12015-01-15 06:51:15 -0500921
922 return 0;
923}