/****************************************************************************** * * Copyright (C) 2014 Google, Inc. * * Licensed 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. * ******************************************************************************/ #define LOG_TAG "bt_osi_config" #include "osi/include/config.h" #include <base/logging.h> #include <ctype.h> #include <errno.h> #include <fcntl.h> #include <libgen.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/stat.h> #include <unistd.h> #include "osi/include/allocator.h" #include "osi/include/list.h" #include "osi/include/log.h" typedef struct { char* key; char* value; } entry_t; typedef struct { char* name; list_t* entries; } section_t; struct config_t { list_t* sections; }; // Empty definition; this type is aliased to list_node_t. struct config_section_iter_t {}; static bool config_parse(FILE* fp, config_t* config); static section_t* section_new(const char* name); static void section_free(void* ptr); static section_t* section_find(const config_t* config, const char* section); static entry_t* entry_new(const char* key, const char* value); static void entry_free(void* ptr); static entry_t* entry_find(const config_t* config, const char* section, const char* key); config_t* config_new_empty(void) { config_t* config = static_cast<config_t*>(osi_calloc(sizeof(config_t))); config->sections = list_new(section_free); if (!config->sections) { LOG_ERROR(LOG_TAG, "%s unable to allocate list for sections.", __func__); goto error; } return config; error:; config_free(config); return NULL; } config_t* config_new(const char* filename) { CHECK(filename != NULL); config_t* config = config_new_empty(); if (!config) return NULL; FILE* fp = fopen(filename, "rt"); if (!fp) { LOG_ERROR(LOG_TAG, "%s unable to open file '%s': %s", __func__, filename, strerror(errno)); config_free(config); return NULL; } if (!config_parse(fp, config)) { config_free(config); config = NULL; } fclose(fp); return config; } config_t* config_new_clone(const config_t* src) { CHECK(src != NULL); config_t* ret = config_new_empty(); CHECK(ret != NULL); for (const list_node_t* node = list_begin(src->sections); node != list_end(src->sections); node = list_next(node)) { section_t* sec = static_cast<section_t*>(list_node(node)); for (const list_node_t* node_entry = list_begin(sec->entries); node_entry != list_end(sec->entries); node_entry = list_next(node_entry)) { entry_t* entry = static_cast<entry_t*>(list_node(node_entry)); config_set_string(ret, sec->name, entry->key, entry->value); } } return ret; } void config_free(config_t* config) { if (!config) return; list_free(config->sections); osi_free(config); } bool config_has_section(const config_t* config, const char* section) { CHECK(config != NULL); CHECK(section != NULL); return (section_find(config, section) != NULL); } bool config_has_key(const config_t* config, const char* section, const char* key) { CHECK(config != NULL); CHECK(section != NULL); CHECK(key != NULL); return (entry_find(config, section, key) != NULL); } int config_get_int(const config_t* config, const char* section, const char* key, int def_value) { CHECK(config != NULL); CHECK(section != NULL); CHECK(key != NULL); entry_t* entry = entry_find(config, section, key); if (!entry) return def_value; char* endptr; int ret = strtol(entry->value, &endptr, 0); return (*endptr == '\0') ? ret : def_value; } bool config_get_bool(const config_t* config, const char* section, const char* key, bool def_value) { CHECK(config != NULL); CHECK(section != NULL); CHECK(key != NULL); entry_t* entry = entry_find(config, section, key); if (!entry) return def_value; if (!strcmp(entry->value, "true")) return true; if (!strcmp(entry->value, "false")) return false; return def_value; } const char* config_get_string(const config_t* config, const char* section, const char* key, const char* def_value) { CHECK(config != NULL); CHECK(section != NULL); CHECK(key != NULL); entry_t* entry = entry_find(config, section, key); if (!entry) return def_value; return entry->value; } void config_set_int(config_t* config, const char* section, const char* key, int value) { CHECK(config != NULL); CHECK(section != NULL); CHECK(key != NULL); char value_str[32] = {0}; snprintf(value_str, sizeof(value_str), "%d", value); config_set_string(config, section, key, value_str); } void config_set_bool(config_t* config, const char* section, const char* key, bool value) { CHECK(config != NULL); CHECK(section != NULL); CHECK(key != NULL); config_set_string(config, section, key, value ? "true" : "false"); } void config_set_string(config_t* config, const char* section, const char* key, const char* value) { section_t* sec = section_find(config, section); if (!sec) { sec = section_new(section); list_append(config->sections, sec); } for (const list_node_t* node = list_begin(sec->entries); node != list_end(sec->entries); node = list_next(node)) { entry_t* entry = static_cast<entry_t*>(list_node(node)); if (!strcmp(entry->key, key)) { osi_free(entry->value); entry->value = osi_strdup(value); return; } } entry_t* entry = entry_new(key, value); list_append(sec->entries, entry); } bool config_remove_section(config_t* config, const char* section) { CHECK(config != NULL); CHECK(section != NULL); section_t* sec = section_find(config, section); if (!sec) return false; return list_remove(config->sections, sec); } bool config_remove_key(config_t* config, const char* section, const char* key) { CHECK(config != NULL); CHECK(section != NULL); CHECK(key != NULL); section_t* sec = section_find(config, section); entry_t* entry = entry_find(config, section, key); if (!sec || !entry) return false; return list_remove(sec->entries, entry); } const config_section_node_t* config_section_begin(const config_t* config) { CHECK(config != NULL); return (const config_section_node_t*)list_begin(config->sections); } const config_section_node_t* config_section_end(const config_t* config) { CHECK(config != NULL); return (const config_section_node_t*)list_end(config->sections); } const config_section_node_t* config_section_next( const config_section_node_t* node) { CHECK(node != NULL); return (const config_section_node_t*)list_next((const list_node_t*)node); } const char* config_section_name(const config_section_node_t* node) { CHECK(node != NULL); const list_node_t* lnode = (const list_node_t*)node; const section_t* section = (const section_t*)list_node(lnode); return section->name; } bool config_save(const config_t* config, const char* filename) { CHECK(config != NULL); CHECK(filename != NULL); CHECK(*filename != '\0'); // Steps to ensure content of config file gets to disk: // // 1) Open and write to temp file (e.g. bt_config.conf.new). // 2) Sync the temp file to disk with fsync(). // 3) Rename temp file to actual config file (e.g. bt_config.conf). // This ensures atomic update. // 4) Sync directory that has the conf file with fsync(). // This ensures directory entries are up-to-date. int dir_fd = -1; FILE* fp = NULL; // Build temp config file based on config file (e.g. bt_config.conf.new). static const char* temp_file_ext = ".new"; const int filename_len = strlen(filename); const int temp_filename_len = filename_len + strlen(temp_file_ext) + 1; char* temp_filename = static_cast<char*>(osi_calloc(temp_filename_len)); snprintf(temp_filename, temp_filename_len, "%s%s", filename, temp_file_ext); // Extract directory from file path (e.g. /data/misc/bluedroid). char* temp_dirname = osi_strdup(filename); const char* directoryname = dirname(temp_dirname); if (!directoryname) { LOG_ERROR(LOG_TAG, "%s error extracting directory from '%s': %s", __func__, filename, strerror(errno)); goto error; } dir_fd = open(directoryname, O_RDONLY); if (dir_fd < 0) { LOG_ERROR(LOG_TAG, "%s unable to open dir '%s': %s", __func__, directoryname, strerror(errno)); goto error; } fp = fopen(temp_filename, "wt"); if (!fp) { LOG_ERROR(LOG_TAG, "%s unable to write file '%s': %s", __func__, temp_filename, strerror(errno)); goto error; } for (const list_node_t* node = list_begin(config->sections); node != list_end(config->sections); node = list_next(node)) { const section_t* section = (const section_t*)list_node(node); if (fprintf(fp, "[%s]\n", section->name) < 0) { LOG_ERROR(LOG_TAG, "%s unable to write to file '%s': %s", __func__, temp_filename, strerror(errno)); goto error; } for (const list_node_t* enode = list_begin(section->entries); enode != list_end(section->entries); enode = list_next(enode)) { const entry_t* entry = (const entry_t*)list_node(enode); if (fprintf(fp, "%s = %s\n", entry->key, entry->value) < 0) { LOG_ERROR(LOG_TAG, "%s unable to write to file '%s': %s", __func__, temp_filename, strerror(errno)); goto error; } } // Only add a separating newline if there are more sections. if (list_next(node) != list_end(config->sections)) { if (fputc('\n', fp) == EOF) { LOG_ERROR(LOG_TAG, "%s unable to write to file '%s': %s", __func__, temp_filename, strerror(errno)); goto error; } } } // Sync written temp file out to disk. fsync() is blocking until data makes it // to disk. if (fsync(fileno(fp)) < 0) { LOG_WARN(LOG_TAG, "%s unable to fsync file '%s': %s", __func__, temp_filename, strerror(errno)); } if (fclose(fp) == EOF) { LOG_ERROR(LOG_TAG, "%s unable to close file '%s': %s", __func__, temp_filename, strerror(errno)); goto error; } fp = NULL; // Change the file's permissions to Read/Write by User and Group if (chmod(temp_filename, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP) == -1) { LOG_ERROR(LOG_TAG, "%s unable to change file permissions '%s': %s", __func__, filename, strerror(errno)); goto error; } // Rename written temp file to the actual config file. if (rename(temp_filename, filename) == -1) { LOG_ERROR(LOG_TAG, "%s unable to commit file '%s': %s", __func__, filename, strerror(errno)); goto error; } // This should ensure the directory is updated as well. if (fsync(dir_fd) < 0) { LOG_WARN(LOG_TAG, "%s unable to fsync dir '%s': %s", __func__, directoryname, strerror(errno)); } if (close(dir_fd) < 0) { LOG_ERROR(LOG_TAG, "%s unable to close dir '%s': %s", __func__, directoryname, strerror(errno)); goto error; } osi_free(temp_filename); osi_free(temp_dirname); return true; error: // This indicates there is a write issue. Unlink as partial data is not // acceptable. unlink(temp_filename); if (fp) fclose(fp); if (dir_fd != -1) close(dir_fd); osi_free(temp_filename); osi_free(temp_dirname); return false; } static char* trim(char* str) { while (isspace(*str)) ++str; if (!*str) return str; char* end_str = str + strlen(str) - 1; while (end_str > str && isspace(*end_str)) --end_str; end_str[1] = '\0'; return str; } static bool config_parse(FILE* fp, config_t* config) { CHECK(fp != NULL); CHECK(config != NULL); int line_num = 0; char line[1024]; char section[1024]; strcpy(section, CONFIG_DEFAULT_SECTION); while (fgets(line, sizeof(line), fp)) { char* line_ptr = trim(line); ++line_num; // Skip blank and comment lines. if (*line_ptr == '\0' || *line_ptr == '#') continue; if (*line_ptr == '[') { size_t len = strlen(line_ptr); if (line_ptr[len - 1] != ']') { LOG_DEBUG(LOG_TAG, "%s unterminated section name on line %d.", __func__, line_num); return false; } strncpy(section, line_ptr + 1, len - 2); section[len - 2] = '\0'; } else { char* split = strchr(line_ptr, '='); if (!split) { LOG_DEBUG(LOG_TAG, "%s no key/value separator found on line %d.", __func__, line_num); return false; } *split = '\0'; config_set_string(config, section, trim(line_ptr), trim(split + 1)); } } return true; } static section_t* section_new(const char* name) { section_t* section = static_cast<section_t*>(osi_calloc(sizeof(section_t))); section->name = osi_strdup(name); section->entries = list_new(entry_free); return section; } static void section_free(void* ptr) { if (!ptr) return; section_t* section = static_cast<section_t*>(ptr); osi_free(section->name); list_free(section->entries); osi_free(section); } static section_t* section_find(const config_t* config, const char* section) { for (const list_node_t* node = list_begin(config->sections); node != list_end(config->sections); node = list_next(node)) { section_t* sec = static_cast<section_t*>(list_node(node)); if (!strcmp(sec->name, section)) return sec; } return NULL; } static entry_t* entry_new(const char* key, const char* value) { entry_t* entry = static_cast<entry_t*>(osi_calloc(sizeof(entry_t))); entry->key = osi_strdup(key); entry->value = osi_strdup(value); return entry; } static void entry_free(void* ptr) { if (!ptr) return; entry_t* entry = static_cast<entry_t*>(ptr); osi_free(entry->key); osi_free(entry->value); osi_free(entry); } static entry_t* entry_find(const config_t* config, const char* section, const char* key) { section_t* sec = section_find(config, section); if (!sec) return NULL; for (const list_node_t* node = list_begin(sec->entries); node != list_end(sec->entries); node = list_next(node)) { entry_t* entry = static_cast<entry_t*>(list_node(node)); if (!strcmp(entry->key, key)) return entry; } return NULL; }