/**************************************************************************** * 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 #include #include #include #include #include #include #include #include /**************************************************************************** * 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; }