Lines Matching +full:top +full:- +full:ctrl

1 // SPDX-License-Identifier: GPL-2.0-only
6 * Copyright (C) 2011, Javier Martin <javier.martin@vista-silicon.com>
27 #include <media/v4l2-async.h>
28 #include <media/v4l2-ctrls.h>
29 #include <media/v4l2-device.h>
30 #include <media/v4l2-fwnode.h>
31 #include <media/v4l2-subdev.h>
33 #include "aptina-pll.h"
165 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in mt9p031_set_output_control()
166 u16 value = (mt9p031->output_control & ~clear) | set; in mt9p031_set_output_control()
173 mt9p031->output_control = value; in mt9p031_set_output_control()
179 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in mt9p031_set_mode2()
180 u16 value = (mt9p031->mode2 & ~clear) | set; in mt9p031_set_mode2()
187 mt9p031->mode2 = value; in mt9p031_set_mode2()
193 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in mt9p031_reset()
205 MT9P031_PIXEL_CLOCK_DIVIDE(mt9p031->clk_div)); in mt9p031_reset()
231 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in mt9p031_clk_setup()
232 struct mt9p031_platform_data *pdata = mt9p031->pdata; in mt9p031_clk_setup()
236 mt9p031->clk = devm_clk_get(&client->dev, NULL); in mt9p031_clk_setup()
237 if (IS_ERR(mt9p031->clk)) in mt9p031_clk_setup()
238 return PTR_ERR(mt9p031->clk); in mt9p031_clk_setup()
240 ret = clk_set_rate(mt9p031->clk, pdata->ext_freq); in mt9p031_clk_setup()
244 ext_freq = clk_get_rate(mt9p031->clk); in mt9p031_clk_setup()
252 div = DIV_ROUND_UP(ext_freq, pdata->target_freq); in mt9p031_clk_setup()
255 mt9p031->clk_div = min_t(unsigned int, div, 64); in mt9p031_clk_setup()
256 mt9p031->use_pll = false; in mt9p031_clk_setup()
261 mt9p031->pll.ext_clock = ext_freq; in mt9p031_clk_setup()
262 mt9p031->pll.pix_clock = pdata->target_freq; in mt9p031_clk_setup()
263 mt9p031->use_pll = true; in mt9p031_clk_setup()
265 return aptina_pll_calculate(&client->dev, &limits, &mt9p031->pll); in mt9p031_clk_setup()
270 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in mt9p031_pll_enable()
273 if (!mt9p031->use_pll) in mt9p031_pll_enable()
282 (mt9p031->pll.m << 8) | (mt9p031->pll.n - 1)); in mt9p031_pll_enable()
286 ret = mt9p031_write(client, MT9P031_PLL_CONFIG_2, mt9p031->pll.p1 - 1); in mt9p031_pll_enable()
299 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in mt9p031_pll_disable()
301 if (!mt9p031->use_pll) in mt9p031_pll_disable()
313 if (mt9p031->reset) { in mt9p031_power_on()
314 gpiod_set_value(mt9p031->reset, 1); in mt9p031_power_on()
319 ret = regulator_bulk_enable(ARRAY_SIZE(mt9p031->regulators), in mt9p031_power_on()
320 mt9p031->regulators); in mt9p031_power_on()
325 if (mt9p031->clk) { in mt9p031_power_on()
326 ret = clk_prepare_enable(mt9p031->clk); in mt9p031_power_on()
328 regulator_bulk_disable(ARRAY_SIZE(mt9p031->regulators), in mt9p031_power_on()
329 mt9p031->regulators); in mt9p031_power_on()
335 if (mt9p031->reset) { in mt9p031_power_on()
336 gpiod_set_value(mt9p031->reset, 0); in mt9p031_power_on()
345 if (mt9p031->reset) { in mt9p031_power_off()
346 gpiod_set_value(mt9p031->reset, 1); in mt9p031_power_off()
350 regulator_bulk_disable(ARRAY_SIZE(mt9p031->regulators), in mt9p031_power_off()
351 mt9p031->regulators); in mt9p031_power_off()
353 clk_disable_unprepare(mt9p031->clk); in mt9p031_power_off()
358 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in __mt9p031_set_power()
372 dev_err(&client->dev, "Failed to reset the camera\n"); in __mt9p031_set_power()
377 if (mt9p031->pdata && mt9p031->pdata->pixclk_pol) { in __mt9p031_set_power()
384 return v4l2_ctrl_handler_setup(&mt9p031->ctrls); in __mt9p031_set_power()
387 /* -----------------------------------------------------------------------------
393 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in mt9p031_set_params()
394 struct v4l2_mbus_framefmt *format = &mt9p031->format; in mt9p031_set_params()
395 const struct v4l2_rect *crop = &mt9p031->crop; in mt9p031_set_params()
410 ret = mt9p031_write(client, MT9P031_COLUMN_START, crop->left); in mt9p031_set_params()
413 ret = mt9p031_write(client, MT9P031_ROW_START, crop->top); in mt9p031_set_params()
416 ret = mt9p031_write(client, MT9P031_WINDOW_WIDTH, crop->width - 1); in mt9p031_set_params()
419 ret = mt9p031_write(client, MT9P031_WINDOW_HEIGHT, crop->height - 1); in mt9p031_set_params()
426 xskip = DIV_ROUND_CLOSEST(crop->width, format->width); in mt9p031_set_params()
427 yskip = DIV_ROUND_CLOSEST(crop->height, format->height); in mt9p031_set_params()
428 xbin = 1 << (ffs(xskip) - 1); in mt9p031_set_params()
429 ybin = 1 << (ffs(yskip) - 1); in mt9p031_set_params()
432 ((xbin - 1) << 4) | (xskip - 1)); in mt9p031_set_params()
436 ((ybin - 1) << 4) | (yskip - 1)); in mt9p031_set_params()
440 /* Blanking - use minimum value for horizontal blanking and default in mt9p031_set_params()
446 ret = mt9p031_write(client, MT9P031_HORIZONTAL_BLANK, hblank - 1); in mt9p031_set_params()
449 ret = mt9p031_write(client, MT9P031_VERTICAL_BLANK, vblank - 1); in mt9p031_set_params()
496 * - clear pause restart in mt9p031_s_stream()
497 * - don't clear restart as clearing restart manually can cause in mt9p031_s_stream()
514 if (code->pad || code->index) in mt9p031_enum_mbus_code()
515 return -EINVAL; in mt9p031_enum_mbus_code()
517 code->code = mt9p031->format.code; in mt9p031_enum_mbus_code()
527 if (fse->index >= 8 || fse->code != mt9p031->format.code) in mt9p031_enum_frame_size()
528 return -EINVAL; in mt9p031_enum_frame_size()
530 fse->min_width = MT9P031_WINDOW_WIDTH_DEF in mt9p031_enum_frame_size()
531 / min_t(unsigned int, 7, fse->index + 1); in mt9p031_enum_frame_size()
532 fse->max_width = fse->min_width; in mt9p031_enum_frame_size()
533 fse->min_height = MT9P031_WINDOW_HEIGHT_DEF / (fse->index + 1); in mt9p031_enum_frame_size()
534 fse->max_height = fse->min_height; in mt9p031_enum_frame_size()
546 return v4l2_subdev_get_try_format(&mt9p031->subdev, sd_state, in __mt9p031_get_pad_format()
549 return &mt9p031->format; in __mt9p031_get_pad_format()
562 return v4l2_subdev_get_try_crop(&mt9p031->subdev, sd_state, in __mt9p031_get_pad_crop()
565 return &mt9p031->crop; in __mt9p031_get_pad_crop()
577 fmt->format = *__mt9p031_get_pad_format(mt9p031, sd_state, fmt->pad, in mt9p031_get_format()
578 fmt->which); in mt9p031_get_format()
594 __crop = __mt9p031_get_pad_crop(mt9p031, sd_state, format->pad, in mt9p031_set_format()
595 format->which); in mt9p031_set_format()
598 width = clamp_t(unsigned int, ALIGN(format->format.width, 2), in mt9p031_set_format()
599 max_t(unsigned int, __crop->width / 7, in mt9p031_set_format()
601 __crop->width); in mt9p031_set_format()
602 height = clamp_t(unsigned int, ALIGN(format->format.height, 2), in mt9p031_set_format()
603 max_t(unsigned int, __crop->height / 8, in mt9p031_set_format()
605 __crop->height); in mt9p031_set_format()
607 hratio = DIV_ROUND_CLOSEST(__crop->width, width); in mt9p031_set_format()
608 vratio = DIV_ROUND_CLOSEST(__crop->height, height); in mt9p031_set_format()
610 __format = __mt9p031_get_pad_format(mt9p031, sd_state, format->pad, in mt9p031_set_format()
611 format->which); in mt9p031_set_format()
612 __format->width = __crop->width / hratio; in mt9p031_set_format()
613 __format->height = __crop->height / vratio; in mt9p031_set_format()
615 format->format = *__format; in mt9p031_set_format()
626 switch (sel->target) { in mt9p031_get_selection()
628 sel->r.left = MT9P031_COLUMN_START_MIN; in mt9p031_get_selection()
629 sel->r.top = MT9P031_ROW_START_MIN; in mt9p031_get_selection()
630 sel->r.width = MT9P031_WINDOW_WIDTH_MAX; in mt9p031_get_selection()
631 sel->r.height = MT9P031_WINDOW_HEIGHT_MAX; in mt9p031_get_selection()
635 sel->r = *__mt9p031_get_pad_crop(mt9p031, sd_state, in mt9p031_get_selection()
636 sel->pad, sel->which); in mt9p031_get_selection()
640 return -EINVAL; in mt9p031_get_selection()
653 if (sel->target != V4L2_SEL_TGT_CROP) in mt9p031_set_selection()
654 return -EINVAL; in mt9p031_set_selection()
659 rect.left = clamp(ALIGN(sel->r.left, 2), MT9P031_COLUMN_START_MIN, in mt9p031_set_selection()
661 rect.top = clamp(ALIGN(sel->r.top, 2), MT9P031_ROW_START_MIN, in mt9p031_set_selection()
663 rect.width = clamp_t(unsigned int, ALIGN(sel->r.width, 2), in mt9p031_set_selection()
666 rect.height = clamp_t(unsigned int, ALIGN(sel->r.height, 2), in mt9p031_set_selection()
671 MT9P031_PIXEL_ARRAY_WIDTH - rect.left); in mt9p031_set_selection()
673 MT9P031_PIXEL_ARRAY_HEIGHT - rect.top); in mt9p031_set_selection()
675 __crop = __mt9p031_get_pad_crop(mt9p031, sd_state, sel->pad, in mt9p031_set_selection()
676 sel->which); in mt9p031_set_selection()
678 if (rect.width != __crop->width || rect.height != __crop->height) { in mt9p031_set_selection()
683 sel->pad, in mt9p031_set_selection()
684 sel->which); in mt9p031_set_selection()
685 __format->width = rect.width; in mt9p031_set_selection()
686 __format->height = rect.height; in mt9p031_set_selection()
690 sel->r = rect; in mt9p031_set_selection()
706 crop->left = MT9P031_COLUMN_START_DEF; in mt9p031_init_cfg()
707 crop->top = MT9P031_ROW_START_DEF; in mt9p031_init_cfg()
708 crop->width = MT9P031_WINDOW_WIDTH_DEF; in mt9p031_init_cfg()
709 crop->height = MT9P031_WINDOW_HEIGHT_DEF; in mt9p031_init_cfg()
713 if (mt9p031->model == MT9P031_MODEL_MONOCHROME) in mt9p031_init_cfg()
714 format->code = MEDIA_BUS_FMT_Y12_1X12; in mt9p031_init_cfg()
716 format->code = MEDIA_BUS_FMT_SGRBG12_1X12; in mt9p031_init_cfg()
718 format->width = MT9P031_WINDOW_WIDTH_DEF; in mt9p031_init_cfg()
719 format->height = MT9P031_WINDOW_HEIGHT_DEF; in mt9p031_init_cfg()
720 format->field = V4L2_FIELD_NONE; in mt9p031_init_cfg()
721 format->colorspace = V4L2_COLORSPACE_SRGB; in mt9p031_init_cfg()
726 /* -----------------------------------------------------------------------------
737 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in mt9p031_restore_blc()
740 if (mt9p031->blc_auto->cur.val != 0) { in mt9p031_restore_blc()
747 if (mt9p031->blc_offset->cur.val != 0) { in mt9p031_restore_blc()
749 mt9p031->blc_offset->cur.val); in mt9p031_restore_blc()
757 static int mt9p031_s_ctrl(struct v4l2_ctrl *ctrl) in mt9p031_s_ctrl() argument
760 container_of(ctrl->handler, struct mt9p031, ctrls); in mt9p031_s_ctrl()
761 struct i2c_client *client = v4l2_get_subdevdata(&mt9p031->subdev); in mt9p031_s_ctrl()
765 if (ctrl->flags & V4L2_CTRL_FLAG_INACTIVE) in mt9p031_s_ctrl()
768 switch (ctrl->id) { in mt9p031_s_ctrl()
771 (ctrl->val >> 16) & 0xffff); in mt9p031_s_ctrl()
776 ctrl->val & 0xffff); in mt9p031_s_ctrl()
783 * ------------------------------------------ in mt9p031_s_ctrl()
793 if (ctrl->val <= 32) { in mt9p031_s_ctrl()
794 data = ctrl->val; in mt9p031_s_ctrl()
795 } else if (ctrl->val <= 64) { in mt9p031_s_ctrl()
796 ctrl->val &= ~1; in mt9p031_s_ctrl()
797 data = (1 << 6) | (ctrl->val >> 1); in mt9p031_s_ctrl()
799 ctrl->val &= ~7; in mt9p031_s_ctrl()
800 data = ((ctrl->val - 64) << 5) | (1 << 6) | 32; in mt9p031_s_ctrl()
806 if (ctrl->val) in mt9p031_s_ctrl()
814 if (ctrl->val) in mt9p031_s_ctrl()
824 * in the image. Activate (deactivate) the BLC-related controls in mt9p031_s_ctrl()
827 v4l2_ctrl_activate(mt9p031->blc_auto, ctrl->val == 0); in mt9p031_s_ctrl()
828 v4l2_ctrl_activate(mt9p031->blc_offset, ctrl->val == 0); in mt9p031_s_ctrl()
830 if (!ctrl->val) { in mt9p031_s_ctrl()
860 ((ctrl->val - 1) << MT9P031_TEST_PATTERN_SHIFT) in mt9p031_s_ctrl()
865 ctrl->val ? 0 : MT9P031_READ_MODE_2_ROW_BLC, in mt9p031_s_ctrl()
866 ctrl->val ? MT9P031_READ_MODE_2_ROW_BLC : 0); in mt9p031_s_ctrl()
871 ctrl->val ? 0 : MT9P031_BLC_MANUAL_BLC); in mt9p031_s_ctrl()
875 ctrl->val); in mt9p031_s_ctrl()
878 data = ctrl->val & ((1 << 9) - 1); in mt9p031_s_ctrl()
893 ctrl->val & ((1 << 12) - 1)); in mt9p031_s_ctrl()
942 .min = -255,
952 .min = -2048,
960 /* -----------------------------------------------------------------------------
969 mutex_lock(&mt9p031->power_lock); in mt9p031_set_power()
974 if (mt9p031->power_count == !on) { in mt9p031_set_power()
981 mt9p031->power_count += on ? 1 : -1; in mt9p031_set_power()
982 WARN_ON(mt9p031->power_count < 0); in mt9p031_set_power()
985 mutex_unlock(&mt9p031->power_lock); in mt9p031_set_power()
989 /* -----------------------------------------------------------------------------
1002 dev_err(&client->dev, "MT9P031 power up failed\n"); in mt9p031_registered()
1011 dev_err(&client->dev, "MT9P031 not detected, wrong version " in mt9p031_registered()
1013 return -ENODEV; in mt9p031_registered()
1016 dev_info(&client->dev, "MT9P031 detected at address 0x%02x\n", in mt9p031_registered()
1017 client->addr); in mt9p031_registered()
1062 /* -----------------------------------------------------------------------------
1075 if (!IS_ENABLED(CONFIG_OF) || !client->dev.of_node) in mt9p031_get_pdata()
1076 return client->dev.platform_data; in mt9p031_get_pdata()
1078 np = of_graph_get_next_endpoint(client->dev.of_node, NULL); in mt9p031_get_pdata()
1085 pdata = devm_kzalloc(&client->dev, sizeof(*pdata), GFP_KERNEL); in mt9p031_get_pdata()
1089 of_property_read_u32(np, "input-clock-frequency", &pdata->ext_freq); in mt9p031_get_pdata()
1090 of_property_read_u32(np, "pixel-clock-frequency", &pdata->target_freq); in mt9p031_get_pdata()
1092 pdata->pixclk_pol = !!(endpoint.bus.parallel.flags & in mt9p031_get_pdata()
1104 struct i2c_adapter *adapter = client->adapter; in mt9p031_probe()
1110 dev_err(&client->dev, "No platform data\n"); in mt9p031_probe()
1111 return -EINVAL; in mt9p031_probe()
1115 dev_warn(&client->dev, in mt9p031_probe()
1116 "I2C-Adapter doesn't support I2C_FUNC_SMBUS_WORD\n"); in mt9p031_probe()
1117 return -EIO; in mt9p031_probe()
1120 mt9p031 = devm_kzalloc(&client->dev, sizeof(*mt9p031), GFP_KERNEL); in mt9p031_probe()
1122 return -ENOMEM; in mt9p031_probe()
1124 mt9p031->pdata = pdata; in mt9p031_probe()
1125 mt9p031->output_control = MT9P031_OUTPUT_CONTROL_DEF; in mt9p031_probe()
1126 mt9p031->mode2 = MT9P031_READ_MODE_2_ROW_BLC; in mt9p031_probe()
1127 mt9p031->model = did->driver_data; in mt9p031_probe()
1129 mt9p031->regulators[0].supply = "vdd"; in mt9p031_probe()
1130 mt9p031->regulators[1].supply = "vdd_io"; in mt9p031_probe()
1131 mt9p031->regulators[2].supply = "vaa"; in mt9p031_probe()
1133 ret = devm_regulator_bulk_get(&client->dev, 3, mt9p031->regulators); in mt9p031_probe()
1135 dev_err(&client->dev, "Unable to get regulators\n"); in mt9p031_probe()
1139 mutex_init(&mt9p031->power_lock); in mt9p031_probe()
1141 v4l2_ctrl_handler_init(&mt9p031->ctrls, ARRAY_SIZE(mt9p031_ctrls) + 6); in mt9p031_probe()
1143 v4l2_ctrl_new_std(&mt9p031->ctrls, &mt9p031_ctrl_ops, in mt9p031_probe()
1147 v4l2_ctrl_new_std(&mt9p031->ctrls, &mt9p031_ctrl_ops, in mt9p031_probe()
1150 v4l2_ctrl_new_std(&mt9p031->ctrls, &mt9p031_ctrl_ops, in mt9p031_probe()
1152 v4l2_ctrl_new_std(&mt9p031->ctrls, &mt9p031_ctrl_ops, in mt9p031_probe()
1154 v4l2_ctrl_new_std(&mt9p031->ctrls, &mt9p031_ctrl_ops, in mt9p031_probe()
1155 V4L2_CID_PIXEL_RATE, pdata->target_freq, in mt9p031_probe()
1156 pdata->target_freq, 1, pdata->target_freq); in mt9p031_probe()
1157 v4l2_ctrl_new_std_menu_items(&mt9p031->ctrls, &mt9p031_ctrl_ops, in mt9p031_probe()
1159 ARRAY_SIZE(mt9p031_test_pattern_menu) - 1, 0, in mt9p031_probe()
1163 v4l2_ctrl_new_custom(&mt9p031->ctrls, &mt9p031_ctrls[i], NULL); in mt9p031_probe()
1165 mt9p031->subdev.ctrl_handler = &mt9p031->ctrls; in mt9p031_probe()
1167 if (mt9p031->ctrls.error) { in mt9p031_probe()
1169 __func__, mt9p031->ctrls.error); in mt9p031_probe()
1170 ret = mt9p031->ctrls.error; in mt9p031_probe()
1174 mt9p031->blc_auto = v4l2_ctrl_find(&mt9p031->ctrls, V4L2_CID_BLC_AUTO); in mt9p031_probe()
1175 mt9p031->blc_offset = v4l2_ctrl_find(&mt9p031->ctrls, in mt9p031_probe()
1178 v4l2_i2c_subdev_init(&mt9p031->subdev, client, &mt9p031_subdev_ops); in mt9p031_probe()
1179 mt9p031->subdev.internal_ops = &mt9p031_subdev_internal_ops; in mt9p031_probe()
1181 mt9p031->subdev.entity.function = MEDIA_ENT_F_CAM_SENSOR; in mt9p031_probe()
1182 mt9p031->pad.flags = MEDIA_PAD_FL_SOURCE; in mt9p031_probe()
1183 ret = media_entity_pads_init(&mt9p031->subdev.entity, 1, &mt9p031->pad); in mt9p031_probe()
1187 mt9p031->subdev.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE; in mt9p031_probe()
1189 ret = mt9p031_init_cfg(&mt9p031->subdev, NULL); in mt9p031_probe()
1193 mt9p031->reset = devm_gpiod_get_optional(&client->dev, "reset", in mt9p031_probe()
1200 ret = v4l2_async_register_subdev(&mt9p031->subdev); in mt9p031_probe()
1204 v4l2_ctrl_handler_free(&mt9p031->ctrls); in mt9p031_probe()
1205 media_entity_cleanup(&mt9p031->subdev.entity); in mt9p031_probe()
1206 mutex_destroy(&mt9p031->power_lock); in mt9p031_probe()
1217 v4l2_ctrl_handler_free(&mt9p031->ctrls); in mt9p031_remove()
1219 media_entity_cleanup(&subdev->entity); in mt9p031_remove()
1220 mutex_destroy(&mt9p031->power_lock); in mt9p031_remove()