1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-18 10:04:20 +03:00

dmstats: start dmfilemapd when creating or updating file maps

Launch an instance of the filemap monitoring daemon when creating,
or updating, a file mapped group, unless the --nomonitor switch is
given.

Unless --foreground is given the daemon will detach from the
terminal and run in the background until it is signaled or the
daemon termination conditions are met.

The --follow={inode|path} switch is added to control the daemon
behaviour when files are moved, unlinked, or renamed while they
are being monitored.

The daemon runs with the same verbosity as the dmstats command
that starts it.
This commit is contained in:
Bryn M. Reeves 2016-12-16 13:42:08 +00:00
parent 044f92e466
commit baa8a9be4a

View File

@ -172,7 +172,9 @@ enum {
SELECT_ARG, SELECT_ARG,
EXEC_ARG, EXEC_ARG,
FILEMAP_ARG, FILEMAP_ARG,
FOLLOW_ARG,
FORCE_ARG, FORCE_ARG,
FOREGROUND_ARG,
GID_ARG, GID_ARG,
GROUP_ARG, GROUP_ARG,
GROUP_ID_ARG, GROUP_ID_ARG,
@ -196,6 +198,7 @@ enum {
NOTABLE_ARG, NOTABLE_ARG,
NOTIMESUFFIX_ARG, NOTIMESUFFIX_ARG,
UDEVCOOKIE_ARG, UDEVCOOKIE_ARG,
NOMONITOR_ARG,
NOUDEVRULES_ARG, NOUDEVRULES_ARG,
NOUDEVSYNC_ARG, NOUDEVSYNC_ARG,
OPTIONS_ARG, OPTIONS_ARG,
@ -4999,15 +5002,25 @@ static int _stats_check_filemap_switches(void)
return 1; return 1;
} }
static dm_filemapd_mode_t _stats_get_filemapd_mode(void)
{
if (!_switches[FOLLOW_ARG])
return DM_FILEMAPD_FOLLOW_INODE;
return dm_filemapd_mode_from_string(_string_args[FOLLOW_ARG]);
}
static int _stats_create_file(CMD_ARGS) static int _stats_create_file(CMD_ARGS)
{ {
const char *alias, *program_id = DM_STATS_PROGRAM_ID; const char *alias, *program_id = DM_STATS_PROGRAM_ID;
const char *bounds_str = _string_args[BOUNDS_ARG]; const char *bounds_str = _string_args[BOUNDS_ARG];
int foreground = _switches[FOREGROUND_ARG];
int verbose = _switches[VERBOSE_ARG];
uint64_t *regions, *region, count = 0; uint64_t *regions, *region, count = 0;
struct dm_histogram *bounds = NULL; struct dm_histogram *bounds = NULL;
char *path, *abspath = NULL; char *path, *abspath = NULL;
struct dm_stats *dms = NULL; struct dm_stats *dms = NULL;
int group, fd = -1, precise; int group, fd = -1, precise;
dm_filemapd_mode_t mode;
if (names) { if (names) {
err("Device names are not compatibile with --filemap."); err("Device names are not compatibile with --filemap.");
@ -5060,6 +5073,10 @@ static int _stats_create_file(CMD_ARGS)
precise = _int_args[PRECISE_ARG]; precise = _int_args[PRECISE_ARG];
group = !_switches[NOGROUP_ARG]; group = !_switches[NOGROUP_ARG];
if (!_switches[NOMONITOR_ARG] && group)
if ((mode = _stats_get_filemapd_mode()) == -1)
goto bad;
if (!(dms = dm_stats_create(DM_STATS_PROGRAM_ID))) if (!(dms = dm_stats_create(DM_STATS_PROGRAM_ID)))
goto_bad; goto_bad;
@ -5091,6 +5108,12 @@ static int _stats_create_file(CMD_ARGS)
regions = dm_stats_create_regions_from_fd(dms, fd, group, precise, regions = dm_stats_create_regions_from_fd(dms, fd, group, precise,
bounds, alias); bounds, alias);
if (!_switches[NOMONITOR_ARG] && group) {
if (!dm_stats_start_filemapd(fd, regions[0], abspath, mode,
foreground, verbose))
log_warn("Failed to start filemap monitoring daemon.");
}
if (close(fd)) if (close(fd))
log_error("Error closing %s", abspath); log_error("Error closing %s", abspath);
@ -5620,12 +5643,16 @@ out:
static int _stats_update_file(CMD_ARGS) static int _stats_update_file(CMD_ARGS)
{ {
uint64_t group_id, *region, *regions, count = 0; uint64_t group_id, *region, *regions = NULL, count = 0;
const char *program_id = DM_STATS_PROGRAM_ID; const char *program_id = DM_STATS_PROGRAM_ID;
int foreground = _switches[FOREGROUND_ARG];
int verbose = _switches[VERBOSE_ARG];
char *path, *abspath = NULL;
dm_filemapd_mode_t mode;
struct dm_stats *dms; struct dm_stats *dms;
char *path, *abspath;
int fd = -1; int fd = -1;
if (names) { if (names) {
err("Device names are not compatibile with update_filemap."); err("Device names are not compatibile with update_filemap.");
return 0; return 0;
@ -5654,6 +5681,10 @@ static int _stats_update_file(CMD_ARGS)
group_id = (uint64_t) _int_args[GROUP_ID_ARG]; group_id = (uint64_t) _int_args[GROUP_ID_ARG];
if (!_switches[NOMONITOR_ARG])
if ((mode = _stats_get_filemapd_mode()) < 0)
goto bad;
if (_switches[PROGRAM_ID_ARG]) if (_switches[PROGRAM_ID_ARG])
program_id = _string_args[PROGRAM_ID_ARG]; program_id = _string_args[PROGRAM_ID_ARG];
if (!strlen(program_id) && !_switches[FORCE_ARG]) if (!strlen(program_id) && !_switches[FORCE_ARG])
@ -5676,6 +5707,25 @@ static int _stats_update_file(CMD_ARGS)
/* force creation of a region with no id */ /* force creation of a region with no id */
dm_stats_set_program_id(dms, 1, NULL); dm_stats_set_program_id(dms, 1, NULL);
/*
* Start dmfilemapd - it will test the file descriptor to determine
* whether it is necessary to call dm_stats_update_regions_from_fd().
*
* If starting the daemon fails, fall back to a direct update.
*/
if (!_switches[NOMONITOR_ARG]) {
if (!dm_stats_start_filemapd(fd, group_id, abspath, mode,
foreground, verbose)) {
log_warn("Failed to start filemap monitoring daemon.");
goto fallback;
}
goto out;
}
fallback:
/*
* --nomonitor case - perform a one-shot update directly from dmstats.
*/
regions = dm_stats_update_regions_from_fd(dms, fd, group_id); regions = dm_stats_update_regions_from_fd(dms, fd, group_id);
if (close(fd)) if (close(fd))
@ -5700,6 +5750,7 @@ static int _stats_update_file(CMD_ARGS)
printf("%s: Updated group ID " FMTu64 " with "FMTu64" region(s).\n", printf("%s: Updated group ID " FMTu64 " with "FMTu64" region(s).\n",
path, group_id, count); path, group_id, count);
out:
dm_free(regions); dm_free(regions);
dm_free(abspath); dm_free(abspath);
dm_stats_destroy(dms); dm_stats_destroy(dms);
@ -5732,7 +5783,7 @@ static int _stats_help(CMD_ARGS);
* [--programid <id>] [--userdata <data> ] * [--programid <id>] [--userdata <data> ]
* [--bounds histogram_boundaries] [--precise] * [--bounds histogram_boundaries] [--precise]
* [--alldevices|<device>...] * [--alldevices|<device>...]
* create --filemap [--nogroup] * create --filemap [--nogroup] [--nomonitor] [--follow=mode]
* [--programid <id>] [--userdata <data> ] * [--programid <id>] [--userdata <data> ]
* [--bounds histogram_boundaries] [--precise] [<file_path>] * [--bounds histogram_boundaries] [--precise] [<file_path>]
* delete [--allprograms|--programid id] * delete [--allprograms|--programid id]
@ -5764,6 +5815,8 @@ static int _stats_help(CMD_ARGS);
#define PRECISE_OPTS "[--precise] " #define PRECISE_OPTS "[--precise] "
#define SEGMENTS_OPT "[--segments] " #define SEGMENTS_OPT "[--segments] "
#define EXTRA_OPTS HIST_OPTS PRECISE_OPTS #define EXTRA_OPTS HIST_OPTS PRECISE_OPTS
#define FILE_MONITOR_OPTS "[--nomonitor] [--follow mode]"
#define GROUP_ID_OPT "--groupid <id> "
#define ALL_PROGS_OPT "[--allprograms|--programid id] " #define ALL_PROGS_OPT "[--allprograms|--programid id] "
#define ALL_REGIONS_OPT "[--allregions|--regionid id] " #define ALL_REGIONS_OPT "[--allregions|--regionid id] "
#define ALL_DEVICES_OPT "[--alldevices|<device>...] " #define ALL_DEVICES_OPT "[--alldevices|<device>...] "
@ -5774,12 +5827,13 @@ static int _stats_help(CMD_ARGS);
/* command options */ /* command options */
#define CREATE_OPTS REGION_OPTS INDENT ID_OPTS INDENT EXTRA_OPTS INDENT SEGMENTS_OPT #define CREATE_OPTS REGION_OPTS INDENT ID_OPTS INDENT EXTRA_OPTS INDENT SEGMENTS_OPT
#define FILEMAP_OPTS "--filemap [--nogroup]" INDENT ID_OPTS INDENT EXTRA_OPTS #define FILEMAP_OPTS "--filemap [--nogroup] " FILE_MONITOR_OPTS INDENT ID_OPTS INDENT EXTRA_OPTS
#define PRINT_OPTS "[--clear] " ALL_PROGS_REGIONS_DEVICES #define PRINT_OPTS "[--clear] " ALL_PROGS_REGIONS_DEVICES
#define REPORT_OPTS "[--interval <seconds>] [--count <cnt>]" INDENT \ #define REPORT_OPTS "[--interval <seconds>] [--count <cnt>]" INDENT \
"[--units <u>] " SELECT_OPTS INDENT DM_REPORT_OPTS INDENT ALL_PROGS_OPT "[--units <u>] " SELECT_OPTS INDENT DM_REPORT_OPTS INDENT ALL_PROGS_OPT
#define GROUP_OPTS "[--alias NAME] --regions <regions>" INDENT ALL_PROGS_OPT ALL_DEVICES_OPT #define GROUP_OPTS "[--alias NAME] --regions <regions>" INDENT ALL_PROGS_OPT ALL_DEVICES_OPT
#define UNGROUP_OPTS ALL_PROGS_OPT INDENT ALL_DEVICES_OPT #define UNGROUP_OPTS GROUP_ID_OPT ALL_PROGS_OPT INDENT ALL_DEVICES_OPT
#define UPDATE_OPTS GROUP_ID_OPT INDENT FILE_MONITOR_OPTS " <file_path>"
/* /*
* The 'create' command has two entries in the table, to allow for the * The 'create' command has two entries in the table, to allow for the
@ -5790,14 +5844,14 @@ static struct command _stats_subcommands[] = {
{"help", "", 0, 0, 0, 0, _stats_help}, {"help", "", 0, 0, 0, 0, _stats_help},
{"clear", ALL_REGIONS_OPT ALL_DEVICES_OPT, 0, -1, 1, 0, _stats_clear}, {"clear", ALL_REGIONS_OPT ALL_DEVICES_OPT, 0, -1, 1, 0, _stats_clear},
{"create", CREATE_OPTS ALL_DEVICES_OPT, 0, -1, 1, 0, _stats_create}, {"create", CREATE_OPTS ALL_DEVICES_OPT, 0, -1, 1, 0, _stats_create},
{"create", FILEMAP_OPTS "[<file_path>]", 0, -1, 1, 0, _stats_create}, {"create", FILEMAP_OPTS "<file_path>", 0, -1, 1, 0, _stats_create},
{"delete", ALL_PROGS_REGIONS_DEVICES, 1, -1, 1, 0, _stats_delete}, {"delete", ALL_PROGS_REGIONS_DEVICES, 1, -1, 1, 0, _stats_delete},
{"group", GROUP_OPTS, 1, -1, 1, 0, _stats_group}, {"group", GROUP_OPTS, 1, -1, 1, 0, _stats_group},
{"list", ALL_PROGS_OPT ALL_REGIONS_OPT, 0, -1, 1, 0, _stats_report}, {"list", ALL_PROGS_OPT ALL_REGIONS_OPT, 0, -1, 1, 0, _stats_report},
{"print", PRINT_OPTS, 0, -1, 1, 0, _stats_print}, {"print", PRINT_OPTS, 0, -1, 1, 0, _stats_print},
{"report", REPORT_OPTS "[<device>...]", 0, -1, 1, 0, _stats_report}, {"report", REPORT_OPTS "[<device>...]", 0, -1, 1, 0, _stats_report},
{"ungroup", "--groupid <id> " UNGROUP_OPTS, 1, -1, 1, 0, _stats_ungroup}, {"ungroup", UNGROUP_OPTS, 1, -1, 1, 0, _stats_ungroup},
{"update_filemap", "--groupid <id> <file_path>", 1, 1, 0, 0, _stats_update_file}, {"update_filemap", UPDATE_OPTS, 1, 1, 0, 0, _stats_update_file},
{"version", "", 0, -1, 1, 0, _version}, {"version", "", 0, -1, 1, 0, _version},
{NULL, NULL, 0, 0, 0, 0, NULL} {NULL, NULL, 0, 0, 0, 0, NULL}
}; };
@ -6053,6 +6107,11 @@ static int _stats(CMD_ARGS)
return 0; return 0;
} }
if (_switches[FOLLOW_ARG] && _switches[NOMONITOR_ARG]) {
log_error("Use of --follow is incompatible with --nomonitor.");
return 0;
}
/* /*
* Pass the sub-command through to allow a single function to be * Pass the sub-command through to allow a single function to be
* used to implement several distinct sub-commands (e.g. 'report' * used to implement several distinct sub-commands (e.g. 'report'
@ -6418,7 +6477,9 @@ static int _process_switches(int *argcp, char ***argvp, const char *dev_dir)
{"select", 1, &ind, SELECT_ARG}, {"select", 1, &ind, SELECT_ARG},
{"exec", 1, &ind, EXEC_ARG}, {"exec", 1, &ind, EXEC_ARG},
{"filemap", 0, &ind, FILEMAP_ARG}, {"filemap", 0, &ind, FILEMAP_ARG},
{"follow", 1, &ind, FOLLOW_ARG},
{"force", 0, &ind, FORCE_ARG}, {"force", 0, &ind, FORCE_ARG},
{"foreground", 0, &ind, FOREGROUND_ARG},
{"gid", 1, &ind, GID_ARG}, {"gid", 1, &ind, GID_ARG},
{"group", 0, &ind, GROUP_ARG}, {"group", 0, &ind, GROUP_ARG},
{"groupid", 1, &ind, GROUP_ID_ARG}, {"groupid", 1, &ind, GROUP_ID_ARG},
@ -6441,6 +6502,7 @@ static int _process_switches(int *argcp, char ***argvp, const char *dev_dir)
{"notable", 0, &ind, NOTABLE_ARG}, {"notable", 0, &ind, NOTABLE_ARG},
{"notimesuffix", 0, &ind, NOTIMESUFFIX_ARG}, {"notimesuffix", 0, &ind, NOTIMESUFFIX_ARG},
{"udevcookie", 1, &ind, UDEVCOOKIE_ARG}, {"udevcookie", 1, &ind, UDEVCOOKIE_ARG},
{"nomonitor", 0, &ind, NOMONITOR_ARG},
{"noudevrules", 0, &ind, NOUDEVRULES_ARG}, {"noudevrules", 0, &ind, NOUDEVRULES_ARG},
{"noudevsync", 0, &ind, NOUDEVSYNC_ARG}, {"noudevsync", 0, &ind, NOUDEVSYNC_ARG},
{"options", 1, &ind, OPTIONS_ARG}, {"options", 1, &ind, OPTIONS_ARG},
@ -6584,8 +6646,14 @@ static int _process_switches(int *argcp, char ***argvp, const char *dev_dir)
_switches[COLS_ARG]++; _switches[COLS_ARG]++;
if (ind == FILEMAP_ARG) if (ind == FILEMAP_ARG)
_switches[FILEMAP_ARG]++; _switches[FILEMAP_ARG]++;
if (ind == FOLLOW_ARG) {
_switches[FOLLOW_ARG]++;
_string_args[FOLLOW_ARG] = optarg;
}
if (c == 'f' || ind == FORCE_ARG) if (c == 'f' || ind == FORCE_ARG)
_switches[FORCE_ARG]++; _switches[FORCE_ARG]++;
if (ind == FOREGROUND_ARG)
_switches[FOREGROUND_ARG]++;
if (c == 'r' || ind == READ_ONLY) if (c == 'r' || ind == READ_ONLY)
_switches[READ_ONLY]++; _switches[READ_ONLY]++;
if (ind == HISTOGRAM_ARG) if (ind == HISTOGRAM_ARG)
@ -6678,6 +6746,8 @@ static int _process_switches(int *argcp, char ***argvp, const char *dev_dir)
_switches[UDEVCOOKIE_ARG]++; _switches[UDEVCOOKIE_ARG]++;
_udev_cookie = _get_cookie_value(optarg); _udev_cookie = _get_cookie_value(optarg);
} }
if (ind == NOMONITOR_ARG)
_switches[NOMONITOR_ARG]++;
if (ind == NOUDEVRULES_ARG) if (ind == NOUDEVRULES_ARG)
_switches[NOUDEVRULES_ARG]++; _switches[NOUDEVRULES_ARG]++;
if (ind == NOUDEVSYNC_ARG) if (ind == NOUDEVSYNC_ARG)