blob: c5b6b897384e80e945d12787e2459330f059ac08 [file] [log] [blame]
/******************************************************************************
*
* Copyright 2017 The Android Open Source Project
*
* 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.
*
******************************************************************************/
#include "osi/include/config.h"
#include <base/files/file_path.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 <sstream>
#include <type_traits>
// Empty definition; this type is aliased to list_node_t.
struct config_section_iter_t {};
static bool config_parse(FILE* fp, config_t* config);
template <typename T,
class = typename std::enable_if<std::is_same<
config_t, typename std::remove_const<T>::type>::value>>
static auto section_find(T& config, const std::string& section) {
return std::find_if(
config.sections.begin(), config.sections.end(),
[&section](const section_t& sec) { return sec.name == section; });
}
static const entry_t* entry_find(const config_t& config,
const std::string& section,
const std::string& key) {
auto sec = section_find(config, section);
if (sec == config.sections.end()) return nullptr;
for (const entry_t& entry : sec->entries) {
if (entry.key == key) return &entry;
}
return nullptr;
}
std::unique_ptr<config_t> config_new_empty(void) {
return std::make_unique<config_t>();
}
std::unique_ptr<config_t> config_new(const char* filename) {
CHECK(filename != nullptr);
std::unique_ptr<config_t> config = config_new_empty();
FILE* fp = fopen(filename, "rt");
if (!fp) {
LOG(ERROR) << __func__ << ": unable to open file '" << filename
<< "': " << strerror(errno);
return nullptr;
}
if (!config_parse(fp, config.get())) {
config.reset();
}
fclose(fp);
return config;
}
std::unique_ptr<config_t> config_new_clone(const config_t& src) {
std::unique_ptr<config_t> ret = config_new_empty();
for (const section_t& sec : src.sections) {
for (const entry_t& entry : sec.entries) {
config_set_string(ret.get(), sec.name, entry.key, entry.value);
}
}
return ret;
}
bool config_has_section(const config_t& config, const std::string& section) {
return (section_find(config, section) != config.sections.end());
}
bool config_has_key(const config_t& config, const std::string& section,
const std::string& key) {
return (entry_find(config, section, key) != nullptr);
}
int config_get_int(const config_t& config, const std::string& section,
const std::string& key, int def_value) {
const entry_t* entry = entry_find(config, section, key);
if (!entry) return def_value;
char* endptr;
int ret = strtol(entry->value.c_str(), &endptr, 0);
return (*endptr == '\0') ? ret : def_value;
}
bool config_get_bool(const config_t& config, const std::string& section,
const std::string& key, bool def_value) {
const entry_t* entry = entry_find(config, section, key);
if (!entry) return def_value;
if (entry->value == "true") return true;
if (entry->value == "false") return false;
return def_value;
}
const std::string* config_get_string(const config_t& config,
const std::string& section,
const std::string& key,
const std::string* def_value) {
const entry_t* entry = entry_find(config, section, key);
if (!entry) return def_value;
return &entry->value;
}
void config_set_int(config_t* config, const std::string& section,
const std::string& key, int value) {
config_set_string(config, section, key, std::to_string(value));
}
void config_set_bool(config_t* config, const std::string& section,
const std::string& key, bool value) {
config_set_string(config, section, key, value ? "true" : "false");
}
void config_set_string(config_t* config, const std::string& section,
const std::string& key, const std::string& value) {
CHECK(config);
auto sec = section_find(*config, section);
if (sec == config->sections.end()) {
config->sections.emplace_back(section_t{.name = section});
sec = std::prev(config->sections.end());
}
for (entry_t& entry : sec->entries) {
if (entry.key == key) {
entry.value = value;
return;
}
}
sec->entries.emplace_back(entry_t{.key = key, .value = value});
}
bool config_remove_section(config_t* config, const std::string& section) {
CHECK(config);
auto sec = section_find(*config, section);
if (sec == config->sections.end()) return false;
config->sections.erase(sec);
return true;
}
bool config_remove_key(config_t* config, const std::string& section,
const std::string& key) {
CHECK(config);
auto sec = section_find(*config, section);
if (sec == config->sections.end()) return false;
for (auto entry = sec->entries.begin(); entry != sec->entries.end();
++entry) {
if (entry->key == key) {
sec->entries.erase(entry);
return true;
}
}
return false;
}
bool config_save(const config_t& config, const std::string& filename) {
CHECK(!filename.empty());
// 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 = nullptr;
std::stringstream serialized;
// Build temp config file based on config file (e.g. bt_config.conf.new).
const std::string temp_filename = filename + ".new";
// Extract directory from file path (e.g. /data/misc/bluedroid).
const std::string& directoryname = base::FilePath(filename).DirName().value();
if (directoryname.empty()) {
LOG(ERROR) << __func__ << ": error extracting directory from '" << filename
<< "': " << strerror(errno);
goto error;
}
dir_fd = open(directoryname.c_str(), O_RDONLY);
if (dir_fd < 0) {
LOG(ERROR) << __func__ << ": unable to open dir '" << directoryname
<< "': " << strerror(errno);
goto error;
}
fp = fopen(temp_filename.c_str(), "wt");
if (!fp) {
LOG(ERROR) << __func__ << ": unable to write to file '" << temp_filename
<< "': " << strerror(errno);
goto error;
}
for (const section_t& section : config.sections) {
serialized << "[" << section.name << "]" << std::endl;
for (const entry_t& entry : section.entries)
serialized << entry.key << " = " << entry.value << std::endl;
serialized << std::endl;
}
if (fprintf(fp, "%s", serialized.str().c_str()) < 0) {
LOG(ERROR) << __func__ << ": unable to write to file '" << 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(WARNING) << __func__ << ": unable to fsync file '" << temp_filename
<< "': " << strerror(errno);
}
if (fclose(fp) == EOF) {
LOG(ERROR) << __func__ << ": unable to close file '" << temp_filename
<< "': " << strerror(errno);
goto error;
}
fp = nullptr;
// Change the file's permissions to Read/Write by User and Group
if (chmod(temp_filename.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP) ==
-1) {
LOG(ERROR) << __func__ << ": unable to change file permissions '"
<< filename << "': " << strerror(errno);
goto error;
}
// Rename written temp file to the actual config file.
if (rename(temp_filename.c_str(), filename.c_str()) == -1) {
LOG(ERROR) << __func__ << ": unable to commit file '" << filename
<< "': " << strerror(errno);
goto error;
}
// This should ensure the directory is updated as well.
if (fsync(dir_fd) < 0) {
LOG(WARNING) << __func__ << ": unable to fsync dir '" << directoryname
<< "': " << strerror(errno);
}
if (close(dir_fd) < 0) {
LOG(ERROR) << __func__ << ": unable to close dir '" << directoryname
<< "': " << strerror(errno);
goto error;
}
return true;
error:
// This indicates there is a write issue. Unlink as partial data is not
// acceptable.
unlink(temp_filename.c_str());
if (fp) fclose(fp);
if (dir_fd != -1) close(dir_fd);
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 != nullptr);
CHECK(config != nullptr);
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] != ']') {
VLOG(1) << __func__ << ": unterminated section name on line "
<< line_num;
return false;
}
strncpy(section, line_ptr + 1, len - 2); // NOLINT (len < 1024)
section[len - 2] = '\0';
} else {
char* split = strchr(line_ptr, '=');
if (!split) {
VLOG(1) << __func__ << ": no key/value separator found on line "
<< line_num;
return false;
}
*split = '\0';
config_set_string(config, section, trim(line_ptr), trim(split + 1));
}
}
return true;
}