Lines Matching +full:g +full:- +full:scaler
1 // SPDX-License-Identifier: GPL-2.0-only
2 /* gain-time-scale conversion helpers for IIO light sensors
18 #include <linux/iio/iio-gts-helper.h>
22 * iio_gts_get_gain - Convert scale to total gain
27 * in magnitude of NANOs and max scale is 64.1 - The linearized
31 * Return: (floored) gain corresponding to the scale. -EINVAL if scale
39 return -EINVAL; in iio_gts_get_gain()
45 * gain_get_scale_fraction - get the gain or time based on scale and known one
48 * in magnitude of NANOs and max scale is 64.1 - The linearized
73 return -EINVAL; in gain_get_scale_fraction()
78 static int iio_gts_delinearize(u64 lin_scale, unsigned long scaler, in iio_gts_delinearize() argument
83 if (scaler > NANO) in iio_gts_delinearize()
84 return -EOVERFLOW; in iio_gts_delinearize()
86 if (!scaler) in iio_gts_delinearize()
87 return -EINVAL; in iio_gts_delinearize()
89 frac = do_div(lin_scale, scaler); in iio_gts_delinearize()
92 *scale_nano = frac * (NANO / scaler); in iio_gts_delinearize()
98 unsigned long scaler, u64 *lin_scale) in iio_gts_linearize() argument
104 if (scaler > NANO || !scaler) in iio_gts_linearize()
105 return -EINVAL; in iio_gts_linearize()
107 *lin_scale = (u64)scale_whole * (u64)scaler + in iio_gts_linearize()
108 (u64)(scale_nano / (NANO / scaler)); in iio_gts_linearize()
114 * iio_gts_total_gain_to_scale - convert gain to scale
121 * generated by HW-gain or integration time. It is up to caller to decide what
122 * part of the total gain is due to integration time and what due to HW-gain.
131 tmp = gts->max_scale; in iio_gts_total_gain_to_scale()
140 * iio_gts_purge_avail_scale_table - free-up the available scale tables
149 if (gts->per_time_avail_scale_tables) { in iio_gts_purge_avail_scale_table()
150 for (i = 0; i < gts->num_itime; i++) in iio_gts_purge_avail_scale_table()
151 kfree(gts->per_time_avail_scale_tables[i]); in iio_gts_purge_avail_scale_table()
153 kfree(gts->per_time_avail_scale_tables); in iio_gts_purge_avail_scale_table()
154 gts->per_time_avail_scale_tables = NULL; in iio_gts_purge_avail_scale_table()
157 kfree(gts->avail_all_scales_table); in iio_gts_purge_avail_scale_table()
158 gts->avail_all_scales_table = NULL; in iio_gts_purge_avail_scale_table()
160 gts->num_avail_all_scales = 0; in iio_gts_purge_avail_scale_table()
165 return *(int *)a - *(int *)b; in iio_gts_gain_cmp()
174 for (i = 0; i < gts->num_itime; i++) { in gain_to_scaletables()
179 sort(gains[i], gts->num_hwgain, sizeof(int), iio_gts_gain_cmp, in gain_to_scaletables()
183 for (j = 0; j < gts->num_hwgain; j++) { in gain_to_scaletables()
192 gain_bytes = array_size(gts->num_hwgain, sizeof(int)); in gain_to_scaletables()
193 all_gains = kcalloc(gts->num_itime, gain_bytes, GFP_KERNEL); in gain_to_scaletables()
195 return -ENOMEM; in gain_to_scaletables()
204 time_idx = gts->num_itime - 1; in gain_to_scaletables()
206 new_idx = gts->num_hwgain; in gain_to_scaletables()
208 while (time_idx--) { in gain_to_scaletables()
209 for (j = 0; j < gts->num_hwgain; j++) { in gain_to_scaletables()
213 if (candidate > all_gains[new_idx - 1]) { in gain_to_scaletables()
227 (new_idx - chk) * sizeof(int)); in gain_to_scaletables()
233 gts->avail_all_scales_table = kcalloc(new_idx, 2 * sizeof(int), in gain_to_scaletables()
235 if (!gts->avail_all_scales_table) { in gain_to_scaletables()
236 ret = -ENOMEM; in gain_to_scaletables()
239 gts->num_avail_all_scales = new_idx; in gain_to_scaletables()
241 for (i = 0; i < gts->num_avail_all_scales; i++) { in gain_to_scaletables()
243 >s->avail_all_scales_table[i * 2], in gain_to_scaletables()
244 >s->avail_all_scales_table[i * 2 + 1]); in gain_to_scaletables()
247 kfree(gts->avail_all_scales_table); in gain_to_scaletables()
248 gts->num_avail_all_scales = 0; in gain_to_scaletables()
260 * iio_gts_build_avail_scale_table - create tables of available scales
278 int **per_time_gains, **per_time_scales, i, j, ret = -ENOMEM; in iio_gts_build_avail_scale_table()
280 per_time_gains = kcalloc(gts->num_itime, sizeof(*per_time_gains), GFP_KERNEL); in iio_gts_build_avail_scale_table()
284 per_time_scales = kcalloc(gts->num_itime, sizeof(*per_time_scales), GFP_KERNEL); in iio_gts_build_avail_scale_table()
288 for (i = 0; i < gts->num_itime; i++) { in iio_gts_build_avail_scale_table()
289 per_time_scales[i] = kcalloc(gts->num_hwgain, 2 * sizeof(int), in iio_gts_build_avail_scale_table()
294 per_time_gains[i] = kcalloc(gts->num_hwgain, sizeof(int), in iio_gts_build_avail_scale_table()
301 for (j = 0; j < gts->num_hwgain; j++) in iio_gts_build_avail_scale_table()
302 per_time_gains[i][j] = gts->hwgain_table[j].gain * in iio_gts_build_avail_scale_table()
303 gts->itime_table[i].mul; in iio_gts_build_avail_scale_table()
310 for (i = 0; i < gts->num_itime; i++) in iio_gts_build_avail_scale_table()
313 gts->per_time_avail_scale_tables = per_time_scales; in iio_gts_build_avail_scale_table()
318 for (i--; i >= 0; i--) { in iio_gts_build_avail_scale_table()
341 * iio_gts_build_avail_time_table - build table of available integration times
345 * to users using the read_avail-callback.
356 if (!gts->num_itime) in iio_gts_build_avail_time_table()
359 times = kcalloc(gts->num_itime, sizeof(int), GFP_KERNEL); in iio_gts_build_avail_time_table()
361 return -ENOMEM; in iio_gts_build_avail_time_table()
364 for (i = gts->num_itime - 1; i >= 0; i--) { in iio_gts_build_avail_time_table()
365 int new = gts->itime_table[i].time_us; in iio_gts_build_avail_time_table()
367 if (idx == 0 || times[idx - 1] < new) { in iio_gts_build_avail_time_table()
377 (idx - j) * sizeof(int)); in iio_gts_build_avail_time_table()
389 * This is just to survive a unlikely corner-case where times in in iio_gts_build_avail_time_table()
391 * trust the gts->num_itime. in iio_gts_build_avail_time_table()
393 gts->num_avail_time_tables = idx; in iio_gts_build_avail_time_table()
397 gts->avail_time_tables = int_micro_times; in iio_gts_build_avail_time_table()
401 return -ENOMEM; in iio_gts_build_avail_time_table()
407 * iio_gts_purge_avail_time_table - free-up the available integration time table
414 if (gts->num_avail_time_tables) { in iio_gts_purge_avail_time_table()
415 kfree(gts->avail_time_tables); in iio_gts_purge_avail_time_table()
416 gts->avail_time_tables = NULL; in iio_gts_purge_avail_time_table()
417 gts->num_avail_time_tables = 0; in iio_gts_purge_avail_time_table()
422 * iio_gts_build_avail_tables - create tables of available scales and int times
462 * iio_gts_purge_avail_tables - free-up the availability tables
480 * devm_iio_gts_build_avail_tables - manged add availability tables
517 if (t->sel < 0 || t->time_us < 0 || t->mul <= 0) in sanity_check_time()
518 return -EINVAL; in sanity_check_time()
523 static int sanity_check_gain(const struct iio_gain_sel_pair *g) in sanity_check_gain() argument
525 if (g->sel < 0 || g->gain <= 0) in sanity_check_gain()
526 return -EINVAL; in sanity_check_gain()
533 int g, t, ret; in iio_gts_sanity_check() local
535 if (!gts->num_hwgain && !gts->num_itime) in iio_gts_sanity_check()
536 return -EINVAL; in iio_gts_sanity_check()
538 for (t = 0; t < gts->num_itime; t++) { in iio_gts_sanity_check()
539 ret = sanity_check_time(>s->itime_table[t]); in iio_gts_sanity_check()
544 for (g = 0; g < gts->num_hwgain; g++) { in iio_gts_sanity_check()
545 ret = sanity_check_gain(>s->hwgain_table[g]); in iio_gts_sanity_check()
550 for (g = 0; g < gts->num_hwgain; g++) { in iio_gts_sanity_check()
551 for (t = 0; t < gts->num_itime; t++) { in iio_gts_sanity_check()
554 gain = gts->hwgain_table[g].gain; in iio_gts_sanity_check()
555 mul = gts->itime_table[t].mul; in iio_gts_sanity_check()
558 return -EOVERFLOW; in iio_gts_sanity_check()
575 >s->max_scale); in iio_init_iio_gts()
579 gts->hwgain_table = gain_tbl; in iio_init_iio_gts()
580 gts->num_hwgain = num_gain; in iio_init_iio_gts()
581 gts->itime_table = tim_tbl; in iio_init_iio_gts()
582 gts->num_itime = num_times; in iio_init_iio_gts()
588 * devm_iio_init_iio_gts - Initialize the gain-time-scale helper
604 * Initialize the gain-time-scale helper for use. Note, gains, times, selectors
628 * iio_gts_all_avail_scales - helper for listing all available scales
639 if (!gts->num_avail_all_scales) in iio_gts_all_avail_scales()
640 return -EINVAL; in iio_gts_all_avail_scales()
642 *vals = gts->avail_all_scales_table; in iio_gts_all_avail_scales()
644 *length = gts->num_avail_all_scales * 2; in iio_gts_all_avail_scales()
651 * iio_gts_avail_scales_for_time - list scales for integration time
669 for (i = 0; i < gts->num_itime; i++) in iio_gts_avail_scales_for_time()
670 if (gts->itime_table[i].time_us == time) in iio_gts_avail_scales_for_time()
673 if (i == gts->num_itime) in iio_gts_avail_scales_for_time()
674 return -EINVAL; in iio_gts_avail_scales_for_time()
676 *vals = gts->per_time_avail_scale_tables[i]; in iio_gts_avail_scales_for_time()
678 *length = gts->num_hwgain * 2; in iio_gts_avail_scales_for_time()
685 * iio_gts_avail_times - helper for listing available integration times
696 if (!gts->num_avail_time_tables) in iio_gts_avail_times()
697 return -EINVAL; in iio_gts_avail_times()
699 *vals = gts->avail_time_tables; in iio_gts_avail_times()
701 *length = gts->num_avail_time_tables * 2; in iio_gts_avail_times()
708 * iio_gts_find_sel_by_gain - find selector corresponding to a HW-gain
710 * @gain: HW-gain for which matching selector is searched for
712 * Return: a selector matching given HW-gain or -EINVAL if selector was
719 for (i = 0; i < gts->num_hwgain; i++) in iio_gts_find_sel_by_gain()
720 if (gts->hwgain_table[i].gain == gain) in iio_gts_find_sel_by_gain()
721 return gts->hwgain_table[i].sel; in iio_gts_find_sel_by_gain()
723 return -EINVAL; in iio_gts_find_sel_by_gain()
728 * iio_gts_find_gain_by_sel - find HW-gain corresponding to a selector
730 * @sel: selector for which matching HW-gain is searched for
732 * Return: a HW-gain matching given selector or -EINVAL if HW-gain was not
739 for (i = 0; i < gts->num_hwgain; i++) in iio_gts_find_gain_by_sel()
740 if (gts->hwgain_table[i].sel == sel) in iio_gts_find_gain_by_sel()
741 return gts->hwgain_table[i].gain; in iio_gts_find_gain_by_sel()
743 return -EINVAL; in iio_gts_find_gain_by_sel()
748 * iio_gts_get_min_gain - find smallest valid HW-gain
751 * Return: The smallest HW-gain -EINVAL if no HW-gains were in the tables.
755 int i, min = -EINVAL; in iio_gts_get_min_gain()
757 for (i = 0; i < gts->num_hwgain; i++) { in iio_gts_get_min_gain()
758 int gain = gts->hwgain_table[i].gain; in iio_gts_get_min_gain()
760 if (min == -EINVAL) in iio_gts_get_min_gain()
771 * iio_find_closest_gain_low - Find the closest lower matching gain
773 * @gain: HW-gain for which the closest match is searched
783 * Return: The closest matching supported gain or -EINVAL if @gain
789 int best = -1; in iio_find_closest_gain_low()
793 for (i = 0; i < gts->num_hwgain; i++) { in iio_find_closest_gain_low()
794 if (gain == gts->hwgain_table[i].gain) { in iio_find_closest_gain_low()
799 if (gain > gts->hwgain_table[i].gain) { in iio_find_closest_gain_low()
801 diff = gain - gts->hwgain_table[i].gain; in iio_find_closest_gain_low()
804 int tmp = gain - gts->hwgain_table[i].gain; in iio_find_closest_gain_low()
813 * We found valid HW-gain which is greater than in iio_find_closest_gain_low()
815 * will have found an in-range gain in iio_find_closest_gain_low()
824 return -EINVAL; in iio_find_closest_gain_low()
827 return gts->hwgain_table[best].gain; in iio_find_closest_gain_low()
838 return -EINVAL; in iio_gts_get_int_time_gain_multiplier_by_sel()
840 return time->mul; in iio_gts_get_int_time_gain_multiplier_by_sel()
844 * iio_gts_find_gain_for_scale_using_time - Find gain by time and scale
856 * Return: 0 on success. -EINVAL if gain matching the parameters is not
876 ret = gain_get_scale_fraction(gts->max_scale, scale_linear, mul, gain); in iio_gts_find_gain_for_scale_using_time()
881 return -EINVAL; in iio_gts_find_gain_for_scale_using_time()
887 * iio_gts_find_gain_sel_for_scale_using_time - Fetch gain selector.
923 return -EINVAL; in iio_gts_get_total_gain()
925 if (!gts->num_itime) in iio_gts_get_total_gain()
930 return -EINVAL; in iio_gts_get_total_gain()
932 return gain * itime->mul; in iio_gts_get_total_gain()
945 tmp = gts->max_scale; in iio_gts_get_scale_linear()
955 * iio_gts_get_scale - get scale based on integration time and HW-gain
957 * @gain: HW-gain for which the scale is computed
962 * Compute scale matching the integration time and HW-gain given as parameter.
981 * iio_gts_find_new_gain_sel_by_old_gain_time - compensate for time change
993 * non-zero value is returned, the @new_gain will be set to a negative or
1010 *new_gain = -1; in iio_gts_find_new_gain_sel_by_old_gain_time()
1014 return -EINVAL; in iio_gts_find_new_gain_sel_by_old_gain_time()
1018 return -EINVAL; in iio_gts_find_new_gain_sel_by_old_gain_time()
1020 ret = iio_gts_get_scale_linear(gts, old_gain, itime_old->time_us, in iio_gts_find_new_gain_sel_by_old_gain_time()
1025 ret = gain_get_scale_fraction(gts->max_scale, scale, itime_new->mul, in iio_gts_find_new_gain_sel_by_old_gain_time()
1031 return -EINVAL; in iio_gts_find_new_gain_sel_by_old_gain_time()
1038 * iio_gts_find_new_gain_by_old_gain_time - compensate for time change
1050 * non-zero value is returned, the @new_gain will be set to a negative or
1067 *new_gain = -1; in iio_gts_find_new_gain_by_old_gain_time()
1071 return -EINVAL; in iio_gts_find_new_gain_by_old_gain_time()
1077 ret = gain_get_scale_fraction(gts->max_scale, scale, itime_new->mul, in iio_gts_find_new_gain_by_old_gain_time()
1083 return -EINVAL; in iio_gts_find_new_gain_by_old_gain_time()
1091 MODULE_DESCRIPTION("IIO light sensor gain-time-scale helpers");