nuttx-apps/system/settings/storage_bin.c
2024-03-03 02:36:56 +08:00

286 lines
7.3 KiB
C

/****************************************************************************
* apps/system/settings/storage_bin.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include "system/settings.h"
#include <unistd.h>
#include <errno.h>
#include <fcntl.h>
#include <nuttx/crc32.h>
#include <stdlib.h>
#include <string.h>
#include <stdint.h>
#include <nuttx/config.h>
#include <sys/types.h>
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
#define BUFFER_SIZE 256 /* Note alignment for Flash writes! */
/****************************************************************************
* Private Types
****************************************************************************/
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
/****************************************************************************
* Name: getsetting
*
* Description:
* Gets the setting information from a given key.
*
* Input Parameters:
* key - key of the required setting
*
* Returned Value:
* The setting
*
****************************************************************************/
FAR static setting_t *getsetting(char *key);
/****************************************************************************
* Private Data
****************************************************************************/
/****************************************************************************
* Public Data
****************************************************************************/
extern setting_t map[CONFIG_SYSTEM_SETTINGS_MAP_SIZE];
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Name: getsetting
*
* Description:
* Gets the setting information from a given key.
*
* Input Parameters:
* key - key of the required setting
*
* Returned Value:
* The setting
*
****************************************************************************/
FAR setting_t *getsetting(FAR char *key)
{
int i;
for (i = 0; i < CONFIG_SYSTEM_SETTINGS_MAP_SIZE; i++)
{
FAR setting_t *setting = &map[i];
if (strcmp(key, setting->key) == 0)
{
return setting;
}
if (setting->type == SETTING_EMPTY)
{
return setting;
}
}
return NULL;
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: load_bin
*
* Description:
* Loads binary data from a storage file.
*
* Input Parameters:
* file - the filename of the storage to use
*
* Returned Value:
* Success or negated failure code
*
****************************************************************************/
int load_bin(FAR char *file)
{
int fd;
int i;
int ret = OK;
uint16_t valid;
uint16_t count;
uint32_t calc_crc = 0;
uint32_t exp_crc = 0;
setting_t setting;
FAR setting_t *slot;
fd = open(file, O_RDONLY);
if (fd < 0)
{
return -ENOENT;
}
valid = 0;
read(fd, &valid, sizeof(uint16_t));
if (valid != VALID)
{
ret = -EBADMSG;
goto abort; /* Just exit - the settings aren't valid */
}
count = 0;
read(fd, &count, sizeof(uint16_t));
for (i = 0; i < count; i++)
{
read(fd, &setting, sizeof(setting_t));
calc_crc = crc32part((FAR uint8_t *)&setting, sizeof(setting_t),
calc_crc);
}
read(fd, &exp_crc, sizeof(uint32_t));
if (calc_crc != exp_crc)
{
ret = -EBADMSG;
goto abort;
}
lseek(fd, (sizeof(uint16_t) * 2), SEEK_SET); /* Get after valid & size */
for (i = 0; i < count; i++)
{
read(fd, &setting, sizeof(setting_t));
slot = getsetting(setting.key);
if (slot == NULL)
{
continue;
}
memcpy(slot, &setting, sizeof(setting_t));
}
abort:
close(fd);
return ret;
}
/****************************************************************************
* Name: save_bin
*
* Description:
* Saves binary data to a storage file.
*
* Input Parameters:
* file - the filename of the storage to use
*
* Returned Value:
* Success or negated failure code
*
****************************************************************************/
int save_bin(FAR char *file)
{
int count;
int fd;
int ret = OK;
off_t offset = sizeof(uint16_t) * 2; /* Valid & count */
size_t data_size;
size_t rem_data;
uint32_t crc;
size_t rem_crc;
FAR uint8_t *buffer = malloc(BUFFER_SIZE);
if (buffer == NULL)
{
return -ENOMEM;
}
count = 0;
int i;
for (i = 0; i < CONFIG_SYSTEM_SETTINGS_MAP_SIZE; i++)
{
if (map[i].type == SETTING_EMPTY)
{
break;
}
count++;
}
fd = open(file, (O_RDWR | O_TRUNC), 0666);
if (fd < 0)
{
ret = -ENODEV;
goto abort;
}
memset(buffer, 0xff, BUFFER_SIZE);
*((FAR uint16_t *)buffer) = VALID;
*(((FAR uint16_t *)buffer) + 1) = count;
data_size = count *sizeof(setting_t);
rem_data = data_size;
crc = crc32((FAR uint8_t *)map, data_size);
rem_crc = sizeof(crc);
while ((offset + rem_data + rem_crc) > 0)
{
size_t to_write = ((offset + rem_data) > BUFFER_SIZE) ?
(size_t)(BUFFER_SIZE - offset) : rem_data;
memcpy((buffer + offset), (((FAR uint8_t *)map) +
(data_size - rem_data)), to_write);
size_t j;
for (j = (to_write + offset);
(j < BUFFER_SIZE) && (rem_crc > 0); j++, rem_crc--)
{
off_t crc_byte = (off_t)(sizeof(crc) - rem_crc);
buffer[j] = (crc >> (8 * crc_byte)) & 0xff;
}
write(fd, buffer, BUFFER_SIZE);
rem_data -= to_write;
offset = 0;
}
close(fd);
abort:
free(buffer);
return ret;
}