Hello, I am trying to combine light switch client example with fds functionality.
For these I did next steps:
-
Included all ‘c’ files from the fds example to the light switch project.
-
Added links for all ‘h’ files to the "User Include Directories" in light switch project
-
Activated necessary variables in the sdk_config.c
-
Copy main.c from the fds_flash to the light switch project.
-
Changed all NRF_ messages to __LOG l.
As a result there aren’t any mistakes after compilation project, but program sticks in wait_for_fds_ready() function in the loop.
My main.c.:
#include <stdint.h>
#include <string.h>
#include "nrf.h"
#include "nordic_common.h"
#ifdef SOFTDEVICE_PRESENT
#include "nrf_sdh.h"
#include "nrf_sdh_ble.h"
#else
#include "nrf_drv_clock.h"
#endif
#include "fds.h"
#include "app_timer.h"
#include "app_error.h"
#include "nrf_cli.h"
#include "fds_example.h"
#define NRF_LOG_MODULE_NAME app
#include "nrf_log.h"
#include "nrf_log_ctrl.h"
#include "log.h"
/* A tag identifying the SoftDevice BLE configuration. */
#define APP_BLE_CONN_CFG_TAG 1
/* Array to map FDS return values to strings. */
char const * fds_err_str[] =
{
"FDS_SUCCESS",
"FDS_ERR_OPERATION_TIMEOUT",
"FDS_ERR_NOT_INITIALIZED",
"FDS_ERR_UNALIGNED_ADDR",
"FDS_ERR_INVALID_ARG",
"FDS_ERR_NULL_ARG",
"FDS_ERR_NO_OPEN_RECORDS",
"FDS_ERR_NO_SPACE_IN_FLASH",
"FDS_ERR_NO_SPACE_IN_QUEUES",
"FDS_ERR_RECORD_TOO_LARGE",
"FDS_ERR_NOT_FOUND",
"FDS_ERR_NO_PAGES",
"FDS_ERR_USER_LIMIT_REACHED",
"FDS_ERR_CRC_CHECK_FAILED",
"FDS_ERR_BUSY",
"FDS_ERR_INTERNAL",
};
/* Array to map FDS events to strings. */
static char const * fds_evt_str[] =
{
"FDS_EVT_INIT",
"FDS_EVT_WRITE",
"FDS_EVT_UPDATE",
"FDS_EVT_DEL_RECORD",
"FDS_EVT_DEL_FILE",
"FDS_EVT_GC",
};
/* Dummy configuration data. */
static configuration_t m_dummy_cfg =
{
.config1_on = false,
.config2_on = true,
.boot_count = 0x0,
.device_name = "dummy",
};
/* A record containing dummy configuration data. */
static fds_record_t const m_dummy_record =
{
.file_id = CONFIG_FILE,
.key = CONFIG_REC_KEY,
.data.p_data = &m_dummy_cfg,
/* The length of a record is always expressed in 4-byte units (words). */
.data.length_words = (sizeof(m_dummy_cfg) + 3) / sizeof(uint32_t),
};
/* Keep track of the progress of a delete_all operation. */
static struct
{
bool delete_next; //!< Delete next record.
bool pending; //!< Waiting for an fds FDS_EVT_DEL_RECORD event, to delete the next record.
} m_delete_all;
/* Flag to check fds initialization. */
static bool volatile m_fds_initialized;
static void fds_evt_handler(fds_evt_t const * p_evt)
{
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Event: %s received (%s)\n",
fds_evt_str[p_evt->id],
fds_err_str[p_evt->result]);
switch (p_evt->id)
{
case FDS_EVT_INIT:
if (p_evt->result == FDS_SUCCESS)
{
m_fds_initialized = true;
}
break;
case FDS_EVT_WRITE:
{
if (p_evt->result == FDS_SUCCESS)
{
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Record ID:\t0x%04x\n", p_evt->write.record_id);
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "File ID:\t0x%04x\n", p_evt->write.file_id);
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Record key:\t0x%04x\n", p_evt->write.record_key);
}
} break;
case FDS_EVT_DEL_RECORD:
{
if (p_evt->result == FDS_SUCCESS)
{
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Record ID:\t0x%04x\n", p_evt->del.record_id);
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "File ID:\t0x%04x\n", p_evt->del.file_id);
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Record key:\t0x%04x\n", p_evt->del.record_key);
}
m_delete_all.pending = false;
} break;
default:
break;
}
}
/**@brief Begin deleting all records, one by one. */
void delete_all_begin(void)
{
m_delete_all.delete_next = true;
}
/**@brief Process a delete all command.
*
* Delete records, one by one, until no records are left.
*/
void delete_all_process(void)
{
if ( m_delete_all.delete_next
& !m_delete_all.pending)
{
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Deleting next record.\n");
m_delete_all.delete_next = record_delete_next();
if (!m_delete_all.delete_next)
{
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "No records left to delete.\n");
}
}
}
#ifdef SOFTDEVICE_PRESENT
/**@brief Function for initializing the SoftDevice and enabling the BLE stack. */
static void ble_stack_init(void)
{
ret_code_t rc;
uint32_t ram_start;
/* Enable the SoftDevice. */
rc = nrf_sdh_enable_request();
APP_ERROR_CHECK(rc);
rc = nrf_sdh_ble_default_cfg_set(APP_BLE_CONN_CFG_TAG, &ram_start);
APP_ERROR_CHECK(rc);
rc = nrf_sdh_ble_enable(&ram_start);
APP_ERROR_CHECK(rc);
}
#else
static void clock_init(void)
{
/* Initialize the clock. */
ret_code_t rc = nrf_drv_clock_init();
APP_ERROR_CHECK(rc);
nrf_drv_clock_lfclk_request(NULL);
/* Wait for the clock to be ready. */
while (!nrf_clock_lf_is_running()) {;}
}
#endif
/**@brief Initialize the timer. */
static void timer_init1(void)
{
ret_code_t err_code = app_timer_init1();
APP_ERROR_CHECK(err_code);
}
/**@brief Initialize logging. */
static void log_init1(void)
{
ret_code_t rc = NRF_LOG_INIT(NULL);
APP_ERROR_CHECK(rc);
}
/**@brief Sleep until an event is received. */
static void power_manage(void)
{
#ifdef SOFTDEVICE_PRESENT
(void) sd_app_evt_wait();
#else
__WFE();
#endif
}
/**@brief Wait for fds to initialize. */
static void wait_for_fds_ready(void)
{
while (!m_fds_initialized)
{
power_manage();
}
}
int main(void)
{
ret_code_t rc;
//ble_stack_init();
timer_init1();
log_init1();
// cli_init();
__LOG_INIT(LOG_SRC_APP | LOG_SRC_ACCESS | LOG_SRC_BEARER, LOG_LEVEL_INFO, LOG_CALLBACK_DEFAULT);
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "----- BLE Mesh Light Switch Client Demo -----\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "FDS example started.\n")
/* Register first to receive an event when initialization is complete. */
(void) fds_register(fds_evt_handler);
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Initializing fds...\n");
rc = fds_init();
APP_ERROR_CHECK(rc);
/* Wait for fds to initialize. */
wait_for_fds_ready();
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Available commands:\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "- print all\t\tprint records\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "- print config\tprint configuration\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "- update\t\tupdate configuration\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "- stat\t\tshow statistics\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "- write\t\twrite a new record\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "- delete\t\tdelete a record\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "- delete_all\tdelete all records\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "- gc\t\trun garbage collection\n");
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Reading flash usage statistics...\n");
fds_stat_t stat = {0};
rc = fds_stat(&stat);
APP_ERROR_CHECK(rc);
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO,"Found %d valid records.\n", stat.valid_records);
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO,"Found %d dirty records (ready to be garbage collected).\n", stat.dirty_records);
fds_record_desc_t desc = {0};
fds_find_token_t tok = {0};
rc = fds_record_find(CONFIG_FILE, CONFIG_REC_KEY, &desc, &tok);
if (rc == FDS_SUCCESS)
{
/* A config file is in flash. Let's update it. */
fds_flash_record_t config = {0};
/* Open the record and read its contents. */
rc = fds_record_open(&desc, &config);
APP_ERROR_CHECK(rc);
/* Copy the configuration from flash into m_dummy_cfg. */
memcpy(&m_dummy_cfg, config.p_data, sizeof(configuration_t));
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Config file found, updating boot count to %d.\n", m_dummy_cfg.boot_count);
/* Update boot count. */
m_dummy_cfg.boot_count++;
/* Close the record when done reading. */
rc = fds_record_close(&desc);
APP_ERROR_CHECK(rc);
/* Write the updated record to flash. */
rc = fds_record_update(&desc, &m_dummy_record);
APP_ERROR_CHECK(rc);
}
else
{
/* System config not found; write a new one. */
__LOG(LOG_SRC_APP, LOG_LEVEL_INFO, "Writing config file...\n");
rc = fds_record_write(&desc, &m_dummy_record);
APP_ERROR_CHECK(rc);
}
//cli_start();
/* Enter main loop. */
for (;;)
{
if (!NRF_LOG_PROCESS())
{
power_manage();
}
// cli_process();
delete_all_process();
}
}