Seeking more information on NRF52833 ADC, reference, and buffers

Hi,

We are developing a product which is reading RTDs and thermistors from the NRF52833's internal ADC. We're working on compensating for the intrinsic errors of the ADC and gain buffers and have some questions not covered in the product specifications.

Do the ADC gain buffers have offsets associated with them? The datasheet says the ADC has an intrinsic offset of ±2LSB @ 10bit res but doesn't mention whether there are voltage offsets associated with the buffers.

Is there any procedure to internally determine the ADC buffer gains? For example, connecting the buffer inputs directly to the reference voltage?

Can you provide any more detail on what the ADC's internal calibration routine does or what it accounts for?

Are there any details on how the ADC input buffer gain is implemented?

Are there any specs on channel isolation in sequential measurement mode?

Is there a spec for temperature drift of the VDDH/5 divider?

Thank you!

Parents
  • Sorry, another question:

    In Product Spec v1.7, section 6.21.2.3 (SAADC Scan mode) it says "The time it takes to sample all channels is less than the sum of the conversion time of all enabled channels. The conversion time for a channel is defined as the sum of the acquisition time tACQ and the conversion time tCONV."

    Why is the total conversion time less than the sum of the channels? What is skipped? What is the expected conversion time?

    Thanks

  • Do the ADC gain buffers have offsets associated with them? The datasheet says the ADC has an intrinsic offset of ±2LSB @ 10bit res but doesn't mention whether there are voltage offsets associated with the buffers.

    1) The input buffer of the ADC has auto-zero for offset. The offset of the ADC core should be scaled with the gain. 

    Is there any procedure to internally determine the ADC buffer gains? For example, connecting the buffer inputs directly to the reference voltage?

    2) Only for VDD as reference. There is an undocumented TASKS_CALIBRATEGAIN that connects the input to VDD/4 and uses gain 1/4

    Can you provide any more detail on what the ADC's internal calibration routine does or what it accounts for?

    3) The offset shorts the ADC input, and then a digital feedback loop tunes a current DAC in the SAR comparator to compensate for the offset. The offset may introduce a noise component if the TASKS_CALIBRATEOFFSET is triggered often

    Are there any details on how the ADC input buffer gain is implemented?

    4) Sorry, no details, but it's a switched capacitor gain stage with high common mode rejection. 

    Are there any specs on channel isolation in sequential measurement mode?

    5) There is only one core ADC, the muxing is done through T-gates (transmission gate, pull down, transmission gate), so there should be very little cross coupling between channels. 

    Is there a spec for temperature drift of the VDDH/5 divider?

    6) No, but it's a resistive divider, as such, as long as the ADC input settles, should not change with temperature (although the resistors do change with temperature, the relative size does not). The ADC temperature coefficient will dominate.

    jdub said:
    Also - any details on the accuracy of the internal VDD/4 divider which can be used as an ADC reference? We are also finding that when using the VDD/4 reference, our signals are much noisier. Any specs on that noise or advice for reducing it?

    7)  The noise that you're seeing is probably coming from VDD. There is a low pass filter inside the ADC, but that is in the MHz range. 

    jdub said:
    Is the VDDH/5 input only enabled when REG0 is on? I'm working on a device where VDDH and VDD are tied together. The 1% spec on the VDDH/5 divider is better than the 3% spec on the ADC's input gain buffers, so I have been trying to use the VDDH5 input. But it behaves like it's floating (the reading seems arbitrary and the value decreases with sample rate, implying a loading effect)

    8) That does not matter. VDDH/5 divider goes to the input gain buffers. I'm not sure it's enabled without the 5V regulator is enabled.

    jdub said:
    Does the ADC sequence converter fail if you use the CH[X}.CONFIG register's RESP and RESN settings to set inputs at VDD/2 while setting CH[X}.PSEL and NSEL to Not Connected? I can take a valid measurement in this configuration if I sample as a one-off, but if I make this measurement part of a sequence the ADC never finishes.

    9) You need to enable the PSEL/NSEL, but not connect them to anything. Try writing 0xFE to both.

    jdub said:

    In Product Spec v1.7, section 6.21.2.3 (SAADC Scan mode) it says "The time it takes to sample all channels is less than the sum of the conversion time of all enabled channels. The conversion time for a channel is defined as the sum of the acquisition time tACQ and the conversion time tCONV."

    Why is the total conversion time less than the sum of the channels? What is skipped? What is the expected conversion time?

    10) Not sure why it's written like this as it's very confusing. We expect the conversion time to be N x (TACQ + t_conv).

  • Good to hear from you, hopefully you got a good break.

    "The resistance offset is consistent over time, and it varies from unit to unit" is almost certainly a calibration issue, more about this below.

    "the RTD has one leg tied to GND, so it must be measured single-ended. I don't expect this to be significant" This is a huge issue, given that reversing the current flow allowing automatic residual offset compensation is not available. This means that SAADC calibration is essential, and the calibration must be performed at the same gain and sampling values as each measurement. SAADC offset calibration is known to be slightly dodgy, and must be repeated for any gain or temperature change. The best solution is to not tie the RTD to GND, unless the RTD itself is physically manufactured with a GND connection, in itself highly suspicious.

    "it's soldered to the same circuit board so the intermediate resistance should be small and constant" Pretty irrelevant, what matters is any common-mode current (current to/from other devices) using the same copper/vias as the RTD and reference resistor measurement. The SAADC, in addition, does not have a dedicated Gnd which makes things worse.

    "We have a multi-microfarad cap in parallel with the RTD" How many uF and what type? Any capacitor across the RTD requires an identical capacitor (repeat identical) across the reference; it can not be Tant or electrolytic and if ceramic must nave a voltage rating at least double the anticipated working voltage. Anything above 100nF is pointless; balanced capacitance on RTD and reference is much more important. Internal RTD capacitance must also be accounted for.

    "We adjust the PGA gain of the ADC for maximum dynamic range" The best performance (my opinion) is with gain at 1/6; anything higher degrades SAADC performance. Regardless, the SAADC calibration must be performed at the chosen gain as the SAADC offset changes with gain.

    "So creepage/clearance/leakage currents might be a factor" doubtful.

    Assuming the Gnd is fixed in the RTD and is immovable, use 3-wire sensing instead. I'll change my circuit to RTD Gnd and 3-wire and post the results and code later if I get time.

    As an endnote, ratiometric doesn't just mean comparing reference with RTD, but also comparing forward- and reverse-current measurements of reference and RTD since the SAADC (as with all ADCs) offsets are never really eliminated without doing this. Also it is often preferred to use VDD/4 and remove all capacitance across RTD and reference resistors to get true ratiometric results, regardless of reference voltage. In all cases, best to trigger the SAADC to start while the CPU and Radio are both quiet, a case for PPI.

    It would be helpful to see the front-end schematic and SAADC settings. Good luck :-)

  • I tested with 2-wire RTD and as expected the results are much poorer and vary from calibration to calibration as the residual offset is not handled. There are two ways to address that.

    Edit: 3 ways. I got quite good results by simply by alternating reversing the differential pin selection on the SAADC (no hardware change) to compensate for the residual SAADC offset: Updated results

    Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 904.9 Ohms with -0.01% error and much better stability/repeatability:

    // Internal 0.6v Ref 14-bit S0S1 2-Wire RTD actual value 905.0 Ohm (reference 2001.4 Ohm) - Pos-Neg ADC
    // VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error
    // ====== ======= ======  ===== ===== =====  ======  ====== ====== ======
    //   2973    1008     38   2934   911    -3  3110.1  6885.8  904.0 -0.11%
    //   2974    1008     39   2934   911    -3  3109.1  6887.3  903.5 -0.17%
    //   2979    1007     41   2937   916     2  3108.0  6881.3  904.0 -0.12%
    //   2974    1007     41   2932   911    -2  3108.3  6881.4  904.0 -0.11%
    //   2978    1008     39   2938   916     1  3107.9  6884.1  903.5 -0.16%
    //   2975    1007     41   2933   911    -3  3107.6  6882.4  903.7 -0.14%
    //   2974    1008     40   2933   911    -3  3108.0  6883.0  903.7 -0.14%
    //   2976    1008     42   2933   909    -2  3109.5  6883.9  904.0 -0.11%
    //
    // Internal 0.6v Ref 14-bit S0S1 2-Wire RTD actual value 905.0 Ohm (reference 2001.4 Ohm) - Neg-Pos ADC
    // VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error
    // ====== ======= ======  ===== ===== =====  ======  ====== ====== ======
    //   2973    1010     39   2933   909    -4  3122.9  6897.1  906.2  0.13%
    //   2973    1010     39   2933   910    -3  3123.3  6897.5  906.3  0.14%
    //   2975    1010     41   2933   912    -4  3121.4  6896.1  905.9  0.10%
    //   2972    1009     39   2932   911    -3  3120.0  6895.0  905.6  0.07%
    //   2975    1010     39   2935   909    -3  3122.3  6897.5  906.0  0.11%
    //   2972    1010     39   2932   910    -3  3122.3  6897.4  906.0  0.11%
    //   2971    1009     35   2935   910    -3  3120.9  6895.6  905.8  0.09%
    //   2975    1010     41   2933   911    -3  3122.3  6896.8  906.1  0.12%
    //
    // Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 904.9 Ohms with -0.01% error
    

    The schematic/Circuit diagram:

    // RTD Measurement - 2- and 3-wire Unidirectional current
    //                          + ----------------------------------------------------------------------+
    //                          |  nRF52832/nRF52833/nRF52840                                           |
    //                          |                                                                       |
    //                          |    VDD               VDD                 VDD                          |
    //                          |   --#--             --#--               --#--                         |
    //                          |     |                 |                   |                           |
    //                          |    _|__               |   Excitation      |   Option                  |
    //                          |    / \                +-|   On            +-|   13k                   |
    //                          |   /-+-\                 |<- H0   H1         |<- Pullup                |
    //                          |     |                 +-|   160R 35R      +-|   (not used)            |
    //         Excitation  P0.04|     |                 |                   |                           |
    //         +----------------O-----#------#----------#-------------------#------------ In            |
    //         |                |     |      |          |   Excitation      |   Option                  |
    //         |                |   __|__  -----        +-|   Off           +-|   13k                   |
    //         |                |    / \   -----          |<- L0   L1         |<- Pulldown              |
    //         |                |   /-+-\    |3pF       +-|   160R 35R      +-|   (not used)            |
    //         |                |     |      |          |                   |                           |
    //         |                |   =====  =====      =====               =====                         |
    //         |                |    ===    ===        ===                 ===                          |
    //         |                |     =      =          =                   =                           |
    //         |                |                                                                       |
    //         |                |    VDD                                         VDD                    |
    //         |                |   --#--                                       --#--                   |
    //         |                |    _|__                    Auto-adjust:         |   160k              |
    //         |                |    / \                     Bias inputs at VDD/2 +-| Bias Hi/Lo/Both   |
    //         |                |   /-+-\                     VDD1_2                |<-                 |
    //         |                |     |   PIN_VOLTAGE_1                           +-|                   |
    //         |          P0.28 |     |   AIN4          |----'T'-Mux-P---|        |            P SAADC  |
    //         #--------#-------O-----#------#-------#--+ +----#-------+ +--#-----#------#------#-->    |
    //         |        |       |     |      |       |  | |    |       | |  |     |      |      |       |
    //         |        |       |   __|__  -----     |  ----   +-|    ----- |     +-|   +++     |       |
    //         |        |       |    / \   -----     |   |       |<--+  |   |       |<- | |   -----     |
    //  Rref  +++ 100nF |       |   /-+-\    | 3pF   |   |     +-|   |  |   |     +-|   | |   -----     |
    //  2k0   | |     -----     |     |      |       |   +---- |--------#   |     |     +++     | 2.5pF |
    //  1%    | |     -----     |   =====  =====     |         |     |  |   |     |      |1M0   |       |
    // (0.1%) +++       |       |    ===    ===      |       =====   |  |   |   =====  =====  =====     |
    //         |        |       |     =      =       |        ===    |  |   |    ===    ===    ===      |
    //         |        |       |                    |         =    Select  |     =      =      =       |
    //         |        |       |                    |                      |                           |
    //         |        |       |                    |                      |    VDD                    |
    //         |        |       |                    |  |----'T'-Mux-N---|  |   --#--                   |
    //         |        |       |                    +--+ +----#-------+ +----+   |   160k              |
    //         |        |       |                                           | |   +-| Bias Hi/Lo/Both   |
    //         |        |       |         PIN_VOLTAGE_2                     | |     |<-                 |
    //         |        | P0.29 |         AIN5          |----'T'-Mux-----|  | |   +-|                   |
    //         #--------#-------O-----#------#-------#--+ +----#-------+ +--# |   |            N SAADC  |
    //         |        |       |                    +--+ +----#-------+ +----#---#------#------#-->    |
    //    RTD +++ 100nF |       |                                           | |   |      |      |       |
    //        | |     -----     |                                           | |   +-|   +++     |       |
    //        | |     -----     |          Note V3 can be negative          | |     |<- | |   -----     |
    //        +++       |       |          PIN_VOLTAGE_3                    | |   +-|   | |   -----     |
    //         |        | P0.31 |          AIN7         |----'T'-Mux-----|  | |   |     +++     | 2.5pF |
    //         #--------#-------O-----#------#-------#--+ +----#-------+ +--# |   |      |1M0   |       |
    //         |                |                    +--+ +----#-------+ +----# =====  =====  =====     |
    //         |                |                                           | |  ===    ===    ===      |
    //         |                |                                           | |   =      =      =       |
    //         |                |                                           | |                         |
    //         |        VDD     |    VDD                                    | |   P: Single-ended       |
    //         |       --#--    |     |                                     | |   P&N: Differential     |
    //         |         +------O-----#   PIN_VOLTAGE_0                     | |                         |
    //         |                |     |   AIN_VDD       |----'T'-Mux-----|  | |                         |
    //         |                |     +-----#--------#--+ +----#-------+ +--+ |                         |
    //         |                |                    +--+ +----#-------+ +----#                         |
    //         |                |                                             |                         |
    //         |                |    GND                |----'T'-Mux-----|    |                         |
    //         #----------------O-----#-----------------+ +----#-------+ +----+                         |
    //         |                |     |                                                                 |
    //       =====              |   =====                                                               |
    //        ===               |    ===                                                                |
    //         =                |     =                                                                 |
    //                          +-----------------------------------------------------------------------+
    //
    // Capacitor min value = (3pF+2.5pF)*2^14=80nF
    // Vref = differential voltage V1 and V2 in SAADC counts
    // Vrtd = differential voltage V2 and V3 in SAADC counts
    // Rrtd = (Vrtd * Rref)/Vref Ohms

    The code:

    #define SAADC_RESOLUTION_VAL   SAADC_RESOLUTION_VAL_14bit  // 12- or 14-bit in this code
    #define PIN_EXCITATION_A  4 // Red lead A to remote RTD
    //      PIN_VOLTAGE_0   VDD // VDD as source of excitation
    #define PIN_VOLTAGE_1    28 // AIN4 White lead from remote RTD (PIN_EXCITATION_A end)
    #define PIN_VOLTAGE_2    29 // AIN5 White lead from remote RTD (PIN_EXCITATION_B end)
    #define PIN_VOLTAGE_3    31 // AIN7 Red lead B to remote RTD and Reference resistor
    #define PIN_VOLTAGE_4    30 // AIN6 Reference resistor and PIN_EXCITATION_B
    
    // Input range = (VDD/4V)/(1/6) = 4.5 V with VDD=3V VDD/4 Ref do not use 100nF caps
    // Input range = (0.6 V)/(1/6)  = 3.6 V with 100nF caps regardless of VDD
    // Differential mode has 1 bit less than single-ended due to sign bit
    #if (SAADC_RESOLUTION_VAL == SAADC_RESOLUTION_VAL_12bit)
     #define SAADC_COUNTS_PER_VOLT_SE   1138  // 12-bit Mode Singe-ended:  4096/3.6V
     #define SAADC_COUNTS_PER_VOLT_DIFF  569  // 12-bit Mode Differential: 2048/3.6V
     const uint8_t SAADC_ResolutionStr[] = "12-bit";
    #elif (SAADC_RESOLUTION_VAL == SAADC_RESOLUTION_VAL_14bit)
     #define SAADC_RESOLUTION_VAL SAADC_RESOLUTION_VAL_14bit
     #define SAADC_COUNTS_PER_VOLT_SE   4551  // 14-bit Mode Singe-ended: 16384/3.6V
    // #define SAADC_COUNTS_PER_VOLT_DIFF 2276  // 14-bit Mode Differential: 8192/3.6V
     #define SAADC_COUNTS_PER_VOLT_DIFF 3413  // 14-bit Mode Differential: 8192/2.4V
     const uint8_t SAADC_ResolutionStr[] = "14-bit";
    #else
     #error SAADC_RESOLUTION_VAL   not 12- or 14-bit in this code
    #endif
    
    // Do an initial calibration as the offset shows up in comparing +ve and -ve current measurements
    static bool mAmCalibrated = false;
    static float mMeasurementError = 0.0f;
    
    static void RequestCalibration(void)
    {
       // Request a recalibration on any gain or mode changes
       mAmCalibrated = false;
    }
    
    // The accumulator is controlled in the OVERSAMPLE register. When using oversampling, 2^OVERSAMPLE input
    // samples are averaged before the sample result is transferred to memory. Hence, the SAMPLE task must be
    // triggered 2^OVERSAMPLE times for each output value
    static int16_t GetRTD_Voltage(uint32_t AnalogueChannelP, uint32_t AnalogueChannelN)
    {
       uint16_t result = 9999;
       uint32_t timeout = 512;
       volatile int16_t buffer[8];
       uint32_t i=0;  // SAADC Channel
       // Can use internal ref SAADC_CH_CONFIG_REFSEL_Internal with 100nF caps or VDD ratiometric SAADC_CH_CONFIG_REFSEL_VDD1_4 with no caps
       // Check if single-ended or differential SAADC mode
       if (AnalogueChannelN == SAADC_CH_PSELN_PSELN_NC)
       {
          // Configure SAADC singled-ended channel, Internal reference (0.6V) and 1/6 gain (range 3.6V), no pull-up or pull-down
          NRF_SAADC->CH[i].CONFIG = (SAADC_CH_CONFIG_GAIN_Gain1_6    << SAADC_CH_CONFIG_GAIN_Pos)   |
                                    (SAADC_CH_CONFIG_MODE_SE         << SAADC_CH_CONFIG_MODE_Pos)   |
                                    (SAADC_CH_CONFIG_REFSEL_Internal << SAADC_CH_CONFIG_REFSEL_Pos) |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESN_Pos)   |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESP_Pos)   |
                                    (SAADC_CH_CONFIG_TACQ_10us       << SAADC_CH_CONFIG_TACQ_Pos);
       }
       else
       {
          // Configure SAADC differential channel, Internal reference (0.6V) and 1/4 gain (range 2.4V), no pull-up or pull-down
          NRF_SAADC->CH[i].CONFIG = (SAADC_CH_CONFIG_GAIN_Gain1_4    << SAADC_CH_CONFIG_GAIN_Pos)   |
                                    (SAADC_CH_CONFIG_MODE_Diff       << SAADC_CH_CONFIG_MODE_Pos)   |
                                    (SAADC_CH_CONFIG_REFSEL_Internal << SAADC_CH_CONFIG_REFSEL_Pos) |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESN_Pos)   |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESP_Pos)   |
                                    (SAADC_CH_CONFIG_TACQ_10us       << SAADC_CH_CONFIG_TACQ_Pos);
       }
       NRF_SAADC->RESOLUTION = SAADC_RESOLUTION_VAL << SAADC_RESOLUTION_VAL_Pos;
       // Analog positive and negative input channels have strange numbering:
       //  PSELP_NC           0UL  Not connected
       //  PSELP_AnalogInput0 1UL  AIN0
       //  PSELP_AnalogInput1 2UL  AIN1
       //  PSELP_AnalogInput2 3UL  AIN2
       //  PSELP_AnalogInput3 4UL  AIN3
       //  PSELP_AnalogInput4 5UL  AIN4
       //  PSELP_AnalogInput5 6UL  AIN5
       //  PSELP_AnalogInput6 7UL  AIN6
       //  PSELP_AnalogInput7 8UL  AIN7
       //  PSELP_VDD          9UL  VDD
       NRF_SAADC->CH[i].PSELP = AnalogueChannelP << SAADC_CH_PSELP_PSELP_Pos;
       NRF_SAADC->CH[i].PSELN = AnalogueChannelN << SAADC_CH_PSELN_PSELN_Pos;
       // Note the SAMPLE task must be triggered 2^OVERSAMPLE times for each output value
       NRF_SAADC->OVERSAMPLE = SAADC_OVERSAMPLE_OVERSAMPLE_Over32x;
       // Enable SAADC
       NRF_SAADC->ENABLE = 1;
       NRF_SAADC->RESULT.PTR = (uint32_t)buffer;
       NRF_SAADC->RESULT.MAXCNT = 1;
       //NRF_SAADC->RESULT.AMOUNT
       // Do an initial calibration as the offset shows up in comparing +ve and -ve current measurements
       // Any gain changes require recalibration
       if(!mAmCalibrated)
       {
          NRF_SAADC->EVENTS_CALIBRATEDONE = 0;
          NRF_SAADC->TASKS_CALIBRATEOFFSET = 1;
          while (!NRF_SAADC->EVENTS_CALIBRATEDONE) ;
          mAmCalibrated = true;
          // Clear event, though maybe not actually necessary
          NRF_SAADC->EVENTS_CALIBRATEDONE = 0;
       }
       NRF_SAADC->EVENTS_END = 0;
       NRF_SAADC->EVENTS_DONE = 0;
       NRF_SAADC->EVENTS_RESULTDONE = 0;
       NRF_SAADC->TASKS_START = 1;
       // Assume could be up to 256 samples, but don't bother to check actual number
       mSaadcSampleCount = 0;
       while(true)
       {
          NRF_SAADC->TASKS_SAMPLE = 1;
          __DSB();
          while(!NRF_SAADC->EVENTS_DONE) ;
          // Update sample count for reports
          mSaadcSampleCount++;
          if (NRF_SAADC->EVENTS_RESULTDONE || timeout == 0) break;
          NRF_SAADC->EVENTS_DONE = 0;
          timeout--;
       }
       // Don't need NRF_SAADC->EVENTS_END for now
       NRF_SAADC->TASKS_STOP = 1;
       NRF_SAADC->EVENTS_STARTED = 0;
       NRF_SAADC->EVENTS_END = 0;
       NRF_SAADC->EVENTS_DONE = 0;
       NRF_SAADC->EVENTS_RESULTDONE = 0;
       // Disable command
       NRF_SAADC->ENABLE = 0;
       if (timeout != 0)
       {
          // Calculate mVolt value with rounding, assume 16-bit unsigned for prints
          result = (uint16_t)((((int32_t)buffer[0] * 1000L)+(SAADC_COUNTS_PER_VOLT_SE/2)) / SAADC_COUNTS_PER_VOLT_SE);
       }
       if (AnalogueChannelN == SAADC_CH_PSELN_PSELN_NC)
       {
          // Display in mVolts
          return result;
       }
       else
       {
          // Display and calculate in signed SAADC counts for best resolution
          return buffer[0];
       }
    }
    
    #define AVERAGING_NUMBER  4 // Averaging now also done with SAADC oversampling
    #define REPORTS_NUMBER    8 // Numer of printed results for analysis
    const float Rref_Ohm  = 2001.4f;  // Ohm
    const float Rtest_Ohm =  905.0f;  //  905.0 Ohm, 1597.3 Ohm, 2984.6 Ohm.
    float Rrtd = 0.0f;
    float Rrtd_Sum = 0.0f;
    float Vrtd_f = 0.0f;
    float Vref_f = 0.0f;
    
    void Test2WireRTD_ResistanceMeasurement(void)
    {
       char RtdInfoPacket[240] = "?";
       int16_t BatteryVolts0 = 0, BatteryVolts1 = 0, BatteryVolts2 = 0, BatteryVolts3 = 0;
       mClocksInit();
       mRtcInit();
       // Take some samples forward current and then some reverse
       for (uint32_t SampleCount=0; SampleCount<REPORTS_NUMBER; SampleCount++)
       {
          int32_t Vrtd = 0;
          int32_t Vref = 0;
          uint32_t FET_RdsOnA = 1;
          // Configuration               Direction                Input                          Pullup                 Drive Level        Sense Level
          // ==========================  =======================  =============================  =====================  =================  =======================
          nrf_gpio_cfg(PIN_EXCITATION_A, NRF_GPIO_PIN_DIR_OUTPUT, NRF_GPIO_PIN_INPUT_DISCONNECT, NRF_GPIO_PIN_NOPULL,   NRF_GPIO_PIN_H0H1, NRF_GPIO_PIN_NOSENSE);
          // Turn on forward current through RTD and Reference
          nrf_gpio_pin_set(PIN_EXCITATION_A);
          if (SampleCount == 0)
          {
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "\r\n// Internal 0.6v Ref %s S0S1 2-Wire RTD actual value %0.1f Ohm (reference %0.1f Ohm) - Pos-Neg ADC\r\n", SAADC_ResolutionStr, Rtest_Ohm, Rref_Ohm);
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// ====== ======= ======  ===== ===== =====  ======  ====== ====== ======\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
          }
          else if (SampleCount == REPORTS_NUMBER/2)
          {
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "//\r\n// Internal 0.6v Ref %s S0S1 2-Wire RTD actual value %0.1f Ohm (reference %0.1f Ohm) - Neg-Pos ADC\r\n", SAADC_ResolutionStr, Rtest_Ohm, Rref_Ohm);
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// ====== ======= ======  ===== ===== =====  ======  ====== ====== ======\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
          }
          // Short arbitrary delay to allow current to stabalise after turning on
          for (uint32_t i=0; i<10000UL; i++) {}
          // Differential RTD and reference resistor voltage values
          Vrtd = 0;
          Vref = 0;
          // Request a recalibration on any gain or mode changes
          RequestCalibration();
          for (uint32_t i=0; i<AVERAGING_NUMBER; i++)
          {
             // Differential measurements Pos-Neg or Neg-Pos
             if (SampleCount < REPORTS_NUMBER/2)
             {
                Vrtd += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput5, SAADC_CH_PSELP_PSELP_AnalogInput7);
                Vref += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput4, SAADC_CH_PSELP_PSELP_AnalogInput5);
             }
             else
             {
                Vrtd += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput7, SAADC_CH_PSELP_PSELP_AnalogInput5);
                Vref += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput5, SAADC_CH_PSELP_PSELP_AnalogInput4);
             }
          }
          Vrtd_f = Vrtd /(float)AVERAGING_NUMBER;
          Vref_f = Vref /(float)AVERAGING_NUMBER;
          // Keep using integer value for current
          Vref /= AVERAGING_NUMBER;
          // Differential measurements Pos-Neg or Neg-Pos, compensate for pseudo-negative
          if (SampleCount >= REPORTS_NUMBER/2)
          {
             Vrtd_f = 0.0f - Vrtd_f;
             Vref_f = 0.0f - Vref_f;
             Vref   = 0 - Vref;
          }
          // Request a recalibration on any gain or mode changes
          RequestCalibration();
          // Single-ended measurement voltages for testing
          BatteryVolts0 = GetRTD_Voltage(SAADC_CH_PSELP_PSELP_VDD, SAADC_CH_PSELN_PSELN_NC);
          BatteryVolts1 = GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput4, SAADC_CH_PSELN_PSELN_NC);
          BatteryVolts2 = GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput5, SAADC_CH_PSELN_PSELN_NC);
          BatteryVolts3 = GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput7, SAADC_CH_PSELN_PSELN_NC);
          // Rrtd = (Vrtd_f * Rref)/Vref Ohms
          float Rrtd = (Vrtd_f * Rref_Ohm)/Vref_f;
          // Accumulate for sum of both forward and reverse measurements
          Rrtd_Sum += Rrtd;
          // Calculate % error
          mMeasurementError = (Rrtd - Rtest_Ohm) * 100.0f/Rtest_Ohm;
          // Calculate mVolt value of Vref using rounding, differential mode
          uint32_t Vref_mV = (((Vref * 1000)+(SAADC_COUNTS_PER_VOLT_DIFF/2)) / SAADC_COUNTS_PER_VOLT_DIFF);
          // Current Iref = Vref/Rref  uA
          uint32_t Iref_uA = (uint32_t)roundf((Vref_mV * 1000)/Rref_Ohm);
          // RdsOn = Vref_mV/Iref_uA Ohms
          FET_RdsOnA = ((BatteryVolts0 - BatteryVolts1) * 1000) / Iref_uA;
          snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "//  %5hd   %5u  %5u  %5hd %5hd %5hd  %6.1f  %6.1f %6.1f % 0.2f%%\r\n", BatteryVolts0, Iref_uA, FET_RdsOnA, BatteryVolts1, BatteryVolts2, BatteryVolts3, Vrtd_f, Vref_f, Rrtd, mMeasurementError);
          uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
       }
       // Accumulate for sum of both forward and reverse measurements
       Rrtd_Sum /= (float)REPORTS_NUMBER;
       // Calculate % error
       mMeasurementError = (Rrtd_Sum - Rtest_Ohm) * 100.0f/Rtest_Ohm;
       // Measured 12-bit value of 905.0 Ohm test resistor is 905.2 Ohms with  0.02% error
       // Measured 12-bit value of 2984.6 Ohm test resistor is 2984.3 Ohms with -0.01% error
       snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "//\r\n// Measured %s value of %4.1f Ohm 2-Wire test resistor using %u samples is %4.1f Ohms with % 0.2f%% error\r\n", SAADC_ResolutionStr, Rtest_Ohm, mSaadcSampleCount*AVERAGING_NUMBER, Rrtd_Sum, mMeasurementError);
       uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
       // Hold
       while(true) {}
    }
    

    So the (other) two ways to improve 2-wire results are to either have a separate differential channel to measure another calibration resistor in both forward and reverse current directions which allows calculation of the residual offset, and the best way I'll post separately with a schematic/circuit diagram as this post is getting a bit long. This last method gives the best results available for the SAADC in 4-wire or 2-wire mode (tomorrow maybe).

  • I updated the schematic on the previous post to include both P & N SAADC paths plus pin capacitance to explain the choice of 100nF.

    The final schematic/circuit diagram shows the use of flying capacitors to further isolate external influences and provide the option of using bias to re-center input voltages, which can now be beyond rail, for example at 11 volts. 2- and 3-wire modes require a triple SPDT and 4-wire mode requires a quad SPDT, although I would suggest using a quad SPDT as that converts 2- and 3-wire into 4-wire mode pretty well. If I get around to it I'll post some results.

    // RTD Measurement - 2- and 3-wire Unidirectional current with flying Capacitor Sample/Hold
    //
    //                                +-----------------------------------------------------------------------+
    //                                |  nRF52832/nRF52833/nRF52840                                           |
    //                                |                                                                       |
    //                                |    VDD               VDD                 VDD  Excitation Supply       |
    //                                |   --#--             --#--               --#--                         |
    //                                |     |                 |                   |                           |
    //                                |    _|__               |   Excitation      |   Option                  |
    //                                |    / \                +-|   On            +-|   13k                   |
    //                                |   /-+-\                 |<- H0   H1         |<- Pullup                |
    //                                |     |                 +-|   160R 35R      +-|   (not used)            |
    //         Excitation       P0.04 |     |                 |                   |                           |
    //         +----------------------O-----#------#----------#-------------------#------------ In            |
    //         |                      |     |      |          |   Excitation      |   Option                  |
    //         |                      |   __|__  -----        +-|   Off           +-|   13k                   |
    //         |                      |    / \   -----          |<- L0   L1         |<- Pulldown              |
    //         |                      |   /-+-\    | 3pF      +-|   160R 35R      +-|   (not used)            |
    //         |                      |     |      |          |                   |                           |
    //         |                      |   =====  =====      =====               =====                         |
    //         |                      |    ===    ===        ===                 ===                          |
    //         |                      |     =      =          =                   =                           |
    //         |        S/H     P0.03 |                                                 Flying Capacitor      |
    //         |        +-------------O-----#------#----------#-------------------#---< Sample/Hold           |
    //         |        |             |                                                                       |
    //         |        |             |    VDD                                         VDD                    |
    //         |        |             |   --#--                                       --#--                   |
    //         |        |             |    _|__                    Auto-adjust:         |   160k              |
    //         |        |             |    / \                     Bias inputs at VDD/2 +-| Bias Hi/Lo/Both   |
    //         |        |             |   /-+-\                     VDD1_2                |<-                 |
    //         |        |             |     |   PIN_VOLTAGE_1                           +-|                   |
    //         |        |       P0.28 |     |   AIN4          |----'T'-Mux-P---|        |            P SAADC  |
    //         #------0 | 0-----------O-----#------#-------#--+ +----#-------+ +--#-----#------#------#-->    |
    //         |       \'             |     |      |       |  | |    |       | |  |     |      |      |       |
    //         |        \             |   __|__  -----     |  ----   +-|    ----- |     +-|   +++     |       |
    //         |        0             |    / \   -----     |   |       |<--+  |   |       |<- | |   -----     |
    //  Rref  +++       |             |   /-+-\    | 3pF   |   |     +-|   |  |   |     +-|   | |   -----     |
    //  2k0   | |     -----           |     |      |       |   +---- |--------#   |     |     +++     | 2.5pF |
    //  1%    | |     -----           |   =====  =====     |         |     |  |   |     |      |1M0   |       |
    // (0.1%) +++       | 100nF       |    ===    ===      |       =====   |  |   |   =====  =====  =====     |
    //         |        |             |     =      =       |        ===    |  |   |    ===    ===    ===      |
    //         |        +---+         |                    |         =    Select  |     =      =      =       |
    //         |            |         |                    |                      |                           |
    //         |            |         |                    |                      |    VDD                    |
    //         |            |         |                    |  |----'T'-Mux-N---|  |   --#--                   |
    //         |            |         |                    +--+ +----#-------+ +----+   |   160k              |
    //         |            |         |                                           | |   +-| Bias Hi/Lo/Both   |
    //         |            |         |         PIN_VOLTAGE_2                     | |     |<-                 |
    //         |            |   P0.29 |         AIN5          |----'T'-Mux-----|  | |   +-|                   |
    //         #------0   0-----------O-----#------#-------#--+ +----#-------+ +--# |   |            N SAADC  |
    //         |       \    |         |                    +--+ +----#-------+ +----#---#------#------#-->    |
    //         |        \   |         |                                           | |   |      |      |       |
    //         |        0   |         |                                           | |   +-|   +++     |       |
    //         |        |   |         |                                           | |     |<- | |   -----     |
    //         |        #---+         |                                           | |   +-|   | |   -----     |
    //    RTD +++       |             |                                           | |   |     +++     | 2.5pF |
    //        | |     -----           |                                           | |   |      |1M0   |       |
    //        | |     -----           |                                           | | =====  =====  =====     |
    //        +++       | 100nF       |                                           | |  ===    ===    ===      |
    //         |        0             |         Note V3 can be negative           | |   =      =      =       |
    //         |        /             |         PIN_VOLTAGE_3                     | |   P: Single-ended       |
    //         |       /        P0.31 |         AIN7          |----'T'-Mux-----|  | |   P&N: Differential     |
    //         #------0   0-----------O-----#-----#--------#--+ +----#-------+ +--# |                         |
    //         |                      |                    +--+ +----#-------+ +----#                         |
    //         |                      |                                           | |                         |
    //         |              VDD     |    VDD                                    | |                         |
    //         |             --#--    |     |                                     | |                         |
    //         |               +------O-----#   PIN_VOLTAGE_0                     | |                         |
    //         |                      |     |   AIN_VDD       |----'T'-Mux-----|  | |                         |
    //         |                      |     +-----#--------#--+ +----#-------+ +--+ |                         |
    //         |                      |                    +--+ +----#-------+ +----#                         |
    //         |                      |                                             |                         |
    //         |                      |    GND                |----'T'-Mux-----|    |                         |
    //         #----------------------O-----#-----------------+ +----#-------+ +----+                         |
    //         |                      |     |                                                                 |
    //       =====                    |   =====                                                               |
    //        ===                     |    ===                                                                |
    //         =                      |     =                                                                 |
    //                                +-----------------------------------------------------------------------+
    //
    // Capacitor min value = (3pF+2.5pF)*2^14=80nF
    // Vref = differential voltage V1 and V2 in SAADC counts
    // Vrtd = differential voltage V2 and V3 in SAADC counts
    // Rrtd = (Vrtd * Rref)/Vref Ohms
    //
    // SPDT Switch, select one with single-supply, rail-to-rail inputs (and negative input voltage capability if needed)
    //  https://www.analog.com/media/en/technical-documentation/data-sheets/max4762-max4765.pdf
    //  https://www.analog.com/media/en/technical-documentation/data-sheets/MAX4781-MAX4783.pdf
    //  https://www.analog.com/media/en/technical-documentation/data-sheets/MAX4780-MAX4784.pdf
    //  https://www.analog.com/media/en/technical-documentation/data-sheets/MAX4780-MAX4784.pdf
    //

  • I added the ENOB value for the 2-wire mode, looks pretty good though 4-wire still better; I'll test with the flying capacitors later this week.

    Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 905.1 Ohms with  0.02% error (12.6 ENOB)

    // Internal 0.6v Ref 14-bit S0S1 2-Wire RTD actual value 905.0 Ohm (reference 2001.4 Ohm) - Pos-Neg ADC
    // VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error
    // ====== ======= ======  ===== ===== =====  ======  ====== ====== ======
    //   2978    1008     41   2936   911     0  3110.6  6883.4  904.4 -0.06%
    //   2977    1008     41   2935   913    -2  3109.0  6883.9  903.9 -0.12%
    //   2978    1008     41   2936   911    -4  3110.5  6887.0  903.9 -0.12%
    //   2976    1008     38   2937   911    -3  3112.1  6888.9  904.2 -0.09%
    //   2979    1009     40   2938   913    -1  3112.0  6892.5  903.6 -0.15%
    //   2981    1008     41   2939   918     1  3110.9  6885.8  904.2 -0.09%
    //   2978    1008     42   2935   912    -2  3109.6  6884.8  904.0 -0.11%
    //   2982    1008     40   2941   918     2  3108.1  6883.4  903.7 -0.14%
    //
    // Internal 0.6v Ref 14-bit S0S1 2-Wire RTD actual value 905.0 Ohm (reference 2001.4 Ohm) - Neg-Pos ADC
    // VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error
    // ====== ======= ======  ===== ===== =====  ======  ====== ====== ======
    //   2978    1010     41   2936   911    -3  3125.9  6902.3  906.4  0.15%
    //   2975    1010     38   2936   912    -3  3125.0  6900.1  906.4  0.16%
    //   2982    1010     41   2940   918     2  3124.4  6899.5  906.3  0.15%
    //   2975    1009     39   2935   913    -3  3123.5  6895.6  906.6  0.17%
    //   2975    1010     39   2935   914    -2  3126.0  6901.4  906.5  0.17%
    //   2978    1010     41   2936   912    -2  3123.9  6901.5  905.9  0.10%
    //   2978    1010     38   2939   915    -1  3124.1  6901.6  906.0  0.11%
    //   2976    1010     39   2936   913    -3  3124.8  6900.9  906.2  0.14%
    //
    // Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 905.1 Ohms with  0.02% error (12.6 ENOB)
    

  • Signing off on this; I tested with the flying-capacitor S/H but with long leads the results are no better than the simple 2- or 3-wire presented previously. I left the code in however.

    2- or 3-wire schematic

    // RTD Measurement - 2- and 3-wire Unidirectional current
    //                          + ----------------------------------------------------------------------+
    //                          |  nRF52832/nRF52833/nRF52840                                           |
    //                          |                                                                       |
    //                          |    VDD               VDD                 VDD                          |
    //                          |   --#--             --#--               --#--                         |
    //                          |     |                 |                   |                           |
    //                          |    _|__               |   Excitation      |   Option                  |
    //                          |    / \                +-|   On            +-|   13k                   |
    //                          |   /-+-\                 |<- H0   H1         |<- Pullup                |
    //                          |     |                 +-|   160R 35R      +-|   (not used)            |
    //         Excitation  P0.04|     |                 |                   |                           |
    //         +----------------O-----#------#----------#-------------------#------------ In            |
    //         |                |     |      |          |   Excitation      |   Option                  |
    //         |                |   __|__  -----        +-|   Off           +-|   13k                   |
    //         |                |    / \   -----          |<- L0   L1         |<- Pulldown              |
    //         |                |   /-+-\    |3pF       +-|   160R 35R      +-|   (not used)            |
    //         |                |     |      |          |                   |                           |
    //         |                |   =====  =====      =====               =====                         |
    //         |                |    ===    ===        ===                 ===                          |
    //         |                |     =      =          =                   =                           |
    //         |                |                                                                       |
    //         |                |    VDD                                         VDD                    |
    //         |                |   --#--                                       --#--                   |
    //         |                |    _|__                    Auto-adjust:         |   160k              |
    //         |                |    / \                     Bias inputs at VDD/2 +-| Bias Hi/Lo/Both   |
    //         |                |   /-+-\                     VDD1_2                |<-                 |
    //         |                |     |   PIN_VOLTAGE_1                           +-|                   |
    //         |          P0.28 |     |   AIN4          |----'T'-Mux-P---|        |            P SAADC  |
    //         #--------#-------O-----#------#-------#--+ +----#-------+ +--#-----#------#------#-->    |
    //         |        |       |     |      |       |  | |    |       | |  |     |      |      |       |
    //         |        |       |   __|__  -----     |  ----   +-|    ----- |     +-|   +++     |       |
    //         |        |       |    / \   -----     |   |       |<--+  |   |       |<- | |   -----     |
    //  Rref  +++ 100nF |       |   /-+-\    | 3pF   |   |     +-|   |  |   |     +-|   | |   -----     |
    //  2k0   | |     -----     |     |      |       |   +---- |--------#   |     |     +++     | 2.5pF |
    //  1%    | |     -----     |   =====  =====     |         |     |  |   |     |      |1M0   |       |
    // (0.1%) +++       |       |    ===    ===      |       =====   |  |   |   =====  =====  =====     |
    //         |        |       |     =      =       |        ===    |  |   |    ===    ===    ===      |
    //         |        |       |                    |         =    Select  |     =      =      =       |
    //         |        |       |                    |                      |                           |
    //         |        |       |                    |                      |    VDD                    |
    //         |        |       |                    |  |----'T'-Mux-N---|  |   --#--                   |
    //         |        |       |                    +--+ +----#-------+ +----+   |   160k              |
    //         |        |       |                                           | |   +-| Bias Hi/Lo/Both   |
    //         |        |       |         PIN_VOLTAGE_2                     | |     |<-                 |
    //         |        | P0.29 |         AIN5          |----'T'-Mux-----|  | |   +-|                   |
    //         #--------#-------O-----#------#-------#--+ +----#-------+ +--# |   |            N SAADC  |
    //         |        |       |                    +--+ +----#-------+ +----#---#------#------#-->    |
    //    RTD +++ 100nF |       |                                           | |   |      |      |       |
    //        | |     -----     |                                           | |   +-|   +++     |       |
    //        | |     -----     |          Note V3 can be negative          | |     |<- | |   -----     |
    //        +++       |       |          PIN_VOLTAGE_3                    | |   +-|   | |   -----     |
    //         |        | P0.31 |          AIN7         |----'T'-Mux-----|  | |   |     +++     | 2.5pF |
    //         #--------#-------O-----#------#-------#--+ +----#-------+ +--# |   |      |1M0   |       |
    //         |                |                    +--+ +----#-------+ +----# =====  =====  =====     |
    //         |                |                                           | |  ===    ===    ===      |
    //         |                |                                           | |   =      =      =       |
    //         |        VDD     |    VDD                                    | |                         |
    //         |       --#--    |   --#--                                   | |   P: Single-ended       |
    //         |         |      |     |                                     | |   P&N: Differential     |
    //         |         +------O-----#   PIN_VOLTAGE_0                     | |                         |
    //         |                |     |   AIN_VDD       |----'T'-Mux-----|  | |                         |
    //         |                |     +-----#--------#--+ +----#-------+ +--+ |                         |
    //         |                |                    +--+ +----#-------+ +----#                         |
    //         |                |                                             |                         |
    //         |                |    GND                |----'T'-Mux-----|    |                         |
    //         #----------------O-----#-----------------+ +----#-------+ +----+                         |
    //         |                |     |                                                                 |
    //       =====              |   =====                                                               |
    //        ===               |    ===                                                                |
    //         =                |     =                                                                 |
    //                          +-----------------------------------------------------------------------+
    //
    // Capacitor min value = (3pF+2.5pF)*2^14=80nF
    // Vref = differential voltage V1 and V2 in SAADC counts
    // Vrtd = differential voltage V2 and V3 in SAADC counts
    // Rrtd = (Vrtd * Rref)/Vref Ohms
    

    Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 905.2 Ohms with 0.02% error (ENOB is 12.2)

    // Internal 0.6v Ref 14-bit S0S1 2-Wire RTD actual value 905.0 Ohm (reference 2001.4 Ohm) - Pos-Neg ADC
    // VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error
    // ====== ======= ======  ===== ===== =====  ======  ====== ====== ======
    //   2973    1008     37   2935   910    -3  3112.3  6884.7  904.8 -0.03%
    //   2973    1007     39   2933   911    -4  3110.3  6879.0  904.9 -0.01%
    //   2973    1007     45   2927   910    -3  3108.5  6881.6  904.1 -0.10%
    //   2975    1008     43   2931   911    -4  3108.5  6883.4  903.8 -0.13%
    //   2973    1007     41   2931   913    -2  3110.6  6881.3  904.7 -0.03%
    //   2973    1007     39   2933   908    -4  3108.4  6881.2  904.1 -0.10%
    //   2972    1007     38   2933   911    -2  3107.4  6882.4  903.6 -0.15%
    //   2976    1008     41   2934   911    -3  3110.3  6883.1  904.4 -0.07%
    //
    // Internal 0.6v Ref 14-bit S0S1 2-Wire RTD actual value 905.0 Ohm (reference 2001.4 Ohm) - Neg-Pos ADC
    // VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error
    // ====== ======= ======  ===== ===== =====  ======  ====== ====== ======
    //   2972    1008     39   2932   911    -5  3119.1  6888.9  906.2  0.13%
    //   2972    1009     37   2934   913    -3  3120.1  6892.4  906.0  0.11%
    //   2975    1009     39   2935   910    -2  3120.8  6890.1  906.5  0.17%
    //   2971    1009     43   2927   911    -6  3119.6  6890.3  906.1  0.12%
    //   2971    1009     38   2932   912    -3  3119.5  6891.5  906.0  0.11%
    //   2972    1009     38   2933   913    -1  3120.9  6892.0  906.3  0.14%
    //   2975    1009     41   2933   911    -4  3121.0  6894.6  906.0  0.11%
    //   2975    1009     40   2934   914    -6  3118.6  6891.1  905.7  0.08%
    //
    // Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 905.2 Ohms with 0.02% error (ENOB is 12.2)
    

    The code:

    //#define FEATURE_FLYING_CAPACITOR // Uncomment this line to use eternal S/H with TS3A5018 or similaar
    #define SAADC_RESOLUTION   SAADC_RESOLUTION_VAL_14bit  // 12- or 14-bit in this code
    #define PIN_EXCITATION_A   4  // Red lead A to remote RTD
    #define PIN_EXCITATION_B   3  // Connected via Reference resistor to Red lead B to remote RTD (4-wire)
    //      PIN_VOLTAGE_0    VDD  // VDD as source of excitation
    #define PIN_VOLTAGE_1     28  // AIN4 White lead from remote RTD (PIN_EXCITATION_A end)
    #define PIN_VOLTAGE_2     29  // AIN5 White lead from remote RTD (PIN_EXCITATION_B end)
    #define PIN_VOLTAGE_3     31  // AIN7 Red lead B to remote RTD and Reference resistor
    #define PIN_VOLTAGE_4     30  // AIN6 Reference resistor and PIN_EXCITATION_B                 (4-wire)
    #if defined(FEATURE_FLYING_CAPACITOR)
     #define PIN_SAMPLE_HOLD  25  // Flying capacitor SPDT switch controls
     #define SH_DELAY        10UL // Flying capacitor settle time, may not need
    #endif
    
    // Input range = (VDD/4V)/(1/6) = 4.5 V with VDD=3V VDD/4 Ref do not use 100nF caps
    // Input range = (0.6 V)/(1/6)  = 3.6 V with 100nF caps regardless of VDD
    // Differential mode has 1 bit less than single-ended due to sign bit
    #if (SAADC_RESOLUTION_VAL == SAADC_RESOLUTION_VAL_12bit)
     #define SAADC_COUNTS_PER_VOLT_SE   1138L  // 12-bit Mode Singe-ended:  4096/3.6V
     #define SAADC_COUNTS_PER_VOLT_DIFF  569L  // 12-bit Mode Differential: 2048/3.6V
     const uint8_t SAADC_ResolutionStr[] = "12-bit";
    #elif (SAADC_RESOLUTION == SAADC_RESOLUTION_VAL_14bit)
     #define SAADC_COUNTS_PER_VOLT_SE   4551L  // 14-bit Mode Singe-ended: 16384/3.6V
    // #define SAADC_COUNTS_PER_VOLT_DIFF 2276L  // 14-bit Mode Differential: 8192/3.6V
     #define SAADC_COUNTS_PER_VOLT_DIFF 3413L  // 14-bit Mode Differential: 8192/2.4V
     const uint8_t SAADC_ResolutionStr[] = "14-bit";
    #else
     #error SAADC_RESOLUTION  not 12- or 14-bit in this code
    #endif
    
    // Do an initial calibration as the offset shows up in comparing +ve and -ve current measurements
    static bool mAmCalibrated = false;
    static float mMeasurementError = 0.0f;
    static uint32_t mSaadcSampleCount = 1UL;
    static void RequestCalibration(void)
    {
       // Request a recalibration on any gain or mode changes
       mAmCalibrated = false;
    }
    
    // The accumulator is controlled in the OVERSAMPLE register. When using oversampling, 2^OVERSAMPLE input
    // samples are averaged before the sample result is transferred to memory. Hence, the SAMPLE task must be
    // triggered 2^OVERSAMPLE times for each output value
    //
    // SAADC on nRF54L15
    // =================
    // CH[n].CONFIG.BURST can be enabled to avoid setting SAMPLE task 2^OVERSAMPLE times. With BURST = 1 the
    // ADC will sample the input 2^OVERSAMPLE times as fast as it can (actual timing: <(tACQ+tCONV)*2^OVERSAMPLE).
    // ie similar to one-shot mode
    // Noise shaping is implemented by using the SAR-ADC in a first-order delta-sigma loop and the filtering
    // is done with the use of FIR filters. The sampling rate in these modes is 1MS/s, and only 12 and 14 bit
    // resolution are supported. To enable noise shaping use NOISESHAPE
    // The ADC can use different reference voltages VREF, controlled in the REFSEL field of the CH[n].CONFIG.
    // These are:
    //   Internal reference, VREF = 0.9 V
    //   External reference, VREF provided by the EXTREF pin
    //    Note: The external reference voltage should be close the internal reference voltage, typ 1.2V
    static uint32_t mSAADC_InterruptCount = 0UL;
    void SAADC_IRQHandler(void)
    {
       NRF_SAADC->EVENTS_END = 0;
       __DSB();
       mSAADC_InterruptCount++;
    }
    static int32_t GetRTD_Voltage(uint32_t AnalogueChannelP, uint32_t AnalogueChannelN)
    {
       int32_t result = 9999L;
       uint32_t timeout = 512UL;
       volatile int16_t buffer[8] = {0};
       uint32_t i=0;  // SAADC Channel
    #if defined(FEATURE_FLYING_CAPACITOR)
       // Short arbitrary delay to allow sample 100nF to charge
       //for (uint32_t i=0; i<SH_DELAY; i++) {}
       // Connnect Flying Capacitor SPDT switch to SAADC - NO contacts
       nrf_gpio_pin_set(PIN_SAMPLE_HOLD);
    #endif
       // Can use internal ref SAADC_CH_CONFIG_REFSEL_Internal with 100nF caps or VDD ratiometric SAADC_CH_CONFIG_REFSEL_VDD1_4 with no caps
       // Check if single-ended or differential SAADC mode
       if (AnalogueChannelN == SAADC_CH_PSELN_PSELN_NC)
       {
          // Configure SAADC singled-ended channel, Internal reference (0.6V) and 1/6 gain (range 3.6V), no pull-up or pull-down
          // nRF54L15 has different settings ToDo
          NRF_SAADC->CH[i].CONFIG = (SAADC_CH_CONFIG_GAIN_Gain1_6    << SAADC_CH_CONFIG_GAIN_Pos)   |
                                    (SAADC_CH_CONFIG_MODE_SE         << SAADC_CH_CONFIG_MODE_Pos)   |
                                    (SAADC_CH_CONFIG_REFSEL_Internal << SAADC_CH_CONFIG_REFSEL_Pos) |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESN_Pos)   |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESP_Pos)   |
                                    (SAADC_CH_CONFIG_TACQ_3us        << SAADC_CH_CONFIG_TACQ_Pos);
       }
       else
       {
          // Configure SAADC differential channel, Internal reference (0.6V) and 1/4 gain (range 2.4V), no pull-up or pull-down
          NRF_SAADC->CH[i].CONFIG = (SAADC_CH_CONFIG_GAIN_Gain1_4    << SAADC_CH_CONFIG_GAIN_Pos)   |
                                    (SAADC_CH_CONFIG_MODE_Diff       << SAADC_CH_CONFIG_MODE_Pos)   |
                                    (SAADC_CH_CONFIG_REFSEL_Internal << SAADC_CH_CONFIG_REFSEL_Pos) |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESN_Pos)   |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESP_Pos)   |
                                    (SAADC_CH_CONFIG_TACQ_3us        << SAADC_CH_CONFIG_TACQ_Pos);
       }
       NRF_SAADC->RESOLUTION = SAADC_RESOLUTION << SAADC_RESOLUTION_VAL_Pos;
       // Analog positive and negative input channels have strange numbering:
       //  PSELP_NC           0UL  Not connected
       //  PSELP_AnalogInput0 1UL  AIN0
       //  PSELP_AnalogInput1 2UL  AIN1
       //  PSELP_AnalogInput2 3UL  AIN2
       //  PSELP_AnalogInput3 4UL  AIN3
       //  PSELP_AnalogInput4 5UL  AIN4
       //  PSELP_AnalogInput5 6UL  AIN5
       //  PSELP_AnalogInput6 7UL  AIN6
       //  PSELP_AnalogInput7 8UL  AIN7
       //  PSELP_VDD          9UL  VDD
       NRF_SAADC->CH[i].PSELP = AnalogueChannelP << SAADC_CH_PSELP_PSELP_Pos;
       NRF_SAADC->CH[i].PSELN = AnalogueChannelN << SAADC_CH_PSELN_PSELN_Pos;
       // Note the SAMPLE task must be triggered 2^OVERSAMPLE times for each output value
       NRF_SAADC->OVERSAMPLE = SAADC_OVERSAMPLE_OVERSAMPLE_Over8x;
       // Setup interrupts if required
       NRF_SAADC->INTENSET = SAADC_INTENSET_END_Set << SAADC_INTENSET_END_Pos;
       NVIC_SetPriority(SAADC_IRQn, 6);
       NVIC_ClearPendingIRQ(SAADC_IRQn);
       NVIC_EnableIRQ(SAADC_IRQn);
       // Enable SAADC
       NRF_SAADC->ENABLE = 1;
       NRF_SAADC->RESULT.PTR = (uint32_t)buffer;
       NRF_SAADC->RESULT.MAXCNT = 1;
       //NRF_SAADC->RESULT.AMOUNT
       // Do an initial calibration as the offset shows up in comparing +ve and -ve current measurements
       // Any gain changes require recalibration. This reduces but does not eliminate SAADC offset
       if(!mAmCalibrated)
       {
          NRF_SAADC->EVENTS_CALIBRATEDONE = 0;
          NRF_SAADC->TASKS_CALIBRATEOFFSET = 1;
          while (!NRF_SAADC->EVENTS_CALIBRATEDONE) ;
          mAmCalibrated = true;
          // Clear event, though maybe not actually necessary
          NRF_SAADC->EVENTS_CALIBRATEDONE = 0;
       }
       NRF_SAADC->EVENTS_END = 0;
       NRF_SAADC->EVENTS_DONE = 0;
       NRF_SAADC->EVENTS_RESULTDONE = 0;
       NRF_SAADC->TASKS_START = 1;
       // Assume could be up to 256 samples, but don't bother to check actual number
       mSaadcSampleCount = 0;
       while(true)
       {
          NRF_SAADC->TASKS_SAMPLE = 1;
          __DSB();
          while(!NRF_SAADC->EVENTS_DONE) ;
          // Update sample count for reports
          mSaadcSampleCount++;
          if (NRF_SAADC->EVENTS_RESULTDONE || timeout == 0) break;
          NRF_SAADC->EVENTS_DONE = 0;
          timeout--;
       }
       // Don't need NRF_SAADC->EVENTS_END for now
       NRF_SAADC->TASKS_STOP = 1;
       NRF_SAADC->EVENTS_STARTED = 0;
       NRF_SAADC->EVENTS_END = 0;
       NRF_SAADC->EVENTS_DONE = 0;
       NRF_SAADC->EVENTS_RESULTDONE = 0;
       // Disable command
       NRF_SAADC->ENABLE = 0;
    #if defined(FEATURE_FLYING_CAPACITOR)
       // Connnect Flying Capacitor SPDT switch to Rrtd and Rref - NC contacts
       nrf_gpio_pin_clear(PIN_SAMPLE_HOLD);
    #endif
       // Process result
       if (AnalogueChannelN == SAADC_CH_PSELN_PSELN_NC)
       {
          // Display in mVolts, single-ended values are only for info
          if (timeout != 0)
          {
             // Calculate mVolt value with rounding, later assume convert to 16-bit signed for prints
             result = (((int32_t)buffer[0] * 1000L)+(SAADC_COUNTS_PER_VOLT_SE)/2L) / (SAADC_COUNTS_PER_VOLT_SE);
          }
          return result;
       }
       else
       {
          // Display and calculate differential values in signed SAADC counts for best resolution
          return (int32_t)buffer[0];
       }
    }
    
    #define AVERAGING_NUMBER  16 // Averaging now also done with SAADC oversampling
    #define REPORTS_NUMBER    16 // Numer of printed results for analysis
    const float Rref_Ohm  = 2001.4f;  // Ohm
    const float Rtest_Ohm =  905.0f;  //  905.0 Ohm, 1597.3 Ohm, 2984.6 Ohm.
    float Rrtd     = 0.0f;
    float Rrtd_Sum = 0.0f;
    float Vrtd_f = 0.0f;
    float Vref_f = 0.0f;
    
    #include <math.h>
    // x^y = pow(x,y)
    inline int root(int input, int n)
    {
       return round(pow(input, 1./n));
    }
    
    void Test3WireRTD_ResistanceMeasurement(void)
    {
       char RtdInfoPacket[240] = "?";
       int16_t BatteryVolts0 = 0, BatteryVolts1 = 0, BatteryVolts2 = 0, BatteryVolts3 = 0;
       // Take some differential samples normal and then some samples with reversed SAADC connection
       for (uint32_t SampleCount=0; SampleCount<REPORTS_NUMBER; SampleCount++)
       {
          int32_t Vrtd = 0;
          int32_t Vref = 0;
          uint32_t FET_RdsOnA = 1;
          // Configuration               Direction                Input                          Pullup                 Drive Level        Sense Level
          // ==========================  =======================  =============================  =====================  =================  =======================
          nrf_gpio_cfg(PIN_EXCITATION_A, NRF_GPIO_PIN_DIR_OUTPUT, NRF_GPIO_PIN_INPUT_DISCONNECT, NRF_GPIO_PIN_NOPULL,   NRF_GPIO_PIN_H0H1, NRF_GPIO_PIN_NOSENSE);
          // Turn on forward current through RTD and Reference
          nrf_gpio_pin_set(PIN_EXCITATION_A);
    #if defined(FEATURE_FLYING_CAPACITOR)
          nrf_gpio_cfg(PIN_SAMPLE_HOLD,  NRF_GPIO_PIN_DIR_OUTPUT, NRF_GPIO_PIN_INPUT_DISCONNECT, NRF_GPIO_PIN_NOPULL,   NRF_GPIO_PIN_S0S1, NRF_GPIO_PIN_NOSENSE);
          // Connnect Flying Capacitor SPDT switch to Rrtd and Rref - NC contacts
          nrf_gpio_pin_clear(PIN_SAMPLE_HOLD);
    #endif
          if (SampleCount == 0)
          {
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "\r\n// Internal 0.6v Ref %s S0S1 2-Wire RTD actual value %0.1f Ohm (reference %0.1f Ohm) - Pos-Neg ADC\r\n", SAADC_ResolutionStr, Rtest_Ohm, Rref_Ohm);
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// ====== ======= ======  ===== ===== =====  ======  ====== ====== ======\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
          }
          else if (SampleCount == REPORTS_NUMBER/2)
          {
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "//\r\n// Internal 0.6v Ref %s S0S1 2-Wire RTD actual value %0.1f Ohm (reference %0.1f Ohm) - Neg-Pos ADC\r\n", SAADC_ResolutionStr, Rtest_Ohm, Rref_Ohm);
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// ====== ======= ======  ===== ===== =====  ======  ====== ====== ======\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
          }
          // Differential RTD and reference resistor voltage values
          Vrtd = 0;
          Vref = 0;
          // Request a recalibration on any gain or mode changes
          RequestCalibration();
          for (uint32_t i=0; i<AVERAGING_NUMBER; i++)
          {
             // Differential measurements Pos-Neg or Neg-Pos
             if (SampleCount < REPORTS_NUMBER/2)
             {
                Vrtd += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput5, SAADC_CH_PSELN_PSELN_AnalogInput7);
                Vref += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput4, SAADC_CH_PSELN_PSELN_AnalogInput5);
             }
             else
             {
                Vrtd += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput7, SAADC_CH_PSELN_PSELN_AnalogInput5);
                Vref += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput5, SAADC_CH_PSELN_PSELN_AnalogInput4);
             }
          }
          Vrtd_f = Vrtd /(float)AVERAGING_NUMBER;
          Vref_f = Vref /(float)AVERAGING_NUMBER;
          // Keep using integer value for current
          Vref /= AVERAGING_NUMBER;
          // Differential measurements Pos-Neg or Neg-Pos, compensate for pseudo-negative
          if (SampleCount >= REPORTS_NUMBER/2)
          {
             Vrtd_f = 0.0f - Vrtd_f;
             Vref_f = 0.0f - Vref_f;
             Vref   = 0 - Vref;
          }
          // Request a recalibration on any gain or mode changes
          RequestCalibration();
          // Single-ended measurement voltages for testing
          BatteryVolts0 = (int16_t)GetRTD_Voltage(SAADC_CH_PSELP_PSELP_VDD, SAADC_CH_PSELN_PSELN_NC);
          BatteryVolts1 = (int16_t)GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput4, SAADC_CH_PSELN_PSELN_NC);
          BatteryVolts2 = (int16_t)GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput5, SAADC_CH_PSELN_PSELN_NC);
          BatteryVolts3 = (int16_t)GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput7, SAADC_CH_PSELN_PSELN_NC);
          // Rrtd = (Vrtd_f * Rref)/Vref Ohms
          float Rrtd = (Vrtd_f * Rref_Ohm)/Vref_f;
          // Accumulate for sum of both forward and reverse measurements
          Rrtd_Sum += Rrtd;
          // Calculate % error
          mMeasurementError = (Rrtd - Rtest_Ohm) * 100.0f/Rtest_Ohm;
          // Calculate mVolt value of Vref using rounding, differential mode
          uint32_t Vref_mV = (((Vref * 1000)+(SAADC_COUNTS_PER_VOLT_DIFF/2)) / SAADC_COUNTS_PER_VOLT_DIFF);
          // Current Iref = Vref/Rref  uA
          uint32_t Iref_uA = (uint32_t)roundf((Vref_mV * 1000)/Rref_Ohm);
          // RdsOn = Vref_mV/Iref_uA Ohms
          FET_RdsOnA = ((BatteryVolts0 - BatteryVolts1) * 1000) / Iref_uA;
          snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "//  %5hd   %5u  %5u  %5hd %5hd %5hd  %6.1f  %6.1f %6.1f % 0.2f%%\r\n", BatteryVolts0, Iref_uA, FET_RdsOnA, BatteryVolts1, BatteryVolts2, BatteryVolts3, Vrtd_f, Vref_f, Rrtd, mMeasurementError);
          uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
       }
       // Accumulate for sum of both forward and reverse measurements
       Rrtd_Sum /= (float)REPORTS_NUMBER;
       // Calculate % error
       mMeasurementError = (Rrtd_Sum - Rtest_Ohm) * 100.0f/Rtest_Ohm;
       // Check ENOB using x^y = pow(x,y)
       float fENOB_Estimate, fIteration = 0.01f, fLimit = 100.0f/mMeasurementError;
       if (fLimit < 0.0F) fLimit = 0.0f - fLimit;
       for (fENOB_Estimate=10.0f; fENOB_Estimate<=14.0f; fENOB_Estimate += fIteration)
       {
          if (pow(2.0f, fENOB_Estimate) > fLimit)
          {
              fENOB_Estimate -= fIteration;
              break;
          }
       }
       // Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 905.2 Ohms with 0.02% error (ENOB is 12.1)
       snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "//\r\n// Measured %s value of %4.1f Ohm 2-Wire test resistor using %u samples is %4.1f Ohms with %0.2f%% error (ENOB is %0.1f)\r\n", SAADC_ResolutionStr, Rtest_Ohm, mSaadcSampleCount*AVERAGING_NUMBER, Rrtd_Sum, mMeasurementError, fENOB_Estimate);
       uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
       // Hold in sleep
       // Turn off forward current through RTD and Reference
       nrf_gpio_pin_clear(PIN_EXCITATION_A);
       while(true)
       {
         __WFE(); __NOP(); __NOP(); __NOP(); __NOP();
       }
    }
    

Reply
  • Signing off on this; I tested with the flying-capacitor S/H but with long leads the results are no better than the simple 2- or 3-wire presented previously. I left the code in however.

    2- or 3-wire schematic

    // RTD Measurement - 2- and 3-wire Unidirectional current
    //                          + ----------------------------------------------------------------------+
    //                          |  nRF52832/nRF52833/nRF52840                                           |
    //                          |                                                                       |
    //                          |    VDD               VDD                 VDD                          |
    //                          |   --#--             --#--               --#--                         |
    //                          |     |                 |                   |                           |
    //                          |    _|__               |   Excitation      |   Option                  |
    //                          |    / \                +-|   On            +-|   13k                   |
    //                          |   /-+-\                 |<- H0   H1         |<- Pullup                |
    //                          |     |                 +-|   160R 35R      +-|   (not used)            |
    //         Excitation  P0.04|     |                 |                   |                           |
    //         +----------------O-----#------#----------#-------------------#------------ In            |
    //         |                |     |      |          |   Excitation      |   Option                  |
    //         |                |   __|__  -----        +-|   Off           +-|   13k                   |
    //         |                |    / \   -----          |<- L0   L1         |<- Pulldown              |
    //         |                |   /-+-\    |3pF       +-|   160R 35R      +-|   (not used)            |
    //         |                |     |      |          |                   |                           |
    //         |                |   =====  =====      =====               =====                         |
    //         |                |    ===    ===        ===                 ===                          |
    //         |                |     =      =          =                   =                           |
    //         |                |                                                                       |
    //         |                |    VDD                                         VDD                    |
    //         |                |   --#--                                       --#--                   |
    //         |                |    _|__                    Auto-adjust:         |   160k              |
    //         |                |    / \                     Bias inputs at VDD/2 +-| Bias Hi/Lo/Both   |
    //         |                |   /-+-\                     VDD1_2                |<-                 |
    //         |                |     |   PIN_VOLTAGE_1                           +-|                   |
    //         |          P0.28 |     |   AIN4          |----'T'-Mux-P---|        |            P SAADC  |
    //         #--------#-------O-----#------#-------#--+ +----#-------+ +--#-----#------#------#-->    |
    //         |        |       |     |      |       |  | |    |       | |  |     |      |      |       |
    //         |        |       |   __|__  -----     |  ----   +-|    ----- |     +-|   +++     |       |
    //         |        |       |    / \   -----     |   |       |<--+  |   |       |<- | |   -----     |
    //  Rref  +++ 100nF |       |   /-+-\    | 3pF   |   |     +-|   |  |   |     +-|   | |   -----     |
    //  2k0   | |     -----     |     |      |       |   +---- |--------#   |     |     +++     | 2.5pF |
    //  1%    | |     -----     |   =====  =====     |         |     |  |   |     |      |1M0   |       |
    // (0.1%) +++       |       |    ===    ===      |       =====   |  |   |   =====  =====  =====     |
    //         |        |       |     =      =       |        ===    |  |   |    ===    ===    ===      |
    //         |        |       |                    |         =    Select  |     =      =      =       |
    //         |        |       |                    |                      |                           |
    //         |        |       |                    |                      |    VDD                    |
    //         |        |       |                    |  |----'T'-Mux-N---|  |   --#--                   |
    //         |        |       |                    +--+ +----#-------+ +----+   |   160k              |
    //         |        |       |                                           | |   +-| Bias Hi/Lo/Both   |
    //         |        |       |         PIN_VOLTAGE_2                     | |     |<-                 |
    //         |        | P0.29 |         AIN5          |----'T'-Mux-----|  | |   +-|                   |
    //         #--------#-------O-----#------#-------#--+ +----#-------+ +--# |   |            N SAADC  |
    //         |        |       |                    +--+ +----#-------+ +----#---#------#------#-->    |
    //    RTD +++ 100nF |       |                                           | |   |      |      |       |
    //        | |     -----     |                                           | |   +-|   +++     |       |
    //        | |     -----     |          Note V3 can be negative          | |     |<- | |   -----     |
    //        +++       |       |          PIN_VOLTAGE_3                    | |   +-|   | |   -----     |
    //         |        | P0.31 |          AIN7         |----'T'-Mux-----|  | |   |     +++     | 2.5pF |
    //         #--------#-------O-----#------#-------#--+ +----#-------+ +--# |   |      |1M0   |       |
    //         |                |                    +--+ +----#-------+ +----# =====  =====  =====     |
    //         |                |                                           | |  ===    ===    ===      |
    //         |                |                                           | |   =      =      =       |
    //         |        VDD     |    VDD                                    | |                         |
    //         |       --#--    |   --#--                                   | |   P: Single-ended       |
    //         |         |      |     |                                     | |   P&N: Differential     |
    //         |         +------O-----#   PIN_VOLTAGE_0                     | |                         |
    //         |                |     |   AIN_VDD       |----'T'-Mux-----|  | |                         |
    //         |                |     +-----#--------#--+ +----#-------+ +--+ |                         |
    //         |                |                    +--+ +----#-------+ +----#                         |
    //         |                |                                             |                         |
    //         |                |    GND                |----'T'-Mux-----|    |                         |
    //         #----------------O-----#-----------------+ +----#-------+ +----+                         |
    //         |                |     |                                                                 |
    //       =====              |   =====                                                               |
    //        ===               |    ===                                                                |
    //         =                |     =                                                                 |
    //                          +-----------------------------------------------------------------------+
    //
    // Capacitor min value = (3pF+2.5pF)*2^14=80nF
    // Vref = differential voltage V1 and V2 in SAADC counts
    // Vrtd = differential voltage V2 and V3 in SAADC counts
    // Rrtd = (Vrtd * Rref)/Vref Ohms
    

    Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 905.2 Ohms with 0.02% error (ENOB is 12.2)

    // Internal 0.6v Ref 14-bit S0S1 2-Wire RTD actual value 905.0 Ohm (reference 2001.4 Ohm) - Pos-Neg ADC
    // VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error
    // ====== ======= ======  ===== ===== =====  ======  ====== ====== ======
    //   2973    1008     37   2935   910    -3  3112.3  6884.7  904.8 -0.03%
    //   2973    1007     39   2933   911    -4  3110.3  6879.0  904.9 -0.01%
    //   2973    1007     45   2927   910    -3  3108.5  6881.6  904.1 -0.10%
    //   2975    1008     43   2931   911    -4  3108.5  6883.4  903.8 -0.13%
    //   2973    1007     41   2931   913    -2  3110.6  6881.3  904.7 -0.03%
    //   2973    1007     39   2933   908    -4  3108.4  6881.2  904.1 -0.10%
    //   2972    1007     38   2933   911    -2  3107.4  6882.4  903.6 -0.15%
    //   2976    1008     41   2934   911    -3  3110.3  6883.1  904.4 -0.07%
    //
    // Internal 0.6v Ref 14-bit S0S1 2-Wire RTD actual value 905.0 Ohm (reference 2001.4 Ohm) - Neg-Pos ADC
    // VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error
    // ====== ======= ======  ===== ===== =====  ======  ====== ====== ======
    //   2972    1008     39   2932   911    -5  3119.1  6888.9  906.2  0.13%
    //   2972    1009     37   2934   913    -3  3120.1  6892.4  906.0  0.11%
    //   2975    1009     39   2935   910    -2  3120.8  6890.1  906.5  0.17%
    //   2971    1009     43   2927   911    -6  3119.6  6890.3  906.1  0.12%
    //   2971    1009     38   2932   912    -3  3119.5  6891.5  906.0  0.11%
    //   2972    1009     38   2933   913    -1  3120.9  6892.0  906.3  0.14%
    //   2975    1009     41   2933   911    -4  3121.0  6894.6  906.0  0.11%
    //   2975    1009     40   2934   914    -6  3118.6  6891.1  905.7  0.08%
    //
    // Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 905.2 Ohms with 0.02% error (ENOB is 12.2)
    

    The code:

    //#define FEATURE_FLYING_CAPACITOR // Uncomment this line to use eternal S/H with TS3A5018 or similaar
    #define SAADC_RESOLUTION   SAADC_RESOLUTION_VAL_14bit  // 12- or 14-bit in this code
    #define PIN_EXCITATION_A   4  // Red lead A to remote RTD
    #define PIN_EXCITATION_B   3  // Connected via Reference resistor to Red lead B to remote RTD (4-wire)
    //      PIN_VOLTAGE_0    VDD  // VDD as source of excitation
    #define PIN_VOLTAGE_1     28  // AIN4 White lead from remote RTD (PIN_EXCITATION_A end)
    #define PIN_VOLTAGE_2     29  // AIN5 White lead from remote RTD (PIN_EXCITATION_B end)
    #define PIN_VOLTAGE_3     31  // AIN7 Red lead B to remote RTD and Reference resistor
    #define PIN_VOLTAGE_4     30  // AIN6 Reference resistor and PIN_EXCITATION_B                 (4-wire)
    #if defined(FEATURE_FLYING_CAPACITOR)
     #define PIN_SAMPLE_HOLD  25  // Flying capacitor SPDT switch controls
     #define SH_DELAY        10UL // Flying capacitor settle time, may not need
    #endif
    
    // Input range = (VDD/4V)/(1/6) = 4.5 V with VDD=3V VDD/4 Ref do not use 100nF caps
    // Input range = (0.6 V)/(1/6)  = 3.6 V with 100nF caps regardless of VDD
    // Differential mode has 1 bit less than single-ended due to sign bit
    #if (SAADC_RESOLUTION_VAL == SAADC_RESOLUTION_VAL_12bit)
     #define SAADC_COUNTS_PER_VOLT_SE   1138L  // 12-bit Mode Singe-ended:  4096/3.6V
     #define SAADC_COUNTS_PER_VOLT_DIFF  569L  // 12-bit Mode Differential: 2048/3.6V
     const uint8_t SAADC_ResolutionStr[] = "12-bit";
    #elif (SAADC_RESOLUTION == SAADC_RESOLUTION_VAL_14bit)
     #define SAADC_COUNTS_PER_VOLT_SE   4551L  // 14-bit Mode Singe-ended: 16384/3.6V
    // #define SAADC_COUNTS_PER_VOLT_DIFF 2276L  // 14-bit Mode Differential: 8192/3.6V
     #define SAADC_COUNTS_PER_VOLT_DIFF 3413L  // 14-bit Mode Differential: 8192/2.4V
     const uint8_t SAADC_ResolutionStr[] = "14-bit";
    #else
     #error SAADC_RESOLUTION  not 12- or 14-bit in this code
    #endif
    
    // Do an initial calibration as the offset shows up in comparing +ve and -ve current measurements
    static bool mAmCalibrated = false;
    static float mMeasurementError = 0.0f;
    static uint32_t mSaadcSampleCount = 1UL;
    static void RequestCalibration(void)
    {
       // Request a recalibration on any gain or mode changes
       mAmCalibrated = false;
    }
    
    // The accumulator is controlled in the OVERSAMPLE register. When using oversampling, 2^OVERSAMPLE input
    // samples are averaged before the sample result is transferred to memory. Hence, the SAMPLE task must be
    // triggered 2^OVERSAMPLE times for each output value
    //
    // SAADC on nRF54L15
    // =================
    // CH[n].CONFIG.BURST can be enabled to avoid setting SAMPLE task 2^OVERSAMPLE times. With BURST = 1 the
    // ADC will sample the input 2^OVERSAMPLE times as fast as it can (actual timing: <(tACQ+tCONV)*2^OVERSAMPLE).
    // ie similar to one-shot mode
    // Noise shaping is implemented by using the SAR-ADC in a first-order delta-sigma loop and the filtering
    // is done with the use of FIR filters. The sampling rate in these modes is 1MS/s, and only 12 and 14 bit
    // resolution are supported. To enable noise shaping use NOISESHAPE
    // The ADC can use different reference voltages VREF, controlled in the REFSEL field of the CH[n].CONFIG.
    // These are:
    //   Internal reference, VREF = 0.9 V
    //   External reference, VREF provided by the EXTREF pin
    //    Note: The external reference voltage should be close the internal reference voltage, typ 1.2V
    static uint32_t mSAADC_InterruptCount = 0UL;
    void SAADC_IRQHandler(void)
    {
       NRF_SAADC->EVENTS_END = 0;
       __DSB();
       mSAADC_InterruptCount++;
    }
    static int32_t GetRTD_Voltage(uint32_t AnalogueChannelP, uint32_t AnalogueChannelN)
    {
       int32_t result = 9999L;
       uint32_t timeout = 512UL;
       volatile int16_t buffer[8] = {0};
       uint32_t i=0;  // SAADC Channel
    #if defined(FEATURE_FLYING_CAPACITOR)
       // Short arbitrary delay to allow sample 100nF to charge
       //for (uint32_t i=0; i<SH_DELAY; i++) {}
       // Connnect Flying Capacitor SPDT switch to SAADC - NO contacts
       nrf_gpio_pin_set(PIN_SAMPLE_HOLD);
    #endif
       // Can use internal ref SAADC_CH_CONFIG_REFSEL_Internal with 100nF caps or VDD ratiometric SAADC_CH_CONFIG_REFSEL_VDD1_4 with no caps
       // Check if single-ended or differential SAADC mode
       if (AnalogueChannelN == SAADC_CH_PSELN_PSELN_NC)
       {
          // Configure SAADC singled-ended channel, Internal reference (0.6V) and 1/6 gain (range 3.6V), no pull-up or pull-down
          // nRF54L15 has different settings ToDo
          NRF_SAADC->CH[i].CONFIG = (SAADC_CH_CONFIG_GAIN_Gain1_6    << SAADC_CH_CONFIG_GAIN_Pos)   |
                                    (SAADC_CH_CONFIG_MODE_SE         << SAADC_CH_CONFIG_MODE_Pos)   |
                                    (SAADC_CH_CONFIG_REFSEL_Internal << SAADC_CH_CONFIG_REFSEL_Pos) |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESN_Pos)   |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESP_Pos)   |
                                    (SAADC_CH_CONFIG_TACQ_3us        << SAADC_CH_CONFIG_TACQ_Pos);
       }
       else
       {
          // Configure SAADC differential channel, Internal reference (0.6V) and 1/4 gain (range 2.4V), no pull-up or pull-down
          NRF_SAADC->CH[i].CONFIG = (SAADC_CH_CONFIG_GAIN_Gain1_4    << SAADC_CH_CONFIG_GAIN_Pos)   |
                                    (SAADC_CH_CONFIG_MODE_Diff       << SAADC_CH_CONFIG_MODE_Pos)   |
                                    (SAADC_CH_CONFIG_REFSEL_Internal << SAADC_CH_CONFIG_REFSEL_Pos) |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESN_Pos)   |
                                    (SAADC_CH_CONFIG_RESN_Bypass     << SAADC_CH_CONFIG_RESP_Pos)   |
                                    (SAADC_CH_CONFIG_TACQ_3us        << SAADC_CH_CONFIG_TACQ_Pos);
       }
       NRF_SAADC->RESOLUTION = SAADC_RESOLUTION << SAADC_RESOLUTION_VAL_Pos;
       // Analog positive and negative input channels have strange numbering:
       //  PSELP_NC           0UL  Not connected
       //  PSELP_AnalogInput0 1UL  AIN0
       //  PSELP_AnalogInput1 2UL  AIN1
       //  PSELP_AnalogInput2 3UL  AIN2
       //  PSELP_AnalogInput3 4UL  AIN3
       //  PSELP_AnalogInput4 5UL  AIN4
       //  PSELP_AnalogInput5 6UL  AIN5
       //  PSELP_AnalogInput6 7UL  AIN6
       //  PSELP_AnalogInput7 8UL  AIN7
       //  PSELP_VDD          9UL  VDD
       NRF_SAADC->CH[i].PSELP = AnalogueChannelP << SAADC_CH_PSELP_PSELP_Pos;
       NRF_SAADC->CH[i].PSELN = AnalogueChannelN << SAADC_CH_PSELN_PSELN_Pos;
       // Note the SAMPLE task must be triggered 2^OVERSAMPLE times for each output value
       NRF_SAADC->OVERSAMPLE = SAADC_OVERSAMPLE_OVERSAMPLE_Over8x;
       // Setup interrupts if required
       NRF_SAADC->INTENSET = SAADC_INTENSET_END_Set << SAADC_INTENSET_END_Pos;
       NVIC_SetPriority(SAADC_IRQn, 6);
       NVIC_ClearPendingIRQ(SAADC_IRQn);
       NVIC_EnableIRQ(SAADC_IRQn);
       // Enable SAADC
       NRF_SAADC->ENABLE = 1;
       NRF_SAADC->RESULT.PTR = (uint32_t)buffer;
       NRF_SAADC->RESULT.MAXCNT = 1;
       //NRF_SAADC->RESULT.AMOUNT
       // Do an initial calibration as the offset shows up in comparing +ve and -ve current measurements
       // Any gain changes require recalibration. This reduces but does not eliminate SAADC offset
       if(!mAmCalibrated)
       {
          NRF_SAADC->EVENTS_CALIBRATEDONE = 0;
          NRF_SAADC->TASKS_CALIBRATEOFFSET = 1;
          while (!NRF_SAADC->EVENTS_CALIBRATEDONE) ;
          mAmCalibrated = true;
          // Clear event, though maybe not actually necessary
          NRF_SAADC->EVENTS_CALIBRATEDONE = 0;
       }
       NRF_SAADC->EVENTS_END = 0;
       NRF_SAADC->EVENTS_DONE = 0;
       NRF_SAADC->EVENTS_RESULTDONE = 0;
       NRF_SAADC->TASKS_START = 1;
       // Assume could be up to 256 samples, but don't bother to check actual number
       mSaadcSampleCount = 0;
       while(true)
       {
          NRF_SAADC->TASKS_SAMPLE = 1;
          __DSB();
          while(!NRF_SAADC->EVENTS_DONE) ;
          // Update sample count for reports
          mSaadcSampleCount++;
          if (NRF_SAADC->EVENTS_RESULTDONE || timeout == 0) break;
          NRF_SAADC->EVENTS_DONE = 0;
          timeout--;
       }
       // Don't need NRF_SAADC->EVENTS_END for now
       NRF_SAADC->TASKS_STOP = 1;
       NRF_SAADC->EVENTS_STARTED = 0;
       NRF_SAADC->EVENTS_END = 0;
       NRF_SAADC->EVENTS_DONE = 0;
       NRF_SAADC->EVENTS_RESULTDONE = 0;
       // Disable command
       NRF_SAADC->ENABLE = 0;
    #if defined(FEATURE_FLYING_CAPACITOR)
       // Connnect Flying Capacitor SPDT switch to Rrtd and Rref - NC contacts
       nrf_gpio_pin_clear(PIN_SAMPLE_HOLD);
    #endif
       // Process result
       if (AnalogueChannelN == SAADC_CH_PSELN_PSELN_NC)
       {
          // Display in mVolts, single-ended values are only for info
          if (timeout != 0)
          {
             // Calculate mVolt value with rounding, later assume convert to 16-bit signed for prints
             result = (((int32_t)buffer[0] * 1000L)+(SAADC_COUNTS_PER_VOLT_SE)/2L) / (SAADC_COUNTS_PER_VOLT_SE);
          }
          return result;
       }
       else
       {
          // Display and calculate differential values in signed SAADC counts for best resolution
          return (int32_t)buffer[0];
       }
    }
    
    #define AVERAGING_NUMBER  16 // Averaging now also done with SAADC oversampling
    #define REPORTS_NUMBER    16 // Numer of printed results for analysis
    const float Rref_Ohm  = 2001.4f;  // Ohm
    const float Rtest_Ohm =  905.0f;  //  905.0 Ohm, 1597.3 Ohm, 2984.6 Ohm.
    float Rrtd     = 0.0f;
    float Rrtd_Sum = 0.0f;
    float Vrtd_f = 0.0f;
    float Vref_f = 0.0f;
    
    #include <math.h>
    // x^y = pow(x,y)
    inline int root(int input, int n)
    {
       return round(pow(input, 1./n));
    }
    
    void Test3WireRTD_ResistanceMeasurement(void)
    {
       char RtdInfoPacket[240] = "?";
       int16_t BatteryVolts0 = 0, BatteryVolts1 = 0, BatteryVolts2 = 0, BatteryVolts3 = 0;
       // Take some differential samples normal and then some samples with reversed SAADC connection
       for (uint32_t SampleCount=0; SampleCount<REPORTS_NUMBER; SampleCount++)
       {
          int32_t Vrtd = 0;
          int32_t Vref = 0;
          uint32_t FET_RdsOnA = 1;
          // Configuration               Direction                Input                          Pullup                 Drive Level        Sense Level
          // ==========================  =======================  =============================  =====================  =================  =======================
          nrf_gpio_cfg(PIN_EXCITATION_A, NRF_GPIO_PIN_DIR_OUTPUT, NRF_GPIO_PIN_INPUT_DISCONNECT, NRF_GPIO_PIN_NOPULL,   NRF_GPIO_PIN_H0H1, NRF_GPIO_PIN_NOSENSE);
          // Turn on forward current through RTD and Reference
          nrf_gpio_pin_set(PIN_EXCITATION_A);
    #if defined(FEATURE_FLYING_CAPACITOR)
          nrf_gpio_cfg(PIN_SAMPLE_HOLD,  NRF_GPIO_PIN_DIR_OUTPUT, NRF_GPIO_PIN_INPUT_DISCONNECT, NRF_GPIO_PIN_NOPULL,   NRF_GPIO_PIN_S0S1, NRF_GPIO_PIN_NOSENSE);
          // Connnect Flying Capacitor SPDT switch to Rrtd and Rref - NC contacts
          nrf_gpio_pin_clear(PIN_SAMPLE_HOLD);
    #endif
          if (SampleCount == 0)
          {
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "\r\n// Internal 0.6v Ref %s S0S1 2-Wire RTD actual value %0.1f Ohm (reference %0.1f Ohm) - Pos-Neg ADC\r\n", SAADC_ResolutionStr, Rtest_Ohm, Rref_Ohm);
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// ====== ======= ======  ===== ===== =====  ======  ====== ====== ======\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
          }
          else if (SampleCount == REPORTS_NUMBER/2)
          {
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "//\r\n// Internal 0.6v Ref %s S0S1 2-Wire RTD actual value %0.1f Ohm (reference %0.1f Ohm) - Neg-Pos ADC\r\n", SAADC_ResolutionStr, Rtest_Ohm, Rref_Ohm);
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// VDD mV Iref uA  RdsOn  V1 mV V2 mV V3 mV    Vrtd    Vref   Rrtd  Error\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
             snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "// ====== ======= ======  ===== ===== =====  ======  ====== ====== ======\r\n");
             uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
          }
          // Differential RTD and reference resistor voltage values
          Vrtd = 0;
          Vref = 0;
          // Request a recalibration on any gain or mode changes
          RequestCalibration();
          for (uint32_t i=0; i<AVERAGING_NUMBER; i++)
          {
             // Differential measurements Pos-Neg or Neg-Pos
             if (SampleCount < REPORTS_NUMBER/2)
             {
                Vrtd += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput5, SAADC_CH_PSELN_PSELN_AnalogInput7);
                Vref += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput4, SAADC_CH_PSELN_PSELN_AnalogInput5);
             }
             else
             {
                Vrtd += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput7, SAADC_CH_PSELN_PSELN_AnalogInput5);
                Vref += GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput5, SAADC_CH_PSELN_PSELN_AnalogInput4);
             }
          }
          Vrtd_f = Vrtd /(float)AVERAGING_NUMBER;
          Vref_f = Vref /(float)AVERAGING_NUMBER;
          // Keep using integer value for current
          Vref /= AVERAGING_NUMBER;
          // Differential measurements Pos-Neg or Neg-Pos, compensate for pseudo-negative
          if (SampleCount >= REPORTS_NUMBER/2)
          {
             Vrtd_f = 0.0f - Vrtd_f;
             Vref_f = 0.0f - Vref_f;
             Vref   = 0 - Vref;
          }
          // Request a recalibration on any gain or mode changes
          RequestCalibration();
          // Single-ended measurement voltages for testing
          BatteryVolts0 = (int16_t)GetRTD_Voltage(SAADC_CH_PSELP_PSELP_VDD, SAADC_CH_PSELN_PSELN_NC);
          BatteryVolts1 = (int16_t)GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput4, SAADC_CH_PSELN_PSELN_NC);
          BatteryVolts2 = (int16_t)GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput5, SAADC_CH_PSELN_PSELN_NC);
          BatteryVolts3 = (int16_t)GetRTD_Voltage(SAADC_CH_PSELP_PSELP_AnalogInput7, SAADC_CH_PSELN_PSELN_NC);
          // Rrtd = (Vrtd_f * Rref)/Vref Ohms
          float Rrtd = (Vrtd_f * Rref_Ohm)/Vref_f;
          // Accumulate for sum of both forward and reverse measurements
          Rrtd_Sum += Rrtd;
          // Calculate % error
          mMeasurementError = (Rrtd - Rtest_Ohm) * 100.0f/Rtest_Ohm;
          // Calculate mVolt value of Vref using rounding, differential mode
          uint32_t Vref_mV = (((Vref * 1000)+(SAADC_COUNTS_PER_VOLT_DIFF/2)) / SAADC_COUNTS_PER_VOLT_DIFF);
          // Current Iref = Vref/Rref  uA
          uint32_t Iref_uA = (uint32_t)roundf((Vref_mV * 1000)/Rref_Ohm);
          // RdsOn = Vref_mV/Iref_uA Ohms
          FET_RdsOnA = ((BatteryVolts0 - BatteryVolts1) * 1000) / Iref_uA;
          snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "//  %5hd   %5u  %5u  %5hd %5hd %5hd  %6.1f  %6.1f %6.1f % 0.2f%%\r\n", BatteryVolts0, Iref_uA, FET_RdsOnA, BatteryVolts1, BatteryVolts2, BatteryVolts3, Vrtd_f, Vref_f, Rrtd, mMeasurementError);
          uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
       }
       // Accumulate for sum of both forward and reverse measurements
       Rrtd_Sum /= (float)REPORTS_NUMBER;
       // Calculate % error
       mMeasurementError = (Rrtd_Sum - Rtest_Ohm) * 100.0f/Rtest_Ohm;
       // Check ENOB using x^y = pow(x,y)
       float fENOB_Estimate, fIteration = 0.01f, fLimit = 100.0f/mMeasurementError;
       if (fLimit < 0.0F) fLimit = 0.0f - fLimit;
       for (fENOB_Estimate=10.0f; fENOB_Estimate<=14.0f; fENOB_Estimate += fIteration)
       {
          if (pow(2.0f, fENOB_Estimate) > fLimit)
          {
              fENOB_Estimate -= fIteration;
              break;
          }
       }
       // Measured 14-bit value of 905.0 Ohm 2-Wire test resistor using 128 samples is 905.2 Ohms with 0.02% error (ENOB is 12.1)
       snprintf(RtdInfoPacket, sizeof(RtdInfoPacket)-1, "//\r\n// Measured %s value of %4.1f Ohm 2-Wire test resistor using %u samples is %4.1f Ohms with %0.2f%% error (ENOB is %0.1f)\r\n", SAADC_ResolutionStr, Rtest_Ohm, mSaadcSampleCount*AVERAGING_NUMBER, Rrtd_Sum, mMeasurementError, fENOB_Estimate);
       uartSend(RtdInfoPacket, strlen(RtdInfoPacket));
       // Hold in sleep
       // Turn off forward current through RTD and Reference
       nrf_gpio_pin_clear(PIN_EXCITATION_A);
       while(true)
       {
         __WFE(); __NOP(); __NOP(); __NOP(); __NOP();
       }
    }
    

Children
No Data
Related