Lines Matching +full:calibration +full:- +full:data

1 // SPDX-License-Identifier: GPL-2.0-only
7 * YAS530 MS-3E (2011 Samsung Galaxy S Advance)
8 * YAS532 MS-3R (2011 Samsung Galaxy S4)
9 * YAS533 MS-3F (Vivo 1633, 1707, V3, Y21L)
11 * YAS535 MS-6C
12 * YAS536 MS-3W
13 * YAS537 MS-3T (2015 Samsung Galaxy S6, Note 5, Galaxy S7)
14 * YAS539 MS-3S (2018 Samsung Galaxy A7 SM-A750FN)
57 #define YAS530_OFFSET_X 0x85 /* [-31 .. 31] */
58 #define YAS530_OFFSET_Y1 0x86 /* [-31 .. 31] */
59 #define YAS530_OFFSET_Y2 0x87 /* [-31 .. 31] */
94 /* Bits in the measure data register */
97 #define YAS530_DEVICE_ID 0x01 /* YAS530 (MS-3E) */
98 #define YAS530_VERSION_A 0 /* YAS530 (MS-3E A) */
99 #define YAS530_VERSION_B 1 /* YAS530B (MS-3E B) */
103 #define YAS530_DATA_CENTER BIT(YAS530_DATA_BITS - 1)
104 #define YAS530_DATA_OVERFLOW (BIT(YAS530_DATA_BITS) - 1)
106 #define YAS532_DEVICE_ID 0x02 /* YAS532/YAS533 (MS-3R/F) */
107 #define YAS532_VERSION_AB 0 /* YAS532/533 AB (MS-3R/F AB) */
108 #define YAS532_VERSION_AC 1 /* YAS532/533 AC (MS-3R/F AC) */
114 #define YAS532_DATA_CENTER BIT(YAS532_DATA_BITS - 1)
115 #define YAS532_DATA_OVERFLOW (BIT(YAS532_DATA_BITS) - 1)
117 #define YAS537_DEVICE_ID 0x07 /* YAS537 (MS-3T) */
153 /* Linearization calibration x, y1, y2 */
156 /* Temperature compensation calibration */
158 /* Misc calibration coefficients */
169 * struct yas5xx_chip_info - device-specific data and function pointers
173 * @volatile_reg: device-specific volatile registers
174 * @volatile_reg_qty: quantity of device-specific volatile registers
179 * @get_calibration_data: function pointer to get calibration data
180 * @dump_calibration: function pointer to dump calibration for debugging
182 * @power_on: function pointer to power-on procedure
188 * given in the data sheets.
207 * struct yas5xx - state container for the YAS5xx driver
209 * @chip_info: device-specific data and function pointers
211 * @calibration: calibration settings from the OTP storage
219 * so that measurements get serialized in a first-come-first serve manner
226 struct yas5xx_calibration calibration; member
244 static u16 yas530_extract_axis(u8 *data) in yas530_extract_axis() argument
253 val = get_unaligned_be16(&data[0]); in yas530_extract_axis()
259 static u16 yas532_extract_axis(u8 *data) in yas532_extract_axis() argument
268 val = get_unaligned_be16(&data[0]); in yas532_extract_axis()
274 * yas530_measure() - Make a measure from the hardware
286 const struct yas5xx_chip_info *ci = yas5xx->chip_info; in yas530_measure()
288 u8 data[8]; in yas530_measure() local
292 mutex_lock(&yas5xx->lock); in yas530_measure()
293 ret = regmap_write(yas5xx->map, YAS530_MEASURE, YAS5XX_MEASURE_START); in yas530_measure()
302 ret = regmap_read_poll_timeout(yas5xx->map, YAS5XX_MEASURE_DATA, busy, in yas530_measure()
306 dev_err(yas5xx->dev, "timeout waiting for measurement\n"); in yas530_measure()
310 ret = regmap_bulk_read(yas5xx->map, YAS5XX_MEASURE_DATA, in yas530_measure()
311 data, sizeof(data)); in yas530_measure()
315 mutex_unlock(&yas5xx->lock); in yas530_measure()
317 switch (ci->devid) { in yas530_measure()
325 val = get_unaligned_be16(&data[0]); in yas530_measure()
328 *x = yas530_extract_axis(&data[2]); in yas530_measure()
329 *y1 = yas530_extract_axis(&data[4]); in yas530_measure()
330 *y2 = yas530_extract_axis(&data[6]); in yas530_measure()
339 val = get_unaligned_be16(&data[0]); in yas530_measure()
342 *x = yas532_extract_axis(&data[2]); in yas530_measure()
343 *y1 = yas532_extract_axis(&data[4]); in yas530_measure()
344 *y2 = yas532_extract_axis(&data[6]); in yas530_measure()
347 dev_err(yas5xx->dev, "unknown data format\n"); in yas530_measure()
348 ret = -EINVAL; in yas530_measure()
355 mutex_unlock(&yas5xx->lock); in yas530_measure()
360 * yas537_measure() - Make a measure from the hardware
370 struct yas5xx_calibration *c = &yas5xx->calibration; in yas537_measure()
372 u8 data[8]; in yas537_measure() local
377 mutex_lock(&yas5xx->lock); in yas537_measure()
380 ret = regmap_write(yas5xx->map, YAS537_MEASURE, YAS5XX_MEASURE_START | in yas537_measure()
385 /* Use same timeout like YAS530/532 but the bit is in data row 2 */ in yas537_measure()
386 ret = regmap_read_poll_timeout(yas5xx->map, YAS5XX_MEASURE_DATA + 2, busy, in yas537_measure()
390 dev_err(yas5xx->dev, "timeout waiting for measurement\n"); in yas537_measure()
394 ret = regmap_bulk_read(yas5xx->map, YAS5XX_MEASURE_DATA, in yas537_measure()
395 data, sizeof(data)); in yas537_measure()
399 mutex_unlock(&yas5xx->lock); in yas537_measure()
401 *t = get_unaligned_be16(&data[0]); in yas537_measure()
402 xy1y2[0] = FIELD_GET(GENMASK(13, 0), get_unaligned_be16(&data[2])); in yas537_measure()
403 xy1y2[1] = get_unaligned_be16(&data[4]); in yas537_measure()
404 xy1y2[2] = get_unaligned_be16(&data[6]); in yas537_measure()
406 /* The second version of YAS537 needs to include calibration coefficients */ in yas537_measure()
407 if (yas5xx->version == YAS537_VERSION_1) { in yas537_measure()
409 s[i] = xy1y2[i] - BIT(13); in yas537_measure()
410 h[0] = (c->k * (128 * s[0] + c->a2 * s[1] + c->a3 * s[2])) / BIT(13); in yas537_measure()
411 h[1] = (c->k * (c->a4 * s[0] + c->a5 * s[1] + c->a6 * s[2])) / BIT(13); in yas537_measure()
412 h[2] = (c->k * (c->a7 * s[0] + c->a8 * s[1] + c->a9 * s[2])) / BIT(13); in yas537_measure()
414 clamp_val(h[i], -BIT(13), BIT(13) - 1); in yas537_measure()
426 mutex_unlock(&yas5xx->lock); in yas537_measure()
433 const struct yas5xx_chip_info *ci = yas5xx->chip_info; in yas530_linearize()
434 struct yas5xx_calibration *c = &yas5xx->calibration; in yas530_linearize()
443 switch (ci->devid) { in yas530_linearize()
445 if (yas5xx->version == YAS530_VERSION_A) in yas530_linearize()
451 if (yas5xx->version == YAS532_VERSION_AB) in yas530_linearize()
458 dev_err(yas5xx->dev, "unknown device type\n"); in yas530_linearize()
464 * x' = x - (3721 + 50 * f) + (xoffset - r) * c in yas530_linearize()
466 * Where f and r are calibration values, c is a per-device in yas530_linearize()
467 * and sometimes per-axis coefficient. in yas530_linearize()
469 return val - (3721 + 50 * c->f[axis]) + in yas530_linearize()
470 (yas5xx->hard_offsets[axis] - c->r[axis]) * coef; in yas530_linearize()
475 const struct yas5xx_chip_info *ci = yas5xx->chip_info; in yas5xx_calc_temperature()
481 t_ref = ci->t_ref; in yas5xx_calc_temperature()
482 min_temp_x10 = ci->min_temp_x10; in yas5xx_calc_temperature()
485 to = (min_temp_x10 + ((ref_temp_x10 - min_temp_x10) * t / t_ref)) * 100; in yas5xx_calc_temperature()
490 * yas530_get_measure() - Measure a sample of all axis and process
502 const struct yas5xx_chip_info *ci = yas5xx->chip_info; in yas530_get_measure()
503 struct yas5xx_calibration *c = &yas5xx->calibration; in yas530_get_measure()
509 /* We first get raw data that needs to be translated to [x,y,z] */ in yas530_get_measure()
524 t_ref = ci->t_ref; in yas530_get_measure()
525 if (ci->devid == YAS532_DEVICE_ID && in yas530_get_measure()
526 yas5xx->version == YAS532_VERSION_AC) { in yas530_get_measure()
527 t_comp = t - t_ref; in yas530_get_measure()
536 * x' = x - ----------- in yas530_get_measure()
539 sx = sx - (c->Cx * t_comp) / 100; in yas530_get_measure()
540 sy1 = sy1 - (c->Cy1 * t_comp) / 100; in yas530_get_measure()
541 sy2 = sy2 - (c->Cy2 * t_comp) / 100; in yas530_get_measure()
547 sy = sy1 - sy2; in yas530_get_measure()
548 sz = -sy1 - sy2; in yas530_get_measure()
557 * x' = k * --------------------------- in yas530_get_measure()
561 * y' = k * --------------------------- in yas530_get_measure()
565 * z' = k * --------------------------- in yas530_get_measure()
568 *xo = c->k * ((100 * sx + c->a2 * sy + c->a3 * sz) / 10); in yas530_get_measure()
569 *yo = c->k * ((c->a4 * sx + c->a5 * sy + c->a6 * sz) / 10); in yas530_get_measure()
570 *zo = c->k * ((c->a7 * sx + c->a8 * sy + c->a9 * sz) / 10); in yas530_get_measure()
576 * yas537_get_measure() - Measure a sample of all axis and process
589 /* We first get raw data that needs to be translated to [x,y,z] */ in yas537_get_measure()
603 *xo = (x - BIT(13)) * 300; in yas537_get_measure()
604 *yo = (y1 - y2) * 1732 / 10; in yas537_get_measure()
605 *zo = (-y1 - y2 + BIT(14)) * 300; in yas537_get_measure()
616 const struct yas5xx_chip_info *ci = yas5xx->chip_info; in yas5xx_read_raw()
623 pm_runtime_get_sync(yas5xx->dev); in yas5xx_read_raw()
624 ret = ci->get_measure(yas5xx, &t, &x, &y, &z); in yas5xx_read_raw()
625 pm_runtime_mark_last_busy(yas5xx->dev); in yas5xx_read_raw()
626 pm_runtime_put_autosuspend(yas5xx->dev); in yas5xx_read_raw()
629 switch (chan->address) { in yas5xx_read_raw()
643 dev_err(yas5xx->dev, "unknown channel\n"); in yas5xx_read_raw()
644 return -EINVAL; in yas5xx_read_raw()
649 *val2 = ci->scaling_val2; in yas5xx_read_raw()
653 return -EINVAL; in yas5xx_read_raw()
660 const struct yas5xx_chip_info *ci = yas5xx->chip_info; in yas5xx_fill_buffer()
664 pm_runtime_get_sync(yas5xx->dev); in yas5xx_fill_buffer()
665 ret = ci->get_measure(yas5xx, &t, &x, &y, &z); in yas5xx_fill_buffer()
666 pm_runtime_mark_last_busy(yas5xx->dev); in yas5xx_fill_buffer()
667 pm_runtime_put_autosuspend(yas5xx->dev); in yas5xx_fill_buffer()
669 dev_err(yas5xx->dev, "error refilling buffer\n"); in yas5xx_fill_buffer()
672 yas5xx->scan.channels[0] = t; in yas5xx_fill_buffer()
673 yas5xx->scan.channels[1] = x; in yas5xx_fill_buffer()
674 yas5xx->scan.channels[2] = y; in yas5xx_fill_buffer()
675 yas5xx->scan.channels[3] = z; in yas5xx_fill_buffer()
676 iio_push_to_buffers_with_timestamp(indio_dev, &yas5xx->scan, in yas5xx_fill_buffer()
683 struct iio_dev *indio_dev = pf->indio_dev; in yas5xx_handle_trigger()
686 iio_trigger_notify_done(indio_dev->trig); in yas5xx_handle_trigger()
698 return &yas5xx->orientation; in yas5xx_get_mount_matrix()
753 const struct yas5xx_chip_info *ci = yas5xx->chip_info; in yas5xx_volatile_reg()
764 reg_qty = ci->volatile_reg_qty; in yas5xx_volatile_reg()
766 if (reg == ci->volatile_reg[i]) in yas5xx_volatile_reg()
782 * yas530_extract_calibration() - extracts the a2-a9 and k calibration
783 * @data: the bitfield to use
784 * @c: the calibration to populate
788 static void yas530_extract_calibration(u8 *data, struct yas5xx_calibration *c) in yas530_extract_calibration() argument
790 u64 val = get_unaligned_be64(data); in yas530_extract_calibration()
793 * Bitfield layout for the axis calibration data, for factor in yas530_extract_calibration()
806 c->a2 = FIELD_GET(GENMASK_ULL(63, 58), val) - 32; in yas530_extract_calibration()
807 c->a3 = FIELD_GET(GENMASK_ULL(57, 54), val) - 8; in yas530_extract_calibration()
808 c->a4 = FIELD_GET(GENMASK_ULL(53, 48), val) - 32; in yas530_extract_calibration()
809 c->a5 = FIELD_GET(GENMASK_ULL(47, 42), val) + 38; in yas530_extract_calibration()
810 c->a6 = FIELD_GET(GENMASK_ULL(41, 36), val) - 32; in yas530_extract_calibration()
811 c->a7 = FIELD_GET(GENMASK_ULL(35, 29), val) - 64; in yas530_extract_calibration()
812 c->a8 = FIELD_GET(GENMASK_ULL(28, 23), val) - 32; in yas530_extract_calibration()
813 c->a9 = FIELD_GET(GENMASK_ULL(22, 15), val); in yas530_extract_calibration()
814 c->k = FIELD_GET(GENMASK_ULL(14, 10), val) + 10; in yas530_extract_calibration()
815 c->dck = FIELD_GET(GENMASK_ULL(9, 7), val); in yas530_extract_calibration()
820 struct yas5xx_calibration *c = &yas5xx->calibration; in yas530_get_calibration_data()
821 u8 data[16]; in yas530_get_calibration_data() local
826 ret = regmap_bulk_read(yas5xx->map, YAS530_CAL, data, sizeof(data)); in yas530_get_calibration_data()
830 /* Actual calibration readout */ in yas530_get_calibration_data()
831 ret = regmap_bulk_read(yas5xx->map, YAS530_CAL, data, sizeof(data)); in yas530_get_calibration_data()
834 dev_dbg(yas5xx->dev, "calibration data: %16ph\n", data); in yas530_get_calibration_data()
836 /* Contribute calibration data to the input pool for kernel entropy */ in yas530_get_calibration_data()
837 add_device_randomness(data, sizeof(data)); in yas530_get_calibration_data()
840 yas5xx->version = data[15] & GENMASK(1, 0); in yas530_get_calibration_data()
842 /* Extract the calibration from the bitfield */ in yas530_get_calibration_data()
843 c->Cx = data[0] * 6 - 768; in yas530_get_calibration_data()
844 c->Cy1 = data[1] * 6 - 768; in yas530_get_calibration_data()
845 c->Cy2 = data[2] * 6 - 768; in yas530_get_calibration_data()
846 yas530_extract_calibration(&data[3], c); in yas530_get_calibration_data()
859 val = get_unaligned_be32(&data[11]); in yas530_get_calibration_data()
860 c->f[0] = FIELD_GET(GENMASK(22, 21), val); in yas530_get_calibration_data()
861 c->f[1] = FIELD_GET(GENMASK(14, 13), val); in yas530_get_calibration_data()
862 c->f[2] = FIELD_GET(GENMASK(6, 5), val); in yas530_get_calibration_data()
863 c->r[0] = sign_extend32(FIELD_GET(GENMASK(28, 23), val), 5); in yas530_get_calibration_data()
864 c->r[1] = sign_extend32(FIELD_GET(GENMASK(20, 15), val), 5); in yas530_get_calibration_data()
865 c->r[2] = sign_extend32(FIELD_GET(GENMASK(12, 7), val), 5); in yas530_get_calibration_data()
872 struct yas5xx_calibration *c = &yas5xx->calibration; in yas532_get_calibration_data()
873 u8 data[14]; in yas532_get_calibration_data() local
878 ret = regmap_bulk_read(yas5xx->map, YAS530_CAL, data, sizeof(data)); in yas532_get_calibration_data()
881 /* Actual calibration readout */ in yas532_get_calibration_data()
882 ret = regmap_bulk_read(yas5xx->map, YAS530_CAL, data, sizeof(data)); in yas532_get_calibration_data()
885 dev_dbg(yas5xx->dev, "calibration data: %14ph\n", data); in yas532_get_calibration_data()
888 if (!memchr_inv(data, 0x00, 13) && !(data[13] & BIT(7))) in yas532_get_calibration_data()
889 dev_warn(yas5xx->dev, "calibration is blank!\n"); in yas532_get_calibration_data()
891 /* Contribute calibration data to the input pool for kernel entropy */ in yas532_get_calibration_data()
892 add_device_randomness(data, sizeof(data)); in yas532_get_calibration_data()
895 yas5xx->version = data[13] & BIT(0); in yas532_get_calibration_data()
897 /* Extract calibration from the bitfield */ in yas532_get_calibration_data()
898 c->Cx = data[0] * 10 - 1280; in yas532_get_calibration_data()
899 c->Cy1 = data[1] * 10 - 1280; in yas532_get_calibration_data()
900 c->Cy2 = data[2] * 10 - 1280; in yas532_get_calibration_data()
901 yas530_extract_calibration(&data[3], c); in yas532_get_calibration_data()
914 val = get_unaligned_be32(&data[10]); in yas532_get_calibration_data()
915 c->f[0] = FIELD_GET(GENMASK(24, 23), val); in yas532_get_calibration_data()
916 c->f[1] = FIELD_GET(GENMASK(16, 15), val); in yas532_get_calibration_data()
917 c->f[2] = FIELD_GET(GENMASK(8, 7), val); in yas532_get_calibration_data()
918 c->r[0] = sign_extend32(FIELD_GET(GENMASK(30, 25), val), 5); in yas532_get_calibration_data()
919 c->r[1] = sign_extend32(FIELD_GET(GENMASK(22, 17), val), 5); in yas532_get_calibration_data()
920 c->r[2] = sign_extend32(FIELD_GET(GENMASK(14, 7), val), 5); in yas532_get_calibration_data()
927 struct yas5xx_calibration *c = &yas5xx->calibration; in yas537_get_calibration_data()
928 u8 data[17]; in yas537_get_calibration_data() local
933 ret = regmap_write(yas5xx->map, YAS537_SRST, BIT(1)); in yas537_get_calibration_data()
937 /* Calibration readout, YAS537 needs one readout only */ in yas537_get_calibration_data()
938 ret = regmap_bulk_read(yas5xx->map, YAS537_CAL, data, sizeof(data)); in yas537_get_calibration_data()
941 dev_dbg(yas5xx->dev, "calibration data: %17ph\n", data); in yas537_get_calibration_data()
944 if (!memchr_inv(data, 0x00, 16) && !FIELD_GET(GENMASK(5, 0), data[16])) in yas537_get_calibration_data()
945 dev_warn(yas5xx->dev, "calibration is blank!\n"); in yas537_get_calibration_data()
947 /* Contribute calibration data to the input pool for kernel entropy */ in yas537_get_calibration_data()
948 add_device_randomness(data, sizeof(data)); in yas537_get_calibration_data()
951 yas5xx->version = FIELD_GET(GENMASK(7, 6), data[16]); in yas537_get_calibration_data()
954 switch (yas5xx->version) { in yas537_get_calibration_data()
957 * The first version simply writes data back into registers: in yas537_get_calibration_data()
959 * data[0] YAS537_MTC 0x93 in yas537_get_calibration_data()
960 * data[1] 0x94 in yas537_get_calibration_data()
961 * data[2] 0x95 in yas537_get_calibration_data()
962 * data[3] 0x96 in yas537_get_calibration_data()
963 * data[4] 0x97 in yas537_get_calibration_data()
964 * data[5] 0x98 in yas537_get_calibration_data()
965 * data[6] 0x99 in yas537_get_calibration_data()
966 * data[7] 0x9a in yas537_get_calibration_data()
967 * data[8] 0x9b in yas537_get_calibration_data()
968 * data[9] 0x9c in yas537_get_calibration_data()
969 * data[10] 0x9d in yas537_get_calibration_data()
970 * data[11] YAS537_OC 0x9e in yas537_get_calibration_data()
972 * data[12] YAS537_OFFSET_X 0x84 in yas537_get_calibration_data()
973 * data[13] YAS537_OFFSET_Y1 0x85 in yas537_get_calibration_data()
974 * data[14] YAS537_OFFSET_Y2 0x86 in yas537_get_calibration_data()
976 * data[15] YAS537_HCK 0x88 in yas537_get_calibration_data()
977 * data[16] YAS537_LCK 0x89 in yas537_get_calibration_data()
980 ret = regmap_write(yas5xx->map, YAS537_MTC + i, in yas537_get_calibration_data()
981 data[i]); in yas537_get_calibration_data()
986 ret = regmap_write(yas5xx->map, YAS537_OFFSET_X + i, in yas537_get_calibration_data()
987 data[i + 12]); in yas537_get_calibration_data()
990 yas5xx->hard_offsets[i] = data[i + 12]; in yas537_get_calibration_data()
993 ret = regmap_write(yas5xx->map, YAS537_HCK + i, in yas537_get_calibration_data()
994 data[i + 15]); in yas537_get_calibration_data()
1001 * The second version writes some data into registers but also in yas537_get_calibration_data()
1002 * extracts calibration coefficients. in yas537_get_calibration_data()
1006 * data[0] YAS537_MTC 0x93 in yas537_get_calibration_data()
1007 * data[1] YAS537_MTC+1 0x94 in yas537_get_calibration_data()
1008 * data[2] YAS537_MTC+2 0x95 in yas537_get_calibration_data()
1009 * data[3] YAS537_MTC+3 (partially) 0x96 in yas537_get_calibration_data()
1011 * data[12] YAS537_OFFSET_X 0x84 in yas537_get_calibration_data()
1012 * data[13] YAS537_OFFSET_Y1 0x85 in yas537_get_calibration_data()
1013 * data[14] YAS537_OFFSET_Y2 0x86 in yas537_get_calibration_data()
1015 * data[15] YAS537_HCK (partially) 0x88 in yas537_get_calibration_data()
1017 * data[16] YAS537_OC (partially) 0x9e in yas537_get_calibration_data()
1020 ret = regmap_write(yas5xx->map, YAS537_MTC + i, in yas537_get_calibration_data()
1021 data[i]); in yas537_get_calibration_data()
1026 ret = regmap_write(yas5xx->map, YAS537_OFFSET_X + i, in yas537_get_calibration_data()
1027 data[i + 12]); in yas537_get_calibration_data()
1030 yas5xx->hard_offsets[i] = data[i + 12]; in yas537_get_calibration_data()
1033 * Visualization of partially taken data: in yas537_get_calibration_data()
1035 * data[3] n 7 6 5 4 3 2 1 0 in yas537_get_calibration_data()
1038 * data[15] n 7 6 5 4 3 2 1 0 in yas537_get_calibration_data()
1041 * data[15] n 7 6 5 4 3 2 1 0 in yas537_get_calibration_data()
1044 * data[16] n 7 6 5 4 3 2 1 0 in yas537_get_calibration_data()
1047 ret = regmap_write(yas5xx->map, YAS537_MTC + 3, in yas537_get_calibration_data()
1049 FIELD_GET(YAS537_MTC3_MASK_GET, data[3])) | in yas537_get_calibration_data()
1053 ret = regmap_write(yas5xx->map, YAS537_HCK, in yas537_get_calibration_data()
1055 FIELD_GET(YAS537_HCK_MASK_GET, data[15]))); in yas537_get_calibration_data()
1058 ret = regmap_write(yas5xx->map, YAS537_LCK, in yas537_get_calibration_data()
1060 FIELD_GET(YAS537_LCK_MASK_GET, data[15]))); in yas537_get_calibration_data()
1063 ret = regmap_write(yas5xx->map, YAS537_OC, in yas537_get_calibration_data()
1064 FIELD_GET(YAS537_OC_MASK_GET, data[16])); in yas537_get_calibration_data()
1068 * For data extraction, build some blocks. Four 32-bit blocks in yas537_get_calibration_data()
1072 * data[0] 0 [ Cx Cx Cx Cx Cx Cx Cx Cx ] bits 31 .. 24 in yas537_get_calibration_data()
1073 * data[1] 1 [ Cx C1 C1 C1 C1 C1 C1 C1 ] bits 23 .. 16 in yas537_get_calibration_data()
1074 * data[2] 2 [ C1 C1 C2 C2 C2 C2 C2 C2 ] bits 15 .. 8 in yas537_get_calibration_data()
1075 * data[3] 3 [ C2 C2 C2 ] bits 7 .. 0 in yas537_get_calibration_data()
1078 * data[3] 0 [ a2 a2 a2 a2 a2 ] bits 31 .. 24 in yas537_get_calibration_data()
1079 * data[4] 1 [ a2 a2 a3 a3 a3 a3 a3 a3 ] bits 23 .. 16 in yas537_get_calibration_data()
1080 * data[5] 2 [ a3 a4 a4 a4 a4 a4 a4 a4 ] bits 15 .. 8 in yas537_get_calibration_data()
1081 * data[6] 3 [ a4 ] bits 7 .. 0 in yas537_get_calibration_data()
1084 * data[6] 0 [ a5 a5 a5 a5 a5 a5 a5 ] bits 31 .. 24 in yas537_get_calibration_data()
1085 * data[7] 1 [ a5 a5 a6 a6 a6 a6 a6 a6 ] bits 23 .. 16 in yas537_get_calibration_data()
1086 * data[8] 2 [ a6 a7 a7 a7 a7 a7 a7 a7 ] bits 15 .. 8 in yas537_get_calibration_data()
1087 * data[9] 3 [ a7 ] bits 7 .. 0 in yas537_get_calibration_data()
1090 * data[9] 0 [ a8 a8 a8 a8 a8 a8 a8 ] bits 31 .. 24 in yas537_get_calibration_data()
1091 * data[10] 1 [ a9 a9 a9 a9 a9 a9 a9 a9 ] bits 23 .. 16 in yas537_get_calibration_data()
1092 * data[11] 2 [ a9 k k k k k k k ] bits 15 .. 8 in yas537_get_calibration_data()
1093 * data[12] 3 [ ] bits 7 .. 0 in yas537_get_calibration_data()
1095 val1 = get_unaligned_be32(&data[0]); in yas537_get_calibration_data()
1096 val2 = get_unaligned_be32(&data[3]); in yas537_get_calibration_data()
1097 val3 = get_unaligned_be32(&data[6]); in yas537_get_calibration_data()
1098 val4 = get_unaligned_be32(&data[9]); in yas537_get_calibration_data()
1099 /* Extract calibration coefficients and modify */ in yas537_get_calibration_data()
1100 c->Cx = FIELD_GET(GENMASK(31, 23), val1) - 256; in yas537_get_calibration_data()
1101 c->Cy1 = FIELD_GET(GENMASK(22, 14), val1) - 256; in yas537_get_calibration_data()
1102 c->Cy2 = FIELD_GET(GENMASK(13, 5), val1) - 256; in yas537_get_calibration_data()
1103 c->a2 = FIELD_GET(GENMASK(28, 22), val2) - 64; in yas537_get_calibration_data()
1104 c->a3 = FIELD_GET(GENMASK(21, 15), val2) - 64; in yas537_get_calibration_data()
1105 c->a4 = FIELD_GET(GENMASK(14, 7), val2) - 128; in yas537_get_calibration_data()
1106 c->a5 = FIELD_GET(GENMASK(30, 22), val3) - 112; in yas537_get_calibration_data()
1107 c->a6 = FIELD_GET(GENMASK(21, 15), val3) - 64; in yas537_get_calibration_data()
1108 c->a7 = FIELD_GET(GENMASK(14, 7), val3) - 128; in yas537_get_calibration_data()
1109 c->a8 = FIELD_GET(GENMASK(30, 24), val4) - 64; in yas537_get_calibration_data()
1110 c->a9 = FIELD_GET(GENMASK(23, 15), val4) - 112; in yas537_get_calibration_data()
1111 c->k = FIELD_GET(GENMASK(14, 8), val4); in yas537_get_calibration_data()
1114 dev_err(yas5xx->dev, "unknown version of YAS537\n"); in yas537_get_calibration_data()
1115 return -EINVAL; in yas537_get_calibration_data()
1124 struct yas5xx_calibration *c = &yas5xx->calibration; in yas530_dump_calibration()
1126 dev_dbg(yas5xx->dev, "f[] = [%d, %d, %d]\n", in yas530_dump_calibration()
1127 c->f[0], c->f[1], c->f[2]); in yas530_dump_calibration()
1128 dev_dbg(yas5xx->dev, "r[] = [%d, %d, %d]\n", in yas530_dump_calibration()
1129 c->r[0], c->r[1], c->r[2]); in yas530_dump_calibration()
1130 dev_dbg(yas5xx->dev, "Cx = %d\n", c->Cx); in yas530_dump_calibration()
1131 dev_dbg(yas5xx->dev, "Cy1 = %d\n", c->Cy1); in yas530_dump_calibration()
1132 dev_dbg(yas5xx->dev, "Cy2 = %d\n", c->Cy2); in yas530_dump_calibration()
1133 dev_dbg(yas5xx->dev, "a2 = %d\n", c->a2); in yas530_dump_calibration()
1134 dev_dbg(yas5xx->dev, "a3 = %d\n", c->a3); in yas530_dump_calibration()
1135 dev_dbg(yas5xx->dev, "a4 = %d\n", c->a4); in yas530_dump_calibration()
1136 dev_dbg(yas5xx->dev, "a5 = %d\n", c->a5); in yas530_dump_calibration()
1137 dev_dbg(yas5xx->dev, "a6 = %d\n", c->a6); in yas530_dump_calibration()
1138 dev_dbg(yas5xx->dev, "a7 = %d\n", c->a7); in yas530_dump_calibration()
1139 dev_dbg(yas5xx->dev, "a8 = %d\n", c->a8); in yas530_dump_calibration()
1140 dev_dbg(yas5xx->dev, "a9 = %d\n", c->a9); in yas530_dump_calibration()
1141 dev_dbg(yas5xx->dev, "k = %d\n", c->k); in yas530_dump_calibration()
1142 dev_dbg(yas5xx->dev, "dck = %d\n", c->dck); in yas530_dump_calibration()
1147 struct yas5xx_calibration *c = &yas5xx->calibration; in yas537_dump_calibration()
1149 if (yas5xx->version == YAS537_VERSION_1) { in yas537_dump_calibration()
1150 dev_dbg(yas5xx->dev, "Cx = %d\n", c->Cx); in yas537_dump_calibration()
1151 dev_dbg(yas5xx->dev, "Cy1 = %d\n", c->Cy1); in yas537_dump_calibration()
1152 dev_dbg(yas5xx->dev, "Cy2 = %d\n", c->Cy2); in yas537_dump_calibration()
1153 dev_dbg(yas5xx->dev, "a2 = %d\n", c->a2); in yas537_dump_calibration()
1154 dev_dbg(yas5xx->dev, "a3 = %d\n", c->a3); in yas537_dump_calibration()
1155 dev_dbg(yas5xx->dev, "a4 = %d\n", c->a4); in yas537_dump_calibration()
1156 dev_dbg(yas5xx->dev, "a5 = %d\n", c->a5); in yas537_dump_calibration()
1157 dev_dbg(yas5xx->dev, "a6 = %d\n", c->a6); in yas537_dump_calibration()
1158 dev_dbg(yas5xx->dev, "a7 = %d\n", c->a7); in yas537_dump_calibration()
1159 dev_dbg(yas5xx->dev, "a8 = %d\n", c->a8); in yas537_dump_calibration()
1160 dev_dbg(yas5xx->dev, "a9 = %d\n", c->a9); in yas537_dump_calibration()
1161 dev_dbg(yas5xx->dev, "k = %d\n", c->k); in yas537_dump_calibration()
1170 ret = regmap_write(yas5xx->map, YAS530_OFFSET_X, ox); in yas530_set_offsets()
1173 ret = regmap_write(yas5xx->map, YAS530_OFFSET_Y1, oy1); in yas530_set_offsets()
1176 return regmap_write(yas5xx->map, YAS530_OFFSET_Y2, oy2); in yas530_set_offsets()
1185 return old - BIT(bit); in yas530_adjust_offset()
1192 const struct yas5xx_chip_info *ci = yas5xx->chip_info; in yas530_measure_offsets()
1200 ret = regmap_write(yas5xx->map, YAS530_ACTUATE_INIT_COIL, 0); in yas530_measure_offsets()
1205 switch (ci->devid) { in yas530_measure_offsets()
1213 dev_err(yas5xx->dev, "unknown device type\n"); in yas530_measure_offsets()
1214 return -EINVAL; in yas530_measure_offsets()
1218 * We set offsets in the interval +-31 by iterating in yas530_measure_offsets()
1219 * +-16, +-8, +-4, +-2, +-1 adjusting the offsets each in yas530_measure_offsets()
1224 * as the values for [x, y1, y2]. The value is +/-31 in yas530_measure_offsets()
1233 for (i = 4; i >= 0; i--) { in yas530_measure_offsets()
1241 dev_dbg(yas5xx->dev, "measurement %d: x=%d, y1=%d, y2=%d\n", in yas530_measure_offsets()
1242 5-i, x, y1, y2); in yas530_measure_offsets()
1249 /* Needed for calibration algorithm */ in yas530_measure_offsets()
1250 yas5xx->hard_offsets[0] = ox; in yas530_measure_offsets()
1251 yas5xx->hard_offsets[1] = oy1; in yas530_measure_offsets()
1252 yas5xx->hard_offsets[2] = oy2; in yas530_measure_offsets()
1257 dev_info(yas5xx->dev, "discovered hard offsets: x=%d, y1=%d, y2=%d\n", in yas530_measure_offsets()
1269 ret = regmap_write(yas5xx->map, YAS530_TEST1, 0); in yas530_power_on()
1272 ret = regmap_write(yas5xx->map, YAS530_TEST2, 0); in yas530_power_on()
1277 val = FIELD_PREP(YAS5XX_CONFIG_CCK_MASK, yas5xx->calibration.dck); in yas530_power_on()
1278 ret = regmap_write(yas5xx->map, YAS530_CONFIG, val); in yas530_power_on()
1282 /* Measure interval 0 (back-to-back?) */ in yas530_power_on()
1283 return regmap_write(yas5xx->map, YAS530_MEASURE_INTERVAL, 0); in yas530_power_on()
1294 ret = regmap_bulk_write(yas5xx->map, YAS537_ADCCAL, &buf, sizeof(buf)); in yas537_power_on()
1297 ret = regmap_write(yas5xx->map, YAS537_TRM, GENMASK(7, 0)); in yas537_power_on()
1303 - YAS537_MEASURE_TIME_WORST_US) / 4100; in yas537_power_on()
1304 ret = regmap_write(yas5xx->map, YAS537_MEASURE_INTERVAL, intrvl); in yas537_power_on()
1309 ret = regmap_write(yas5xx->map, YAS537_AVR, YAS537_MAG_AVERAGE_32_MASK); in yas537_power_on()
1314 ret = regmap_write(yas5xx->map, YAS537_CONFIG, BIT(3)); in yas537_power_on()
1327 .product_name = "YAS530 MS-3E",
1333 .min_temp_x10 = -620, /* 1/10:s degrees Celsius */
1342 .product_name = "YAS532 MS-3R",
1348 .min_temp_x10 = -500, /* 1/10:s degrees Celsius */
1357 .product_name = "YAS533 MS-3F",
1363 .min_temp_x10 = -500, /* 1/10:s degrees Celsius */
1372 .product_name = "YAS537 MS-3T",
1378 .min_temp_x10 = -3860, /* 1/10:s degrees Celsius */
1391 struct device *dev = &i2c->dev; in yas5xx_probe()
1399 return -ENOMEM; in yas5xx_probe()
1403 yas5xx->dev = dev; in yas5xx_probe()
1404 mutex_init(&yas5xx->lock); in yas5xx_probe()
1406 ret = iio_read_mount_matrix(dev, &yas5xx->orientation); in yas5xx_probe()
1410 yas5xx->regs[0].supply = "vdd"; in yas5xx_probe()
1411 yas5xx->regs[1].supply = "iovdd"; in yas5xx_probe()
1412 ret = devm_regulator_bulk_get(dev, ARRAY_SIZE(yas5xx->regs), in yas5xx_probe()
1413 yas5xx->regs); in yas5xx_probe()
1417 ret = regulator_bulk_enable(ARRAY_SIZE(yas5xx->regs), yas5xx->regs); in yas5xx_probe()
1425 yas5xx->reset = devm_gpiod_get_optional(dev, "reset", GPIOD_OUT_LOW); in yas5xx_probe()
1426 if (IS_ERR(yas5xx->reset)) { in yas5xx_probe()
1427 ret = dev_err_probe(dev, PTR_ERR(yas5xx->reset), "failed to get reset line\n"); in yas5xx_probe()
1431 yas5xx->map = devm_regmap_init_i2c(i2c, &yas5xx_regmap_config); in yas5xx_probe()
1432 if (IS_ERR(yas5xx->map)) { in yas5xx_probe()
1433 ret = dev_err_probe(dev, PTR_ERR(yas5xx->map), "failed to allocate register map\n"); in yas5xx_probe()
1439 ci = (const struct yas5xx_chip_info *)id->driver_data; in yas5xx_probe()
1440 yas5xx->chip_info = ci; in yas5xx_probe()
1442 ret = regmap_read(yas5xx->map, YAS5XX_DEVICE_ID, &id_check); in yas5xx_probe()
1446 if (id_check != ci->devid) { in yas5xx_probe()
1447 ret = dev_err_probe(dev, -ENODEV, in yas5xx_probe()
1449 id_check, id->name); in yas5xx_probe()
1453 ret = ci->get_calibration_data(yas5xx); in yas5xx_probe()
1457 dev_info(dev, "detected %s %s\n", ci->product_name, in yas5xx_probe()
1458 ci->version_names[yas5xx->version]); in yas5xx_probe()
1460 ci->dump_calibration(yas5xx); in yas5xx_probe()
1462 ret = ci->power_on(yas5xx); in yas5xx_probe()
1466 if (ci->measure_offsets) { in yas5xx_probe()
1467 ret = ci->measure_offsets(yas5xx); in yas5xx_probe()
1472 indio_dev->info = &yas5xx_info; in yas5xx_probe()
1473 indio_dev->available_scan_masks = yas5xx_scan_masks; in yas5xx_probe()
1474 indio_dev->modes = INDIO_DIRECT_MODE; in yas5xx_probe()
1475 indio_dev->name = id->name; in yas5xx_probe()
1476 indio_dev->channels = yas5xx_channels; in yas5xx_probe()
1477 indio_dev->num_channels = ARRAY_SIZE(yas5xx_channels); in yas5xx_probe()
1507 gpiod_set_value_cansleep(yas5xx->reset, 1); in yas5xx_probe()
1509 regulator_bulk_disable(ARRAY_SIZE(yas5xx->regs), yas5xx->regs); in yas5xx_probe()
1518 struct device *dev = &i2c->dev; in yas5xx_remove()
1530 gpiod_set_value_cansleep(yas5xx->reset, 1); in yas5xx_remove()
1531 regulator_bulk_disable(ARRAY_SIZE(yas5xx->regs), yas5xx->regs); in yas5xx_remove()
1539 gpiod_set_value_cansleep(yas5xx->reset, 1); in yas5xx_runtime_suspend()
1540 regulator_bulk_disable(ARRAY_SIZE(yas5xx->regs), yas5xx->regs); in yas5xx_runtime_suspend()
1549 const struct yas5xx_chip_info *ci = yas5xx->chip_info; in yas5xx_runtime_resume()
1552 ret = regulator_bulk_enable(ARRAY_SIZE(yas5xx->regs), yas5xx->regs); in yas5xx_runtime_resume()
1564 gpiod_set_value_cansleep(yas5xx->reset, 0); in yas5xx_runtime_resume()
1566 ret = ci->power_on(yas5xx); in yas5xx_runtime_resume()
1575 gpiod_set_value_cansleep(yas5xx->reset, 1); in yas5xx_runtime_resume()
1576 regulator_bulk_disable(ARRAY_SIZE(yas5xx->regs), yas5xx->regs); in yas5xx_runtime_resume()
1614 MODULE_DESCRIPTION("Yamaha YAS53x 3-axis magnetometer driver");