2 * Functions for auto gain.
4 * Copyright (C) 2010-2012 Hans de Goede <hdegoede@redhat.com>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 /* auto gain and exposure algorithm based on the knee algorithm described here:
23 http://ytse.tricolour.net/docs/LowLightOptimization.html
25 Returns 0 if no changes were made, 1 if the gain and or exposure settings
27 int gspca_expo_autogain(
28 struct gspca_dev
*gspca_dev
,
35 s32 gain
, orig_gain
, exposure
, orig_exposure
;
36 int i
, steps
, retval
= 0;
38 if (v4l2_ctrl_g_ctrl(gspca_dev
->autogain
) == 0)
41 orig_gain
= gain
= v4l2_ctrl_g_ctrl(gspca_dev
->gain
);
42 orig_exposure
= exposure
= v4l2_ctrl_g_ctrl(gspca_dev
->exposure
);
44 /* If we are of a multiple of deadzone, do multiple steps to reach the
45 desired lumination fast (with the risc of a slight overshoot) */
46 steps
= abs(desired_avg_lum
- avg_lum
) / deadzone
;
48 PDEBUG(D_FRAM
, "autogain: lum: %d, desired: %d, steps: %d",
49 avg_lum
, desired_avg_lum
, steps
);
51 for (i
= 0; i
< steps
; i
++) {
52 if (avg_lum
> desired_avg_lum
) {
55 else if (exposure
> exposure_knee
)
57 else if (gain
> gspca_dev
->gain
->default_value
)
59 else if (exposure
> gspca_dev
->exposure
->minimum
)
61 else if (gain
> gspca_dev
->gain
->minimum
)
66 if (gain
< gspca_dev
->gain
->default_value
)
68 else if (exposure
< exposure_knee
)
70 else if (gain
< gain_knee
)
72 else if (exposure
< gspca_dev
->exposure
->maximum
)
74 else if (gain
< gspca_dev
->gain
->maximum
)
81 if (gain
!= orig_gain
) {
82 v4l2_ctrl_s_ctrl(gspca_dev
->gain
, gain
);
85 if (exposure
!= orig_exposure
) {
86 v4l2_ctrl_s_ctrl(gspca_dev
->exposure
, exposure
);
91 PDEBUG(D_FRAM
, "autogain: changed gain: %d, expo: %d",
95 EXPORT_SYMBOL(gspca_expo_autogain
);
97 /* Autogain + exposure algorithm for cameras with a coarse exposure control
98 (usually this means we can only control the clockdiv to change exposure)
99 As changing the clockdiv so that the fps drops from 30 to 15 fps for
100 example, will lead to a huge exposure change (it effectively doubles),
101 this algorithm normally tries to only adjust the gain (between 40 and
102 80 %) and if that does not help, only then changes exposure. This leads
103 to a much more stable image then using the knee algorithm which at
104 certain points of the knee graph will only try to adjust exposure,
105 which leads to oscilating as one exposure step is huge.
107 Returns 0 if no changes were made, 1 if the gain and or exposure settings
109 int gspca_coarse_grained_expo_autogain(
110 struct gspca_dev
*gspca_dev
,
115 s32 gain_low
, gain_high
, gain
, orig_gain
, exposure
, orig_exposure
;
116 int steps
, retval
= 0;
118 if (v4l2_ctrl_g_ctrl(gspca_dev
->autogain
) == 0)
121 orig_gain
= gain
= v4l2_ctrl_g_ctrl(gspca_dev
->gain
);
122 orig_exposure
= exposure
= v4l2_ctrl_g_ctrl(gspca_dev
->exposure
);
124 gain_low
= (s32
)(gspca_dev
->gain
->maximum
- gspca_dev
->gain
->minimum
) /
125 5 * 2 + gspca_dev
->gain
->minimum
;
126 gain_high
= (s32
)(gspca_dev
->gain
->maximum
- gspca_dev
->gain
->minimum
) /
127 5 * 4 + gspca_dev
->gain
->minimum
;
129 /* If we are of a multiple of deadzone, do multiple steps to reach the
130 desired lumination fast (with the risc of a slight overshoot) */
131 steps
= (desired_avg_lum
- avg_lum
) / deadzone
;
133 PDEBUG(D_FRAM
, "autogain: lum: %d, desired: %d, steps: %d",
134 avg_lum
, desired_avg_lum
, steps
);
136 if ((gain
+ steps
) > gain_high
&&
137 exposure
< gspca_dev
->exposure
->maximum
) {
139 gspca_dev
->exp_too_low_cnt
++;
140 gspca_dev
->exp_too_high_cnt
= 0;
141 } else if ((gain
+ steps
) < gain_low
&&
142 exposure
> gspca_dev
->exposure
->minimum
) {
144 gspca_dev
->exp_too_high_cnt
++;
145 gspca_dev
->exp_too_low_cnt
= 0;
148 if (gain
> gspca_dev
->gain
->maximum
)
149 gain
= gspca_dev
->gain
->maximum
;
150 else if (gain
< gspca_dev
->gain
->minimum
)
151 gain
= gspca_dev
->gain
->minimum
;
152 gspca_dev
->exp_too_high_cnt
= 0;
153 gspca_dev
->exp_too_low_cnt
= 0;
156 if (gspca_dev
->exp_too_high_cnt
> 3) {
158 gspca_dev
->exp_too_high_cnt
= 0;
159 } else if (gspca_dev
->exp_too_low_cnt
> 3) {
161 gspca_dev
->exp_too_low_cnt
= 0;
164 if (gain
!= orig_gain
) {
165 v4l2_ctrl_s_ctrl(gspca_dev
->gain
, gain
);
168 if (exposure
!= orig_exposure
) {
169 v4l2_ctrl_s_ctrl(gspca_dev
->exposure
, exposure
);
174 PDEBUG(D_FRAM
, "autogain: changed gain: %d, expo: %d",
178 EXPORT_SYMBOL(gspca_coarse_grained_expo_autogain
);