mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-22 17:35:59 +03:00
1697 lines
45 KiB
C
1697 lines
45 KiB
C
/*
|
|
* Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
|
|
* Copyright (C) 2004-2011 Red Hat, Inc. All rights reserved.
|
|
*
|
|
* This file is part of LVM2.
|
|
*
|
|
* This copyrighted material is made available to anyone wishing to use,
|
|
* modify, copy, or redistribute it subject to the terms and conditions
|
|
* of the GNU Lesser General Public License v.2.1.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
|
|
#include "lib.h"
|
|
|
|
#include "config.h"
|
|
#include "crc.h"
|
|
#include "device.h"
|
|
#include "str_list.h"
|
|
#include "toolcontext.h"
|
|
#include "lvm-file.h"
|
|
#include "memlock.h"
|
|
|
|
#include <sys/stat.h>
|
|
#include <sys/mman.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <assert.h>
|
|
#include <ctype.h>
|
|
|
|
static const char *_config_source_names[] = {
|
|
[CONFIG_UNDEFINED] = "undefined",
|
|
[CONFIG_FILE] = "file",
|
|
[CONFIG_MERGED_FILES] = "merged files",
|
|
[CONFIG_STRING] = "string",
|
|
[CONFIG_PROFILE] = "profile"
|
|
};
|
|
|
|
struct config_file {
|
|
off_t st_size;
|
|
char *filename;
|
|
int exists;
|
|
int keep_open;
|
|
struct device *dev;
|
|
};
|
|
|
|
struct config_source {
|
|
config_source_t type;
|
|
time_t timestamp;
|
|
union {
|
|
struct config_file *file;
|
|
struct config_file *profile;
|
|
} source;
|
|
};
|
|
|
|
/*
|
|
* Map each ID to respective definition of the configuration item.
|
|
*/
|
|
static struct cfg_def_item _cfg_def_items[CFG_COUNT + 1] = {
|
|
#define cfg_section(id, name, parent, flags, since_version, comment) {id, parent, name, CFG_TYPE_SECTION, {0}, flags, since_version, comment},
|
|
#define cfg(id, name, parent, flags, type, default_value, since_version, comment) {id, parent, name, type, {.v_##type = default_value}, flags, since_version, comment},
|
|
#define cfg_runtime(id, name, parent, flags, type, since_version, comment) {id, parent, name, type, {.fn_##type = get_default_##id}, flags | CFG_DEFAULT_RUN_TIME, since_version, comment},
|
|
#define cfg_array(id, name, parent, flags, types, default_value, since_version, comment) {id, parent, name, CFG_TYPE_ARRAY | types, {.v_CFG_TYPE_STRING = default_value}, flags, since_version, comment},
|
|
#define cfg_array_runtime(id, name, parent, flags, types, since_version, comment) {id, parent, name, CFG_TYPE_ARRAY | types, {.fn_CFG_TYPE_STRING = get_default_##id}, flags | CFG_DEFAULT_RUN_TIME, since_version, comment},
|
|
#include "config_settings.h"
|
|
#undef cfg_section
|
|
#undef cfg
|
|
#undef cfg_runtime
|
|
#undef cfg_array
|
|
#undef cfg_array_runtime
|
|
};
|
|
|
|
config_source_t config_get_source_type(struct dm_config_tree *cft)
|
|
{
|
|
struct config_source *cs = dm_config_get_custom(cft);
|
|
return cs ? cs->type : CONFIG_UNDEFINED;
|
|
}
|
|
|
|
/*
|
|
* public interface
|
|
*/
|
|
struct dm_config_tree *config_open(config_source_t source,
|
|
const char *filename,
|
|
int keep_open)
|
|
{
|
|
struct dm_config_tree *cft = dm_config_create();
|
|
struct config_source *cs;
|
|
struct config_file *cf;
|
|
|
|
if (!cft)
|
|
return NULL;
|
|
|
|
if (!(cs = dm_pool_zalloc(cft->mem, sizeof(struct config_source)))) {
|
|
log_error("Failed to allocate config source.");
|
|
goto fail;
|
|
}
|
|
|
|
if ((source == CONFIG_FILE) || (source == CONFIG_PROFILE)) {
|
|
if (!(cf = dm_pool_zalloc(cft->mem, sizeof(struct config_file)))) {
|
|
log_error("Failed to allocate config file.");
|
|
goto fail;
|
|
}
|
|
|
|
cf->keep_open = keep_open;
|
|
if (filename &&
|
|
!(cf->filename = dm_pool_strdup(cft->mem, filename))) {
|
|
log_error("Failed to duplicate filename.");
|
|
goto fail;
|
|
}
|
|
|
|
cs->source.file = cf;
|
|
}
|
|
|
|
cs->type = source;
|
|
dm_config_set_custom(cft, cs);
|
|
return cft;
|
|
fail:
|
|
dm_config_destroy(cft);
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Doesn't populate filename if the file is empty.
|
|
*/
|
|
int config_file_check(struct dm_config_tree *cft, const char **filename, struct stat *info)
|
|
{
|
|
struct config_source *cs = dm_config_get_custom(cft);
|
|
struct config_file *cf;
|
|
struct stat _info;
|
|
|
|
if ((cs->type != CONFIG_FILE) && (cs->type != CONFIG_PROFILE)) {
|
|
log_error(INTERNAL_ERROR "config_file_check: expected file or profile config source, "
|
|
"found %s config source.", _config_source_names[cs->type]);
|
|
return 0;
|
|
}
|
|
|
|
if (!info)
|
|
info = &_info;
|
|
|
|
cf = cs->source.file;
|
|
|
|
if (stat(cf->filename, info)) {
|
|
log_sys_error("stat", cf->filename);
|
|
cf->exists = 0;
|
|
return 0;
|
|
}
|
|
|
|
if (!S_ISREG(info->st_mode)) {
|
|
log_error("%s is not a regular file", cf->filename);
|
|
cf->exists = 0;
|
|
return 0;
|
|
}
|
|
|
|
cs->timestamp = info->st_ctime;
|
|
cf->exists = 1;
|
|
cf->st_size = info->st_size;
|
|
|
|
if (info->st_size == 0)
|
|
log_verbose("%s is empty", cf->filename);
|
|
else if (filename)
|
|
*filename = cf->filename;
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Return 1 if config files ought to be reloaded
|
|
*/
|
|
int config_file_changed(struct dm_config_tree *cft)
|
|
{
|
|
struct config_source *cs = dm_config_get_custom(cft);
|
|
struct config_file *cf;
|
|
struct stat info;
|
|
|
|
if (cs->type != CONFIG_FILE) {
|
|
log_error(INTERNAL_ERROR "config_file_changed: expected file config source, "
|
|
"found %s config source.", _config_source_names[cs->type]);
|
|
return 0;
|
|
}
|
|
|
|
cf = cs->source.file;
|
|
|
|
if (!cf->filename)
|
|
return 0;
|
|
|
|
if (stat(cf->filename, &info) == -1) {
|
|
/* Ignore a deleted config file: still use original data */
|
|
if (errno == ENOENT) {
|
|
if (!cf->exists)
|
|
return 0;
|
|
log_very_verbose("Config file %s has disappeared!",
|
|
cf->filename);
|
|
goto reload;
|
|
}
|
|
log_sys_error("stat", cf->filename);
|
|
log_error("Failed to reload configuration files");
|
|
return 0;
|
|
}
|
|
|
|
if (!S_ISREG(info.st_mode)) {
|
|
log_error("Configuration file %s is not a regular file",
|
|
cf->filename);
|
|
goto reload;
|
|
}
|
|
|
|
/* Unchanged? */
|
|
if (cs->timestamp == info.st_ctime && cf->st_size == info.st_size)
|
|
return 0;
|
|
|
|
reload:
|
|
log_verbose("Detected config file change to %s", cf->filename);
|
|
return 1;
|
|
}
|
|
|
|
void config_destroy(struct dm_config_tree *cft)
|
|
{
|
|
struct config_source *cs;
|
|
struct config_file *cf;
|
|
|
|
if (!cft)
|
|
return;
|
|
|
|
cs = dm_config_get_custom(cft);
|
|
|
|
if ((cs->type == CONFIG_FILE) || (cs->type == CONFIG_PROFILE)) {
|
|
cf = cs->source.file;
|
|
if (cf && cf->dev)
|
|
if (!dev_close(cf->dev))
|
|
stack;
|
|
}
|
|
|
|
dm_config_destroy(cft);
|
|
}
|
|
|
|
struct dm_config_tree *config_file_open_and_read(const char *config_file,
|
|
config_source_t source)
|
|
{
|
|
struct dm_config_tree *cft;
|
|
struct stat info;
|
|
|
|
if (!(cft = config_open(source, config_file, 0))) {
|
|
log_error("config_tree allocation failed");
|
|
return NULL;
|
|
}
|
|
|
|
/* Is there a config file? */
|
|
if (stat(config_file, &info) == -1) {
|
|
/* Profile file must be present! */
|
|
if (errno == ENOENT && (source != CONFIG_PROFILE))
|
|
return cft;
|
|
log_sys_error("stat", config_file);
|
|
goto bad;
|
|
}
|
|
|
|
log_very_verbose("Loading config file: %s", config_file);
|
|
if (!config_file_read(cft)) {
|
|
log_error("Failed to load config file %s", config_file);
|
|
goto bad;
|
|
}
|
|
|
|
return cft;
|
|
bad:
|
|
config_destroy(cft);
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Returns config tree if it was removed.
|
|
*/
|
|
struct dm_config_tree *remove_config_tree_by_source(struct cmd_context *cmd,
|
|
config_source_t source)
|
|
{
|
|
struct dm_config_tree *previous_cft = NULL;
|
|
struct dm_config_tree *cft = cmd->cft;
|
|
struct config_source *cs;
|
|
|
|
while (cft) {
|
|
cs = dm_config_get_custom(cft);
|
|
if (cs && (cs->type == source)) {
|
|
if (previous_cft) {
|
|
previous_cft->cascade = cft->cascade;
|
|
cmd->cft = previous_cft;
|
|
} else
|
|
cmd->cft = cft->cascade;
|
|
cft->cascade = NULL;
|
|
break;
|
|
}
|
|
previous_cft = cft;
|
|
cft = cft->cascade;
|
|
}
|
|
|
|
return cft;
|
|
}
|
|
|
|
int override_config_tree_from_string(struct cmd_context *cmd,
|
|
const char *config_settings)
|
|
{
|
|
struct dm_config_tree *cft_new;
|
|
struct config_source *cs = dm_config_get_custom(cmd->cft);
|
|
|
|
/*
|
|
* Follow this sequence:
|
|
* CONFIG_STRING -> CONFIG_PROFILE -> CONFIG_FILE/CONFIG_MERGED_FILES
|
|
*/
|
|
|
|
if (cs->type == CONFIG_STRING) {
|
|
log_error(INTERNAL_ERROR "override_config_tree_from_string: "
|
|
"config cascade already contains a string config.");
|
|
return 0;
|
|
}
|
|
|
|
if (!(cft_new = dm_config_from_string(config_settings))) {
|
|
log_error("Failed to set overridden configuration entries.");
|
|
return 0;
|
|
}
|
|
|
|
if (!(cs = dm_pool_zalloc(cft_new->mem, sizeof(struct config_source)))) {
|
|
log_error("Failed to allocate config source.");
|
|
dm_config_destroy(cft_new);
|
|
return 0;
|
|
}
|
|
|
|
cs->type = CONFIG_STRING;
|
|
dm_config_set_custom(cft_new, cs);
|
|
|
|
cmd->cft = dm_config_insert_cascaded_tree(cft_new, cmd->cft);
|
|
|
|
return 1;
|
|
}
|
|
|
|
int override_config_tree_from_profile(struct cmd_context *cmd,
|
|
struct profile *profile)
|
|
{
|
|
struct dm_config_tree *cft = cmd->cft, *cft_string = NULL;
|
|
struct config_source *cs = dm_config_get_custom(cft);
|
|
|
|
/*
|
|
* Follow this sequence:
|
|
* CONFIG_STRING -> CONFIG_PROFILE -> CONFIG_FILE/CONFIG_MERGED_FILES
|
|
*/
|
|
|
|
if (!profile->cft && !load_profile(cmd, profile))
|
|
return_0;
|
|
|
|
if (cs->type == CONFIG_STRING) {
|
|
cft_string = cft;
|
|
cft = cft->cascade;
|
|
cs = dm_config_get_custom(cft);
|
|
if (cs->type == CONFIG_PROFILE) {
|
|
log_error(INTERNAL_ERROR "override_config_tree_from_profile: "
|
|
"config cascade already contains a profile config.");
|
|
return 0;
|
|
}
|
|
dm_config_insert_cascaded_tree(cft_string, profile->cft);
|
|
}
|
|
|
|
cmd->cft = dm_config_insert_cascaded_tree(profile->cft, cft);
|
|
|
|
cmd->cft = cft_string ? : profile->cft;
|
|
|
|
return 1;
|
|
}
|
|
|
|
int config_file_read_fd(struct dm_config_tree *cft, struct device *dev,
|
|
off_t offset, size_t size, off_t offset2, size_t size2,
|
|
checksum_fn_t checksum_fn, uint32_t checksum)
|
|
{
|
|
char *fb, *fe;
|
|
int r = 0;
|
|
int use_mmap = 1;
|
|
off_t mmap_offset = 0;
|
|
char *buf = NULL;
|
|
struct config_source *cs = dm_config_get_custom(cft);
|
|
|
|
if ((cs->type != CONFIG_FILE) && (cs->type != CONFIG_PROFILE)) {
|
|
log_error(INTERNAL_ERROR "config_file_read_fd: expected file or profile config source, "
|
|
"found %s config source.", _config_source_names[cs->type]);
|
|
return 0;
|
|
}
|
|
|
|
/* Only use mmap with regular files */
|
|
if (!(dev->flags & DEV_REGULAR) || size2)
|
|
use_mmap = 0;
|
|
|
|
if (use_mmap) {
|
|
mmap_offset = offset % lvm_getpagesize();
|
|
/* memory map the file */
|
|
fb = mmap((caddr_t) 0, size + mmap_offset, PROT_READ,
|
|
MAP_PRIVATE, dev_fd(dev), offset - mmap_offset);
|
|
if (fb == (caddr_t) (-1)) {
|
|
log_sys_error("mmap", dev_name(dev));
|
|
goto out;
|
|
}
|
|
fb = fb + mmap_offset;
|
|
} else {
|
|
if (!(buf = dm_malloc(size + size2))) {
|
|
log_error("Failed to allocate circular buffer.");
|
|
return 0;
|
|
}
|
|
if (!dev_read_circular(dev, (uint64_t) offset, size,
|
|
(uint64_t) offset2, size2, buf)) {
|
|
goto out;
|
|
}
|
|
fb = buf;
|
|
}
|
|
|
|
if (checksum_fn && checksum !=
|
|
(checksum_fn(checksum_fn(INITIAL_CRC, (const uint8_t *)fb, size),
|
|
(const uint8_t *)(fb + size), size2))) {
|
|
log_error("%s: Checksum error", dev_name(dev));
|
|
goto out;
|
|
}
|
|
|
|
fe = fb + size + size2;
|
|
if (!dm_config_parse(cft, fb, fe))
|
|
goto_out;
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
if (!use_mmap)
|
|
dm_free(buf);
|
|
else {
|
|
/* unmap the file */
|
|
if (munmap(fb - mmap_offset, size + mmap_offset)) {
|
|
log_sys_error("munmap", dev_name(dev));
|
|
r = 0;
|
|
}
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
int config_file_read(struct dm_config_tree *cft)
|
|
{
|
|
const char *filename = NULL;
|
|
struct config_source *cs = dm_config_get_custom(cft);
|
|
struct config_file *cf;
|
|
struct stat info;
|
|
int r;
|
|
|
|
if (!config_file_check(cft, &filename, &info))
|
|
return_0;
|
|
|
|
/* Nothing to do. E.g. empty file. */
|
|
if (!filename)
|
|
return 1;
|
|
|
|
cf = cs->source.file;
|
|
|
|
if (!cf->dev) {
|
|
if (!(cf->dev = dev_create_file(filename, NULL, NULL, 1)))
|
|
return_0;
|
|
|
|
if (!dev_open_readonly_buffered(cf->dev))
|
|
return_0;
|
|
}
|
|
|
|
r = config_file_read_fd(cft, cf->dev, 0, (size_t) info.st_size, 0, 0,
|
|
(checksum_fn_t) NULL, 0);
|
|
|
|
if (!cf->keep_open) {
|
|
if (!dev_close(cf->dev))
|
|
stack;
|
|
cf->dev = NULL;
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
time_t config_file_timestamp(struct dm_config_tree *cft)
|
|
{
|
|
struct config_source *cs = dm_config_get_custom(cft);
|
|
return cs->timestamp;
|
|
}
|
|
|
|
#define cfg_def_get_item_p(id) (&_cfg_def_items[id])
|
|
#define cfg_def_get_default_value_hint(cmd,item,type,profile) ((item->flags & CFG_DEFAULT_RUN_TIME) ? item->default_value.fn_##type(cmd,profile) : item->default_value.v_##type)
|
|
#define cfg_def_get_default_value(cmd,item,type,profile) (item->flags & CFG_DEFAULT_UNDEFINED ? 0 : cfg_def_get_default_value_hint(cmd,item,type,profile))
|
|
|
|
static int _cfg_def_make_path(char *buf, size_t buf_size, int id, cfg_def_item_t *item, int xlate)
|
|
{
|
|
int variable = item->flags & CFG_NAME_VARIABLE;
|
|
int parent_id = item->parent;
|
|
int count, n;
|
|
|
|
if (id == parent_id)
|
|
return 0;
|
|
|
|
count = _cfg_def_make_path(buf, buf_size, parent_id, cfg_def_get_item_p(parent_id), xlate);
|
|
if ((n = dm_snprintf(buf + count, buf_size - count, "%s%s%s%s",
|
|
count ? "/" : "",
|
|
xlate && variable ? "<" : "",
|
|
!xlate && variable ? "#" : item->name,
|
|
xlate && variable ? ">" : "")) < 0) {
|
|
log_error(INTERNAL_ERROR "_cfg_def_make_path: supplied buffer too small for %s/%s",
|
|
cfg_def_get_item_p(parent_id)->name, item->name);
|
|
buf[0] = '\0';
|
|
return 0;
|
|
}
|
|
|
|
return count + n;
|
|
}
|
|
|
|
int config_def_get_path(char *buf, size_t buf_size, int id)
|
|
{
|
|
return _cfg_def_make_path(buf, buf_size, id, cfg_def_get_item_p(id), 0);
|
|
}
|
|
|
|
static void _get_type_name(char *buf, size_t buf_size, cfg_def_type_t type)
|
|
{
|
|
(void) dm_snprintf(buf, buf_size, "%s%s%s%s%s%s",
|
|
(type & CFG_TYPE_ARRAY) ?
|
|
((type & ~CFG_TYPE_ARRAY) ?
|
|
" array with values of type:" : " array") : "",
|
|
(type & CFG_TYPE_SECTION) ? " section" : "",
|
|
(type & CFG_TYPE_BOOL) ? " boolean" : "",
|
|
(type & CFG_TYPE_INT) ? " integer" : "",
|
|
(type & CFG_TYPE_FLOAT) ? " float" : "",
|
|
(type & CFG_TYPE_STRING) ? " string" : "");
|
|
}
|
|
|
|
static void _log_type_error(const char *path, cfg_def_type_t actual,
|
|
cfg_def_type_t expected, int suppress_messages)
|
|
{
|
|
static char actual_type_name[128];
|
|
static char expected_type_name[128];
|
|
|
|
_get_type_name(actual_type_name, sizeof(actual_type_name), actual);
|
|
_get_type_name(expected_type_name, sizeof(expected_type_name), expected);
|
|
|
|
log_warn_suppress(suppress_messages, "Configuration setting \"%s\" has invalid type. "
|
|
"Found%s, expected%s.", path,
|
|
actual_type_name, expected_type_name);
|
|
}
|
|
|
|
static int _config_def_check_node_single_value(const char *rp, const struct dm_config_value *v,
|
|
const cfg_def_item_t *def, int suppress_messages)
|
|
{
|
|
/* Check empty array first if present. */
|
|
if (v->type == DM_CFG_EMPTY_ARRAY) {
|
|
if (!(def->type & CFG_TYPE_ARRAY)) {
|
|
_log_type_error(rp, CFG_TYPE_ARRAY, def->type, suppress_messages);
|
|
return 0;
|
|
}
|
|
if (!(def->flags & CFG_ALLOW_EMPTY)) {
|
|
log_warn_suppress(suppress_messages,
|
|
"Configuration setting \"%s\" invalid. Empty value not allowed.", rp);
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
switch (v->type) {
|
|
case DM_CFG_INT:
|
|
if (!(def->type & CFG_TYPE_INT) && !(def->type & CFG_TYPE_BOOL)) {
|
|
_log_type_error(rp, CFG_TYPE_INT, def->type, suppress_messages);
|
|
return 0;
|
|
}
|
|
break;
|
|
case DM_CFG_FLOAT:
|
|
if (!(def->type & CFG_TYPE_FLOAT)) {
|
|
_log_type_error(rp, CFG_TYPE_FLOAT, def->type, suppress_messages);
|
|
return 0;
|
|
}
|
|
break;
|
|
case DM_CFG_STRING:
|
|
if (def->type & CFG_TYPE_BOOL) {
|
|
if (!dm_config_value_is_bool(v)) {
|
|
log_warn_suppress(suppress_messages,
|
|
"Configuration setting \"%s\" invalid. "
|
|
"Found string value \"%s\", "
|
|
"expected boolean value: 0/1, \"y/n\", "
|
|
"\"yes/no\", \"on/off\", "
|
|
"\"true/false\".", rp, v->v.str);
|
|
return 0;
|
|
}
|
|
} else if (!(def->type & CFG_TYPE_STRING)) {
|
|
_log_type_error(rp, CFG_TYPE_STRING, def->type, suppress_messages);
|
|
return 0;
|
|
}
|
|
break;
|
|
default: ;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _config_def_check_node_value(struct cft_check_handle *handle,
|
|
const char *rp, const struct dm_config_value *v,
|
|
const cfg_def_item_t *def)
|
|
{
|
|
if (!v) {
|
|
if (def->type != CFG_TYPE_SECTION) {
|
|
_log_type_error(rp, CFG_TYPE_SECTION, def->type, handle->suppress_messages);
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
if (v->next) {
|
|
if (!(def->type & CFG_TYPE_ARRAY)) {
|
|
_log_type_error(rp, CFG_TYPE_ARRAY, def->type, handle->suppress_messages);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
do {
|
|
if (!_config_def_check_node_single_value(rp, v, def, handle->suppress_messages))
|
|
return 0;
|
|
v = v->next;
|
|
} while (v);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _config_def_check_node(struct cft_check_handle *handle,
|
|
const char *vp, char *pvp, char *rp, char *prp,
|
|
size_t buf_size, struct dm_config_node *cn,
|
|
struct dm_hash_table *ht)
|
|
{
|
|
cfg_def_item_t *def;
|
|
int sep = vp != pvp; /* don't use '/' separator for top-level node */
|
|
|
|
if (dm_snprintf(pvp, buf_size, "%s%s", sep ? "/" : "", cn->key) < 0 ||
|
|
dm_snprintf(prp, buf_size, "%s%s", sep ? "/" : "", cn->key) < 0) {
|
|
log_error("Failed to construct path for configuration node %s.", cn->key);
|
|
return 0;
|
|
}
|
|
|
|
|
|
if (!(def = (cfg_def_item_t *) dm_hash_lookup(ht, vp))) {
|
|
/* If the node is not a section but a setting, fail now. */
|
|
if (cn->v) {
|
|
log_warn_suppress(handle->suppress_messages,
|
|
"Configuration setting \"%s\" unknown.", rp);
|
|
cn->id = -1;
|
|
return 0;
|
|
}
|
|
|
|
/* If the node is a section, try if the section name is variable. */
|
|
/* Modify virtual path vp in situ and replace the key name with a '#'. */
|
|
/* The real path without '#' is still stored in rp variable. */
|
|
pvp[sep] = '#', pvp[sep + 1] = '\0';
|
|
if (!(def = (cfg_def_item_t *) dm_hash_lookup(ht, vp))) {
|
|
log_warn_suppress(handle->suppress_messages,
|
|
"Configuration section \"%s\" unknown.", rp);
|
|
cn->id = -1;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
handle->status[def->id] |= CFG_USED;
|
|
cn->id = def->id;
|
|
|
|
if (!_config_def_check_node_value(handle, rp, cn->v, def))
|
|
return 0;
|
|
|
|
/*
|
|
* Also check whether this configuration item is allowed
|
|
* in certain types of configuration trees as in some
|
|
* the use of configuration is restricted, e.g. profiles...
|
|
*/
|
|
if (handle->source == CONFIG_PROFILE &&
|
|
!(def->flags & CFG_PROFILABLE)) {
|
|
log_warn_suppress(handle->suppress_messages,
|
|
"Configuration %s \"%s\" is not customizable by "
|
|
"a profile.", cn->v ? "option" : "section", rp);
|
|
return 0;
|
|
}
|
|
|
|
handle->status[def->id] |= CFG_VALID;
|
|
return 1;
|
|
}
|
|
|
|
static int _config_def_check_tree(struct cft_check_handle *handle,
|
|
const char *vp, char *pvp, char *rp, char *prp,
|
|
size_t buf_size, struct dm_config_node *root,
|
|
struct dm_hash_table *ht)
|
|
{
|
|
struct dm_config_node *cn;
|
|
int valid, r = 1;
|
|
size_t len;
|
|
|
|
for (cn = root->child; cn; cn = cn->sib) {
|
|
if ((valid = _config_def_check_node(handle, vp, pvp, rp, prp,
|
|
buf_size, cn, ht)) && !cn->v) {
|
|
len = strlen(rp);
|
|
valid = _config_def_check_tree(handle, vp, pvp + strlen(pvp),
|
|
rp, prp + len, buf_size - len, cn, ht);
|
|
}
|
|
if (!valid)
|
|
r = 0;
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
int config_def_check(struct cmd_context *cmd, struct cft_check_handle *handle)
|
|
{
|
|
cfg_def_item_t *def;
|
|
struct dm_config_node *cn;
|
|
char vp[CFG_PATH_MAX_LEN], rp[CFG_PATH_MAX_LEN];
|
|
size_t rplen;
|
|
int id, r = 1;
|
|
|
|
/*
|
|
* vp = virtual path, it might contain substitutes for variable parts
|
|
* of the path, used while working with the hash
|
|
* rp = real path, the real path of the config element as found in the
|
|
* configuration, used for message output
|
|
*/
|
|
|
|
/*
|
|
* If the check has already been done and 'skip_if_checked' is set,
|
|
* skip the actual check and use last result if available.
|
|
* If not available, we must do the check. The global status
|
|
* is stored in root node.
|
|
*/
|
|
if (handle->skip_if_checked && (handle->status[root_CFG_SECTION] & CFG_USED))
|
|
return handle->status[root_CFG_SECTION] & CFG_VALID;
|
|
|
|
/* Nothing to do if checks are disabled and also not forced. */
|
|
if (!handle->force_check && !find_config_tree_bool(cmd, config_checks_CFG, NULL))
|
|
return 1;
|
|
|
|
/* Clear 'used' and 'valid' status flags. */
|
|
for (id = 0; id < CFG_COUNT; id++)
|
|
handle->status[id] &= ~(CFG_USED | CFG_VALID);
|
|
|
|
/*
|
|
* Create a hash of all possible configuration
|
|
* sections and settings with full path as a key.
|
|
* If section name is variable, use '#' as a substitute.
|
|
*/
|
|
if (!cmd->cft_def_hash) {
|
|
if (!(cmd->cft_def_hash = dm_hash_create(64))) {
|
|
log_error("Failed to create configuration definition hash.");
|
|
r = 0; goto out;
|
|
}
|
|
for (id = 1; id < CFG_COUNT; id++) {
|
|
def = cfg_def_get_item_p(id);
|
|
if (!_cfg_def_make_path(vp, CFG_PATH_MAX_LEN, def->id, def, 0)) {
|
|
dm_hash_destroy(cmd->cft_def_hash);
|
|
cmd->cft_def_hash = NULL;
|
|
r = 0; goto out;
|
|
}
|
|
if (!dm_hash_insert(cmd->cft_def_hash, vp, def)) {
|
|
log_error("Failed to insert configuration to hash.");
|
|
r = 0;
|
|
goto out;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Mark this handle as used so next time we know that the check
|
|
* has already been done and so we can just reuse the previous
|
|
* status instead of running this whole check again.
|
|
*/
|
|
handle->status[root_CFG_SECTION] |= CFG_USED;
|
|
|
|
/*
|
|
* Allow only sections as top-level elements.
|
|
* Iterate top-level sections and dive deeper.
|
|
* If any of subsequent checks fails, the whole check fails.
|
|
*/
|
|
for (cn = handle->cft->root; cn; cn = cn->sib) {
|
|
if (!cn->v) {
|
|
/* top level node: vp=vp, rp=rp */
|
|
if (!_config_def_check_node(handle, vp, vp, rp, rp,
|
|
CFG_PATH_MAX_LEN,
|
|
cn, cmd->cft_def_hash)) {
|
|
r = 0; continue;
|
|
}
|
|
rplen = strlen(rp);
|
|
if (!_config_def_check_tree(handle,
|
|
vp, vp + strlen(vp),
|
|
rp, rp + rplen,
|
|
CFG_PATH_MAX_LEN - rplen,
|
|
cn, cmd->cft_def_hash))
|
|
r = 0;
|
|
} else {
|
|
log_error_suppress(handle->suppress_messages,
|
|
"Configuration setting \"%s\" invalid. "
|
|
"It's not part of any section.", cn->key);
|
|
r = 0;
|
|
}
|
|
}
|
|
out:
|
|
if (r)
|
|
handle->status[root_CFG_SECTION] |= CFG_VALID;
|
|
else
|
|
handle->status[root_CFG_SECTION] &= ~CFG_VALID;
|
|
|
|
return r;
|
|
}
|
|
|
|
const struct dm_config_node *find_config_tree_node(struct cmd_context *cmd, int id, struct profile *profile)
|
|
{
|
|
cfg_def_item_t *item = cfg_def_get_item_p(id);
|
|
char path[CFG_PATH_MAX_LEN];
|
|
int profile_applied = 0;
|
|
const struct dm_config_node *cn;
|
|
|
|
if (profile && !cmd->profile_params->global_profile)
|
|
profile_applied = override_config_tree_from_profile(cmd, profile);
|
|
|
|
_cfg_def_make_path(path, sizeof(path), item->id, item, 0);
|
|
cn = dm_config_tree_find_node(cmd->cft, path);
|
|
|
|
if (profile_applied)
|
|
remove_config_tree_by_source(cmd, CONFIG_PROFILE);
|
|
|
|
return cn;
|
|
}
|
|
|
|
const char *find_config_tree_str(struct cmd_context *cmd, int id, struct profile *profile)
|
|
{
|
|
cfg_def_item_t *item = cfg_def_get_item_p(id);
|
|
char path[CFG_PATH_MAX_LEN];
|
|
int profile_applied = 0;
|
|
const char *str;
|
|
|
|
if (profile && !cmd->profile_params->global_profile)
|
|
profile_applied = override_config_tree_from_profile(cmd, profile);
|
|
|
|
_cfg_def_make_path(path, sizeof(path), item->id, item, 0);
|
|
|
|
if (item->type != CFG_TYPE_STRING)
|
|
log_error(INTERNAL_ERROR "%s cfg tree element not declared as string.", path);
|
|
|
|
str = dm_config_tree_find_str(cmd->cft, path, cfg_def_get_default_value(cmd, item, CFG_TYPE_STRING, profile));
|
|
|
|
if (profile_applied)
|
|
remove_config_tree_by_source(cmd, CONFIG_PROFILE);
|
|
|
|
return str;
|
|
}
|
|
|
|
const char *find_config_tree_str_allow_empty(struct cmd_context *cmd, int id, struct profile *profile)
|
|
{
|
|
cfg_def_item_t *item = cfg_def_get_item_p(id);
|
|
char path[CFG_PATH_MAX_LEN];
|
|
int profile_applied = 0;
|
|
const char *str;
|
|
|
|
if (profile && !cmd->profile_params->global_profile)
|
|
profile_applied = override_config_tree_from_profile(cmd, profile);
|
|
|
|
_cfg_def_make_path(path, sizeof(path), item->id, item, 0);
|
|
|
|
if (item->type != CFG_TYPE_STRING)
|
|
log_error(INTERNAL_ERROR "%s cfg tree element not declared as string.", path);
|
|
if (!(item->flags & CFG_ALLOW_EMPTY))
|
|
log_error(INTERNAL_ERROR "%s cfg tree element not declared to allow empty values.", path);
|
|
|
|
str = dm_config_tree_find_str_allow_empty(cmd->cft, path, cfg_def_get_default_value(cmd, item, CFG_TYPE_STRING, profile));
|
|
|
|
if (profile_applied)
|
|
remove_config_tree_by_source(cmd, CONFIG_PROFILE);
|
|
|
|
return str;
|
|
}
|
|
|
|
int find_config_tree_int(struct cmd_context *cmd, int id, struct profile *profile)
|
|
{
|
|
cfg_def_item_t *item = cfg_def_get_item_p(id);
|
|
char path[CFG_PATH_MAX_LEN];
|
|
int profile_applied = 0;
|
|
int i;
|
|
|
|
if (profile && !cmd->profile_params->global_profile)
|
|
profile_applied = override_config_tree_from_profile(cmd, profile);
|
|
|
|
_cfg_def_make_path(path, sizeof(path), item->id, item, 0);
|
|
|
|
if (item->type != CFG_TYPE_INT)
|
|
log_error(INTERNAL_ERROR "%s cfg tree element not declared as integer.", path);
|
|
|
|
i = dm_config_tree_find_int(cmd->cft, path, cfg_def_get_default_value(cmd, item, CFG_TYPE_INT, profile));
|
|
|
|
if (profile_applied)
|
|
remove_config_tree_by_source(cmd, CONFIG_PROFILE);
|
|
|
|
return i;
|
|
}
|
|
|
|
int64_t find_config_tree_int64(struct cmd_context *cmd, int id, struct profile *profile)
|
|
{
|
|
cfg_def_item_t *item = cfg_def_get_item_p(id);
|
|
char path[CFG_PATH_MAX_LEN];
|
|
int profile_applied = 0;
|
|
int i64;
|
|
|
|
if (profile && !cmd->profile_params->global_profile)
|
|
profile_applied = override_config_tree_from_profile(cmd, profile);
|
|
|
|
_cfg_def_make_path(path, sizeof(path), item->id, item, 0);
|
|
|
|
if (item->type != CFG_TYPE_INT)
|
|
log_error(INTERNAL_ERROR "%s cfg tree element not declared as integer.", path);
|
|
|
|
i64 = dm_config_tree_find_int64(cmd->cft, path, cfg_def_get_default_value(cmd, item, CFG_TYPE_INT, profile));
|
|
|
|
if (profile_applied)
|
|
remove_config_tree_by_source(cmd, CONFIG_PROFILE);
|
|
|
|
return i64;
|
|
}
|
|
|
|
float find_config_tree_float(struct cmd_context *cmd, int id, struct profile *profile)
|
|
{
|
|
cfg_def_item_t *item = cfg_def_get_item_p(id);
|
|
char path[CFG_PATH_MAX_LEN];
|
|
int profile_applied = 0;
|
|
float f;
|
|
|
|
if (profile && !cmd->profile_params->global_profile)
|
|
profile_applied = override_config_tree_from_profile(cmd, profile);
|
|
|
|
_cfg_def_make_path(path, sizeof(path), item->id, item, 0);
|
|
|
|
if (item->type != CFG_TYPE_FLOAT)
|
|
log_error(INTERNAL_ERROR "%s cfg tree element not declared as float.", path);
|
|
|
|
f = dm_config_tree_find_float(cmd->cft, path, cfg_def_get_default_value(cmd, item, CFG_TYPE_FLOAT, profile));
|
|
|
|
if (profile_applied)
|
|
remove_config_tree_by_source(cmd, CONFIG_PROFILE);
|
|
|
|
return f;
|
|
}
|
|
|
|
int find_config_tree_bool(struct cmd_context *cmd, int id, struct profile *profile)
|
|
{
|
|
cfg_def_item_t *item = cfg_def_get_item_p(id);
|
|
char path[CFG_PATH_MAX_LEN];
|
|
int profile_applied = 0;
|
|
int b;
|
|
|
|
if (profile && !cmd->profile_params->global_profile)
|
|
profile_applied = override_config_tree_from_profile(cmd, profile);
|
|
|
|
_cfg_def_make_path(path, sizeof(path), item->id, item, 0);
|
|
|
|
if (item->type != CFG_TYPE_BOOL)
|
|
log_error(INTERNAL_ERROR "%s cfg tree element not declared as boolean.", path);
|
|
|
|
b = dm_config_tree_find_bool(cmd->cft, path, cfg_def_get_default_value(cmd, item, CFG_TYPE_BOOL, profile));
|
|
|
|
if (profile_applied)
|
|
remove_config_tree_by_source(cmd, CONFIG_PROFILE);
|
|
|
|
return b;
|
|
}
|
|
|
|
/* Insert cn2 after cn1 */
|
|
static void _insert_config_node(struct dm_config_node **cn1,
|
|
struct dm_config_node *cn2)
|
|
{
|
|
if (!*cn1) {
|
|
*cn1 = cn2;
|
|
cn2->sib = NULL;
|
|
} else {
|
|
cn2->sib = (*cn1)->sib;
|
|
(*cn1)->sib = cn2;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Merge section cn2 into section cn1 (which has the same name)
|
|
* overwriting any existing cn1 nodes with matching names.
|
|
*/
|
|
static void _merge_section(struct dm_config_node *cn1, struct dm_config_node *cn2,
|
|
config_merge_t merge_type)
|
|
{
|
|
struct dm_config_node *cn, *nextn, *oldn;
|
|
struct dm_config_value *cv;
|
|
|
|
for (cn = cn2->child; cn; cn = nextn) {
|
|
nextn = cn->sib;
|
|
|
|
if (merge_type == CONFIG_MERGE_TYPE_TAGS) {
|
|
/* Skip "tags" */
|
|
if (!strcmp(cn->key, "tags"))
|
|
continue;
|
|
}
|
|
|
|
/* Subsection? */
|
|
if (!cn->v)
|
|
/* Ignore - we don't have any of these yet */
|
|
continue;
|
|
/* Not already present? */
|
|
if (!(oldn = dm_config_find_node(cn1->child, cn->key))) {
|
|
_insert_config_node(&cn1->child, cn);
|
|
continue;
|
|
}
|
|
if (merge_type == CONFIG_MERGE_TYPE_TAGS) {
|
|
/* Merge certain value lists */
|
|
if ((!strcmp(cn1->key, "activation") &&
|
|
!strcmp(cn->key, "volume_list")) ||
|
|
(!strcmp(cn1->key, "devices") &&
|
|
(!strcmp(cn->key, "filter") || !strcmp(cn->key, "types")))) {
|
|
cv = cn->v;
|
|
while (cv->next)
|
|
cv = cv->next;
|
|
cv->next = oldn->v;
|
|
}
|
|
}
|
|
|
|
/* Replace values */
|
|
oldn->v = cn->v;
|
|
}
|
|
}
|
|
|
|
static int _match_host_tags(struct dm_list *tags, const struct dm_config_node *tn)
|
|
{
|
|
const struct dm_config_value *tv;
|
|
const char *str;
|
|
|
|
for (tv = tn->v; tv; tv = tv->next) {
|
|
if (tv->type != DM_CFG_STRING)
|
|
continue;
|
|
str = tv->v.str;
|
|
if (*str == '@')
|
|
str++;
|
|
if (!*str)
|
|
continue;
|
|
if (str_list_match_item(tags, str))
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Destructively merge a new config tree into an existing one */
|
|
int merge_config_tree(struct cmd_context *cmd, struct dm_config_tree *cft,
|
|
struct dm_config_tree *newdata, config_merge_t merge_type)
|
|
{
|
|
struct dm_config_node *root = cft->root;
|
|
struct dm_config_node *cn, *nextn, *oldn, *cn2;
|
|
const struct dm_config_node *tn;
|
|
struct config_source *cs, *csn;
|
|
|
|
for (cn = newdata->root; cn; cn = nextn) {
|
|
nextn = cn->sib;
|
|
if (merge_type == CONFIG_MERGE_TYPE_TAGS) {
|
|
/* Ignore tags section */
|
|
if (!strcmp(cn->key, "tags"))
|
|
continue;
|
|
/* If there's a tags node, skip if host tags don't match */
|
|
if ((tn = dm_config_find_node(cn->child, "tags"))) {
|
|
if (!_match_host_tags(&cmd->tags, tn))
|
|
continue;
|
|
}
|
|
}
|
|
if (!(oldn = dm_config_find_node(root, cn->key))) {
|
|
_insert_config_node(&cft->root, cn);
|
|
if (merge_type == CONFIG_MERGE_TYPE_TAGS) {
|
|
/* Remove any "tags" nodes */
|
|
for (cn2 = cn->child; cn2; cn2 = cn2->sib) {
|
|
if (!strcmp(cn2->key, "tags")) {
|
|
cn->child = cn2->sib;
|
|
continue;
|
|
}
|
|
if (cn2->sib && !strcmp(cn2->sib->key, "tags")) {
|
|
cn2->sib = cn2->sib->sib;
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
_merge_section(oldn, cn, merge_type);
|
|
}
|
|
|
|
/*
|
|
* Persistent filter loading is based on timestamp,
|
|
* so we need to know the newest timestamp to make right decision
|
|
* whether the .cache isn't older then any of configs
|
|
*/
|
|
cs = dm_config_get_custom(cft);
|
|
csn = dm_config_get_custom(newdata);
|
|
|
|
if (cs && csn && (cs->timestamp < csn->timestamp))
|
|
cs->timestamp = csn->timestamp;
|
|
|
|
return 1;
|
|
}
|
|
|
|
struct out_baton {
|
|
FILE *fp;
|
|
struct config_def_tree_spec *tree_spec;
|
|
struct dm_pool *mem;
|
|
};
|
|
|
|
static int _out_prefix_fn(const struct dm_config_node *cn, const char *line, void *baton)
|
|
{
|
|
struct out_baton *out = baton;
|
|
struct cfg_def_item *cfg_def;
|
|
char version[9]; /* 8+1 chars for max version of 7.15.511 */
|
|
const char *node_type_name = cn->v ? "option" : "section";
|
|
char path[CFG_PATH_MAX_LEN];
|
|
|
|
|
|
if (cn->id < 0)
|
|
return 1;
|
|
|
|
if (!cn->id) {
|
|
log_error(INTERNAL_ERROR "Configuration node %s has invalid id.", cn->key);
|
|
return 0;
|
|
}
|
|
|
|
cfg_def = cfg_def_get_item_p(cn->id);
|
|
|
|
if (out->tree_spec->withcomments) {
|
|
_cfg_def_make_path(path, sizeof(path), cfg_def->id, cfg_def, 1);
|
|
fprintf(out->fp, "%s# Configuration %s %s.\n", line, node_type_name, path);
|
|
|
|
if (cfg_def->comment)
|
|
fprintf(out->fp, "%s# %s\n", line, cfg_def->comment);
|
|
|
|
if (cfg_def->flags & CFG_ADVANCED)
|
|
fprintf(out->fp, "%s# This configuration %s is advanced.\n", line, node_type_name);
|
|
|
|
if (cfg_def->flags & CFG_UNSUPPORTED)
|
|
fprintf(out->fp, "%s# This configuration %s is not officially supported.\n", line, node_type_name);
|
|
|
|
if (cfg_def->flags & CFG_NAME_VARIABLE)
|
|
fprintf(out->fp, "%s# This configuration %s has variable name.\n", line, node_type_name);
|
|
|
|
if (cfg_def->flags & CFG_DEFAULT_UNDEFINED)
|
|
fprintf(out->fp, "%s# This configuration %s does not have a default value defined.\n", line, node_type_name);
|
|
}
|
|
|
|
if (out->tree_spec->withversions) {
|
|
if (dm_snprintf(version, 9, "%u.%u.%u",
|
|
(cfg_def->since_version & 0xE000) >> 13,
|
|
(cfg_def->since_version & 0x1E00) >> 9,
|
|
(cfg_def->since_version & 0x1FF)) == -1) {
|
|
log_error("_out_prefix_fn: couldn't create version string");
|
|
return 0;
|
|
}
|
|
fprintf(out->fp, "%s# Since version %s.\n", line, version);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _out_line_fn(const struct dm_config_node *cn, const char *line, void *baton)
|
|
{
|
|
struct out_baton *out = baton;
|
|
struct cfg_def_item *cfg_def = cfg_def_get_item_p(cn->id);
|
|
|
|
fprintf(out->fp, "%s%s\n", (out->tree_spec->type != CFG_DEF_TREE_CURRENT) &&
|
|
(cfg_def->flags & CFG_DEFAULT_UNDEFINED) ? "#" : "", line);
|
|
return 1;
|
|
}
|
|
|
|
static int _out_suffix_fn(const struct dm_config_node *cn, const char *line, void *baton)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
int config_write(struct dm_config_tree *cft,
|
|
struct config_def_tree_spec *tree_spec,
|
|
const char *file, int argc, char **argv)
|
|
{
|
|
static const struct dm_config_node_out_spec _out_spec = {
|
|
.prefix_fn = _out_prefix_fn,
|
|
.line_fn = _out_line_fn,
|
|
.suffix_fn = _out_suffix_fn
|
|
};
|
|
const struct dm_config_node *cn;
|
|
struct out_baton baton = {
|
|
.tree_spec = tree_spec,
|
|
.mem = cft->mem
|
|
};
|
|
int r = 1;
|
|
|
|
if (!file) {
|
|
baton.fp = stdout;
|
|
file = "stdout";
|
|
} else if (!(baton.fp = fopen(file, "w"))) {
|
|
log_sys_error("open", file);
|
|
return 0;
|
|
}
|
|
|
|
log_verbose("Dumping configuration to %s", file);
|
|
if (!argc) {
|
|
if (!dm_config_write_node_out(cft->root, &_out_spec, &baton)) {
|
|
log_error("Failure while writing to %s", file);
|
|
r = 0;
|
|
}
|
|
} else while (argc--) {
|
|
if ((cn = dm_config_find_node(cft->root, *argv))) {
|
|
if (!dm_config_write_one_node_out(cn, &_out_spec, &baton)) {
|
|
log_error("Failure while writing to %s", file);
|
|
r = 0;
|
|
}
|
|
} else {
|
|
log_error("Configuration node %s not found", *argv);
|
|
r = 0;
|
|
}
|
|
argv++;
|
|
}
|
|
|
|
if (baton.fp && baton.fp != stdout && dm_fclose(baton.fp)) {
|
|
stack;
|
|
r = 0;
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
static struct dm_config_value *_get_def_array_values(struct dm_config_tree *cft,
|
|
cfg_def_item_t *def)
|
|
{
|
|
char *enc_value, *token, *p, *r;
|
|
struct dm_config_value *array = NULL, *v = NULL, *oldv = NULL;
|
|
|
|
if (!def->default_value.v_CFG_TYPE_STRING) {
|
|
if (!(array = dm_config_create_value(cft))) {
|
|
log_error("Failed to create default empty array for %s.", def->name);
|
|
return NULL;
|
|
}
|
|
array->type = DM_CFG_EMPTY_ARRAY;
|
|
return array;
|
|
}
|
|
|
|
if (!(p = token = enc_value = dm_strdup(def->default_value.v_CFG_TYPE_STRING))) {
|
|
log_error("_get_def_array_values: dm_strdup failed");
|
|
return NULL;
|
|
}
|
|
/* Proper value always starts with '#'. */
|
|
if (token[0] != '#')
|
|
goto bad;
|
|
|
|
while (token) {
|
|
/* Move to type identifier. Error on no char. */
|
|
token++;
|
|
if (!token[0])
|
|
goto bad;
|
|
|
|
/* Move to the actual value and decode any "##" into "#". */
|
|
p = token + 1;
|
|
while ((p = strchr(p, '#')) && p[1] == '#') {
|
|
memmove(p, p + 1, strlen(p));
|
|
p++;
|
|
}
|
|
/* Separate the value out of the whole string. */
|
|
if (p)
|
|
p[0] = '\0';
|
|
|
|
if (!(v = dm_config_create_value(cft))) {
|
|
log_error("Failed to create default config array value for %s.", def->name);
|
|
dm_free(enc_value);
|
|
return NULL;
|
|
}
|
|
if (oldv)
|
|
oldv->next = v;
|
|
if (!array)
|
|
array = v;
|
|
|
|
switch (toupper(token[0])) {
|
|
case 'I':
|
|
case 'B':
|
|
v->v.i = strtoll(token + 1, &r, 10);
|
|
if (*r)
|
|
goto bad;
|
|
v->type = DM_CFG_INT;
|
|
break;
|
|
case 'F':
|
|
v->v.f = strtod(token + 1, &r);
|
|
if (*r)
|
|
goto bad;
|
|
v->type = DM_CFG_FLOAT;
|
|
break;
|
|
case 'S':
|
|
if (!(r = dm_pool_strdup(cft->mem, token + 1))) {
|
|
dm_free(enc_value);
|
|
log_error("Failed to duplicate token for default "
|
|
"array value of %s.", def->name);
|
|
return NULL;
|
|
}
|
|
v->v.str = r;
|
|
v->type = DM_CFG_STRING;
|
|
break;
|
|
default:
|
|
goto bad;
|
|
}
|
|
|
|
oldv = v;
|
|
token = p;
|
|
}
|
|
|
|
dm_free(enc_value);
|
|
return array;
|
|
bad:
|
|
log_error(INTERNAL_ERROR "Default array value malformed for \"%s\", "
|
|
"value: \"%s\", token: \"%s\".", def->name,
|
|
def->default_value.v_CFG_TYPE_STRING, token);
|
|
dm_free(enc_value);
|
|
return NULL;
|
|
}
|
|
|
|
static struct dm_config_node *_add_def_node(struct dm_config_tree *cft,
|
|
struct config_def_tree_spec *spec,
|
|
struct dm_config_node *parent,
|
|
struct dm_config_node *relay,
|
|
cfg_def_item_t *def)
|
|
{
|
|
struct dm_config_node *cn;
|
|
const char *str;
|
|
|
|
if (!(cn = dm_config_create_node(cft, def->name))) {
|
|
log_error("Failed to create default config setting node.");
|
|
return NULL;
|
|
}
|
|
|
|
if (!(def->type & CFG_TYPE_SECTION) && (!(cn->v = dm_config_create_value(cft)))) {
|
|
log_error("Failed to create default config setting node value.");
|
|
return NULL;
|
|
}
|
|
|
|
cn->id = def->id;
|
|
|
|
if (!(def->type & CFG_TYPE_ARRAY)) {
|
|
switch (def->type) {
|
|
case CFG_TYPE_SECTION:
|
|
cn->v = NULL;
|
|
break;
|
|
case CFG_TYPE_BOOL:
|
|
cn->v->type = DM_CFG_INT;
|
|
cn->v->v.i = cfg_def_get_default_value_hint(spec->cmd, def, CFG_TYPE_BOOL, NULL);
|
|
break;
|
|
case CFG_TYPE_INT:
|
|
cn->v->type = DM_CFG_INT;
|
|
cn->v->v.i = cfg_def_get_default_value_hint(spec->cmd, def, CFG_TYPE_INT, NULL);
|
|
break;
|
|
case CFG_TYPE_FLOAT:
|
|
cn->v->type = DM_CFG_FLOAT;
|
|
cn->v->v.f = cfg_def_get_default_value_hint(spec->cmd, def, CFG_TYPE_FLOAT, NULL);
|
|
break;
|
|
case CFG_TYPE_STRING:
|
|
cn->v->type = DM_CFG_STRING;
|
|
if (!(str = cfg_def_get_default_value_hint(spec->cmd, def, CFG_TYPE_STRING, NULL)))
|
|
str = "";
|
|
cn->v->v.str = str;
|
|
break;
|
|
default:
|
|
log_error(INTERNAL_ERROR "_add_def_node: unknown type");
|
|
return NULL;
|
|
break;
|
|
}
|
|
} else
|
|
cn->v = _get_def_array_values(cft, def);
|
|
|
|
cn->child = NULL;
|
|
if (parent) {
|
|
cn->parent = parent;
|
|
if (!parent->child)
|
|
parent->child = cn;
|
|
} else
|
|
cn->parent = cn;
|
|
|
|
if (relay)
|
|
relay->sib = cn;
|
|
|
|
return cn;
|
|
}
|
|
|
|
static int _should_skip_def_node(struct config_def_tree_spec *spec, int section_id, int id)
|
|
{
|
|
cfg_def_item_t *def = cfg_def_get_item_p(id);
|
|
|
|
if ((def->parent != section_id) ||
|
|
(spec->ignoreadvanced && def->flags & CFG_ADVANCED) ||
|
|
(spec->ignoreunsupported && def->flags & CFG_UNSUPPORTED))
|
|
return 1;
|
|
|
|
switch (spec->type) {
|
|
case CFG_DEF_TREE_MISSING:
|
|
if (!spec->check_status) {
|
|
log_error_once(INTERNAL_ERROR "couldn't determine missing "
|
|
"config nodes - unknown status of last config check.");
|
|
return 1;
|
|
}
|
|
if ((spec->check_status[id] & CFG_USED) ||
|
|
(def->flags & CFG_NAME_VARIABLE) ||
|
|
(def->since_version > spec->version))
|
|
return 1;
|
|
break;
|
|
case CFG_DEF_TREE_NEW:
|
|
if (def->since_version != spec->version)
|
|
return 1;
|
|
break;
|
|
case CFG_DEF_TREE_PROFILABLE:
|
|
if (!(def->flags & CFG_PROFILABLE) ||
|
|
(def->since_version > spec->version))
|
|
return 1;
|
|
break;
|
|
default:
|
|
if (def->since_version > spec->version)
|
|
return 1;
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct dm_config_node *_add_def_section_subtree(struct dm_config_tree *cft,
|
|
struct config_def_tree_spec *spec,
|
|
struct dm_config_node *parent,
|
|
struct dm_config_node *relay,
|
|
int section_id)
|
|
{
|
|
struct dm_config_node *cn = NULL, *relay_sub = NULL, *tmp;
|
|
cfg_def_item_t *def;
|
|
int id;
|
|
|
|
for (id = 0; id < CFG_COUNT; id++) {
|
|
if (_should_skip_def_node(spec, section_id, id))
|
|
continue;
|
|
|
|
if (!cn && !(cn = _add_def_node(cft, spec, parent, relay, cfg_def_get_item_p(section_id))))
|
|
goto bad;
|
|
|
|
def = cfg_def_get_item_p(id);
|
|
if ((tmp = def->type == CFG_TYPE_SECTION ? _add_def_section_subtree(cft, spec, cn, relay_sub, id)
|
|
: _add_def_node(cft, spec, cn, relay_sub, def)))
|
|
relay_sub = tmp;
|
|
}
|
|
|
|
return cn;
|
|
bad:
|
|
log_error("Failed to create default config section node.");
|
|
return NULL;
|
|
}
|
|
|
|
struct dm_config_tree *config_def_create_tree(struct config_def_tree_spec *spec)
|
|
{
|
|
struct dm_config_tree *cft;
|
|
struct dm_config_node *root = NULL, *relay = NULL, *tmp;
|
|
int id;
|
|
|
|
if (!(cft = dm_config_create())) {
|
|
log_error("Failed to create default config tree.");
|
|
return NULL;
|
|
}
|
|
|
|
for (id = root_CFG_SECTION + 1; id < CFG_COUNT; id++) {
|
|
if (cfg_def_get_item_p(id)->parent != root_CFG_SECTION)
|
|
continue;
|
|
|
|
if ((tmp = _add_def_section_subtree(cft, spec, root, relay, id))) {
|
|
relay = tmp;
|
|
if (!root)
|
|
root = relay;
|
|
}
|
|
}
|
|
|
|
cft->root = root;
|
|
return cft;
|
|
}
|
|
|
|
static int _check_profile(struct cmd_context *cmd, struct profile *profile)
|
|
{
|
|
struct cft_check_handle *handle;
|
|
int r;
|
|
|
|
if (!(handle = dm_pool_zalloc(cmd->libmem, sizeof(*handle)))) {
|
|
log_debug("_check_profile: profile check handle allocation failed");
|
|
return 0;
|
|
}
|
|
|
|
handle->cft = profile->cft;
|
|
handle->source = CONFIG_PROFILE;
|
|
/* the check is compulsory - allow only profilable items in a profile config! */
|
|
handle->force_check = 1;
|
|
/* provide warning messages only if config/checks=1 */
|
|
handle->suppress_messages = !find_config_tree_bool(cmd, config_checks_CFG, NULL);
|
|
|
|
r = config_def_check(cmd, handle);
|
|
|
|
dm_pool_free(cmd->libmem, handle);
|
|
return r;
|
|
}
|
|
|
|
struct profile *add_profile(struct cmd_context *cmd, const char *profile_name)
|
|
{
|
|
struct profile *profile;
|
|
|
|
/* Do some sanity checks first. */
|
|
if (!profile_name || !*profile_name) {
|
|
log_error("Undefined profile name.");
|
|
return NULL;
|
|
}
|
|
|
|
if (strchr(profile_name, '/')) {
|
|
log_error("%s: bad profile name, it contains '/'.", profile_name);
|
|
return NULL;
|
|
}
|
|
|
|
/* Check if the profile is added already... */
|
|
dm_list_iterate_items(profile, &cmd->profile_params->profiles_to_load) {
|
|
if (!strcmp(profile->name, profile_name))
|
|
return profile;
|
|
}
|
|
dm_list_iterate_items(profile, &cmd->profile_params->profiles) {
|
|
if (!strcmp(profile->name, profile_name))
|
|
return profile;
|
|
}
|
|
|
|
if (!(profile = dm_pool_zalloc(cmd->libmem, sizeof(*profile)))) {
|
|
log_error("profile allocation failed");
|
|
return NULL;
|
|
}
|
|
|
|
profile->name = dm_pool_strdup(cmd->libmem, profile_name);
|
|
dm_list_add(&cmd->profile_params->profiles_to_load, &profile->list);
|
|
|
|
return profile;
|
|
}
|
|
|
|
int load_profile(struct cmd_context *cmd, struct profile *profile) {
|
|
static char profile_path[PATH_MAX];
|
|
|
|
if (critical_section()) {
|
|
log_error(INTERNAL_ERROR "trying to load profile %s "
|
|
"in critical section.", profile->name);
|
|
return 0;
|
|
}
|
|
|
|
if (profile->cft)
|
|
return 1;
|
|
|
|
if (dm_snprintf(profile_path, sizeof(profile_path), "%s/%s.profile",
|
|
cmd->profile_params->dir, profile->name) < 0) {
|
|
log_error("LVM_SYSTEM_DIR or profile name too long");
|
|
return 0;
|
|
}
|
|
|
|
if (!(profile->cft = config_file_open_and_read(profile_path, CONFIG_PROFILE)))
|
|
return 0;
|
|
|
|
dm_list_move(&cmd->profile_params->profiles, &profile->list);
|
|
|
|
/*
|
|
* *Profile must be valid* otherwise we'd end up with incorrect config!
|
|
* If there were config items present that are not supposed to be
|
|
* customized by a profile, we could end up with non-deterministic
|
|
* behaviour. Therefore, this check is *strictly forced* even if
|
|
* config/checks=0. The config/checks=0 will only cause the warning
|
|
* messages to be suppressed, but the check itself is always done
|
|
* for profiles!
|
|
*/
|
|
if (!_check_profile(cmd, profile)) {
|
|
log_error("Ignoring invalid configuration profile %s.", profile->name);
|
|
/* if invalid, cut the whole tree and leave it empty */
|
|
dm_pool_free(profile->cft->mem, profile->cft->root);
|
|
profile->cft->root = NULL;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
int load_pending_profiles(struct cmd_context *cmd)
|
|
{
|
|
struct profile *profile, *temp_profile;
|
|
|
|
dm_list_iterate_items_safe(profile, temp_profile, &cmd->profile_params->profiles_to_load) {
|
|
if (!load_profile(cmd, profile))
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
const char *get_default_devices_cache_dir_CFG(struct cmd_context *cmd, struct profile *profile)
|
|
{
|
|
static char buf[PATH_MAX];
|
|
|
|
if (dm_snprintf(buf, sizeof(buf), "%s/%s", cmd->system_dir, DEFAULT_CACHE_SUBDIR) < 0) {
|
|
log_error("Persistent cache directory name too long.");
|
|
return NULL;
|
|
}
|
|
|
|
return dm_pool_strdup(cmd->mem, buf);
|
|
}
|
|
|
|
const char *get_default_devices_cache_CFG(struct cmd_context *cmd, struct profile *profile)
|
|
{
|
|
const char *cache_dir = NULL, *cache_file_prefix = NULL;
|
|
static char buf[PATH_MAX];
|
|
|
|
/*
|
|
* If 'cache_dir' or 'cache_file_prefix' is set, ignore 'cache'.
|
|
*/
|
|
if (find_config_tree_node(cmd, devices_cache_dir_CFG, profile))
|
|
cache_dir = find_config_tree_str(cmd, devices_cache_dir_CFG, profile);
|
|
if (find_config_tree_node(cmd, devices_cache_file_prefix_CFG, profile))
|
|
cache_file_prefix = find_config_tree_str_allow_empty(cmd, devices_cache_file_prefix_CFG, profile);
|
|
|
|
if (cache_dir || cache_file_prefix) {
|
|
if (dm_snprintf(buf, sizeof(buf),
|
|
"%s%s%s/%s.cache",
|
|
cache_dir ? "" : cmd->system_dir,
|
|
cache_dir ? "" : "/",
|
|
cache_dir ? : DEFAULT_CACHE_SUBDIR,
|
|
cache_file_prefix ? : DEFAULT_CACHE_FILE_PREFIX) < 0) {
|
|
log_error("Persistent cache filename too long.");
|
|
return NULL;
|
|
}
|
|
return dm_pool_strdup(cmd->mem, buf);
|
|
}
|
|
|
|
if (dm_snprintf(buf, sizeof(buf), "%s/%s/%s.cache", cmd->system_dir,
|
|
DEFAULT_CACHE_SUBDIR, DEFAULT_CACHE_FILE_PREFIX) < 0) {
|
|
log_error("Persistent cache filename too long.");
|
|
return NULL;
|
|
}
|
|
return dm_pool_strdup(cmd->mem, buf);
|
|
}
|
|
|
|
const char *get_default_backup_backup_dir_CFG(struct cmd_context *cmd, struct profile *profile)
|
|
{
|
|
static char buf[PATH_MAX];
|
|
|
|
if (dm_snprintf(buf, sizeof(buf), "%s/%s", cmd->system_dir, DEFAULT_BACKUP_SUBDIR) == -1) {
|
|
log_error("Couldn't create default backup path '%s/%s'.",
|
|
cmd->system_dir, DEFAULT_BACKUP_SUBDIR);
|
|
return NULL;
|
|
}
|
|
|
|
return dm_pool_strdup(cmd->mem, buf);
|
|
}
|
|
|
|
const char *get_default_backup_archive_dir_CFG(struct cmd_context *cmd, struct profile *profile)
|
|
{
|
|
static char buf[PATH_MAX];
|
|
|
|
if (dm_snprintf (buf, sizeof(buf), "%s/%s", cmd->system_dir, DEFAULT_ARCHIVE_SUBDIR) == -1) {
|
|
log_error("Couldn't create default archive path '%s/%s'.",
|
|
cmd->system_dir, DEFAULT_ARCHIVE_SUBDIR);
|
|
return NULL;
|
|
}
|
|
|
|
return dm_pool_strdup(cmd->mem, buf);
|
|
}
|
|
|
|
const char *get_default_config_profile_dir_CFG(struct cmd_context *cmd, struct profile *profile)
|
|
{
|
|
static char buf[PATH_MAX];
|
|
|
|
if (dm_snprintf(buf, sizeof(buf), "%s/%s", cmd->system_dir, DEFAULT_PROFILE_SUBDIR) == -1) {
|
|
log_error("Couldn't create default profile path '%s/%s'.",
|
|
cmd->system_dir, DEFAULT_PROFILE_SUBDIR);
|
|
return NULL;
|
|
}
|
|
|
|
return dm_pool_strdup(cmd->mem, buf);
|
|
}
|
|
|
|
const char *get_default_activation_mirror_image_fault_policy_CFG(struct cmd_context *cmd, struct profile *profile)
|
|
{
|
|
return find_config_tree_str(cmd, activation_mirror_device_fault_policy_CFG, profile);
|
|
}
|
|
|
|
int get_default_allocation_thin_pool_chunk_size_CFG(struct cmd_context *cmd, struct profile *profile)
|
|
{
|
|
const char *str;
|
|
uint32_t chunk_size;
|
|
|
|
str = find_config_tree_str(cmd, allocation_thin_pool_chunk_size_policy_CFG, profile);
|
|
|
|
if (!strcasecmp(str, "generic"))
|
|
chunk_size = DEFAULT_THIN_POOL_CHUNK_SIZE;
|
|
else if (!strcasecmp(str, "performance"))
|
|
chunk_size = DEFAULT_THIN_POOL_CHUNK_SIZE_PERFORMANCE;
|
|
else {
|
|
log_error("Thin pool chunk size calculation policy \"%s\" is unrecognised.", str);
|
|
return 0;
|
|
}
|
|
|
|
return (int) chunk_size;
|
|
}
|
|
|
|
int get_default_allocation_cache_pool_chunk_size_CFG(struct cmd_context *cmd, struct profile *profile)
|
|
{
|
|
return DEFAULT_CACHE_POOL_CHUNK_SIZE;
|
|
}
|