From: Rodrigo Alencar <[email protected]>

Add parallel port channel with frequency scale, frequency offset, phase
offset, and amplitude offset extended attributes for configuring the
parallel data path.

Signed-off-by: Rodrigo Alencar <[email protected]>
---
 drivers/iio/frequency/ad9910.c | 152 +++++++++++++++++++++++++++++++++++++++++
 1 file changed, 152 insertions(+)

diff --git a/drivers/iio/frequency/ad9910.c b/drivers/iio/frequency/ad9910.c
index e9005037db1a..5b4076028a29 100644
--- a/drivers/iio/frequency/ad9910.c
+++ b/drivers/iio/frequency/ad9910.c
@@ -114,9 +114,13 @@
 /* Auxiliary DAC Control Register Bits */
 #define AD9910_AUX_DAC_FSC_MSK                 GENMASK(7, 0)
 
+/* POW Register Bits */
+#define AD9910_POW_PP_LSB_MSK                  GENMASK(7, 0)
+
 /* ASF Register Bits */
 #define AD9910_ASF_RAMP_RATE_MSK               GENMASK(31, 16)
 #define AD9910_ASF_SCALE_FACTOR_MSK            GENMASK(15, 2)
+#define AD9910_ASF_SCALE_FACTOR_PP_LSB_MSK     GENMASK(7, 2)
 #define AD9910_ASF_STEP_SIZE_MSK               GENMASK(1, 0)
 
 /* Multichip Sync Register Bits */
@@ -140,7 +144,9 @@
 #define AD9910_MAX_PHASE_MICRORAD      (AD9910_PI_NANORAD / 500)
 
 #define AD9910_ASF_MAX                 GENMASK(13, 0)
+#define AD9910_ASF_PP_LSB_MAX          GENMASK(5, 0)
 #define AD9910_POW_MAX                 GENMASK(15, 0)
+#define AD9910_POW_PP_LSB_MAX          GENMASK(7, 0)
 #define AD9910_NUM_PROFILES            8
 
 /* PLL constants */
@@ -191,6 +197,7 @@
  * @AD9910_CHANNEL_PROFILE_5: Profile 5 output channel
  * @AD9910_CHANNEL_PROFILE_6: Profile 6 output channel
  * @AD9910_CHANNEL_PROFILE_7: Profile 7 output channel
+ * @AD9910_CHANNEL_PARALLEL_PORT: Parallel port output channel
  */
 enum ad9910_channel {
        AD9910_CHANNEL_PHY = 100,
@@ -202,6 +209,7 @@ enum ad9910_channel {
        AD9910_CHANNEL_PROFILE_5 = 106,
        AD9910_CHANNEL_PROFILE_6 = 107,
        AD9910_CHANNEL_PROFILE_7 = 108,
+       AD9910_CHANNEL_PARALLEL_PORT = 110,
 };
 
 enum {
@@ -224,6 +232,10 @@ enum {
 
 enum {
        AD9910_POWERDOWN,
+       AD9910_PP_FREQ_SCALE,
+       AD9910_PP_FREQ_OFFSET,
+       AD9910_PP_PHASE_OFFSET,
+       AD9910_PP_AMP_OFFSET,
 };
 
 struct ad9910_data {
@@ -478,6 +490,10 @@ static ssize_t ad9910_ext_info_read(struct iio_dev 
*indio_dev,
                val = !!FIELD_GET(AD9910_CFR1_SOFT_POWER_DOWN_MSK,
                                  st->reg[AD9910_REG_CFR1].val32);
                break;
+       case AD9910_PP_FREQ_SCALE:
+               val = BIT(FIELD_GET(AD9910_CFR2_FM_GAIN_MSK,
+                                   st->reg[AD9910_REG_CFR2].val32));
+               break;
        default:
                return -EINVAL;
        }
@@ -508,6 +524,113 @@ static ssize_t ad9910_ext_info_write(struct iio_dev 
*indio_dev,
                                          AD9910_CFR1_SOFT_POWER_DOWN_MSK,
                                          val32, true);
                break;
+       case AD9910_PP_FREQ_SCALE:
+               if (val32 > BIT(15) || !is_power_of_2(val32))
+                       return -EINVAL;
+
+               val32 = FIELD_PREP(AD9910_CFR2_FM_GAIN_MSK, ilog2(val32));
+               ret = ad9910_reg32_update(st, AD9910_REG_CFR2,
+                                         AD9910_CFR2_FM_GAIN_MSK,
+                                         val32, true);
+               break;
+       default:
+               return -EINVAL;
+       }
+
+       return ret ?: len;
+}
+
+static ssize_t ad9910_pp_attrs_read(struct iio_dev *indio_dev,
+                                   uintptr_t private,
+                                   const struct iio_chan_spec *chan,
+                                   char *buf)
+{
+       struct ad9910_state *st = iio_priv(indio_dev);
+       int vals[2];
+       u32 tmp32;
+       u64 tmp64;
+
+       guard(mutex)(&st->lock);
+
+       switch (private) {
+       case AD9910_PP_FREQ_OFFSET:
+               tmp64 = (u64)st->reg[AD9910_REG_FTW].val32 * 
st->data.sysclk_freq_hz;
+               vals[0] = tmp64 >> 32;
+               vals[1] = ((tmp64 & GENMASK_ULL(31, 0)) * MICRO) >> 32;
+               break;
+       case AD9910_PP_PHASE_OFFSET:
+               tmp32 = FIELD_GET(AD9910_POW_PP_LSB_MSK,
+                                 st->reg[AD9910_REG_POW].val16);
+               tmp32 = (tmp32 * AD9910_MAX_PHASE_MICRORAD) >> 16;
+               vals[0] = tmp32 / MICRO;
+               vals[1] = tmp32 % MICRO;
+               break;
+       case AD9910_PP_AMP_OFFSET:
+               tmp32 = FIELD_GET(AD9910_ASF_SCALE_FACTOR_PP_LSB_MSK,
+                                 st->reg[AD9910_REG_ASF].val32);
+               vals[0] = 0;
+               vals[1] = (u64)tmp32 * MICRO >> 14;
+               break;
+       default:
+               return -EINVAL;
+       }
+
+       return iio_format_value(buf, IIO_VAL_INT_PLUS_MICRO, ARRAY_SIZE(vals), 
vals);
+}
+
+static ssize_t ad9910_pp_attrs_write(struct iio_dev *indio_dev,
+                                    uintptr_t private,
+                                    const struct iio_chan_spec *chan,
+                                    const char *buf, size_t len)
+{
+       struct ad9910_state *st = iio_priv(indio_dev);
+       int val, val2;
+       u32 tmp32;
+       int ret;
+
+       ret = iio_str_to_fixpoint(buf, MICRO / 10, &val, &val2);
+       if (ret)
+               return ret;
+
+       guard(mutex)(&st->lock);
+
+       switch (private) {
+       case AD9910_PP_FREQ_OFFSET:
+               if (!in_range(val, 0, st->data.sysclk_freq_hz / 2))
+                       return -EINVAL;
+
+               tmp32 = ad9910_rational_scale((u64)val * MICRO + val2, 
BIT_ULL(32),
+                                             (u64)MICRO * 
st->data.sysclk_freq_hz);
+               ret = ad9910_reg32_write(st, AD9910_REG_FTW, tmp32, true);
+               break;
+       case AD9910_PP_PHASE_OFFSET:
+               if (val)
+                       return -EINVAL;
+
+               if (!in_range(val2, 0, (AD9910_MAX_PHASE_MICRORAD >> 8)))
+                       return -EINVAL;
+
+               tmp32 = DIV_ROUND_CLOSEST((u32)val2 << 16, 
AD9910_MAX_PHASE_MICRORAD);
+               tmp32 = min(tmp32, AD9910_POW_PP_LSB_MAX);
+               tmp32 = FIELD_PREP(AD9910_POW_PP_LSB_MSK, tmp32);
+               ret = ad9910_reg16_update(st, AD9910_REG_POW,
+                                         AD9910_POW_PP_LSB_MSK,
+                                         tmp32, true);
+               break;
+       case AD9910_PP_AMP_OFFSET:
+               if (val)
+                       return -EINVAL;
+
+               if (!in_range(val2, 0, (MICRO >> 8)))
+                       return -EINVAL;
+
+               tmp32 = DIV_ROUND_CLOSEST((u32)val2 << 14, MICRO);
+               tmp32 = min(tmp32, AD9910_ASF_PP_LSB_MAX);
+               tmp32 = FIELD_PREP(AD9910_ASF_SCALE_FACTOR_PP_LSB_MSK, tmp32);
+               ret = ad9910_reg32_update(st, AD9910_REG_ASF,
+                                         AD9910_ASF_SCALE_FACTOR_PP_LSB_MSK,
+                                         tmp32, true);
+               break;
        default:
                return -EINVAL;
        }
@@ -526,11 +649,22 @@ static ssize_t ad9910_ext_info_write(struct iio_dev 
*indio_dev,
 #define AD9910_EXT_INFO(_name, _ident, _shared) \
        AD9910_EXT_INFO_TMPL(_name, _ident, _shared, ext_info)
 
+#define AD9910_PP_EXT_INFO(_name, _ident) \
+       AD9910_EXT_INFO_TMPL(_name, _ident, IIO_SEPARATE, pp_attrs)
+
 static const struct iio_chan_spec_ext_info ad9910_phy_ext_info[] = {
        AD9910_EXT_INFO("powerdown", AD9910_POWERDOWN, IIO_SEPARATE),
        { }
 };
 
+static const struct iio_chan_spec_ext_info ad9910_pp_ext_info[] = {
+       AD9910_EXT_INFO("frequency_scale", AD9910_PP_FREQ_SCALE, IIO_SEPARATE),
+       AD9910_PP_EXT_INFO("frequency_offset", AD9910_PP_FREQ_OFFSET),
+       AD9910_PP_EXT_INFO("phase_offset", AD9910_PP_PHASE_OFFSET),
+       AD9910_PP_EXT_INFO("scale_offset", AD9910_PP_AMP_OFFSET),
+       { }
+};
+
 #define AD9910_PROFILE_CHAN(idx) {                             \
        .type = IIO_ALTVOLTAGE,                                 \
        .indexed = 1,                                           \
@@ -563,6 +697,15 @@ static const struct iio_chan_spec ad9910_channels[] = {
        [AD9910_CHAN_IDX_PROFILE_5] = AD9910_PROFILE_CHAN(5),
        [AD9910_CHAN_IDX_PROFILE_6] = AD9910_PROFILE_CHAN(6),
        [AD9910_CHAN_IDX_PROFILE_7] = AD9910_PROFILE_CHAN(7),
+       [AD9910_CHAN_IDX_PARALLEL_PORT] = {
+               .type = IIO_ALTVOLTAGE,
+               .indexed = 1,
+               .output = 1,
+               .channel = AD9910_CHANNEL_PARALLEL_PORT,
+               .address = AD9910_CHAN_IDX_PARALLEL_PORT,
+               .info_mask_separate = BIT(IIO_CHAN_INFO_ENABLE),
+               .ext_info = ad9910_pp_ext_info,
+       },
 };
 
 static int ad9910_read_raw(struct iio_dev *indio_dev,
@@ -582,6 +725,10 @@ static int ad9910_read_raw(struct iio_dev *indio_dev,
                        tmp32 = (chan->channel - AD9910_CHANNEL_PROFILE_0);
                        *val = (tmp32 == st->profile);
                        break;
+               case AD9910_CHANNEL_PARALLEL_PORT:
+                       *val = FIELD_GET(AD9910_CFR2_PARALLEL_DATA_PORT_EN_MSK,
+                                        st->reg[AD9910_REG_CFR2].val32);
+                       break;
                default:
                        return -EINVAL;
                }
@@ -661,6 +808,11 @@ static int ad9910_write_raw(struct iio_dev *indio_dev,
                        }
 
                        return ad9910_profile_set(st, tmp32);
+               case AD9910_CHANNEL_PARALLEL_PORT:
+                       tmp32 = 
FIELD_PREP(AD9910_CFR2_PARALLEL_DATA_PORT_EN_MSK, !!val);
+                       return ad9910_reg32_update(st, AD9910_REG_CFR2,
+                                                  
AD9910_CFR2_PARALLEL_DATA_PORT_EN_MSK,
+                                                  tmp32, true);
                default:
                        return -EINVAL;
                }

-- 
2.43.0



Reply via email to