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