mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-05 13:18:20 +03:00
678 lines
14 KiB
C
678 lines
14 KiB
C
/*
|
|
* Copyright (C) 2001 Sistina Software (UK) Limited.
|
|
*
|
|
* This file is released under the LGPL.
|
|
*
|
|
*/
|
|
|
|
#include "lib.h"
|
|
#include "toolcontext.h"
|
|
#include "pool.h"
|
|
#include "metadata.h"
|
|
#include "defaults.h"
|
|
#include "lvm-string.h"
|
|
#include "activate.h"
|
|
#include "filter.h"
|
|
#include "filter-composite.h"
|
|
#include "filter-persistent.h"
|
|
#include "filter-regex.h"
|
|
#include "filter-sysfs.h"
|
|
#include "label.h"
|
|
#include "lvm-file.h"
|
|
#include "format-text.h"
|
|
#include "display.h"
|
|
#include "memlock.h"
|
|
#include "str_list.h"
|
|
|
|
#ifdef HAVE_LIBDL
|
|
#include "sharedlib.h"
|
|
#endif
|
|
|
|
#ifdef LVM1_INTERNAL
|
|
#include "format1.h"
|
|
#endif
|
|
|
|
#include <locale.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/utsname.h>
|
|
#include <syslog.h>
|
|
#include <time.h>
|
|
|
|
#ifdef linux
|
|
# include <malloc.h>
|
|
#endif
|
|
|
|
static FILE *_log;
|
|
|
|
static int _get_env_vars(struct cmd_context *cmd)
|
|
{
|
|
const char *e;
|
|
|
|
/* Set to "" to avoid using any system directory */
|
|
if ((e = getenv("LVM_SYSTEM_DIR"))) {
|
|
if (lvm_snprintf(cmd->sys_dir, sizeof(cmd->sys_dir),
|
|
"%s", e) < 0) {
|
|
log_error("LVM_SYSTEM_DIR environment variable "
|
|
"is too long.");
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void _init_logging(struct cmd_context *cmd)
|
|
{
|
|
int append = 1;
|
|
time_t t;
|
|
|
|
const char *log_file;
|
|
|
|
/* Syslog */
|
|
cmd->default_settings.syslog =
|
|
find_config_int(cmd->cft->root, "log/syslog", DEFAULT_SYSLOG);
|
|
if (cmd->default_settings.syslog != 1)
|
|
fin_syslog();
|
|
|
|
if (cmd->default_settings.syslog > 1)
|
|
init_syslog(cmd->default_settings.syslog);
|
|
|
|
/* Debug level for log file output */
|
|
cmd->default_settings.debug =
|
|
find_config_int(cmd->cft->root, "log/level", DEFAULT_LOGLEVEL);
|
|
init_debug(cmd->default_settings.debug);
|
|
|
|
/* Verbose level for tty output */
|
|
cmd->default_settings.verbose =
|
|
find_config_int(cmd->cft->root, "log/verbose", DEFAULT_VERBOSE);
|
|
init_verbose(cmd->default_settings.verbose + VERBOSE_BASE_LEVEL);
|
|
|
|
/* Log message formatting */
|
|
init_indent(find_config_int(cmd->cft->root, "log/indent",
|
|
DEFAULT_INDENT));
|
|
|
|
cmd->default_settings.msg_prefix = find_config_str(cmd->cft->root,
|
|
"log/prefix",
|
|
DEFAULT_MSG_PREFIX);
|
|
init_msg_prefix(cmd->default_settings.msg_prefix);
|
|
|
|
cmd->default_settings.cmd_name = find_config_int(cmd->cft->root,
|
|
"log/command_names",
|
|
DEFAULT_CMD_NAME);
|
|
init_cmd_name(cmd->default_settings.cmd_name);
|
|
|
|
/* Test mode */
|
|
cmd->default_settings.test =
|
|
find_config_int(cmd->cft->root, "global/test", 0);
|
|
|
|
/* Settings for logging to file */
|
|
if (find_config_int(cmd->cft->root, "log/overwrite",
|
|
DEFAULT_OVERWRITE))
|
|
append = 0;
|
|
|
|
log_file = find_config_str(cmd->cft->root, "log/file", 0);
|
|
|
|
if (log_file) {
|
|
release_log_memory();
|
|
fin_log();
|
|
init_log_file(log_file, append);
|
|
}
|
|
|
|
log_file = find_config_str(cmd->cft->root, "log/activate_file", 0);
|
|
if (log_file)
|
|
init_log_direct(log_file, append);
|
|
|
|
init_log_while_suspended(find_config_int(cmd->cft->root,
|
|
"log/activation", 0));
|
|
|
|
t = time(NULL);
|
|
log_verbose("Logging initialised at %s", ctime(&t));
|
|
|
|
/* Tell device-mapper about our logging */
|
|
#ifdef DEVMAPPER_SUPPORT
|
|
dm_log_init(print_log);
|
|
#endif
|
|
}
|
|
|
|
static int _process_config(struct cmd_context *cmd)
|
|
{
|
|
mode_t old_umask;
|
|
|
|
/* umask */
|
|
cmd->default_settings.umask = find_config_int(cmd->cft->root,
|
|
"global/umask",
|
|
DEFAULT_UMASK);
|
|
|
|
if ((old_umask = umask((mode_t) cmd->default_settings.umask)) !=
|
|
(mode_t) cmd->default_settings.umask)
|
|
log_verbose("Set umask to %04o", cmd->default_settings.umask);
|
|
|
|
/* dev dir */
|
|
if (lvm_snprintf(cmd->dev_dir, sizeof(cmd->dev_dir), "%s/",
|
|
find_config_str(cmd->cft->root, "devices/dir",
|
|
DEFAULT_DEV_DIR)) < 0) {
|
|
log_error("Device directory given in config file too long");
|
|
return 0;
|
|
}
|
|
#ifdef DEVMAPPER_SUPPORT
|
|
dm_set_dev_dir(cmd->dev_dir);
|
|
#endif
|
|
|
|
/* proc dir */
|
|
if (lvm_snprintf(cmd->proc_dir, sizeof(cmd->proc_dir), "%s",
|
|
find_config_str(cmd->cft->root, "global/proc",
|
|
DEFAULT_PROC_DIR)) < 0) {
|
|
log_error("Device directory given in config file too long");
|
|
return 0;
|
|
}
|
|
|
|
/* activation? */
|
|
cmd->default_settings.activation = find_config_int(cmd->cft->root,
|
|
"global/activation",
|
|
DEFAULT_ACTIVATION);
|
|
set_activation(cmd->default_settings.activation);
|
|
|
|
cmd->default_settings.suffix = find_config_int(cmd->cft->root,
|
|
"global/suffix",
|
|
DEFAULT_SUFFIX);
|
|
|
|
if (!(cmd->default_settings.unit_factor =
|
|
units_to_bytes(find_config_str(cmd->cft->root,
|
|
"global/units",
|
|
DEFAULT_UNITS),
|
|
&cmd->default_settings.unit_type))) {
|
|
log_error("Invalid units specification");
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* Find and read config file */
|
|
static int _init_config(struct cmd_context *cmd)
|
|
{
|
|
struct stat info;
|
|
char config_file[PATH_MAX] = "";
|
|
|
|
if (!(cmd->cft = create_config_tree())) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
/* No config file if LVM_SYSTEM_DIR is empty */
|
|
if (!*cmd->sys_dir)
|
|
return 1;
|
|
|
|
if (lvm_snprintf(config_file, sizeof(config_file),
|
|
"%s/lvm.conf", cmd->sys_dir) < 0) {
|
|
log_error("LVM_SYSTEM_DIR was too long");
|
|
destroy_config_tree(cmd->cft);
|
|
return 0;
|
|
}
|
|
|
|
/* Is there a config file? */
|
|
if (stat(config_file, &info) == -1) {
|
|
if (errno == ENOENT)
|
|
return 1;
|
|
log_sys_error("stat", config_file);
|
|
destroy_config_tree(cmd->cft);
|
|
return 0;
|
|
}
|
|
|
|
if (!read_config_file(cmd->cft, config_file)) {
|
|
log_error("Failed to load config file %s", config_file);
|
|
destroy_config_tree(cmd->cft);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _init_dev_cache(struct cmd_context *cmd)
|
|
{
|
|
struct config_node *cn;
|
|
struct config_value *cv;
|
|
|
|
if (!dev_cache_init()) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
if (!(cn = find_config_node(cmd->cft->root, "devices/scan"))) {
|
|
if (!dev_cache_add_dir("/dev")) {
|
|
log_error("Failed to add /dev to internal "
|
|
"device cache");
|
|
return 0;
|
|
}
|
|
log_verbose("device/scan not in config file: "
|
|
"Defaulting to /dev");
|
|
return 1;
|
|
}
|
|
|
|
for (cv = cn->v; cv; cv = cv->next) {
|
|
if (cv->type != CFG_STRING) {
|
|
log_error("Invalid string in config file: "
|
|
"devices/scan");
|
|
return 0;
|
|
}
|
|
|
|
if (!dev_cache_add_dir(cv->v.str)) {
|
|
log_error("Failed to add %s to internal device cache",
|
|
cv->v.str);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
#define MAX_FILTERS 3
|
|
|
|
static struct dev_filter *_init_filter_components(struct cmd_context *cmd)
|
|
{
|
|
unsigned nr_filt = 0;
|
|
struct config_node *cn;
|
|
struct dev_filter *filters[MAX_FILTERS];
|
|
|
|
memset(filters, 0, sizeof(filters));
|
|
|
|
/* sysfs filter */
|
|
if (find_config_bool(cmd->cft->root, "devices/sysfs_scan",
|
|
DEFAULT_SYSFS_SCAN)) {
|
|
if ((filters[nr_filt] = sysfs_filter_create(cmd->proc_dir)))
|
|
nr_filt++;
|
|
}
|
|
|
|
/* regex filter */
|
|
if (!(cn = find_config_node(cmd->cft->root, "devices/filter")))
|
|
log_debug("devices/filter not found in config file: no regex "
|
|
"filter installed");
|
|
|
|
else if (!(filters[nr_filt++] = regex_filter_create(cn->v))) {
|
|
log_error("Failed to create regex device filter");
|
|
return NULL;
|
|
}
|
|
|
|
/* device type filter */
|
|
cn = find_config_node(cmd->cft->root, "devices/types");
|
|
if (!(filters[nr_filt++] = lvm_type_filter_create(cmd->proc_dir, cn))) {
|
|
log_error("Failed to create lvm type filter");
|
|
return NULL;
|
|
}
|
|
|
|
/* only build a composite filter if we really need it */
|
|
return (nr_filt == 1) ?
|
|
filters[0] : composite_filter_create(nr_filt, filters);
|
|
}
|
|
|
|
static int _init_filters(struct cmd_context *cmd)
|
|
{
|
|
const char *dev_cache;
|
|
struct dev_filter *f3, *f4;
|
|
struct stat st;
|
|
char cache_file[PATH_MAX];
|
|
|
|
cmd->dump_filter = 0;
|
|
|
|
if (!(f3 = _init_filter_components(cmd)))
|
|
return 0;
|
|
|
|
if (lvm_snprintf(cache_file, sizeof(cache_file),
|
|
"%s/.cache", cmd->sys_dir) < 0) {
|
|
log_error("Persistent cache filename too long ('%s/.cache').",
|
|
cmd->sys_dir);
|
|
return 0;
|
|
}
|
|
|
|
dev_cache =
|
|
find_config_str(cmd->cft->root, "devices/cache", cache_file);
|
|
if (!(f4 = persistent_filter_create(f3, dev_cache))) {
|
|
log_error("Failed to create persistent device filter");
|
|
return 0;
|
|
}
|
|
|
|
/* Should we ever dump persistent filter state? */
|
|
if (find_config_int(cmd->cft->root, "devices/write_cache_state", 1))
|
|
cmd->dump_filter = 1;
|
|
|
|
if (!*cmd->sys_dir)
|
|
cmd->dump_filter = 0;
|
|
|
|
if (!stat(dev_cache, &st) &&
|
|
(st.st_mtime > config_file_timestamp(cmd->cft)) &&
|
|
!persistent_filter_load(f4))
|
|
log_verbose("Failed to load existing device cache from %s",
|
|
dev_cache);
|
|
|
|
cmd->filter = f4;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _init_formats(struct cmd_context *cmd)
|
|
{
|
|
const char *format;
|
|
|
|
struct format_type *fmt;
|
|
struct list *fmth;
|
|
|
|
#ifdef HAVE_LIBDL
|
|
struct config_node *cn;
|
|
#endif
|
|
|
|
label_init();
|
|
|
|
#ifdef LVM1_INTERNAL
|
|
if (!(fmt = init_lvm1_format(cmd)))
|
|
return 0;
|
|
fmt->library = NULL;
|
|
list_add(&cmd->formats, &fmt->list);
|
|
#endif
|
|
|
|
#ifdef HAVE_LIBDL
|
|
/* Load any formats in shared libs */
|
|
if ((cn = find_config_node(cmd->cft->root, "global/format_libraries"))) {
|
|
|
|
struct config_value *cv;
|
|
struct format_type *(*init_format_fn) (struct cmd_context *);
|
|
void *lib;
|
|
|
|
for (cv = cn->v; cv; cv = cv->next) {
|
|
if (cv->type != CFG_STRING) {
|
|
log_error("Invalid string in config file: "
|
|
"global/format_libraries");
|
|
return 0;
|
|
}
|
|
if (!(lib = load_shared_library(cmd->cft, cv->v.str,
|
|
"format"))) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
if (!(init_format_fn = dlsym(lib, "init_format"))) {
|
|
log_error("Shared library %s does not contain "
|
|
"format functions", cv->v.str);
|
|
dlclose(lib);
|
|
return 0;
|
|
}
|
|
|
|
if (!(fmt = init_format_fn(cmd)))
|
|
return 0;
|
|
fmt->library = lib;
|
|
list_add(&cmd->formats, &fmt->list);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if (!(fmt = create_text_format(cmd)))
|
|
return 0;
|
|
fmt->library = NULL;
|
|
list_add(&cmd->formats, &fmt->list);
|
|
|
|
cmd->fmt_backup = fmt;
|
|
|
|
format = find_config_str(cmd->cft->root, "global/format",
|
|
DEFAULT_FORMAT);
|
|
|
|
list_iterate(fmth, &cmd->formats) {
|
|
fmt = list_item(fmth, struct format_type);
|
|
if (!strcasecmp(fmt->name, format) ||
|
|
(fmt->alias && !strcasecmp(fmt->alias, format))) {
|
|
cmd->default_settings.fmt = fmt;
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
log_error("_init_formats: Default format (%s) not found", format);
|
|
return 0;
|
|
}
|
|
|
|
static int _init_hostname(struct cmd_context *cmd)
|
|
{
|
|
struct utsname uts;
|
|
|
|
if (uname(&uts)) {
|
|
log_sys_error("uname", "_init_hostname");
|
|
return 0;
|
|
}
|
|
|
|
if (!(cmd->hostname = pool_strdup(cmd->libmem, uts.nodename))) {
|
|
log_error("_init_hostname: pool_strdup failed");
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _set_tag(struct cmd_context *cmd, const char *tag)
|
|
{
|
|
log_very_verbose("Setting host tag: %s", pool_strdup(cmd->libmem, tag));
|
|
|
|
if (!str_list_add(cmd->libmem, &cmd->tags, tag)) {
|
|
log_error("_init_tags: str_list_add %s failed", tag);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _check_host_filters(struct cmd_context *cmd, struct config_node *hn,
|
|
int *passes)
|
|
{
|
|
struct config_node *cn;
|
|
struct config_value *cv;
|
|
|
|
*passes = 1;
|
|
|
|
for (cn = hn; cn; cn = cn->sib) {
|
|
if (!cn->v)
|
|
continue;
|
|
if (!strcmp(cn->key, "host_list")) {
|
|
*passes = 0;
|
|
if (cn->v->type == CFG_EMPTY_ARRAY)
|
|
continue;
|
|
for (cv = cn->v; cv; cv = cv->next) {
|
|
if (cv->type != CFG_STRING) {
|
|
log_error("Invalid hostname string "
|
|
"for tag %s", cn->key);
|
|
return 0;
|
|
}
|
|
if (!strcmp(cv->v.str, cmd->hostname)) {
|
|
*passes = 1;
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
if (!strcmp(cn->key, "host_filter")) {
|
|
log_error("host_filter not supported yet");
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _init_tags(struct cmd_context *cmd)
|
|
{
|
|
struct config_node *tn, *cn;
|
|
const char *tag;
|
|
int passes;
|
|
|
|
list_init(&cmd->tags);
|
|
|
|
if (!(tn = find_config_node(cmd->cft->root, "tags")) ||
|
|
!tn->child) {
|
|
log_very_verbose("No tags defined in config file");
|
|
return 1;
|
|
}
|
|
|
|
if (find_config_int(cmd->cft->root, "tags/hosttags",
|
|
DEFAULT_HOSTTAGS)) {
|
|
/* FIXME Strip out invalid chars: only A-Za-z0-9_+.- */
|
|
if (!_set_tag(cmd, cmd->hostname)) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
for (cn = tn->child; cn; cn = cn->sib) {
|
|
if (cn->v)
|
|
continue;
|
|
tag = cn->key;
|
|
if (*tag == '@')
|
|
tag++;
|
|
if (!validate_name(tag)) {
|
|
log_error("Invalid tag in config file: %s", cn->key);
|
|
return 0;
|
|
}
|
|
if (cn->child) {
|
|
passes = 0;
|
|
if (!_check_host_filters(cmd, cn->child, &passes)) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
if (!passes)
|
|
continue;
|
|
}
|
|
if (!_set_tag(cmd, tag)) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* Entry point */
|
|
struct cmd_context *create_toolcontext(struct arg *the_args)
|
|
{
|
|
struct cmd_context *cmd;
|
|
|
|
#ifdef M_MMAP_MAX
|
|
mallopt(M_MMAP_MAX, 0);
|
|
#endif
|
|
|
|
if (!setlocale(LC_ALL, ""))
|
|
log_error("setlocale failed");
|
|
|
|
#ifdef INTL_PACKAGE
|
|
bindtextdomain(INTL_PACKAGE, LOCALEDIR);
|
|
#endif
|
|
|
|
init_syslog(DEFAULT_LOG_FACILITY);
|
|
|
|
if (!(cmd = dbg_malloc(sizeof(*cmd)))) {
|
|
log_error("Failed to allocate command context");
|
|
return NULL;
|
|
}
|
|
memset(cmd, 0, sizeof(*cmd));
|
|
cmd->args = the_args;
|
|
list_init(&cmd->formats);
|
|
|
|
strcpy(cmd->sys_dir, DEFAULT_SYS_DIR);
|
|
|
|
if (!_get_env_vars(cmd))
|
|
goto error;
|
|
|
|
/* Create system directory if it doesn't already exist */
|
|
if (*cmd->sys_dir && !create_dir(cmd->sys_dir))
|
|
goto error;
|
|
|
|
if (!_init_config(cmd))
|
|
goto error;
|
|
|
|
_init_logging(cmd);
|
|
|
|
if (!(cmd->libmem = pool_create(4 * 1024))) {
|
|
log_error("Library memory pool creation failed");
|
|
return 0;
|
|
}
|
|
|
|
if (!_process_config(cmd))
|
|
goto error;
|
|
|
|
if (!_init_dev_cache(cmd))
|
|
goto error;
|
|
|
|
if (!_init_filters(cmd))
|
|
goto error;
|
|
|
|
if (!(cmd->mem = pool_create(4 * 1024))) {
|
|
log_error("Command memory pool creation failed");
|
|
return 0;
|
|
}
|
|
|
|
memlock_init(cmd);
|
|
|
|
if (!_init_formats(cmd))
|
|
goto error;
|
|
|
|
if (!_init_hostname(cmd))
|
|
goto error;
|
|
|
|
if (!_init_tags(cmd))
|
|
goto error;
|
|
|
|
cmd->current_settings = cmd->default_settings;
|
|
|
|
return cmd;
|
|
|
|
error:
|
|
dbg_free(cmd);
|
|
return NULL;
|
|
}
|
|
|
|
int refresh_toolcontext(struct cmd_context *cmd)
|
|
{
|
|
_init_logging(cmd);
|
|
_init_tags(cmd);
|
|
|
|
/* FIXME Reset filters and dev_cache */
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void _destroy_formats(struct list *formats)
|
|
{
|
|
struct list *fmtl, *tmp;
|
|
struct format_type *fmt;
|
|
void *lib;
|
|
|
|
list_iterate_safe(fmtl, tmp, formats) {
|
|
fmt = list_item(fmtl, struct format_type);
|
|
list_del(&fmt->list);
|
|
lib = fmt->library;
|
|
fmt->ops->destroy(fmt);
|
|
#ifdef HAVE_LIBDL
|
|
if (lib)
|
|
dlclose(lib);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void destroy_toolcontext(struct cmd_context *cmd)
|
|
{
|
|
if (cmd->dump_filter)
|
|
persistent_filter_dump(cmd->filter);
|
|
|
|
activation_exit();
|
|
lvmcache_destroy();
|
|
label_exit();
|
|
_destroy_formats(&cmd->formats);
|
|
cmd->filter->destroy(cmd->filter);
|
|
pool_destroy(cmd->mem);
|
|
dev_cache_exit();
|
|
destroy_config_tree(cmd->cft);
|
|
pool_destroy(cmd->libmem);
|
|
dbg_free(cmd);
|
|
|
|
release_log_memory();
|
|
dump_memory();
|
|
fin_log();
|
|
fin_syslog();
|
|
|
|
if (_log)
|
|
fclose(_log);
|
|
|
|
}
|