This post is older than 2 years and might not be relevant anymore
More Info: Consider searching for newer posts

SAADC Configuration

Hi All,

I am developing one project and there we are using ADC (i.e., SAADC in nRF). I have designed the hardware where I am using Capacitor ( C14, 2.2µF) which is connected to Analog pin and my requirement is I have to read the voltage level.

Here I am attaching details and image,

The voltage level of the input capacitor C14, 2.2µF / 400V to the Step-Down controller must be monitored. A voltage divider is used across the capacitor from R2, 3M3, and R3, 24k. The tapped voltage is measured at pin port P0.04 / AIN2 of the nRF52840. The expected voltage at the input of the controller is between 2.35V and 0V. The voltage at the capacitor must not fall below the value of 100V.

This is my requirement can anyone help how to configure SAADC with respect to my requirement.

Thanks in advance

Rohit R

Parents
  • Hello Rohit R,

    Have you seen the formulas for calculating SAADC input range in the SAADC documentation?

    The expected voltage at the input of the controller is between 2.35V and 0V.
    This is my requirement can anyone help how to configure SAADC with respect to my requirement.

    How is your nRF52840 supplies with power, are you using the coin cell battery, USB or external power?
    In the case of USB / 3 V power:
    If you would like your SAADC to have a the input range 0 - 2.35 V, you could either use reference VDD/4 and gain 1/3, which yields the input range 0 - 2.27 V, or you could use VDD/4 and gain 1/4 and get 0 - 3 V.
    Have you seen the SAADC Example from the SDK? It demonstrates how to setup and use the SAADC - then you could just input your own configuration, and you should start seeing your measurements.

    Please let me know if anything should be unclear with the forumlas, or how to go about doing the configuration.

    Best regards,
    Karl

  • Hi Karl,

    Yes, I have seen the document and SDK examples too.

    I am modifying SDK example as per my requirement, I have shared the changes in the previous post.

    How is your nRF52840 supplies with power, are you using the coin cell battery, USB or external power?
    In the case of USB / 3 V power:

    - It is basically a light Dimming control circuit we are using 230V and then using the voltage regulator and divider we are powering up MCU with 3.3V. If you make this ticket private then I can share the schematic file here as it confidential.

    If you would like your SAADC to have a the input range 0 - 2.35 V, you could either use reference VDD/4 and gain 1/3, which yields the input range 0 - 2.27 V, or you could use VDD/4 and gain 1/4 and get 0 - 3 V.

    - Okay, let me try this configuration related Gain and check the result.

    - One more question, I am using P0.04 / AIN2, with respect to SDK example as per my understanding AIN2 means the channel 2 and to configure this we need to call,

    nrf_saadc_channel_config_t channel_config =
            NRF_DRV_SAADC_DEFAULT_CHANNEL_CONFIG_SE(NRF_SAADC_INPUT_AIN2);  //single shot and channel config

    I am correct in configuring pin or did I miss something.

    Correct me If i am doing anything wrong.

    Thanks and Regards

    Rohit R

  • Hello Rohit,

    Rohit Rajapure said:
    I will note your suggestion and look into it.

    Great! Please do not hesitate to let me know if anything should be unclear.

    Rohit Rajapure said:

    The voltage level of the input capacitor C14, 2.2µF / 400V to the Step-Down controller must be monitored. A voltage divider is used across the capacitor from R2, 3M3, and R3, 24k. The tapped voltage is measured at pin port P0.04 / AIN2 of the nRF52840. The expected voltage at the input of the controller is between 2.35V and 0V. The voltage at the capacitor must not fall below the value of 100V.

    Capacitor ( C14, 2.2µF) which is connected to Analog pin and my requirement is I have to read the voltage level.

    Yes, like mentioned in the original ticket, but what is your required resolution and sampling frequency?
    If you only need to know when the voltage crosses a certain threshold then you might want to look into using the SAADC limits feature. 

    Best regards,
    Karl

  • Hi Karl,

    As you suggested I tried configuring my custom requirement in SDK example.

    My configurations,

        nrf_saadc_channel_config_t channel_config =
            NRF_DRV_SAADC_DEFAULT_CHANNEL_CONFIG_SE(NRF_SAADC_INPUT_AIN2);  //singale shot and channel config 
            
       channel_config.gain = NRF_SAADC_GAIN1_4;
       channel_config.reference = NRF_SAADC_REFERENCE_VDD4;
       channel_config.mode = NRF_SAADC_MODE_SINGLE_ENDED;
       channel_config.pin_n = NRF_SAADC_INPUT_DISABLED; 
       channel_config.pin_p = NRF_SAADC_INPUT_AIN2;
       channel_config.resistor_n = NRF_SAADC_RESISTOR_DISABLED;
       channel_config.resistor_p = NRF_SAADC_RESISTOR_DISABLED;

    After making changes in SDK example with the above configuration. When I followed the testing steps suggested for SDK example by Connecting pin to VDD and GND, I am still getting 10bit raw value. But I have configured for 12bit.

    Once I get exact raw values then I can apply voltage converter calculation as mentioned by you in the previous post.

    And one more thing I would like share is, my end device is Zigbee based Dimming control hardware. The selection of intensity levels is depended on the ADC values of the capacitor. More details you check mine below ticket where I have shared my schematic and details. This hardware is basically powered by 230V then using internal voltage converter we make 3V to power up nRF MCU.

    Please go through schematic also and let me know if I am making any mistakes in configuring SAADC.

    ticket link - https://devzone.nordicsemi.com/support/255201

    Thanks in advance

    Rohit R

  • Hello Rohit R,

    Rohit Rajapure said:
    My configurations,

    This is fine, with your code, you configure AIN2 channel to have 1/4 gain, VDD/4 reference and 10 us TAQ.
    Be advised that the NRF_DRV_SAADC_DEFAULT_CHANNEL_CONFIG_SE is a macro for setting up the default channel configurations. You may then change those that you do not want to be default. In your case, you are changing some things to the value that they already hold, which is why I am mentioning this.

    Rohit Rajapure said:
    After making changes in SDK example with the above configuration. When I followed the testing steps suggested for SDK example by Connecting pin to VDD and GND, I am still getting 10bit raw value. But I have configured for 12bit.

     The SAADC example is using 10 bit resolution as default. Could you show me where you have changed this to 12 bit? Also, this is not done as part of the channel configuration, but rather as part of the SAADC configuration.

    Rohit Rajapure said:
    Once I get exact raw values then I can apply voltage converter calculation as mentioned by you in the previous post.

    Yes, lets ensure that you are getting the correct raw SAADC output values first. This is a good approach.

    Rohit Rajapure said:
    Please go through schematic also and let me know if I am making any mistakes in configuring SAADC.

    What specifically would you like me to take a look at in your schematics / in your other ticket?
    It seems to me that you are getting good help by my colleague Jørgen in the other ticket.

    As for the SAADC configuration, you still have not told me what your required resolution for the project is, and what your intended sampling frequency is.

    Best regards,
    Karl

  • Hi,

    Be advised that the NRF_DRV_SAADC_DEFAULT_CHANNEL_CONFIG_SE is a macro for setting up the default channel configurations. You may then change those that you do not want to be default. In your case, you are changing some things to the value that they already hold, which is why I am mentioning this.

    - I did configuration as below,

      nrf_saadc_channel_config_t channel_config;     
       channel_config.gain = NRF_SAADC_GAIN1_4;
       channel_config.reference = NRF_SAADC_REFERENCE_VDD4;
       channel_config.mode = NRF_SAADC_MODE_SINGLE_ENDED;
       channel_config.pin_n = NRF_SAADC_INPUT_DISABLED; 
       channel_config.pin_p = NRF_SAADC_INPUT_AIN2;
       channel_config.resistor_n = NRF_SAADC_RESISTOR_DISABLED;
       channel_config.resistor_p = NRF_SAADC_RESISTOR_DISABLED;
    

     The SAADC example is using 10 bit resolution as default. Could you show me where you have changed this to 12 bit? Also, this is not done as part of the channel configuration, but rather as part of the SAADC configuration.

    - I set resolution here and in sdk_config.h file to

        nrf_drv_saadc_config_t saadc_config;
        saadc_config.resolution = NRF_SAADC_RESOLUTION_12BIT;   //12 bit resolution

    // <0=> 8 bit 
    // <1=> 10 bit 
    // <2=> 12 bit 
    // <3=> 14 bit 
    
    #ifndef NRFX_SAADC_CONFIG_RESOLUTION
    #define NRFX_SAADC_CONFIG_RESOLUTION 2
    #endif
    

    What specifically would you like me to take a look at in your schematics / in your other ticket?
    It seems to me that you are getting good help by my colleague Jørgen in the other ticket.

    - In the previous post, you had asked for the powering procedure of the device. So I shared this ticket and asked to go through the schematic more clear picture about powering MCU and ADC input pin related details.

    As for the SAADC configuration, you still have not told me what your required resolution for the project is, and what your intended sampling frequency is.

    - What I was informed is by a customer about capacitor voltage should be as below,

    "a voltage of capacitor<100V  (This corresponds to a voltage of 0.72V on

      Analog input of the nRF528)"

    - So I need an 8bit resolution sufficient for me but I was trying with 12bit and sampling frequency 50Hz.

    - In both 8bit and 12bit resolution also I am getting 10bit values 1023 when connecting to VDD.

    Please correct If am doing wrong, As my capacitor pin is connected to AIN2/P0_4 so I have to short P0_4 to VDD/GND to check values correct?

    As mentioned above snippet, I set the resolution as 12bit in both files still I am getting 1023 raw values when I connect to VDD.

    Can you let me know your views on this?

    Thanks and Regards

    Rohit R

  • Hi Karl,

    It's my fault. I have resolved the resolution issue.

    I am getting 8bit raw values and able to print on putty.

    Now, my next step is to convert  into voltage range.

        err_code = nrf_drv_saadc_init(&saadc_config, saadc_callback);
        APP_ERROR_CHECK(err_code);

    - I have forgot to call saadc_config parameter in init function. I kept it NULL.

    it's my fault.

    Thanks and Regards

    Rohit R

Reply Children
  • Hello Rohit,

    Rohit Rajapure said:
    I have resolved the resolution issue.

    I am happy to hear that you were able to resolve your issue!

    Rohit Rajapure said:
    I am getting 8bit raw values and able to print on putty.

    As I had understood it, you needed 12 bit resolution. Are you able to reconfigure to 12 bit resolution now?

    Rohit Rajapure said:
    - I have forgot to call saadc_config parameter in init function. I kept it NULL.

    I see. To avoid this in the future, I would advise that you remove obsolete code from your project instead of commenting it out - this increases readability tenfold.
    If you are worried that you will need the commented out code later, place it in a temporary work file.

    Please do not hesitate to open a new ticket if you should encounter any other issues or questions in the future.

    Good luck with your development!

    Best regards,
    Karl

  • Hi Karl,

    Yes, I am able to read all resolution values.

    8bit resolution is sufficient for me so I am going ahead with an 8bit resolution.

    Thank you so much for your constant help.

    Thanks and Regards

    Rohit R

  • Hello Rohit,

    Rohit Rajapure said:
    Yes, I am able to read all resolution values.

    Great, I am glad to hear that! 

    Rohit Rajapure said:
    8bit resolution is sufficient for me so I am going ahead with an 8bit resolution.

    All right, that is great! Please keep in mind that you might loose some LSB to noise in your measurements.
    So, if you need 8 NOEB, then I would suggest that you use 10 bit resolution.

    Rohit Rajapure said:
    Thank you so much for your constant help.

    It is no problem at all, I am happy to help!

    Please do not hesitate to open a new ticket if you should encounter any issues or questions in the future.

    Good luck with your development!

    Best regards,
    Karl

  • Hi Karl,

    I need one help,

    I am using saadc_sampling_trigger(); trigger the ADC samples. But I am getting 1 sample.

    I have tried by changing the buffer size than my code not working.

    What I need is, I want to read multiple samples when I trigger ADC?

    Can you please help how to do that? below is my code let me know what I did wrong.

    /**
     * Copyright (c) 2014 - 2019, Nordic Semiconductor ASA
     *
     * All rights reserved.
     *
     * Redistribution and use in source and binary forms, with or without modification,
     * are permitted provided that the following conditions are met:
     *
     * 1. Redistributions of source code must retain the above copyright notice, this
     *    list of conditions and the following disclaimer.
     *
     * 2. Redistributions in binary form, except as embedded into a Nordic
     *    Semiconductor ASA integrated circuit in a product or a software update for
     *    such product, must reproduce the above copyright notice, this list of
     *    conditions and the following disclaimer in the documentation and/or other
     *    materials provided with the distribution.
     *
     * 3. Neither the name of Nordic Semiconductor ASA nor the names of its
     *    contributors may be used to endorse or promote products derived from this
     *    software without specific prior written permission.
     *
     * 4. This software, with or without modification, must only be used with a
     *    Nordic Semiconductor ASA integrated circuit.
     *
     * 5. Any software provided in binary form under this license must not be reverse
     *    engineered, decompiled, modified and/or disassembled.
     *
     * THIS SOFTWARE IS PROVIDED BY NORDIC SEMICONDUCTOR ASA "AS IS" AND ANY EXPRESS
     * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
     * OF MERCHANTABILITY, NONINFRINGEMENT, AND FITNESS FOR A PARTICULAR PURPOSE ARE
     * DISCLAIMED. IN NO EVENT SHALL NORDIC SEMICONDUCTOR ASA OR CONTRIBUTORS BE
     * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
     * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
     * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     *
     */
    /** @file
     * @defgroup nrf_adc_example main.c
     * @{
     * @ingroup nrf_adc_example
     * @brief ADC Example Application main file.
     *
     * This file contains the source code for a sample application using ADC.
     *
     * @image html example_board_setup_a.jpg "Use board setup A for this example."
     */
    
    #include <stdbool.h>
    #include <stdint.h>
    #include <stdio.h>
    #include <string.h>
    #include "nrf.h"
    #include "nrf_drv_saadc.h"
    #include "nrf_drv_ppi.h"
    #include "nrf_drv_timer.h"
    #include "boards.h"
    #include "app_error.h"
    #include "nrf_delay.h"
    #include "app_util_platform.h"
    #include "nrf_pwr_mgmt.h"
    
    #include "nrf_log.h"
    #include "nrf_log_ctrl.h"
    #include "nrf_log_default_backends.h"
    
    
    #define SAMPLES_IN_BUFFER 1
    
    volatile uint8_t state = 1;
    
    static const nrf_drv_timer_t m_timer = NRF_DRV_TIMER_INSTANCE(0);
    static nrf_saadc_value_t     m_buffer_pool[1][SAMPLES_IN_BUFFER];
    static nrf_ppi_channel_t     m_ppi_channel;
    static uint32_t              m_adc_evt_counter;
    
    bool boolEventHandled = true;
    
    void saadc_callback(nrf_drv_saadc_evt_t const * p_event)
    {
        if (p_event->type == NRF_DRV_SAADC_EVT_DONE)
        {
            ret_code_t err_code;
            //ui16CurrentADCValue = p_event->data.done.p_buffer[0]; 
             
            err_code = nrf_drv_saadc_buffer_convert(p_event->data.done.p_buffer, SAMPLES_IN_BUFFER);
            APP_ERROR_CHECK(err_code);
    
            int i;
            NRF_LOG_INFO("ADC event number: %d", (int)m_adc_evt_counter);
    
            for (i = 0; i < SAMPLES_IN_BUFFER; i++)
            {
                NRF_LOG_INFO("%d",p_event->data.done.p_buffer[i]);
            }
            m_adc_evt_counter++;
    
            boolEventHandled = true;
        }
    
    }
    
    
    void saadc_init(void)
    {
            ret_code_t err_code;
        nrf_drv_saadc_config_t saadc_config;
        saadc_config.resolution = NRF_SAADC_RESOLUTION_8BIT;   //8 bit resolution
        saadc_config.interrupt_priority = APP_IRQ_PRIORITY_LOW;
        
        err_code = nrf_drv_saadc_init(&saadc_config, saadc_callback);
        APP_ERROR_CHECK(err_code);
        
       //singale shot and channel config 
       nrf_saadc_channel_config_t channel_config;
       channel_config.reference = NRF_SAADC_REFERENCE_INTERNAL;
       channel_config.gain = NRF_SAADC_GAIN1_4;
       channel_config.acq_time = NRF_SAADC_ACQTIME_10US;
       channel_config.mode = NRF_SAADC_MODE_SINGLE_ENDED;
       channel_config.pin_n = NRF_SAADC_INPUT_DISABLED; 
       channel_config.pin_p = NRF_SAADC_INPUT_AIN2;
       channel_config.resistor_n = NRF_SAADC_RESISTOR_DISABLED;
       channel_config.resistor_p = NRF_SAADC_RESISTOR_DISABLED;
    
        err_code = nrf_drv_saadc_channel_init(2, &channel_config);
        APP_ERROR_CHECK(err_code);
    
        err_code = nrf_drv_saadc_buffer_convert(&m_buffer_pool[0], SAMPLES_IN_BUFFER);
        APP_ERROR_CHECK(err_code);
       
    }
    
    void saadc_sampling_trigger(void)
    {
        ret_code_t err_code;
        err_code = nrf_drv_saadc_sample(); // Check error
        APP_ERROR_CHECK(err_code);
        boolEventHandled = false;
    }
    
    /**
     * @brief Function for main application entry.
     */
    int main(void)
    {
        uint32_t err_code = NRF_LOG_INIT(NULL);
        APP_ERROR_CHECK(err_code);
    
        NRF_LOG_DEFAULT_BACKENDS_INIT();
    
        saadc_init();
    
        NRF_LOG_INFO("SAADC HAL simple example started.");
    
    
        while (1)
        {   
            if(boolEventHandled)
            {
              saadc_sampling_trigger();
            }
            nrf_delay_ms(1000);
            NRF_LOG_FLUSH();
        }
    }
    
    
    /** @} */
    

    Thanks and Regards

    Rohit R

  • Hello Rohit R,

    Rohit Rajapure said:

    I am using saadc_sampling_trigger(); trigger the ADC samples. But I am getting 1 sample.

    I have tried by changing the buffer size than my code not working.

    The nrfx_saadc_sample() function triggers the START task or SAMPLE task register, depending on weather the driver is configured in low power mode or not. If you want to use a timer to trigger ADC sampling yo could trigger the task directly via PPI instead. This way you get precise timing, as it happens in HW and won't be delayed by a potential high priority interrupt.
    If I have misunderstood you, and you would just like to get a single sample by calling a single function, you should instead use the nrfx_saadc_sample_convert function for this.

    Rohit Rajapure said:
    What I need is, I want to read multiple samples when I trigger ADC?

    Have you seen the SAADC example from the SDK? It demonstrates how to set up the SAADC's sampling to trigger by a timer through PPI.
    What are your requirements for the SAADC - do you intend to perform a sampling ( or multiple samples ? ) each second, have I understood this correct?
    If you are doing a periodic sampling, I would highly recommend setting it to trigger by a timer through PPI, as demonstrated in the SDK example.

    On a general note, I would also recommend that you do some power management in your main while loop, so as to reduce power consumption.
    In most examples - including the SAADC peripheral example - you can see this implemented as the idle_state_handler function.
    Currently, it seems that you are using a 1000 ms delay in the main loop to time your measurements - this is not recommended, as the delay NOP CPU instructions are wasted, and the delay function will not give you an exact delay, if you intend to add BLE communication to your project later on. 

    Best regards,
    Karl

Related