Message ID | 20230607164712.63579-4-hdegoede@redhat.com (mailing list archive) |
---|---|
State | Superseded |
Delegated to: | Sakari Ailus |
Headers |
Received: from vger.kernel.org ([23.128.96.18]) by www.linuxtv.org with esmtp (Exim 4.92) (envelope-from <linux-media-owner@vger.kernel.org>) id 1q6wKY-00Gsfe-EH; Wed, 07 Jun 2023 16:48:18 +0000 Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S230298AbjFGQsR (ORCPT <rfc822;mkrufky@linuxtv.org> + 1 other); Wed, 7 Jun 2023 12:48:17 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:35086 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S230189AbjFGQsQ (ORCPT <rfc822;linux-media@vger.kernel.org>); Wed, 7 Jun 2023 12:48:16 -0400 Received: from us-smtp-delivery-124.mimecast.com (us-smtp-delivery-124.mimecast.com [170.10.133.124]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 6B5B41FCF for <linux-media@vger.kernel.org>; Wed, 7 Jun 2023 09:47:29 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=redhat.com; s=mimecast20190719; t=1686156448; h=from:from:reply-to:subject:subject:date:date:message-id:message-id: to:to:cc:cc:mime-version:mime-version: content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references; bh=V//MPuTj6sdbqvPlg9CzoXnHcKPITWOGQhKrhjkntPo=; b=Uy0+MSponEnUwJ9n9VvJ15N3feDT7tRczKabt2zZtSroDRHgYm6zh8JzxTZGOJe+EIHhLD Ldo+k/oeIaoLDGau2EB/X0D42M4YTcmifJNcYjkzjSgIvyraAYM6Ms/XOR9UJyCITKrq0d 1Oj8X+1njqGbHy2ldEraQzA5uGbDKHs= Received: from mimecast-mx02.redhat.com (mimecast-mx02.redhat.com [66.187.233.88]) by relay.mimecast.com with ESMTP with STARTTLS (version=TLSv1.2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id us-mta-634-MhLSOFX3M2qEG-u_NbhAWQ-1; Wed, 07 Jun 2023 12:47:24 -0400 X-MC-Unique: MhLSOFX3M2qEG-u_NbhAWQ-1 Received: from smtp.corp.redhat.com (int-mx08.intmail.prod.int.rdu2.redhat.com [10.11.54.8]) (using TLSv1.2 with cipher AECDH-AES256-SHA (256/256 bits)) (No client certificate requested) by mimecast-mx02.redhat.com (Postfix) with ESMTPS id D950C811E78; Wed, 7 Jun 2023 16:47:23 +0000 (UTC) Received: from localhost.localdomain (unknown [10.39.192.22]) by smtp.corp.redhat.com (Postfix) with ESMTP id C06DDC1603B; Wed, 7 Jun 2023 16:47:22 +0000 (UTC) From: Hans de Goede <hdegoede@redhat.com> To: Daniel Scally <dan.scally@ideasonboard.com>, Laurent Pinchart <laurent.pinchart@ideasonboard.com>, Sakari Ailus <sakari.ailus@linux.intel.com>, Andy Shevchenko <andy@kernel.org> Cc: Hans de Goede <hdegoede@redhat.com>, Mauro Carvalho Chehab <mchehab@kernel.org>, Kate Hsuan <hpa@redhat.com>, linux-media@vger.kernel.org Subject: [PATCH 03/28] media: ov2680: Fix vflip / hflip set functions Date: Wed, 7 Jun 2023 18:46:47 +0200 Message-Id: <20230607164712.63579-4-hdegoede@redhat.com> In-Reply-To: <20230607164712.63579-1-hdegoede@redhat.com> References: <20230607164712.63579-1-hdegoede@redhat.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Scanned-By: MIMEDefang 3.1 on 10.11.54.8 X-Spam-Status: No, score=-2.1 required=5.0 tests=BAYES_00,DKIMWL_WL_HIGH, DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,RCVD_IN_DNSWL_NONE, SPF_HELO_NONE,SPF_NONE,T_SCC_BODY_TEXT_LINE autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: <linux-media.vger.kernel.org> X-Mailing-List: linux-media@vger.kernel.org X-LSpam-Score: -2.5 (--) X-LSpam-Report: No, score=-2.5 required=5.0 tests=BAYES_00=-1.9,DKIMWL_WL_HIGH=0.001,DKIM_SIGNED=0.1,DKIM_VALID=-0.1,DKIM_VALID_AU=-0.1,HEADER_FROM_DIFFERENT_DOMAINS=0.5,MAILING_LIST_MULTI=-1 autolearn=ham autolearn_force=no |
Series |
media: ov2680: Bugfixes + ACPI + selection(crop-tgt) API support
|
|
Commit Message
Hans de Goede
June 7, 2023, 4:46 p.m. UTC
ov2680_vflip_disable() / ov2680_hflip_disable() pass BIT(0) instead of
0 as value to ov2680_mod_reg().
While fixing this also:
1. Stop having separate enable/disable functions for hflip / vflip
2. Move the is_streaming check, which is unique to hflip / vflip
into the ov2680_set_?flip() functions.
for a nice code cleanup.
Fixes: 3ee47cad3e69 ("media: ov2680: Add Omnivision OV2680 sensor driver")
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
---
drivers/media/i2c/ov2680.c | 48 +++++++++-----------------------------
1 file changed, 11 insertions(+), 37 deletions(-)
Comments
Hello Hans (and everybody else) On 07/06/2023 17:46, Hans de Goede wrote: > ov2680_vflip_disable() / ov2680_hflip_disable() pass BIT(0) instead of > 0 as value to ov2680_mod_reg(). > > While fixing this also: > > 1. Stop having separate enable/disable functions for hflip / vflip > 2. Move the is_streaming check, which is unique to hflip / vflip > into the ov2680_set_?flip() functions. I think the code in the patch is good, but I wonder about the right way to deal with the changed pixel ordering following a flip/mirror. None of the other Omnivision sensor drivers I've looked at disallow a flip/mirror whilst streaming, but of course doing so wrecks the colours. I don't think that there's any reason we couldn't make the equivalent calculation to ov2680_set_bayer_order() within libcamera though and configure the ipu3-imgu appropriately, and so properly support a flip-whilst-streaming...but quite possibly I'm missing something. So I'm leaning towards "let's just drop the is_streaming check", but I'm interested in what everyone else thinks. > for a nice code cleanup. > > Fixes: 3ee47cad3e69 ("media: ov2680: Add Omnivision OV2680 sensor driver") > Signed-off-by: Hans de Goede <hdegoede@redhat.com> > --- > drivers/media/i2c/ov2680.c | 48 +++++++++----------------------------- > 1 file changed, 11 insertions(+), 37 deletions(-) > > diff --git a/drivers/media/i2c/ov2680.c b/drivers/media/i2c/ov2680.c > index 9a9c90c8a949..c1b23c5b7818 100644 > --- a/drivers/media/i2c/ov2680.c > +++ b/drivers/media/i2c/ov2680.c > @@ -328,11 +328,14 @@ static void ov2680_set_bayer_order(struct ov2680_dev *sensor) > sensor->fmt.code = ov2680_hv_flip_bayer_order[hv_flip]; > } > > -static int ov2680_vflip_enable(struct ov2680_dev *sensor) > +static int ov2680_set_vflip(struct ov2680_dev *sensor, s32 val) > { > int ret; > > - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), BIT(2)); > + if (sensor->is_streaming) > + return -EBUSY; > + > + ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), val ? BIT(2) : 0); > if (ret < 0) > return ret; > > @@ -340,33 +343,14 @@ static int ov2680_vflip_enable(struct ov2680_dev *sensor) > return 0; > } > > -static int ov2680_vflip_disable(struct ov2680_dev *sensor) > +static int ov2680_set_hflip(struct ov2680_dev *sensor, s32 val) > { > int ret; > > - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), BIT(0)); > - if (ret < 0) > - return ret; > + if (sensor->is_streaming) > + return -EBUSY; > > - return ov2680_bayer_order(sensor); > -} > - > -static int ov2680_hflip_enable(struct ov2680_dev *sensor) > -{ > - int ret; > - > - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), BIT(2)); > - if (ret < 0) > - return ret; > - > - return ov2680_bayer_order(sensor); > -} > - > -static int ov2680_hflip_disable(struct ov2680_dev *sensor) > -{ > - int ret; > - > - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), BIT(0)); > + ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), val ? BIT(2) : 0); > if (ret < 0) > return ret; > > @@ -719,19 +703,9 @@ static int ov2680_s_ctrl(struct v4l2_ctrl *ctrl) > case V4L2_CID_EXPOSURE: > return ov2680_exposure_set(sensor, ctrl->val); > case V4L2_CID_VFLIP: > - if (sensor->is_streaming) > - return -EBUSY; > - if (ctrl->val) > - return ov2680_vflip_enable(sensor); > - else > - return ov2680_vflip_disable(sensor); > + return ov2680_set_vflip(sensor, ctrl->val); > case V4L2_CID_HFLIP: > - if (sensor->is_streaming) > - return -EBUSY; > - if (ctrl->val) > - return ov2680_hflip_enable(sensor); > - else > - return ov2680_hflip_disable(sensor); > + return ov2680_set_hflip(sensor, ctrl->val); > case V4L2_CID_TEST_PATTERN: > return ov2680_test_pattern_set(sensor, ctrl->val); > default:
Hi, On 6/12/23 09:32, Dan Scally wrote: > Hello Hans (and everybody else) > > On 07/06/2023 17:46, Hans de Goede wrote: >> ov2680_vflip_disable() / ov2680_hflip_disable() pass BIT(0) instead of >> 0 as value to ov2680_mod_reg(). >> >> While fixing this also: >> >> 1. Stop having separate enable/disable functions for hflip / vflip >> 2. Move the is_streaming check, which is unique to hflip / vflip >> into the ov2680_set_?flip() functions. > > > I think the code in the patch is good, but I wonder about the right way to deal with the changed pixel ordering following a flip/mirror. None of the other Omnivision sensor drivers I've looked at disallow a flip/mirror whilst streaming, but of course doing so wrecks the colours. I don't think that there's any reason we couldn't make the equivalent calculation to ov2680_set_bayer_order() within libcamera though and configure the ipu3-imgu appropriately, and so properly support a flip-whilst-streaming...but quite possibly I'm missing something. So I'm leaning towards "let's just drop the is_streaming check", but I'm interested in what everyone else thinks. So the story wrt flipping on the ov2680 is complicated. I have done quite a bit of testing wrt the flipping with the atomisp-ov2680.c version of the code and I have made the following observations: 1. By default the ISP window is set to auto which means the ISP will set it so that the bayer order does NOT change when flipping. Except when trying to use the full native sensor resolution because then the ISP has no "room" to adjust the window to fixup the bayer order. So the whole ov2680_set_bayer_order() helper is only relevant when either using manual ISP settings or when using the full native resolution. Note the ISP here is the sensor's internal ISP (which has very little functionality on the ov2680). 2. Independent of auto vs manual ISP window settings changing flipping during mid stream does not work. Instead of the colors getting messed up the sensor simply just stops streaming when changing the flipping mid stream. Because of 2. I have decided to keep ov2680_set_bayer_order() and add support for manually setting the ISP window so that the full native size can be used. Having the full native size available is important for the atomisp ISP since that expects the resolution outputted by the sensor to have some padding (minimum 12 / 14 pixels depending on bayer order) vs the resolution being output to userspace. Manually setting the ISP window makes things actually work as the code expected before doing this (the manually setting the ISP window is also done in the commit adding the mode-dependent registers calculation). 2. Also means we need to keep the is_streaming checks. Changing flipping while streaming simply does not work on this sensor (or at least I don't know how to make it work). As Sakari already mentioned it seems that the current mainline driver has not seen much testing of things like the flip controls. Regards, Hans > >> for a nice code cleanup. >> >> Fixes: 3ee47cad3e69 ("media: ov2680: Add Omnivision OV2680 sensor driver") >> Signed-off-by: Hans de Goede <hdegoede@redhat.com> >> --- >> drivers/media/i2c/ov2680.c | 48 +++++++++----------------------------- >> 1 file changed, 11 insertions(+), 37 deletions(-) >> >> diff --git a/drivers/media/i2c/ov2680.c b/drivers/media/i2c/ov2680.c >> index 9a9c90c8a949..c1b23c5b7818 100644 >> --- a/drivers/media/i2c/ov2680.c >> +++ b/drivers/media/i2c/ov2680.c >> @@ -328,11 +328,14 @@ static void ov2680_set_bayer_order(struct ov2680_dev *sensor) >> sensor->fmt.code = ov2680_hv_flip_bayer_order[hv_flip]; >> } >> -static int ov2680_vflip_enable(struct ov2680_dev *sensor) >> +static int ov2680_set_vflip(struct ov2680_dev *sensor, s32 val) >> { >> int ret; >> - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), BIT(2)); >> + if (sensor->is_streaming) >> + return -EBUSY; >> + >> + ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), val ? BIT(2) : 0); >> if (ret < 0) >> return ret; >> @@ -340,33 +343,14 @@ static int ov2680_vflip_enable(struct ov2680_dev *sensor) >> return 0; >> } >> -static int ov2680_vflip_disable(struct ov2680_dev *sensor) >> +static int ov2680_set_hflip(struct ov2680_dev *sensor, s32 val) >> { >> int ret; >> - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), BIT(0)); >> - if (ret < 0) >> - return ret; >> + if (sensor->is_streaming) >> + return -EBUSY; >> - return ov2680_bayer_order(sensor); >> -} >> - >> -static int ov2680_hflip_enable(struct ov2680_dev *sensor) >> -{ >> - int ret; >> - >> - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), BIT(2)); >> - if (ret < 0) >> - return ret; >> - >> - return ov2680_bayer_order(sensor); >> -} >> - >> -static int ov2680_hflip_disable(struct ov2680_dev *sensor) >> -{ >> - int ret; >> - >> - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), BIT(0)); >> + ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), val ? BIT(2) : 0); >> if (ret < 0) >> return ret; >> @@ -719,19 +703,9 @@ static int ov2680_s_ctrl(struct v4l2_ctrl *ctrl) >> case V4L2_CID_EXPOSURE: >> return ov2680_exposure_set(sensor, ctrl->val); >> case V4L2_CID_VFLIP: >> - if (sensor->is_streaming) >> - return -EBUSY; >> - if (ctrl->val) >> - return ov2680_vflip_enable(sensor); >> - else >> - return ov2680_vflip_disable(sensor); >> + return ov2680_set_vflip(sensor, ctrl->val); >> case V4L2_CID_HFLIP: >> - if (sensor->is_streaming) >> - return -EBUSY; >> - if (ctrl->val) >> - return ov2680_hflip_enable(sensor); >> - else >> - return ov2680_hflip_disable(sensor); >> + return ov2680_set_hflip(sensor, ctrl->val); >> case V4L2_CID_TEST_PATTERN: >> return ov2680_test_pattern_set(sensor, ctrl->val); >> default: >
On Mon, Jun 12, 2023 at 10:06:09AM +0200, Hans de Goede wrote: > On 6/12/23 09:32, Dan Scally wrote: > > Hello Hans (and everybody else) > > > > On 07/06/2023 17:46, Hans de Goede wrote: > >> ov2680_vflip_disable() / ov2680_hflip_disable() pass BIT(0) instead of > >> 0 as value to ov2680_mod_reg(). > >> > >> While fixing this also: > >> > >> 1. Stop having separate enable/disable functions for hflip / vflip > >> 2. Move the is_streaming check, which is unique to hflip / vflip > >> into the ov2680_set_?flip() functions. > > > > I think the code in the patch is good, but I wonder about the right > > way to deal with the changed pixel ordering following a flip/mirror. > > None of the other Omnivision sensor drivers I've looked at disallow > > a flip/mirror whilst streaming, but of course doing so wrecks the > > colours. I don't think that there's any reason we couldn't make the > > equivalent calculation to ov2680_set_bayer_order() within libcamera > > though and configure the ipu3-imgu appropriately, and so properly > > support a flip-whilst-streaming...but quite possibly I'm missing > > something. So I'm leaning towards "let's just drop the is_streaming > > check", but I'm interested in what everyone else thinks. > > So the story wrt flipping on the ov2680 is complicated. The story of flipping is complicated, period :-) We made a mistake in V4L2 when adding support for Bayer formats. We should have created RAW{8,10,12,...} media bus codes and pixel formats, regardless of the CFA pattern, and conveyed the pattern information out-of-band. This would have allowed changing the CFA pattern during streaming. As a result of this bad decision, various sensor drivers have decided to move the crop rectangle by one pixel when flipping, in order to preserve the CFA pattern. In some cases, this crop rectangle adjustement is even performed by the camera sensor itself (and I've seen OV sensors where this behaviour is configurable, you can disable the auto-adjustment through a register). Other drivers change their reported CFA pattern when flipping, and disable flipping changes during streaming. TL;DR: It's a mess. It's probably not too late to fix that, and move towards RAW media bus codes, but it won't be easy. > I have done quite a bit > of testing wrt the flipping with the atomisp-ov2680.c version of the code and > I have made the following observations: > > 1. By default the ISP window is set to auto which means the ISP will set it so that > the bayer order does NOT change when flipping. Except when trying to use the full > native sensor resolution because then the ISP has no "room" to adjust the window to > fixup the bayer order. So the whole ov2680_set_bayer_order() helper is only relevant > when either using manual ISP settings or when using the full native resolution. > Note the ISP here is the sensor's internal ISP (which has very little functionality > on the ov2680). > > 2. Independent of auto vs manual ISP window settings changing flipping during mid > stream does not work. Instead of the colors getting messed up the sensor simply just > stops streaming when changing the flipping mid stream. > > Because of 2. I have decided to keep ov2680_set_bayer_order() and add support > for manually setting the ISP window so that the full native size can be used. > > Having the full native size available is important for the atomisp ISP since > that expects the resolution outputted by the sensor to have some padding > (minimum 12 / 14 pixels depending on bayer order) vs the resolution being > output to userspace. > > Manually setting the ISP window makes things actually work as the code > expected before doing this (the manually setting the ISP window is also done > in the commit adding the mode-dependent registers calculation). > > 2. Also means we need to keep the is_streaming checks. Changing flipping > while streaming simply does not work on this sensor (or at least I don't know > how to make it work). > > As Sakari already mentioned it seems that the current mainline driver has > not seen much testing of things like the flip controls. > > >> for a nice code cleanup. > >> > >> Fixes: 3ee47cad3e69 ("media: ov2680: Add Omnivision OV2680 sensor driver") > >> Signed-off-by: Hans de Goede <hdegoede@redhat.com> > >> --- > >> drivers/media/i2c/ov2680.c | 48 +++++++++----------------------------- > >> 1 file changed, 11 insertions(+), 37 deletions(-) > >> > >> diff --git a/drivers/media/i2c/ov2680.c b/drivers/media/i2c/ov2680.c > >> index 9a9c90c8a949..c1b23c5b7818 100644 > >> --- a/drivers/media/i2c/ov2680.c > >> +++ b/drivers/media/i2c/ov2680.c > >> @@ -328,11 +328,14 @@ static void ov2680_set_bayer_order(struct ov2680_dev *sensor) > >> sensor->fmt.code = ov2680_hv_flip_bayer_order[hv_flip]; > >> } > >> -static int ov2680_vflip_enable(struct ov2680_dev *sensor) > >> +static int ov2680_set_vflip(struct ov2680_dev *sensor, s32 val) > >> { > >> int ret; > >> - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), BIT(2)); > >> + if (sensor->is_streaming) > >> + return -EBUSY; > >> + > >> + ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), val ? BIT(2) : 0); > >> if (ret < 0) > >> return ret; > >> @@ -340,33 +343,14 @@ static int ov2680_vflip_enable(struct ov2680_dev *sensor) > >> return 0; > >> } > >> -static int ov2680_vflip_disable(struct ov2680_dev *sensor) > >> +static int ov2680_set_hflip(struct ov2680_dev *sensor, s32 val) > >> { > >> int ret; > >> - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), BIT(0)); > >> - if (ret < 0) > >> - return ret; > >> + if (sensor->is_streaming) > >> + return -EBUSY; > >> - return ov2680_bayer_order(sensor); > >> -} > >> - > >> -static int ov2680_hflip_enable(struct ov2680_dev *sensor) > >> -{ > >> - int ret; > >> - > >> - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), BIT(2)); > >> - if (ret < 0) > >> - return ret; > >> - > >> - return ov2680_bayer_order(sensor); > >> -} > >> - > >> -static int ov2680_hflip_disable(struct ov2680_dev *sensor) > >> -{ > >> - int ret; > >> - > >> - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), BIT(0)); > >> + ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), val ? BIT(2) : 0); > >> if (ret < 0) > >> return ret; > >> @@ -719,19 +703,9 @@ static int ov2680_s_ctrl(struct v4l2_ctrl *ctrl) > >> case V4L2_CID_EXPOSURE: > >> return ov2680_exposure_set(sensor, ctrl->val); > >> case V4L2_CID_VFLIP: > >> - if (sensor->is_streaming) > >> - return -EBUSY; > >> - if (ctrl->val) > >> - return ov2680_vflip_enable(sensor); > >> - else > >> - return ov2680_vflip_disable(sensor); > >> + return ov2680_set_vflip(sensor, ctrl->val); > >> case V4L2_CID_HFLIP: > >> - if (sensor->is_streaming) > >> - return -EBUSY; > >> - if (ctrl->val) > >> - return ov2680_hflip_enable(sensor); > >> - else > >> - return ov2680_hflip_disable(sensor); > >> + return ov2680_set_hflip(sensor, ctrl->val); > >> case V4L2_CID_TEST_PATTERN: > >> return ov2680_test_pattern_set(sensor, ctrl->val); > >> default:
diff --git a/drivers/media/i2c/ov2680.c b/drivers/media/i2c/ov2680.c index 9a9c90c8a949..c1b23c5b7818 100644 --- a/drivers/media/i2c/ov2680.c +++ b/drivers/media/i2c/ov2680.c @@ -328,11 +328,14 @@ static void ov2680_set_bayer_order(struct ov2680_dev *sensor) sensor->fmt.code = ov2680_hv_flip_bayer_order[hv_flip]; } -static int ov2680_vflip_enable(struct ov2680_dev *sensor) +static int ov2680_set_vflip(struct ov2680_dev *sensor, s32 val) { int ret; - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), BIT(2)); + if (sensor->is_streaming) + return -EBUSY; + + ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), val ? BIT(2) : 0); if (ret < 0) return ret; @@ -340,33 +343,14 @@ static int ov2680_vflip_enable(struct ov2680_dev *sensor) return 0; } -static int ov2680_vflip_disable(struct ov2680_dev *sensor) +static int ov2680_set_hflip(struct ov2680_dev *sensor, s32 val) { int ret; - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT1, BIT(2), BIT(0)); - if (ret < 0) - return ret; + if (sensor->is_streaming) + return -EBUSY; - return ov2680_bayer_order(sensor); -} - -static int ov2680_hflip_enable(struct ov2680_dev *sensor) -{ - int ret; - - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), BIT(2)); - if (ret < 0) - return ret; - - return ov2680_bayer_order(sensor); -} - -static int ov2680_hflip_disable(struct ov2680_dev *sensor) -{ - int ret; - - ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), BIT(0)); + ret = ov2680_mod_reg(sensor, OV2680_REG_FORMAT2, BIT(2), val ? BIT(2) : 0); if (ret < 0) return ret; @@ -719,19 +703,9 @@ static int ov2680_s_ctrl(struct v4l2_ctrl *ctrl) case V4L2_CID_EXPOSURE: return ov2680_exposure_set(sensor, ctrl->val); case V4L2_CID_VFLIP: - if (sensor->is_streaming) - return -EBUSY; - if (ctrl->val) - return ov2680_vflip_enable(sensor); - else - return ov2680_vflip_disable(sensor); + return ov2680_set_vflip(sensor, ctrl->val); case V4L2_CID_HFLIP: - if (sensor->is_streaming) - return -EBUSY; - if (ctrl->val) - return ov2680_hflip_enable(sensor); - else - return ov2680_hflip_disable(sensor); + return ov2680_set_hflip(sensor, ctrl->val); case V4L2_CID_TEST_PATTERN: return ov2680_test_pattern_set(sensor, ctrl->val); default: