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

Problems with flash erase/write while using SD S340

Hi,

I'm trying to write/read from flash in my app. Reading/writing works until I either:

a) run ant_hrm_rx_start() - then after the call to nrf_fstorage_erase I get "00> <info> app: --> Event received: ERROR while executing an fstorage operation." in the log, or

b) connect to my device via bluetooth and try to write some data via service - then the application hangs on wait_for_flash_ready(&fstorage);

Before any of these two, I can read/write/erase without problems, even after I initiate my softdevice.

This is how I init my flash: 

nrf_fstorage_api_t* p_fs_api;
p_fs_api = &nrf_fstorage_sd;
err_code = nrf_fstorage_init(&fstorage, p_fs_api, NULL);
APP_ERROR_CHECK(err_code);

I also have FDS_ENABLED 1 and  #define FDS_BACKEND 2.

Below is my messy main.c file. From BLE service I'm calling the function write_config_ble (see main below for declaration)

/**
 * Copyright (c) 2013 - 2020, 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 ble_sdk_app_ant_hrs_main main.c
 * @{
 * @ingroup ble_sdk_app_ant_hrs
 * @brief HRM sample application using both BLE and ANT.
 *
 * The application uses the BLE Heart Rate Service (and also the Device Information
 * services), and the ANT HRM RX profile.
 *
 * It will open a receive channel which will connect to an ANT HRM TX profile device when the
 * application starts. The received data will be propagated to a BLE central through the
 * BLE Heart Rate Service.
 *
 * The ANT HRM TX profile device simulator SDK application
 * (Board\pca10003\ant\ant_hrm\hrm_tx_buttons) can be used as a peer ANT device. By changing
 * ANT_HRMRX_NETWORK_KEY to the ANT+ Network Key, the application will instead be able to connect to
 * an ANT heart rate belt.
 *
 * @note The ANT+ Network Key is available for ANT+ Adopters. Please refer to
 *       http://thisisant.com to become an ANT+ Adopter and access the key.
 *
 * @note This application is based on the BLE Heart Rate Service Sample Application
 *       (Board\nrf6310\ble\ble_app_hrs). Please refer to this application for additional
 *       documentation.
 */

#include <stdint.h>
#include <string.h>
#include "nordic_common.h"
#include "app_error.h"
#include "app_timer.h"
#include "ble.h"
#include "ble_hci.h"
#include "ble_conn_state.h"
#include "ble_conn_params.h"
#include "ble_srv_common.h"
#include "ble_advdata.h"
#include "ble_hrs.h"
#include "nrf_drv_timer.h"
#include "bsp.h"
#include "fds.h"
#include "peer_manager.h"
#include "peer_manager_handler.h"
#include "sensorsim.h"
#include "nrf_ble_gatt.h"
#include "nrf_sdh.h"
#include "nrf_sdh_ble.h"
#include "nrf_sdh_ant.h"
#include "nrf_sdh_soc.h"
#include "nrf_ble_qwr.h"
#include "nrf_pwr_mgmt.h"
#include "nrf_fstorage_sd.h"

#include "ant_error.h"
#include "ant_key_manager.h"
#include "ant_hrm.h"
#include "ant_parameters.h"
#include "ant_interface.h"
#include "nrf_delay.h"

#include "nrf_log.h"
#include "nrf_log_ctrl.h"
#include "nrf_log_default_backends.h"
#include "nrf_drv_gpiote.h"
#include "my_common.h"
#include "math.h"

// ######################################################################################################
// ################### parameters dependent on the user device and user HR/preference ###################
// ######################################################################################################
uint8_t fan_hi = 0; // in new approach we store the index of array corresponding to the delay above
										// fan_hi is the index for high speed of the fan (for upper_hr)
uint8_t fan_lo = 159; // this variable is the index of the "segments" array with delays corresponding to the low speed (for lower_hr)
uint8_t lower_hr = 70; // this value of HR will be used as a value above which we will start the fan
uint8_t upper_hr = 200; // this is the value of HR above which we will set the delay to minimum = give max power to fan
uint16_t ant_id = 0; // this ant id of hrm should be used; if zero -> find the nearest hr belt instead (default)
// ######################################################################################################

const nrf_drv_timer_t TRIAC_TIMER = NRF_DRV_TIMER_INSTANCE(1);
uint32_t flash_addr;
uint8_t global_hr = 50;
#define PIN_IN BSP_BUTTON_0 // 15 on nrf51 // pin for zero crossing detection
#define PIN_OUT BSP_LED_0 // 20 on nrf51 // pin for contronling the triac
#define uS_IN_PI 10000
#define M_PI 3.14159265358979323846
#define MULTIPLIER uS_IN_PI/M_PI
double segments[SEGMENTS_NO];
double scale;
bool triac_on;
bool turn_off;
uint32_t global_delay = 0; // how long shall we wait till firing the triac
bool test;
uint32_t test_delay;
ret_code_t ant_reinit(ant_channel_config_t const * p_config);

#define WAKEUP_BUTTON_ID                0                                            /**< Button used to wake up the application. */
#define BOND_DELETE_ALL_BUTTON_ID       1                                            /**< Button used for deleting all bonded centrals during startup. */

#define DEVICE_NAME                     "HRM Fan #3"                                 /**< Name of device. Will be included in the advertising data. */
//#define MANUFACTURER_NAME               "NordicSemiconductor"                        /**< Manufacturer. Will be passed to Device Information Service. */
#define APP_ADV_INTERVAL                40                                           /**< The advertising interval (in units of 0.625 ms. This value corresponds to 25 ms). */
#define APP_ADV_DURATION                18000                                        /**< The advertising duration in units of seconds. */

#define APP_BLE_CONN_CFG_TAG            1                                            /**< A tag that refers to the BLE stack configuration we set with @ref sd_ble_cfg_set. Default tag is @ref BLE_CONN_CFG_TAG_DEFAULT. */

#define IS_SRVC_CHANGED_CHARACT_PRESENT 0                                            /**< Whether or not to include the service_changed characteristic. If not enabled, the server's database cannot be changed for the lifetime of the device */

#define SECOND_1_25_MS_UNITS            800                                          /**< Definition of 1 second, when 1 unit is 1.25 ms. */
#define SECOND_10_MS_UNITS              100                                          /**< Definition of 1 second, when 1 unit is 10 ms. */
#define MIN_CONN_INTERVAL               (SECOND_1_25_MS_UNITS / 10)                  /**< Minimum acceptable connection interval (0.5 seconds), Connection interval uses 1.25 ms units. */
#define MAX_CONN_INTERVAL               (SECOND_1_25_MS_UNITS / 5)                       /**< Maximum acceptable connection interval (1 second), Connection interval uses 1.25 ms units. */
#define SLAVE_LATENCY                   0                                            /**< Slave latency. */
#define CONN_SUP_TIMEOUT                (4 * SECOND_10_MS_UNITS)                     /**< Connection supervisory timeout (4 seconds), Supervision Timeout uses 10 ms units. */

#define FIRST_CONN_PARAMS_UPDATE_DELAY  APP_TIMER_TICKS(5000)                        /**< Time from initiating event (connect or start of notification) to first time sd_ble_gap_conn_param_update is called (5 seconds). */
#define NEXT_CONN_PARAMS_UPDATE_DELAY   APP_TIMER_TICKS(30000)                       /**< Time between each call to sd_ble_gap_conn_param_update after the first call (30 seconds). */
#define MAX_CONN_PARAMS_UPDATE_COUNT    3                                            /**< Number of attempts before giving up the connection parameter negotiation. */

#define SEC_PARAM_TIMEOUT               30                                           /**< Timeout for Pairing Request or Security Request (in seconds). */
#define SEC_PARAM_BOND                  1                                            /**< Perform bonding. */
#define SEC_PARAM_MITM                  0                                            /**< Man In The Middle protection not required. */
#define SEC_PARAM_IO_CAPABILITIES       BLE_GAP_IO_CAPS_NONE                         /**< No I/O capabilities. */
#define SEC_PARAM_OOB                   0                                            /**< Out Of Band data not available. */
#define SEC_PARAM_MIN_KEY_SIZE          7                                            /**< Minimum encryption key size. */
#define SEC_PARAM_MAX_KEY_SIZE          16                                           /**< Maximum encryption key size. */

#define DEAD_BEEF                       0xDEADBEEF                                   /**< Value used as error code on stack dump, can be used to identify stack location on stack unwind. */

#define ANT_HRMRX_ANT_CHANNEL           0                                            /**< Default ANT Channel. */
#define ANT_HRMRX_DEVICE_NUMBER         0                                            /**< Device Number. */
#define ANT_HRMRX_TRANS_TYPE            0                                            /**< Transmission Type. */
#define ANTPLUS_NETWORK_NUMBER          0                                            /**< Network number. */

static volatile uint16_t                m_conn_handle = BLE_CONN_HANDLE_INVALID;     /**< Handle of the current connection. */
static uint8_t                          m_adv_handle;                                /**< Advertising handle. */
BLE_HRS_DEF(m_hrs);                                                                  /**< Heart rate service instance. */
NRF_BLE_QWR_DEF(m_qwr);                                                              /**< Context for the Queued Write module.*/
NRF_BLE_GATT_DEF(m_gatt);                                                            /**< GATT module instance. */

ant_hrm_profile_t m_ant_hrm;                                                         /**< ANT HRM profile instance. */
HRM_DISP_CHANNEL_CONFIG_DEF(m_ant_hrm,
                            ANT_HRMRX_ANT_CHANNEL,
                            ANT_HRMRX_TRANS_TYPE,
                            ANT_HRMRX_DEVICE_NUMBER,
                            ANTPLUS_NETWORK_NUMBER,
                            HRM_MSG_PERIOD_4Hz);
NRF_SDH_ANT_OBSERVER(m_ant_hrm_observer, ANT_HRM_ANT_OBSERVER_PRIO, ant_hrm_disp_evt_handler, &m_ant_hrm);

static void fstorage_evt_handler(nrf_fstorage_evt_t * p_evt);
NRF_FSTORAGE_DEF(nrf_fstorage_t fstorage) =
{
    /* Set a handler for fstorage events. */
    .evt_handler = fstorage_evt_handler,

    /* These below are the boundaries of the flash space assigned to this instance of fstorage.
     * You must set these manually, even at runtime, before nrf_fstorage_init() is called.
     * The function nrf5_flash_end_addr_get() can be used to retrieve the last address on the
     * last page of flash available to write data. */
    .start_addr = 0x0FF000,//0x3e000,
    .end_addr   = 0x100000,//0x3ffff,
};

void divide_sinewave(int pieces)
{
		double u;
		float area;
		double offset;
		float two = 2.0;
		// area of the whole half-sine = 2, so area of each piece = 2/pieces 
		area = two / (float)pieces;
	
		//NRF_LOG_INFO("area:" NRF_LOG_FLOAT_MARKER "\r\n", NRF_LOG_FLOAT(area));
		u = area;
		for(int t = 0; t < pieces; t ++)
		{
				segments[t] = acos(1 - u) * MULTIPLIER; // multipy value in radians to match the range of 0-10ms (10k us)
				
				// next 2 lines - shift values to the 0 offset - minimum delay works, maximum won't due to timer inacuracies and incudctive load influence
				if(t == 0) offset = segments[t];
				segments[t] -=  offset;
				
				u += area;
				//if(t<10 || t>190) app_trace_log("segments[%i]=%lf\n", t, segments[t]);
		}		
}

void calculate_scale(int pieces)
{
		// we store the index of "segments" array instead of delay in ms
		scale = (float)(fan_lo - fan_hi) / (float)(upper_hr - lower_hr);
}
void in_pin_handler(nrf_drv_gpiote_pin_t pin, nrf_gpiote_polarity_t action)
{
		//NRF_LOG_INFO("button pressed\n");
    if(triac_on) 
		{
				if(global_delay > 0) 
				{
						turn_off = false;
						nrf_drv_gpiote_out_clear(PIN_OUT);
						nrf_drv_timer_extended_compare(&TRIAC_TIMER, NRF_TIMER_CC_CHANNEL1, global_delay, NRF_TIMER_SHORT_COMPARE1_STOP_MASK, true);
						nrf_drv_timer_enable(&TRIAC_TIMER);
				}
				else
				{
						nrf_drv_gpiote_out_set(PIN_OUT);
				}
		} 
		else 
		{ 
				nrf_drv_gpiote_out_clear(PIN_OUT);
		}
}
/**@brief Callback function for asserts in the SoftDevice.
 *
 * @details This function will be called in case of an assert in the SoftDevice.
 *
 * @warning This handler is an example only and does not fit a final product. You need to analyze
 *          how your product is supposed to react in case of Assert.
 * @warning On assert from the SoftDevice, the system can only recover on reset.
 *
 * @param[in]   line_num   Line number of the failing ASSERT call.
 * @param[in]   file_name  File name of the failing ASSERT call.
 */
void assert_nrf_callback(uint16_t line_num, const uint8_t * p_file_name)
{
    app_error_handler(DEAD_BEEF, line_num, p_file_name);
}

void init_flash(void)
{
	  uint32_t err_code;
	
	  uint32_t pg_size = NRF_FICR->CODEPAGESIZE;
    uint32_t pg_num  = NRF_FICR->CODESIZE - 1;  // Use last page in flash
		//app_trace_log("page_size=%zu, code_size=%zu\n", NRF_FICR->CODEPAGESIZE, NRF_FICR->CODESIZE);
		flash_addr = (uint32_t)(pg_size * pg_num);
	
    nrf_fstorage_api_t* p_fs_api;
    p_fs_api = &nrf_fstorage_sd;
    err_code = nrf_fstorage_init(&fstorage, p_fs_api, NULL);
    APP_ERROR_CHECK(err_code);
}

void wait_for_flash_ready(nrf_fstorage_t const * p_fstorage);
void write_data_to_flash(uint32_t m_address, uint32_t m_data)
{
		uint32_t err_code;
	
		// erase first
		NRF_LOG_INFO("erasing flash @%x", m_address);
		err_code = nrf_fstorage_erase(&fstorage, m_address, 1, NULL);
		APP_ERROR_CHECK(err_code);
	
		NRF_LOG_INFO("waiting after erase ... ");
    wait_for_flash_ready(&fstorage);
    NRF_LOG_INFO("done");
		
    // write to flash
    NRF_LOG_INFO("writing %x to flash @%x", m_data, m_address);
    err_code = nrf_fstorage_write(&fstorage, m_address, &m_data, sizeof(m_data), NULL);
    APP_ERROR_CHECK(err_code);

    NRF_LOG_INFO("waiting after write ... ");
    wait_for_flash_ready(&fstorage);
		NRF_LOG_INFO("done");
}
uint32_t read_data_from_flash(uint32_t m_address)
{
		uint32_t err_code;
		uint32_t read_data = 0;	
		NRF_LOG_INFO("reading data @%x", m_address);
		err_code = nrf_fstorage_read(&fstorage, m_address, &read_data, sizeof(read_data));
		APP_ERROR_CHECK(err_code);
		NRF_LOG_INFO("read data: %x ", read_data);
		return read_data;
}
void read_config(void)
{
		uint32_t read_data;

		// read first word from flash
		//read_data = (uint32_t) *(flash_addr);			
		//app_trace_log("data read: %zu\n", read_data);	
	
		read_data = read_data_from_flash(flash_addr);
    NRF_LOG_INFO("waiting after read? ... ");
    wait_for_flash_ready(&fstorage);
		NRF_LOG_INFO("done, now converting ...");	
	
		fan_lo = (uint8_t)((read_data & 0xFF000000) >> 24);
		fan_hi = (uint8_t)((read_data & 0x00FF0000) >> 16);
		lower_hr = (uint8_t)((read_data & 0x0000FF00) >> 8);
		upper_hr = (uint8_t)(read_data & 0x000000FF);

    NRF_LOG_INFO("done, now reading again ... ");
		// read second word - ant_id
		read_data = read_data_from_flash(flash_addr + 100);
		ant_id = (uint16_t)(read_data & 0x0000FFFF);
		
		NRF_LOG_INFO("FLASH read: fan_lo=%u, fan_hi=%u, lower_hr=%u, upper_hr=%u, ant_id=%u\n", \
									fan_lo, fan_hi, lower_hr, upper_hr, ant_id);
}
void write_config_ble(void)
{
		const uint32_t data_to_write = fan_lo << 24 | fan_hi << 16 | lower_hr << 8 | upper_hr; 
		uint32_t ant_id_32 = (uint32_t) ant_id;
		write_data_to_flash(flash_addr, data_to_write);
		write_data_to_flash(flash_addr + 100, ant_id_32);
}


/**@brief Start advertising.
 */
static void advertising_start(void)
{
    uint32_t err_code;

    err_code = sd_ble_gap_adv_start(m_adv_handle, APP_BLE_CONN_CFG_TAG);
    if (err_code != NRF_SUCCESS && err_code != NRF_ERROR_INVALID_STATE)
    {
        APP_ERROR_HANDLER(err_code);
    }

    err_code = bsp_indication_set(BSP_INDICATE_ADVERTISING);
    APP_ERROR_CHECK(err_code);
}


/**@brief Start receiving the ANT HRM data.
 */
static void ant_hrm_rx_start(void)
{
    uint32_t err_code = ant_hrm_disp_open(&m_ant_hrm);
    APP_ERROR_CHECK(err_code);
}


/**@brief Attempt to both open the ant channel and start ble advertising.
*/
static void ant_and_adv_start(void)
{
    advertising_start();
    ant_hrm_rx_start();
}


/**@brief Timer initialization.
 *
 * @details Initializes the timer module. This creates and starts application timers.
 */
static void peripherals_init(void)
{
    // Initialize timer module
    uint32_t err_code = app_timer_init();
    APP_ERROR_CHECK(err_code);
}


/**@brief Function for initializing the GATT module.
 */
static void gatt_init(void)
{
    ret_code_t err_code = nrf_ble_gatt_init(&m_gatt, NULL);
    APP_ERROR_CHECK(err_code);
}


/**@brief GAP initialization.
 *
 * @details This function sets up all the necessary GAP (Generic Access Profile) parameters of the
 *          device including the device name, appearance, and the preferred connection parameters.
 */
static void gap_params_init(void)
{
    uint32_t                err_code;
    ble_gap_conn_params_t   gap_conn_params;
    ble_gap_conn_sec_mode_t sec_mode;

    BLE_GAP_CONN_SEC_MODE_SET_OPEN(&sec_mode);

    err_code = sd_ble_gap_device_name_set(&sec_mode,
                                          (const uint8_t *)DEVICE_NAME,
                                          strlen(DEVICE_NAME));
    APP_ERROR_CHECK(err_code);

    err_code = sd_ble_gap_appearance_set(BLE_APPEARANCE_HEART_RATE_SENSOR_HEART_RATE_BELT);
    APP_ERROR_CHECK(err_code);

    memset(&gap_conn_params, 0, sizeof(gap_conn_params));

    gap_conn_params.min_conn_interval = MIN_CONN_INTERVAL;
    gap_conn_params.max_conn_interval = MAX_CONN_INTERVAL;
    gap_conn_params.slave_latency     = SLAVE_LATENCY;
    gap_conn_params.conn_sup_timeout  = CONN_SUP_TIMEOUT;

    err_code = sd_ble_gap_ppcp_set(&gap_conn_params);
    APP_ERROR_CHECK(err_code);
}


/**@brief Advertising functionality initialization.
 *
 * @details Encodes the required advertising data and passes it to the stack.
 *          Also builds a structure to be passed to the stack when starting advertising.
 */
static void advertising_init(void)
{
    uint32_t             err_code;
    ble_advdata_t        advdata;
		ble_advdata_t				 scanrsp;
    ble_gap_adv_data_t   advdata_enc;
    ble_gap_adv_params_t adv_params;
    static uint8_t       advdata_buff[BLE_GAP_ADV_SET_DATA_SIZE_MAX];
    uint16_t             advdata_buff_len = BLE_GAP_ADV_SET_DATA_SIZE_MAX;
    uint8_t              flags = BLE_GAP_ADV_FLAGS_LE_ONLY_GENERAL_DISC_MODE;
	
		static uint8_t			srdata_buff[BLE_GAP_ADV_SET_DATA_SIZE_MAX];
		uint16_t						srdata_buff_len = BLE_GAP_ADV_SET_DATA_SIZE_MAX;

		ble_uuid_t ble_uuid;
		ble_uuid128_t base_uuid = {LBS_UUID_BASE};
    err_code = sd_ble_uuid_vs_add(&base_uuid, &ble_uuid.type);
		APP_ERROR_CHECK(err_code);
		ble_uuid_t adv_uuids[] = {{LBS_UUID_SERVICE, ble_uuid.type}};		
		
		// Build and set advertising data.
    memset(&advdata, 0, sizeof(advdata));

    advdata.name_type               = BLE_ADVDATA_FULL_NAME;
    advdata.include_appearance      = true;
    advdata.flags                   = flags;
    //advdata.uuids_complete.uuid_cnt = sizeof(adv_uuids) / sizeof(adv_uuids[0]);
    //advdata.uuids_complete.p_uuids  = adv_uuids;		
		
    err_code = ble_advdata_encode(&advdata, advdata_buff, &advdata_buff_len);
    APP_ERROR_CHECK(err_code);

    m_adv_handle = BLE_GAP_ADV_SET_HANDLE_NOT_SET;

    memset(&advdata_enc, 0, sizeof(advdata_enc));

		// SCAN RESPONSE DATA (vendor id does not fit in standard advertising data)
		memset(&scanrsp, 0, sizeof(scanrsp));
    scanrsp.uuids_complete.uuid_cnt = sizeof(adv_uuids) / sizeof(adv_uuids[0]);
    scanrsp.uuids_complete.p_uuids  = adv_uuids;
		err_code = ble_advdata_encode(&scanrsp, srdata_buff, &srdata_buff_len);
    APP_ERROR_CHECK(err_code);
		
    advdata_enc.adv_data.p_data = advdata_buff;
    advdata_enc.adv_data.len    = advdata_buff_len;
		advdata_enc.scan_rsp_data.p_data = srdata_buff;
		advdata_enc.scan_rsp_data.len = srdata_buff_len;

    // Initialise advertising parameters (used when starting advertising).
    memset(&adv_params, 0, sizeof(adv_params));

    adv_params.primary_phy     = BLE_GAP_PHY_1MBPS;
    adv_params.properties.type = BLE_GAP_ADV_TYPE_CONNECTABLE_SCANNABLE_UNDIRECTED;
    adv_params.filter_policy   = BLE_GAP_ADV_FP_ANY;
    adv_params.interval        = APP_ADV_INTERVAL;
    adv_params.duration        = APP_ADV_DURATION;

    err_code = sd_ble_gap_adv_set_configure(&m_adv_handle, &advdata_enc, &adv_params);
    APP_ERROR_CHECK(err_code);
}


/**@brief Function for handling Queued Write Module errors.
 *
 * @details A pointer to this function will be passed to each service which may need to inform the
 *          application about an error.
 *
 * @param[in]   nrf_error   Error code containing information about what went wrong.
 */
static void nrf_qwr_error_handler(uint32_t nrf_error)
{
    APP_ERROR_HANDLER(nrf_error);
}

static void config_write_handler(ble_hrs_t* p_lbs, const uint8_t config[]) 
{
		uint16_t prev_ant_id = ant_id;

	NRF_LOG_INFO("config [%i %i %i %i %i %i]\n", config[0], config[1], config[2], config[3], config[4], config[5]);

		fan_lo = SEGMENTS_NO - config[0];
		fan_hi = SEGMENTS_NO - config[1];
		lower_hr = config[2];
		upper_hr = config[3];
		ant_id = config[4] << 8 | config[5];

		NRF_LOG_INFO("incoming conf: fan_lo=%i, fan_hi=%i, lower_hr=%i, upper_hr=%i, ant_id=%i\n", \
				fan_lo, fan_hi, lower_hr, upper_hr, ant_id);

		write_config_ble();	
	
		// reinitialize the ant connection if ant_id has changed
		if(prev_ant_id != ant_id) 
		{
				uint32_t err_code;
				//app_trace_log("new ant_id - reconnect\n");
				err_code = ant_reinit(HRM_DISP_CHANNEL_CONFIG(m_ant_hrm));
				APP_ERROR_CHECK(err_code);
				nrf_delay_ms(30);
				triac_on = false;
		}
}
static void test_write_handler(ble_hrs_t* p_lbs, const uint8_t newState[])
{
		uint8_t testState = newState[0];

		//app_trace_log("incoming test %i\n", testState);
		if(testState > 0) {
				uint32_t time_delay;
				int index = SEGMENTS_NO - testState; // incoming value should be between 1 and 201
				if(index < 0) 
				{
						global_delay = 0;
						nrf_drv_gpiote_out_set(PIN_OUT);
				}
				else 
				{
						time_delay = segments[index];
						global_delay = nrf_drv_timer_us_to_ticks(&TRIAC_TIMER, time_delay); 			
				}

				//app_trace_log("index=%i, testState=%i, time_delay=%i, global_delay=%i\n", index, testState, time_delay, global_delay);
				test = true;
				triac_on = true;
		}
		else
		{
				triac_on = false;
				test = false;
		}
}
/**@brief Initialize services that will be used by the application.
 *
 * @details Initialize the Heart Rate and Device Information services.
 */
static void services_init(void)
{
    uint32_t           err_code;
    ble_hrs_init_t     hrs_init;
    nrf_ble_qwr_init_t qwr_init = {0};

    // Initialize the Queued Write module.
    qwr_init.error_handler = nrf_qwr_error_handler;

    err_code = nrf_ble_qwr_init(&m_qwr, &qwr_init);
    APP_ERROR_CHECK(err_code);

    // Initialize Heart Rate Service.
    memset(&hrs_init, 0, sizeof(hrs_init));
    hrs_init.evt_handler                 = NULL;

    // Here the sec level for the Heart Rate Service can be changed/increased.
    hrs_init.wr_sec = SEC_OPEN;
    hrs_init.rd_sec      = SEC_OPEN;
    hrs_init.test_write_handler = test_write_handler;
		hrs_init.config_write_handler = config_write_handler;

    err_code = ble_hrs_init(&m_hrs, &hrs_init);
    APP_ERROR_CHECK(err_code);
}


/**@brief Connection Parameters Module handler.
 *
 * @details This function will be called for all events in the Connection Parameters Module which
 *          are passed to the application.
 *          @note All this function does is to disconnect. This could have been done by simply
 *                setting the disconnect_on_fail config parameter, but instead we use the event
 *                handler mechanism to demonstrate its use.
 *
 * @param[in]   p_evt   Event received from the Connection Parameters Module.
 */
static void on_conn_params_evt(ble_conn_params_evt_t * p_evt)
{
    uint32_t err_code;

    switch (p_evt->evt_type)
    {
        case BLE_CONN_PARAMS_EVT_FAILED:
            err_code = sd_ble_gap_disconnect(m_conn_handle, BLE_HCI_CONN_INTERVAL_UNACCEPTABLE);
            APP_ERROR_CHECK(err_code);
            break;

        default:
            // No implementation needed.
            break;
    }
}


/**@brief Connection Parameters module error handler.
 *
 * @param[in]   nrf_error   Error code containing information about what went wrong.
 */
static void conn_params_error_handler(uint32_t nrf_error)
{
    APP_ERROR_HANDLER(nrf_error);
}


/**@brief Initialize the Connection Parameters module.
 */
static void conn_params_init(void)
{
    uint32_t               err_code;
    ble_conn_params_init_t cp_init;

    memset(&cp_init, 0, sizeof(cp_init));

    cp_init.p_conn_params                  = NULL;
    cp_init.first_conn_params_update_delay = FIRST_CONN_PARAMS_UPDATE_DELAY;
    cp_init.next_conn_params_update_delay  = NEXT_CONN_PARAMS_UPDATE_DELAY;
    cp_init.max_conn_params_update_count   = MAX_CONN_PARAMS_UPDATE_COUNT;
    //cp_init.start_on_notify_cccd_handle    = m_hrs.hrm_handles.cccd_handle;
    cp_init.disconnect_on_fail             = false;
    cp_init.evt_handler                    = on_conn_params_evt;
    cp_init.error_handler                  = conn_params_error_handler;

    err_code = ble_conn_params_init(&cp_init);
    APP_ERROR_CHECK(err_code);
}


/**@brief Function for handling a ANT stack event.
 *
 * @param[in] p_ant_evt  ANT stack event.
 * @param[in] p_context  Context.
 */
void ant_evt_handler(ant_evt_t * p_ant_evt, void * p_context)
{
    if (p_ant_evt->event != EVENT_CHANNEL_CLOSED)
    {
        return;
    }
    if (m_conn_handle != BLE_CONN_HANDLE_INVALID)
    {
        ant_hrm_rx_start();
    }
    else
    {
        ant_and_adv_start();
    }
}


/**@brief Handle received ANT+ HRM data.
 *
 * @param[in]   p_profile       Pointer to the ANT+ HRM profile instance.
 * @param[in]   event           Event related with ANT+ HRM Display profile.
 */
static void ant_hrm_evt_handler(ant_hrm_profile_t * p_profile, ant_hrm_evt_t event)
{

}

/**@brief Function for dispatching a BLE stack event to all modules with a BLE stack event handler.
 *
 * @param[in] p_ble_evt  Bluetooth stack event.
 * @param[in] p_context  Context.
 */
static void ble_evt_handler(ble_evt_t const * p_ble_evt, void * p_context)
{
    ret_code_t err_code = NRF_SUCCESS;

    switch (p_ble_evt->header.evt_id)
    {
        case BLE_GAP_EVT_CONNECTED:
            err_code = bsp_indication_set(BSP_INDICATE_CONNECTED);
            APP_ERROR_CHECK(err_code);
            m_conn_handle = p_ble_evt->evt.gap_evt.conn_handle;
            err_code = nrf_ble_qwr_conn_handle_assign(&m_qwr, m_conn_handle);
            APP_ERROR_CHECK(err_code);
            break;

        case BLE_GAP_EVT_DISCONNECTED:
            err_code = bsp_indication_set(BSP_INDICATE_IDLE);
            APP_ERROR_CHECK(err_code);
            m_conn_handle = BLE_CONN_HANDLE_INVALID;

            // Need to close the ANT channel to make it safe to write bonding information to flash
            err_code = sd_ant_channel_close(ANT_HRMRX_ANT_CHANNEL);
            APP_ERROR_CHECK(err_code);

            // Note: Bonding information will be stored, advertising will be restarted and the
            //       ANT channel will be reopened when ANT event CHANNEL_CLOSED is received.
            break;

#ifndef S140
        case BLE_GAP_EVT_PHY_UPDATE_REQUEST:
        {
            ble_gap_phys_t const phys =
            {
                .rx_phys = BLE_GAP_PHY_AUTO,
                .tx_phys = BLE_GAP_PHY_AUTO,
            };

            err_code = sd_ble_gap_phy_update(p_ble_evt->evt.gap_evt.conn_handle, &phys);
            APP_ERROR_CHECK(err_code);
        } break;
#endif

#ifndef BONDING_ENABLE
        case BLE_GAP_EVT_SEC_INFO_REQUEST:
            err_code = sd_ble_gap_sec_info_reply(m_conn_handle, NULL, NULL, NULL);
            APP_ERROR_CHECK(err_code);
            break;

        case BLE_GAP_EVT_SEC_PARAMS_REQUEST:
            err_code = sd_ble_gap_sec_params_reply(m_conn_handle,
                                                   BLE_GAP_SEC_STATUS_PAIRING_NOT_SUPP,
                                                   NULL,
                                                   NULL);
            APP_ERROR_CHECK(err_code);
            break;
#endif // BONDING_ENABLE

        case BLE_GAP_EVT_ADV_SET_TERMINATED:
            if (p_ble_evt->evt.gap_evt.params.adv_set_terminated.reason ==
                BLE_GAP_EVT_ADV_SET_TERMINATED_REASON_TIMEOUT)
            {
                err_code = bsp_indication_set(BSP_INDICATE_IDLE);
                APP_ERROR_CHECK(err_code);
                // Go to system-off mode (this function will not return; wakeup will cause a reset)
                //err_code = sd_power_system_off();
                //APP_ERROR_CHECK(err_code);
            }
            break;

#ifndef BONDING_ENABLE
            case BLE_GATTS_EVT_SYS_ATTR_MISSING:
                err_code = sd_ble_gatts_sys_attr_set(m_conn_handle,
                                                     NULL,
                                                     0,
                                                     BLE_GATTS_SYS_ATTR_FLAG_SYS_SRVCS | BLE_GATTS_SYS_ATTR_FLAG_USR_SRVCS);
                APP_ERROR_CHECK(err_code);
                break;
#endif // BONDING_ENABLE

        default:
            // No implementation needed.
            break;
    }
}

#ifdef BONDING_ENABLE
/**@brief Function for handling Peer Manager events.
 *
 * @param[in] p_evt  Peer Manager event.
 */
static void pm_evt_handler(pm_evt_t const * p_evt)
{
    pm_handler_on_pm_evt(p_evt);
    pm_handler_flash_clean(p_evt);

    switch (p_evt->evt_id)
    {
        case PM_EVT_PEERS_DELETE_SUCCEEDED:
            advertising_start();
            break;

        default:
            break;
    }
}


/**@brief Function for the Peer Manager initialization.
 *
 * @param[in] erase_bonds  Indicates whether bonding information should be cleared from
 *                         persistent storage during initialization of the Peer Manager.
 */
static void peer_manager_init(bool erase_bonds)
{
    ble_gap_sec_params_t sec_param;
    ret_code_t           err_code;

    err_code = pm_init();
    APP_ERROR_CHECK(err_code);

    if (erase_bonds)
    {
        err_code = pm_peers_delete();
        APP_ERROR_CHECK(err_code);
    }

    memset(&sec_param, 0, sizeof(ble_gap_sec_params_t));

    // Security parameters to be used for all security procedures.
    sec_param.bond           = SEC_PARAM_BOND;
    sec_param.mitm           = SEC_PARAM_MITM;
    sec_param.io_caps        = SEC_PARAM_IO_CAPABILITIES;
    sec_param.oob            = SEC_PARAM_OOB;
    sec_param.min_key_size   = SEC_PARAM_MIN_KEY_SIZE;
    sec_param.max_key_size   = SEC_PARAM_MAX_KEY_SIZE;
    sec_param.kdist_own.enc  = 1;
    sec_param.kdist_own.id   = 1;
    sec_param.kdist_peer.enc = 1;
    sec_param.kdist_peer.id  = 1;

    err_code = pm_sec_params_set(&sec_param);
    APP_ERROR_CHECK(err_code);

    err_code = pm_register(pm_evt_handler);
    APP_ERROR_CHECK(err_code);
}
#endif // BONDING_ENABLE


/**@brief BLE + ANT stack initialization.
 *
 * @details Initializes the SoftDevice and the stack event interrupt.
 */
static void softdevice_setup(void)
{
    ret_code_t err_code = nrf_sdh_enable_request();
    APP_ERROR_CHECK(err_code);

    ASSERT(nrf_sdh_is_enabled());

    // Configure the BLE stack using the default settings.
    // Fetch the start address of the application RAM.
    uint32_t ram_start = 0;
    err_code = nrf_sdh_ble_default_cfg_set(APP_BLE_CONN_CFG_TAG, &ram_start);
    APP_ERROR_CHECK(err_code);

    // Enable BLE stack.
    err_code = nrf_sdh_ble_enable(&ram_start);
    APP_ERROR_CHECK(err_code);

    err_code = nrf_sdh_ant_enable();
    APP_ERROR_CHECK(err_code);

    err_code = ant_plus_key_set(ANTPLUS_NETWORK_NUMBER);
    APP_ERROR_CHECK(err_code);

    // Register a handler for BLE events.
    NRF_SDH_BLE_OBSERVER(m_ble_observer, APP_BLE_OBSERVER_PRIO, ble_evt_handler, NULL);
    NRF_SDH_ANT_OBSERVER(m_ant_observer, APP_ANT_OBSERVER_PRIO, ant_evt_handler, NULL);
}

void TRIAC_TIMER_event_handler(nrf_timer_event_t event_type, void* p_context)
{
		nrf_drv_timer_clear(&TRIAC_TIMER);	
		if(!turn_off)
		{
				nrf_drv_gpiote_out_set(PIN_OUT);    

				turn_off = true;
				nrf_drv_timer_extended_compare(&TRIAC_TIMER, NRF_TIMER_CC_CHANNEL1, 3, NRF_TIMER_SHORT_COMPARE1_STOP_MASK, true);
				nrf_drv_timer_enable(&TRIAC_TIMER);
		}
		else
		{
				nrf_drv_gpiote_out_clear(PIN_OUT);
		}	
}

void nrf_timer_init(void)
{
		uint32_t err_code;
    //uint32_t time_ms = 500; //Time(in miliseconds) between consecutive compare events.
    //uint32_t time_ticks;
		//Configure all leds on board.
    bsp_board_init(BSP_INIT_LEDS);

    //Configure TRIAC_TIMER for generating simple light effect - leds on board will invert his state one after the other.
    nrf_drv_timer_config_t timer_cfg = NRF_DRV_TIMER_DEFAULT_CONFIG;
		timer_cfg.frequency = NRF_TIMER_FREQ_31250Hz;
    err_code = nrf_drv_timer_init(&TRIAC_TIMER, &timer_cfg, TRIAC_TIMER_event_handler);
    APP_ERROR_CHECK(err_code);

    //time_ticks = nrf_drv_timer_ms_to_ticks(&TRIAC_TIMER, time_ms);
    //nrf_drv_timer_extended_compare(
    //     &TRIAC_TIMER, NRF_TIMER_CC_CHANNEL1, time_ticks, NRF_TIMER_SHORT_COMPARE1_CLEAR_MASK, true);
    //nrf_drv_timer_enable(&TRIAC_TIMER);
}
void gpiote_init(void)
{
		uint32_t err_code;  
		err_code = nrf_drv_gpiote_init();
    APP_ERROR_CHECK(err_code);

    nrf_drv_gpiote_out_config_t out_config = GPIOTE_CONFIG_OUT_SIMPLE(false);

    err_code = nrf_drv_gpiote_out_init(PIN_OUT, &out_config);
    APP_ERROR_CHECK(err_code);

    nrf_drv_gpiote_in_config_t in_config = GPIOTE_CONFIG_IN_SENSE_LOTOHI(true);
    in_config.pull = NRF_GPIO_PIN_NOPULL;

    err_code = nrf_drv_gpiote_in_init(PIN_IN, &in_config, in_pin_handler);
    APP_ERROR_CHECK(err_code);

    nrf_drv_gpiote_in_event_enable(PIN_IN, true);
}

/**@brief Application main function.
 */

//#################################################################################################################################################################################
//#################################################################################################################################################################################
//#################################################################################################################################################################################

/**@brief   Sleep until an event is received. */
static void power_manage(void)
{
    (void) sd_app_evt_wait();
}

static void fstorage_evt_handler(nrf_fstorage_evt_t * p_evt)
{
    if (p_evt->result != NRF_SUCCESS)
    {
        NRF_LOG_INFO("--> Event received: ERROR while executing an fstorage operation.");
        return;
    }

    switch (p_evt->id)
    {
        case NRF_FSTORAGE_EVT_WRITE_RESULT:
        {
            NRF_LOG_INFO("--> Event received: wrote %d bytes at address 0x%x.",
                         p_evt->len, p_evt->addr);
        } break;

        case NRF_FSTORAGE_EVT_ERASE_RESULT:
        {
            NRF_LOG_INFO("--> Event received: erased %d page from address 0x%x.",
                         p_evt->len, p_evt->addr);
        } break;

        default:
            break;
    }
}

void wait_for_flash_ready(nrf_fstorage_t const * p_fstorage)
{
    /* While fstorage is busy, sleep and wait for an event. */
    while (nrf_fstorage_is_busy(p_fstorage))
    {
				power_manage();
    }
}

//#################################################################################################################################################################################
//#################################################################################################################################################################################
//#################################################################################################################################################################################

int main(void)
{
    uint32_t err_code;

    err_code = NRF_LOG_INIT(NULL);
    APP_ERROR_CHECK(err_code);
		global_hr = lower_hr - 1;
	
		NRF_LOG_DEFAULT_BACKENDS_INIT();
	
		// FLASH Stuff
		init_flash();
	
		// Uncomment next line only to reset config to predefined values, then compile, run, and comment again
		//write_config_ble();
		//read_config();
	
		// divide sinewave into SEGMENT_NO pieecs, each with equal area
		divide_sinewave(SEGMENTS_NO);
	
    // Initialize peripherals
    peripherals_init();

    err_code = nrf_pwr_mgmt_init();
    APP_ERROR_CHECK(err_code);

    softdevice_setup();
				
    // Initialize Bluetooth stack parameters.
    gatt_init();
    gap_params_init();
    advertising_init();
    services_init();
    conn_params_init();

    // Initialize ANT+ HRM receive channel.
    err_code = ant_hrm_disp_init(&m_ant_hrm,
                                 HRM_DISP_CHANNEL_CONFIG(m_ant_hrm),
                                 ant_hrm_evt_handler);
    APP_ERROR_CHECK(err_code);

		nrf_timer_init();
		triac_on = false;	
		gpiote_init();

#ifdef BONDING_ENABLE
    bool erase_bonds = bsp_button_is_pressed(BOND_DELETE_ALL_BUTTON_ID);

    peer_manager_init(erase_bonds);
    if (erase_bonds == true)
    {
        NRF_LOG_INFO("Bonds erased!");
    }
#endif // BONDING_ENABLE

    NRF_LOG_INFO("ANT and BLE Heart Rate Monitor Relay example started.");

		advertising_start();
	
		// FLASH TEST - HERE IT WORKS!!!
		uint32_t m_data = 0x22334455;
		uint32_t r_data = 0;
		write_data_to_flash(flash_addr, m_data);
		r_data = read_data_from_flash(flash_addr);
		NRF_LOG_INFO("r_data = %x", r_data);
		// END TEST		

		ant_hrm_rx_start();
		
		// FLASH TEST - HERE IT DOESN't WORK!!! - Event received: ERROR while executing an fstorage operation.
		write_data_to_flash(flash_addr, m_data);
		r_data = read_data_from_flash(flash_addr);
		NRF_LOG_INFO("r_data = %x", r_data);
		// END TEST		

		// Enter main loop.
    for (;;)
    {
        if (NRF_LOG_PROCESS() == false)
        {
            nrf_pwr_mgmt_run();
        }
    }
}

/**
 * @}
 */

What can be wrong?

PLEASE HELP

Parents
  • Ok. Short update. I chekced and the error I get in the first case (when "Event received: ERROR while executing an fstorage operation." is displayed) is 13, so NRF_ERROR_TIMEOUT

    I'm guessing this has to do with some priorities, but I can't figure our what and where...

    Anyway, sdk_config has no priority settings. In previous SDK I used for NRF51 dongle I could easily manipulate the priorities in nrf_drv_config.h. Where do I do that now, in SDK 17?

Reply
  • Ok. Short update. I chekced and the error I get in the first case (when "Event received: ERROR while executing an fstorage operation." is displayed) is 13, so NRF_ERROR_TIMEOUT

    I'm guessing this has to do with some priorities, but I can't figure our what and where...

    Anyway, sdk_config has no priority settings. In previous SDK I used for NRF51 dongle I could easily manipulate the priorities in nrf_drv_config.h. Where do I do that now, in SDK 17?

Children
No Data
Related