From e43a8a79ed93daab802f578b3f8ea9c50b52a367 Mon Sep 17 00:00:00 2001 From: Martin Braun Date: Thu, 10 Mar 2022 12:07:24 +0100 Subject: mpm: ad937x: Fix tuning code This fixes a known issue (see TODO) that the N300 and N310 have sub-Hz tuning inaccuracies. In fact, these are explained in the corresponding User Guide (UG-992). This commit modifies MPM to return the actual frequency when calling into ad937x_device::tune() or ad937x_device::get_freq(), instead of the nearest integer value. Co-authored-by: Arthur Moraes do Lago --- mpm/lib/mykonos/ad937x_device.cpp | 55 +++++++++++++++++++++++++++++++++++++-- 1 file changed, 53 insertions(+), 2 deletions(-) (limited to 'mpm/lib') diff --git a/mpm/lib/mykonos/ad937x_device.cpp b/mpm/lib/mykonos/ad937x_device.cpp index 1a1905004..c9e58d2b9 100644 --- a/mpm/lib/mykonos/ad937x_device.cpp +++ b/mpm/lib/mykonos/ad937x_device.cpp @@ -759,6 +759,57 @@ void ad937x_device::set_enable_gain_pins( /****************************************************** Get configuration functions ******************************************************/ + +/* Convert from the value returned by MYKONOS_getRfPllFrequency into the actual + * frequency. + * + * Note that MYKONOS_setRfPllFrequency() and MYKONOS_getRfPllFrequency() only + * deal in integers, and thus will end up with sub-Hz errors. + * + * Depending on the frequency range, ad9371 adopts a set denominator for the + * fractional N PLL: + * From 400MHz to 750MHz : divide by 16 + * From 750MHz to 1500MHz : divide by 8 + * From 1500MHz to 3000MHz : divide by 4 + * From 3000MHz to 6000MHz : divide by 2 + * This table comes straight from the user guide, from section + * RF PLL FREQUENCY CHANGE PROCEDURE. + * + * From the the table in section RF PLL RESOLUTION LIMITATIONS, + * we have some frequency steps depending on the master clock rate (aka device + * clock, or DEV_CLK). + * Dividing the master clock rate by the frequency steps, and combining + * with some experimental data analyzing the possible frequency steps, + * we arrive at the formula for frequency_step below + * + */ +double _get_actual_pll_freq(const double coerced_pll_freq, const double dev_clk) +{ + const unsigned denominator = [&]() { + // This first band is in the datasheet as from 400 MHz to 750 MHz, but + // we'll leave it open-ended. + if (coerced_pll_freq <= 750e6) { + return 16; + } else if (coerced_pll_freq <= 1500e6) { + return 8; + } else if (coerced_pll_freq <= 3000e6) { + return 4; + // This last band is in the datasheet as from 3000 MHz to 6000 MHz, but + // we'll leave it open-ended. + } else { + return 2; + } + }(); + // Note: This value is not listed in the data sheet, but can be derived by + // dividing the master clock rate by the frequency steps from Table 97, + // and verifying with experimental results (see comment above). + constexpr double pll_den_base = 33546240.0; + const double frequency_step = 2.0 * dev_clk / pll_den_base / denominator; + + return std::round(coerced_pll_freq / frequency_step) * frequency_step; +} + + double ad937x_device::get_freq(const direction_t direction) { mykonosRfPllName_t pll; @@ -773,10 +824,10 @@ double ad937x_device::get_freq(const direction_t direction) MPM_THROW_INVALID_CODE_PATH(); } - // TODO: because coerced_pll is returned as an integer, it's not accurate uint64_t coerced_pll; CALL_API(MYKONOS_getRfPllFrequency(mykonos_config.device, pll, &coerced_pll)); - return static_cast(coerced_pll); + return _get_actual_pll_freq(static_cast(coerced_pll), + mykonos_config.device->clocks->deviceClock_kHz * 1000.0); } bool ad937x_device::get_pll_lock_status(const uint8_t pll, const bool wait_for_lock) -- cgit v1.2.3