Beware that this post is related to an SDK in maintenance mode
More Info: Consider nRF Connect SDK for new designs
This post is older than 2 years and might not be relevant anymore
More Info: Consider searching for newer posts

nrf_ringbuf bugs

nrf_ringbuf has many logic bugs that cause it to both corrupt internal structures and return invalid data.  For example one of these calls to nrf_ringbuf_alloc tells me that I can write 100 bytes to a 4-byte buffer:

int i;
uint8_t *data;
size_t len;

NRF_RINGBUF_DEF(ring, 4);
nrf_ringbuf_init(&ring);

for (i = 0; i < 10; i++)
{
    // allocate and put one byte                                                                                                                                                                            
    len = 100;
    ASSERT(nrf_ringbuf_alloc(&ring, &data, &len, true) == NRF_SUCCESS);
    ASSERT(len <= 4);
    ASSERT(nrf_ringbuf_put(&ring, 1) == NRF_SUCCESS);
    
    // allocate and put zero bytes                                                                                                                                                                          
    len = 100;
    ASSERT(nrf_ringbuf_alloc(&ring, &data, &len, true) == NRF_SUCCESS);
    ASSERT(len <= 4); // <------- assert fails, len == 100
    ASSERT(nrf_ringbuf_put(&ring, 0) == NRF_SUCCESS);
    
    // get all available data                                                                                                                                                                               
    len = 100;
    ASSERT(nrf_ringbuf_get(&ring, &data, &len, false) == NRF_SUCCESS);
    ASSERT(nrf_ringbuf_free(&ring, len) == NRF_SUCCESS);
}

This particular usage pattern is triggered all the time by nrf_cli_uart.

Parents
  • I can confirm the bug. Please find the attached fixed version. It will be part of next release.

    /**
     * Copyright (c) 2017 - 2018, 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.
     * 
     */
    #include "nrf_ringbuf.h"
    #include "app_util_platform.h"
    #include "nrf_assert.h"
    
    #define WR_OFFSET 0
    #define RD_OFFSET 1
    
    void nrf_ringbuf_init(nrf_ringbuf_t const * p_ringbuf)
    {
        p_ringbuf->p_cb->wr_idx = 0;
        p_ringbuf->p_cb->rd_idx = 0;
        p_ringbuf->p_cb->tmp_rd_idx = 0;
        p_ringbuf->p_cb->tmp_wr_idx = 0;
        p_ringbuf->p_cb->rd_flag   = 0;
        p_ringbuf->p_cb->wr_flag   = 0;
    }
    
    ret_code_t nrf_ringbuf_alloc(nrf_ringbuf_t const * p_ringbuf, uint8_t * * pp_data, size_t * p_length, bool start)
    {
        ASSERT(pp_data);
        ASSERT(p_length);
    
        if (start)
        {
            if (nrf_atomic_flag_set_fetch(&p_ringbuf->p_cb->wr_flag))
            {
                return NRF_ERROR_BUSY;
            }
        }
    
        if (p_ringbuf->p_cb->tmp_wr_idx - p_ringbuf->p_cb->rd_idx == p_ringbuf->bufsize_mask + 1)
        {
            *p_length = 0;
            if (start)
            {
                UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->wr_flag));
            }
            return NRF_SUCCESS;
        }
    
        uint32_t wr_idx = p_ringbuf->p_cb->tmp_wr_idx & p_ringbuf->bufsize_mask;
        uint32_t rd_idx = p_ringbuf->p_cb->rd_idx & p_ringbuf->bufsize_mask;
        uint32_t available = (wr_idx >= rd_idx) ? p_ringbuf->bufsize_mask + 1 - wr_idx :
                p_ringbuf->p_cb->rd_idx -  (p_ringbuf->p_cb->tmp_wr_idx - (p_ringbuf->bufsize_mask + 1));
        *p_length = *p_length < available ? *p_length : available;
        *pp_data = &p_ringbuf->p_buffer[wr_idx];
        p_ringbuf->p_cb->tmp_wr_idx += *p_length;
    
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_put(nrf_ringbuf_t const * p_ringbuf, size_t length)
    {
        uint32_t available = p_ringbuf->bufsize_mask + 1 -
                (p_ringbuf->p_cb->wr_idx -  p_ringbuf->p_cb->rd_idx);
        if (length > available)
        {
            return NRF_ERROR_NO_MEM;
        }
    
        p_ringbuf->p_cb->wr_idx    += length;
        p_ringbuf->p_cb->tmp_wr_idx = p_ringbuf->p_cb->wr_idx;
        if (nrf_atomic_flag_clear_fetch(&p_ringbuf->p_cb->wr_flag) == 0)
        {
            /* Flag was already cleared. Suggests misuse. */
            return NRF_ERROR_INVALID_STATE;
        }
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_cpy_put(nrf_ringbuf_t const * p_ringbuf,
                                   uint8_t const * p_data,
                                   size_t * p_length)
    {
        ASSERT(p_data);
        ASSERT(p_length);
    
        if (nrf_atomic_flag_set_fetch(&p_ringbuf->p_cb->wr_flag))
        {
            return NRF_ERROR_BUSY;
        }
    
        uint32_t available = p_ringbuf->bufsize_mask + 1 -
                                    (p_ringbuf->p_cb->wr_idx -  p_ringbuf->p_cb->rd_idx);
        *p_length = available > *p_length ? *p_length : available;
        size_t   length        = *p_length;
        uint32_t masked_wr_idx = (p_ringbuf->p_cb->wr_idx & p_ringbuf->bufsize_mask);
        uint32_t trail         = p_ringbuf->bufsize_mask + 1 - masked_wr_idx;
    
        if (length > trail)
        {
            memcpy(&p_ringbuf->p_buffer[masked_wr_idx], p_data, trail);
            length -= trail;
            masked_wr_idx = 0;
            p_data += trail;
        }
        memcpy(&p_ringbuf->p_buffer[masked_wr_idx], p_data, length);
        p_ringbuf->p_cb->wr_idx += *p_length;
    
        UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->wr_flag));
    
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_get(nrf_ringbuf_t const * p_ringbuf, uint8_t * * pp_data, size_t * p_length, bool start)
    {
        ASSERT(pp_data);
        ASSERT(p_length);
    
        if (start)
        {
            if (nrf_atomic_flag_set_fetch(&p_ringbuf->p_cb->rd_flag))
            {
                return NRF_ERROR_BUSY;
            }
        }
    
        uint32_t available = p_ringbuf->p_cb->wr_idx - p_ringbuf->p_cb->tmp_rd_idx;
        if (available == 0)
        {
            *p_length = 0;
            if (start)
            {
                UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->rd_flag));
            }
            return NRF_SUCCESS;
        }
    
        uint32_t masked_tmp_rd_idx = p_ringbuf->p_cb->tmp_rd_idx & p_ringbuf->bufsize_mask;
        uint32_t masked_wr_idx     = p_ringbuf->p_cb->wr_idx & p_ringbuf->bufsize_mask;
    
        if ((masked_wr_idx > masked_tmp_rd_idx) && (available < *p_length))
        {
            *p_length = available;
        }
        else if (masked_wr_idx <= masked_tmp_rd_idx)
        {
            uint32_t trail = p_ringbuf->bufsize_mask + 1 - masked_tmp_rd_idx;
            if (*p_length > trail)
            {
                *p_length = trail;
            }
        }
        *pp_data = &p_ringbuf->p_buffer[masked_tmp_rd_idx];
        p_ringbuf->p_cb->tmp_rd_idx += *p_length;
    
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_cpy_get(nrf_ringbuf_t const * p_ringbuf,
                                   uint8_t * p_data,
                                   size_t * p_length)
    {
        ASSERT(p_data);
        ASSERT(p_length);
    
        if (nrf_atomic_flag_set_fetch(&p_ringbuf->p_cb->rd_flag))
        {
           return NRF_ERROR_BUSY;
        }
    
        uint32_t available = p_ringbuf->p_cb->wr_idx -  p_ringbuf->p_cb->rd_idx;
        *p_length = available > *p_length ? *p_length : available;
        size_t   length        = *p_length;
        uint32_t masked_rd_idx = (p_ringbuf->p_cb->rd_idx & p_ringbuf->bufsize_mask);
        uint32_t masked_wr_idx = (p_ringbuf->p_cb->wr_idx & p_ringbuf->bufsize_mask);
        uint32_t trail         = (masked_wr_idx > masked_rd_idx) ? masked_wr_idx - masked_rd_idx :
                                                          p_ringbuf->bufsize_mask + 1 - masked_rd_idx;
    
        if (length > trail)
        {
            memcpy(p_data, &p_ringbuf->p_buffer[masked_rd_idx], trail);
            length -= trail;
            masked_rd_idx = 0;
            p_data += trail;
        }
        memcpy(p_data, &p_ringbuf->p_buffer[masked_rd_idx], length);
        p_ringbuf->p_cb->rd_idx += *p_length;
    
        UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->rd_flag));
    
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_free(nrf_ringbuf_t const * p_ringbuf, size_t length)
    {
        uint32_t available = (p_ringbuf->p_cb->wr_idx -  p_ringbuf->p_cb->rd_idx);
        if (length > available)
        {
            return NRF_ERROR_NO_MEM;
        }
    
        p_ringbuf->p_cb->rd_idx    += length;
        p_ringbuf->p_cb->tmp_rd_idx = p_ringbuf->p_cb->rd_idx;
        UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->rd_flag));
    
        return NRF_SUCCESS;
    }
    

Reply
  • I can confirm the bug. Please find the attached fixed version. It will be part of next release.

    /**
     * Copyright (c) 2017 - 2018, 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.
     * 
     */
    #include "nrf_ringbuf.h"
    #include "app_util_platform.h"
    #include "nrf_assert.h"
    
    #define WR_OFFSET 0
    #define RD_OFFSET 1
    
    void nrf_ringbuf_init(nrf_ringbuf_t const * p_ringbuf)
    {
        p_ringbuf->p_cb->wr_idx = 0;
        p_ringbuf->p_cb->rd_idx = 0;
        p_ringbuf->p_cb->tmp_rd_idx = 0;
        p_ringbuf->p_cb->tmp_wr_idx = 0;
        p_ringbuf->p_cb->rd_flag   = 0;
        p_ringbuf->p_cb->wr_flag   = 0;
    }
    
    ret_code_t nrf_ringbuf_alloc(nrf_ringbuf_t const * p_ringbuf, uint8_t * * pp_data, size_t * p_length, bool start)
    {
        ASSERT(pp_data);
        ASSERT(p_length);
    
        if (start)
        {
            if (nrf_atomic_flag_set_fetch(&p_ringbuf->p_cb->wr_flag))
            {
                return NRF_ERROR_BUSY;
            }
        }
    
        if (p_ringbuf->p_cb->tmp_wr_idx - p_ringbuf->p_cb->rd_idx == p_ringbuf->bufsize_mask + 1)
        {
            *p_length = 0;
            if (start)
            {
                UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->wr_flag));
            }
            return NRF_SUCCESS;
        }
    
        uint32_t wr_idx = p_ringbuf->p_cb->tmp_wr_idx & p_ringbuf->bufsize_mask;
        uint32_t rd_idx = p_ringbuf->p_cb->rd_idx & p_ringbuf->bufsize_mask;
        uint32_t available = (wr_idx >= rd_idx) ? p_ringbuf->bufsize_mask + 1 - wr_idx :
                p_ringbuf->p_cb->rd_idx -  (p_ringbuf->p_cb->tmp_wr_idx - (p_ringbuf->bufsize_mask + 1));
        *p_length = *p_length < available ? *p_length : available;
        *pp_data = &p_ringbuf->p_buffer[wr_idx];
        p_ringbuf->p_cb->tmp_wr_idx += *p_length;
    
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_put(nrf_ringbuf_t const * p_ringbuf, size_t length)
    {
        uint32_t available = p_ringbuf->bufsize_mask + 1 -
                (p_ringbuf->p_cb->wr_idx -  p_ringbuf->p_cb->rd_idx);
        if (length > available)
        {
            return NRF_ERROR_NO_MEM;
        }
    
        p_ringbuf->p_cb->wr_idx    += length;
        p_ringbuf->p_cb->tmp_wr_idx = p_ringbuf->p_cb->wr_idx;
        if (nrf_atomic_flag_clear_fetch(&p_ringbuf->p_cb->wr_flag) == 0)
        {
            /* Flag was already cleared. Suggests misuse. */
            return NRF_ERROR_INVALID_STATE;
        }
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_cpy_put(nrf_ringbuf_t const * p_ringbuf,
                                   uint8_t const * p_data,
                                   size_t * p_length)
    {
        ASSERT(p_data);
        ASSERT(p_length);
    
        if (nrf_atomic_flag_set_fetch(&p_ringbuf->p_cb->wr_flag))
        {
            return NRF_ERROR_BUSY;
        }
    
        uint32_t available = p_ringbuf->bufsize_mask + 1 -
                                    (p_ringbuf->p_cb->wr_idx -  p_ringbuf->p_cb->rd_idx);
        *p_length = available > *p_length ? *p_length : available;
        size_t   length        = *p_length;
        uint32_t masked_wr_idx = (p_ringbuf->p_cb->wr_idx & p_ringbuf->bufsize_mask);
        uint32_t trail         = p_ringbuf->bufsize_mask + 1 - masked_wr_idx;
    
        if (length > trail)
        {
            memcpy(&p_ringbuf->p_buffer[masked_wr_idx], p_data, trail);
            length -= trail;
            masked_wr_idx = 0;
            p_data += trail;
        }
        memcpy(&p_ringbuf->p_buffer[masked_wr_idx], p_data, length);
        p_ringbuf->p_cb->wr_idx += *p_length;
    
        UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->wr_flag));
    
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_get(nrf_ringbuf_t const * p_ringbuf, uint8_t * * pp_data, size_t * p_length, bool start)
    {
        ASSERT(pp_data);
        ASSERT(p_length);
    
        if (start)
        {
            if (nrf_atomic_flag_set_fetch(&p_ringbuf->p_cb->rd_flag))
            {
                return NRF_ERROR_BUSY;
            }
        }
    
        uint32_t available = p_ringbuf->p_cb->wr_idx - p_ringbuf->p_cb->tmp_rd_idx;
        if (available == 0)
        {
            *p_length = 0;
            if (start)
            {
                UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->rd_flag));
            }
            return NRF_SUCCESS;
        }
    
        uint32_t masked_tmp_rd_idx = p_ringbuf->p_cb->tmp_rd_idx & p_ringbuf->bufsize_mask;
        uint32_t masked_wr_idx     = p_ringbuf->p_cb->wr_idx & p_ringbuf->bufsize_mask;
    
        if ((masked_wr_idx > masked_tmp_rd_idx) && (available < *p_length))
        {
            *p_length = available;
        }
        else if (masked_wr_idx <= masked_tmp_rd_idx)
        {
            uint32_t trail = p_ringbuf->bufsize_mask + 1 - masked_tmp_rd_idx;
            if (*p_length > trail)
            {
                *p_length = trail;
            }
        }
        *pp_data = &p_ringbuf->p_buffer[masked_tmp_rd_idx];
        p_ringbuf->p_cb->tmp_rd_idx += *p_length;
    
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_cpy_get(nrf_ringbuf_t const * p_ringbuf,
                                   uint8_t * p_data,
                                   size_t * p_length)
    {
        ASSERT(p_data);
        ASSERT(p_length);
    
        if (nrf_atomic_flag_set_fetch(&p_ringbuf->p_cb->rd_flag))
        {
           return NRF_ERROR_BUSY;
        }
    
        uint32_t available = p_ringbuf->p_cb->wr_idx -  p_ringbuf->p_cb->rd_idx;
        *p_length = available > *p_length ? *p_length : available;
        size_t   length        = *p_length;
        uint32_t masked_rd_idx = (p_ringbuf->p_cb->rd_idx & p_ringbuf->bufsize_mask);
        uint32_t masked_wr_idx = (p_ringbuf->p_cb->wr_idx & p_ringbuf->bufsize_mask);
        uint32_t trail         = (masked_wr_idx > masked_rd_idx) ? masked_wr_idx - masked_rd_idx :
                                                          p_ringbuf->bufsize_mask + 1 - masked_rd_idx;
    
        if (length > trail)
        {
            memcpy(p_data, &p_ringbuf->p_buffer[masked_rd_idx], trail);
            length -= trail;
            masked_rd_idx = 0;
            p_data += trail;
        }
        memcpy(p_data, &p_ringbuf->p_buffer[masked_rd_idx], length);
        p_ringbuf->p_cb->rd_idx += *p_length;
    
        UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->rd_flag));
    
        return NRF_SUCCESS;
    }
    
    ret_code_t nrf_ringbuf_free(nrf_ringbuf_t const * p_ringbuf, size_t length)
    {
        uint32_t available = (p_ringbuf->p_cb->wr_idx -  p_ringbuf->p_cb->rd_idx);
        if (length > available)
        {
            return NRF_ERROR_NO_MEM;
        }
    
        p_ringbuf->p_cb->rd_idx    += length;
        p_ringbuf->p_cb->tmp_rd_idx = p_ringbuf->p_cb->rd_idx;
        UNUSED_RETURN_VALUE(nrf_atomic_flag_clear(&p_ringbuf->p_cb->rd_flag));
    
        return NRF_SUCCESS;
    }
    

Children
Related