diff options
Diffstat (limited to 'tools/corosync-cmapctl.c')
-rw-r--r-- | tools/corosync-cmapctl.c | 996 |
1 files changed, 996 insertions, 0 deletions
diff --git a/tools/corosync-cmapctl.c b/tools/corosync-cmapctl.c new file mode 100644 index 0000000..6c79c9a --- /dev/null +++ b/tools/corosync-cmapctl.c @@ -0,0 +1,996 @@ +/* + * Copyright (c) 2011-2012 Red Hat, Inc. + * + * All rights reserved. + * + * Author: Jan Friesse (jfriesse@redhat.com) + * + * This software licensed under BSD license, the text of which follows: + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * - Redistributions of source code must retain the above copyright notice, + * this list of conditions and the following disclaimer. + * - Redistributions in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * - Neither the name of the Red Hat, Inc. nor the names of its + * contributors may be used to endorse or promote products derived from this + * software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF + * THE POSSIBILITY OF SUCH DAMAGE. + */ + +#include <config.h> + +#include <ctype.h> +#include <stdio.h> +#include <poll.h> + +#include <corosync/corotypes.h> +#include <corosync/cmap.h> +#include "../lib/util.h" + +#ifndef INFTIM +#define INFTIM -1 +#endif + +#define MAX_TRY_AGAIN 10 + +enum user_action { + ACTION_GET, + ACTION_SET, + ACTION_DELETE, + ACTION_DELETE_PREFIX, + ACTION_PRINT_PREFIX, + ACTION_TRACK, + ACTION_LOAD, + ACTION_CLEARSTATS, +}; + +struct name_to_type_item { + const char *name; + cmap_value_types_t type; +}; + +struct name_to_type_item name_to_type[] = { + {"i8", CMAP_VALUETYPE_INT8}, + {"u8", CMAP_VALUETYPE_UINT8}, + {"i16", CMAP_VALUETYPE_INT16}, + {"u16", CMAP_VALUETYPE_UINT16}, + {"i32", CMAP_VALUETYPE_INT32}, + {"u32", CMAP_VALUETYPE_UINT32}, + {"i64", CMAP_VALUETYPE_INT64}, + {"u64", CMAP_VALUETYPE_UINT64}, + {"flt", CMAP_VALUETYPE_FLOAT}, + {"dbl", CMAP_VALUETYPE_DOUBLE}, + {"str", CMAP_VALUETYPE_STRING}, + {"bin", CMAP_VALUETYPE_BINARY}}; + +int show_binary = 0; +int quiet = 0; + +static int convert_name_to_type(const char *name) +{ + int i; + + for (i = 0; i < sizeof(name_to_type) / sizeof(*name_to_type); i++) { + if (strcmp(name, name_to_type[i].name) == 0) { + return (name_to_type[i].type); + } + } + + return (-1); +} + +static int print_help(void) +{ + printf("\n"); + printf("usage: corosync-cmapctl [-b] [-DdghsqTCt] [-p filename] [-m map] [params...]\n"); + printf("\n"); + printf(" -b show binary values\n"); + printf("\n"); + printf(" -m select map to use\n"); + printf(" The default map is 'icmap' which contains configuration information and some runtime variables used by corosync. \n"); + printf(" A 'stats' map is also available which displays network statistics - in great detail when knet is used as the transport.\n"); + printf("Set key:\n"); + printf(" corosync-cmapctl -s key_name type value\n"); + printf("\n"); + printf(" where type is one of ([i|u][8|16|32|64] | flt | dbl | str | bin)\n"); + printf(" for bin, value is file name (or - for stdin)\n"); + printf("\n"); + printf(" map can be either 'icmap' (the default) which contains corosync\n"); + printf(" configuration information, or 'stats' which contains statistics\n"); + printf(" about the networking and IPC traffic in some detail.\n"); + printf("\n"); + printf("Clear stats:\n"); + printf(" corosync-cmapctl -C [knet|ipc|totem|schedmiss|all]\n"); + printf(" The 'stats' map is implied\n"); + printf("\n"); + printf("Load settings from a file:\n"); + printf(" corosync-cmapctl -p filename\n"); + printf("\n"); + printf(" the format of the file is:\n"); + printf(" [^[^]]<key_name>[ <type> <value>]\n"); + printf(" Keys prefixed with single caret ('^') are deleted (see -d).\n"); + printf(" Keys (actually prefixes) prefixed with double caret ('^^') are deleted by prefix (see -D).\n"); + printf(" <type> and <value> are optional (not checked) in above cases.\n"); + printf(" Other keys are set (see -s) so both <type> and <value> are required.\n"); + printf("\n"); + printf("Delete key:\n"); + printf(" corosync-cmapctl -d key_name...\n"); + printf("\n"); + printf("Delete multiple keys with prefix:\n"); + printf(" corosync-cmapctl -D key_prefix...\n"); + printf("\n"); + printf("Get key:\n"); + printf(" corosync-cmapctl [-b] -g key_name...\n"); + printf("\n"); + printf("Quiet mode:\n"); + printf(" corosync-cmapctl [-b] -q -g key_name...\n"); + printf("\n"); + printf("Display all keys:\n"); + printf(" corosync-cmapctl [-b]\n"); + printf("\n"); + printf("Display keys with prefix key_name:\n"); + printf(" corosync-cmapctl [-b] key_name...\n"); + printf("\n"); + printf("Track changes on keys with key_name:\n"); + printf(" corosync-cmapctl [-b] -t key_name\n"); + printf("\n"); + printf("Track changes on keys with key prefix:\n"); + printf(" corosync-cmapctl [-b] -T key_prefix\n"); + printf("\n"); + + return (0); +} + +static void print_binary_key (char *value, size_t value_len) +{ + size_t i; + char c; + + for (i = 0; i < value_len; i++) { + c = value[i]; + if (c >= ' ' && c < 0x7f && c != '\\') { + fputc (c, stdout); + } else { + if (c == '\\') { + printf ("\\\\"); + } else { + printf ("\\x%02X", c); + } + } + } +} + +static void print_key(cmap_handle_t handle, + const char *key_name, + size_t value_len, + const void *value, + cmap_value_types_t type) +{ + char *str; + char *bin_value = NULL; + cs_error_t err; + int8_t i8; + uint8_t u8; + int16_t i16; + uint16_t u16; + int32_t i32; + uint32_t u32; + int64_t i64; + uint64_t u64; + float flt; + double dbl; + int end_loop; + int no_retries; + size_t bin_value_len; + + end_loop = 0; + no_retries = 0; + + err = CS_OK; + + while (!end_loop) { + switch (type) { + case CMAP_VALUETYPE_INT8: + if (value == NULL) { + err = cmap_get_int8(handle, key_name, &i8); + } else { + i8 = *((int8_t *)value); + } + break; + case CMAP_VALUETYPE_INT16: + if (value == NULL) { + err = cmap_get_int16(handle, key_name, &i16); + } else { + i16 = *((int16_t *)value); + } + break; + case CMAP_VALUETYPE_INT32: + if (value == NULL) { + err = cmap_get_int32(handle, key_name, &i32); + } else { + i32 = *((int32_t *)value); + } + break; + case CMAP_VALUETYPE_INT64: + if (value == NULL) { + err = cmap_get_int64(handle, key_name, &i64); + } else { + i64 = *((int64_t *)value); + } + break; + case CMAP_VALUETYPE_UINT8: + if (value == NULL) { + err = cmap_get_uint8(handle, key_name, &u8); + } else { + u8 = *((uint8_t *)value); + } + break; + case CMAP_VALUETYPE_UINT16: + if (value == NULL) { + err = cmap_get_uint16(handle, key_name, &u16); + } else { + u16 = *((uint16_t *)value); + } + break; + case CMAP_VALUETYPE_UINT32: + if (value == NULL) { + err = cmap_get_uint32(handle, key_name, &u32); + } else { + u32 = *((uint32_t *)value); + } + break; + case CMAP_VALUETYPE_UINT64: + if (value == NULL) { + err = cmap_get_uint64(handle, key_name, &u64); + } else { + u64 = *((uint64_t *)value); + } + break; + case CMAP_VALUETYPE_FLOAT: + if (value == NULL) { + err = cmap_get_float(handle, key_name, &flt); + } else { + flt = *((float *)value); + } + break; + case CMAP_VALUETYPE_DOUBLE: + if (value == NULL) { + err = cmap_get_double(handle, key_name, &dbl); + } else { + dbl = *((double *)value); + } + break; + case CMAP_VALUETYPE_STRING: + if (value == NULL) { + err = cmap_get_string(handle, key_name, &str); + } else { + str = (char *)value; + } + break; + case CMAP_VALUETYPE_BINARY: + if (show_binary) { + if (value == NULL) { + bin_value = malloc(value_len); + if (bin_value == NULL) { + fprintf(stderr, "Can't alloc memory\n"); + exit(EXIT_FAILURE); + } + bin_value_len = value_len; + err = cmap_get(handle, key_name, bin_value, &bin_value_len, NULL); + } else { + bin_value = (char *)value; + } + } + break; + } + + if (err == CS_OK) { + end_loop = 1; + } else if (err == CS_ERR_TRY_AGAIN) { + sleep(1); + no_retries++; + + if (no_retries > MAX_TRY_AGAIN) { + end_loop = 1; + } + } else { + end_loop = 1; + } + }; + + if (err != CS_OK) { + fprintf(stderr, "Can't get value of %s. Error %s\n", key_name, cs_strerror(err)); + + /* + * bin_value was newly allocated + */ + if (bin_value != NULL && value == NULL) { + free(bin_value); + } + return ; + } + + if (!quiet) + printf("%s (", key_name); + + switch (type) { + case CMAP_VALUETYPE_INT8: + if (!quiet) + printf("%s) = %"PRId8, "i8", i8); + else + printf("%"PRId8, i8); + break; + case CMAP_VALUETYPE_UINT8: + if (!quiet) + printf("%s) = %"PRIu8, "u8", u8); + else + printf("%"PRIu8, u8); + break; + case CMAP_VALUETYPE_INT16: + if (!quiet) + printf("%s) = %"PRId16, "i16", i16); + else + printf("%"PRId16, i16); + break; + case CMAP_VALUETYPE_UINT16: + if (!quiet) + printf("%s) = %"PRIu16, "u16", u16); + else + printf("%"PRIu16, u16); + break; + case CMAP_VALUETYPE_INT32: + if (!quiet) + printf("%s) = %"PRId32, "i32", i32); + else + printf("%"PRId32, i32); + break; + case CMAP_VALUETYPE_UINT32: + if (!quiet) + printf("%s) = %"PRIu32, "u32", u32); + else + printf("%"PRIu32, u32); + break; + case CMAP_VALUETYPE_INT64: + if (!quiet) + printf("%s) = %"PRId64, "i64", i64); + else + printf("%"PRId64, i64); + break; + case CMAP_VALUETYPE_UINT64: + if (!quiet) + printf("%s) = %"PRIu64, "u64", u64); + else + printf("%"PRIu64, u64); + break; + case CMAP_VALUETYPE_FLOAT: + if (!quiet) + printf("%s) = %f", "flt", flt); + else + printf("%f", flt); + break; + case CMAP_VALUETYPE_DOUBLE: + if (!quiet) + printf("%s) = %lf", "dbl", dbl); + else + printf("%lf", dbl); + break; + case CMAP_VALUETYPE_STRING: + if (!quiet) + printf("%s) = %s", "str", str); + else + printf("%s", str); + if (value == NULL) { + free(str); + } + break; + case CMAP_VALUETYPE_BINARY: + printf("%s)", "bin"); + if (show_binary) { + printf(" = "); + if (bin_value) { + print_binary_key(bin_value, value_len); + if (value == NULL) { + free(bin_value); + } + } else { + printf("*empty*"); + } + } + break; + } + + printf("\n"); +} + +static int print_iter(cmap_handle_t handle, const char *prefix) +{ + cmap_iter_handle_t iter_handle; + char key_name[CMAP_KEYNAME_MAXLEN + 1]; + size_t value_len; + cmap_value_types_t type; + cs_error_t err; + int no_result = 1; + + err = cmap_iter_init(handle, prefix, &iter_handle); + if (err != CS_OK) { + fprintf (stderr, "Failed to initialize iteration. Error %s\n", cs_strerror(err)); + exit (EXIT_FAILURE); + } + + while ((err = cmap_iter_next(handle, iter_handle, key_name, &value_len, &type)) == CS_OK) { + no_result = 0; + print_key(handle, key_name, value_len, NULL, type); + } + + cmap_iter_finalize(handle, iter_handle); + return no_result; +} + +static void delete_with_prefix(cmap_handle_t handle, const char *prefix) +{ + cmap_iter_handle_t iter_handle; + char key_name[CMAP_KEYNAME_MAXLEN + 1]; + size_t value_len; + cmap_value_types_t type; + cs_error_t err; + cs_error_t err2; + + err = cmap_iter_init(handle, prefix, &iter_handle); + if (err != CS_OK) { + fprintf (stderr, "Failed to initialize iteration. Error %s\n", cs_strerror(err)); + exit (EXIT_FAILURE); + } + + while ((err = cmap_iter_next(handle, iter_handle, key_name, &value_len, &type)) == CS_OK) { + err2 = cmap_delete(handle, key_name); + if (err2 != CS_OK) { + fprintf(stderr, "Can't delete key %s. Error %s\n", key_name, cs_strerror(err2)); + } + } + cmap_iter_finalize(handle, iter_handle); +} + +static void cmap_notify_fn( + cmap_handle_t cmap_handle, + cmap_track_handle_t cmap_track_handle, + int32_t event, + const char *key_name, + struct cmap_notify_value new_val, + struct cmap_notify_value old_val, + void *user_data) +{ + switch (event) { + case CMAP_TRACK_ADD: + printf("create> "); + print_key(cmap_handle, key_name, new_val.len, new_val.data, new_val.type); + break; + case CMAP_TRACK_DELETE: + printf("delete> "); + print_key(cmap_handle, key_name, old_val.len, old_val.data, old_val.type); + break; + case CMAP_TRACK_MODIFY: + printf("modify> "); + print_key(cmap_handle, key_name, new_val.len, new_val.data, new_val.type); + break; + default: + printf("unknown change> "); + break; + } + +} + +static void add_track(cmap_handle_t handle, const char *key_name, int prefix) +{ + cmap_track_handle_t track_handle; + int32_t track_type; + cs_error_t err; + + track_type = CMAP_TRACK_ADD | CMAP_TRACK_DELETE | CMAP_TRACK_MODIFY; + if (prefix) { + track_type |= CMAP_TRACK_PREFIX; + } + + err = cmap_track_add(handle, key_name, track_type, cmap_notify_fn, NULL, &track_handle); + if (err != CS_OK) { + fprintf(stderr, "Failed to add tracking function. Error %s\n", cs_strerror(err)); + exit (EXIT_FAILURE); + } +} + +static void track_changes(cmap_handle_t handle) +{ + struct pollfd pfd[2]; + int cmap_fd; + cs_error_t err; + int poll_res; + char inbuf[3]; + int quit = CS_FALSE; + + err = cmap_fd_get(handle, &cmap_fd); + if (err != CS_OK) { + fprintf(stderr, "Failed to get file handle. Error %s\n", cs_strerror(err)); + exit (EXIT_FAILURE); + } + + pfd[0].fd = cmap_fd; + pfd[1].fd = STDIN_FILENO; + pfd[0].events = pfd[1].events = POLLIN; + + printf("Type \"q\" to finish\n"); + do { + pfd[0].revents = pfd[1].revents = 0; + poll_res = poll(pfd, 2, INFTIM); + if (poll_res == -1) { + perror("poll"); + } + + if (pfd[1].revents & POLLIN) { + if (fgets(inbuf, sizeof(inbuf), stdin) == NULL) { + quit = CS_TRUE; + } else if (strncmp(inbuf, "q", 1) == 0) { + quit = CS_TRUE; + } + } + + if (pfd[0].revents & POLLIN) { + err = cmap_dispatch(handle, CS_DISPATCH_ALL); + if (err != CS_OK) { + fprintf(stderr, "Dispatch error %s\n", cs_strerror(err)); + quit = CS_TRUE; + } + } + } while (poll_res > 0 && !quit); +} + +static cs_error_t set_key_bin(cmap_handle_t handle, const char *key_name, const char *fname) +{ + FILE *f; + char *val; + char buf[4096]; + size_t size; + size_t readed; + size_t pos; + cs_error_t err; + + if (strcmp(fname, "-") == 0) { + f = stdin; + } else { + f = fopen(fname, "rb"); + if (f == NULL) { + perror("Can't open input file"); + exit(EXIT_FAILURE); + } + } + + val = NULL; + size = 0; + pos = 0; + + while ((readed = fread(buf, 1, sizeof(buf), f)) != 0) { + size += readed; + if ((val = realloc(val, size)) == NULL) { + fprintf(stderr, "Can't alloc memory\n"); + exit (EXIT_FAILURE); + } + memcpy(val + pos, buf, readed); + pos += readed; + } + + if (f != stdin) { + fclose(f); + } + + err = cmap_set(handle, key_name, val, size, CMAP_VALUETYPE_BINARY); + free(val); + + return (err); +} + +static void set_key(cmap_handle_t handle, const char *key_name, const char *key_type_s, const char *key_value_s) +{ + int64_t i64; + uint64_t u64; + double dbl; + float flt; + cs_error_t err = CS_OK; + int scanf_res = 0; + + cmap_value_types_t type; + + if (convert_name_to_type(key_type_s) == -1) { + fprintf(stderr, "Unknown type %s\n", key_type_s); + exit (EXIT_FAILURE); + } + + type = convert_name_to_type(key_type_s); + + switch (type) { + case CMAP_VALUETYPE_INT8: + case CMAP_VALUETYPE_INT16: + case CMAP_VALUETYPE_INT32: + case CMAP_VALUETYPE_INT64: + scanf_res = sscanf(key_value_s, "%"PRId64, &i64); + break; + case CMAP_VALUETYPE_UINT8: + case CMAP_VALUETYPE_UINT16: + case CMAP_VALUETYPE_UINT32: + case CMAP_VALUETYPE_UINT64: + scanf_res = sscanf(key_value_s, "%"PRIu64, &u64); + break; + case CMAP_VALUETYPE_FLOAT: + scanf_res = sscanf(key_value_s, "%f", &flt); + break; + case CMAP_VALUETYPE_DOUBLE: + scanf_res = sscanf(key_value_s, "%lf", &dbl); + break; + case CMAP_VALUETYPE_STRING: + case CMAP_VALUETYPE_BINARY: + /* + * Do nothing + */ + scanf_res = 1; + break; + } + + if (scanf_res != 1) { + fprintf(stderr, "%s is not valid %s type value\n", key_value_s, key_type_s); + exit(EXIT_FAILURE); + } + /* + * We have parsed value, so insert value + */ + + switch (type) { + case CMAP_VALUETYPE_INT8: + if (i64 > INT8_MAX || i64 < INT8_MIN) { + fprintf(stderr, "%s is not valid i8 integer\n", key_value_s); + exit(EXIT_FAILURE); + } + err = cmap_set_int8(handle, key_name, i64); + break; + case CMAP_VALUETYPE_INT16: + if (i64 > INT16_MAX || i64 < INT16_MIN) { + fprintf(stderr, "%s is not valid i16 integer\n", key_value_s); + exit(EXIT_FAILURE); + } + err = cmap_set_int16(handle, key_name, i64); + break; + case CMAP_VALUETYPE_INT32: + if (i64 > INT32_MAX || i64 < INT32_MIN) { + fprintf(stderr, "%s is not valid i32 integer\n", key_value_s); + exit(EXIT_FAILURE); + } + err = cmap_set_int32(handle, key_name, i64); + break; + case CMAP_VALUETYPE_INT64: + err = cmap_set_int64(handle, key_name, i64); + break; + + case CMAP_VALUETYPE_UINT8: + if (u64 > UINT8_MAX) { + fprintf(stderr, "%s is not valid u8 integer\n", key_value_s); + exit(EXIT_FAILURE); + } + err = cmap_set_uint8(handle, key_name, u64); + break; + case CMAP_VALUETYPE_UINT16: + if (u64 > UINT16_MAX) { + fprintf(stderr, "%s is not valid u16 integer\n", key_value_s); + exit(EXIT_FAILURE); + } + err = cmap_set_uint16(handle, key_name, u64); + break; + case CMAP_VALUETYPE_UINT32: + if (u64 > UINT32_MAX) { + fprintf(stderr, "%s is not valid u32 integer\n", key_value_s); + exit(EXIT_FAILURE); + } + err = cmap_set_uint32(handle, key_name, u64); + break; + case CMAP_VALUETYPE_UINT64: + err = cmap_set_uint64(handle, key_name, u64); + break; + case CMAP_VALUETYPE_FLOAT: + err = cmap_set_float(handle, key_name, flt); + break; + case CMAP_VALUETYPE_DOUBLE: + err = cmap_set_double(handle, key_name, dbl); + break; + case CMAP_VALUETYPE_STRING: + err = cmap_set_string(handle, key_name, key_value_s); + break; + case CMAP_VALUETYPE_BINARY: + err = set_key_bin(handle, key_name, key_value_s); + break; + } + + if (err != CS_OK) { + fprintf (stderr, "Failed to set key %s. Error %s\n", key_name, cs_strerror(err)); + exit (EXIT_FAILURE); + } +} + + +static void read_in_config_file(cmap_handle_t handle, char * filename) +{ + int ignore; + int c; + FILE* fh; + char buf[1024]; + char * line; + char *key_name; + char *key_type_s; + char *key_value_s; + + fh = fopen(filename, "r"); + if (fh == NULL) { + perror ("Couldn't open file."); + return; + } + + while (fgets (buf, 1024, fh) != NULL) { + /* find the first real character, if it is + * a '#' then ignore this line. + * else process. + * if no real characters then also ignore. + */ + ignore = 1; + for (c = 0; c < 1024; c++) { + if (isblank (buf[c])) { + continue; + } + + if (buf[c] == '#' || buf[c] == '\n') { + ignore = 1; + break; + } + ignore = 0; + line = &buf[c]; + break; + } + if (ignore == 1) { + continue; + } + + /* + * should be: + * [^[^]]<key>[ <type> <value>] + */ + key_name = strtok(line, " \n"); + if (key_name && *key_name == '^') { + key_name++; + if (*key_name == '^') { + key_name++; + delete_with_prefix(handle, key_name); + } else { + cs_error_t err; + + err = cmap_delete(handle, key_name); + if (err != CS_OK) { + fprintf(stderr, "Can't delete key %s. Error %s\n", key_name, cs_strerror(err)); + } + } + } else { + key_type_s = strtok(NULL, " \n"); + key_value_s = strtok(NULL, " \n"); + if (key_type_s == NULL || key_value_s == NULL) { + fprintf(stderr, "Both type and value for key %s are required\n", key_name); + exit (EXIT_FAILURE); + } + set_key(handle, key_name, key_type_s, key_value_s); + } + } + + fclose (fh); +} + +static void clear_stats(cmap_handle_t handle, char *clear_opt) +{ + char key_name[CMAP_KEYNAME_MAXLEN + 1]; + + sprintf(key_name, "stats.clear.%s", clear_opt); + cmap_set_uint32(handle, key_name, 1); +} + +int main(int argc, char *argv[]) +{ + enum user_action action; + int c; + cs_error_t err; + cmap_handle_t handle; + int i; + size_t value_len; + cmap_value_types_t type; + cmap_map_t map = CMAP_MAP_DEFAULT; + int track_prefix; + int map_set = 0; + int no_retries; + char * clear_opt = NULL; + char * settings_file = NULL; + int count_of_no_result = 0; + + action = ACTION_PRINT_PREFIX; + track_prefix = 1; + + while ((c = getopt(argc, argv, "m:hqgsdDtTbp:C:")) != -1) { + switch (c) { + case 'h': + return print_help(); + break; + case 'b': + show_binary++; + break; + case 'q': + quiet = 1; + break; + case 'g': + action = ACTION_GET; + break; + case 's': + action = ACTION_SET; + break; + case 'd': + action = ACTION_DELETE; + break; + case 'D': + action = ACTION_DELETE_PREFIX; + break; + case 'p': + settings_file = optarg; + action = ACTION_LOAD; + break; + case 'C': + if (strcmp(optarg, "knet") == 0 || + strcmp(optarg, "totem") == 0 || + strcmp(optarg, "ipc") == 0 || + strcmp(optarg, "schedmiss") == 0 || + strcmp(optarg, "all") == 0) { + action = ACTION_CLEARSTATS; + clear_opt = optarg; + + /* Force the map to be STATS */ + map = CMAP_MAP_STATS; + } + else { + fprintf(stderr, "argument to -C should be 'knet', 'totem', 'ipc', 'schedmiss' or 'all'\n"); + return (EXIT_FAILURE); + } + break; + case 't': + action = ACTION_TRACK; + track_prefix = 0; + break; + case 'T': + action = ACTION_TRACK; + break; + case 'm': + if (strcmp(optarg, "icmap") == 0 || + strcmp(optarg, "default") == 0) { + map = CMAP_MAP_ICMAP; + map_set = 1; + } + if (strcmp(optarg, "stats") == 0) { + map = CMAP_MAP_STATS; + map_set = 1; + } + if (!map_set) { + fprintf(stderr, "invalid map name, must be 'default', 'icmap' or 'stats'\n"); + return (EXIT_FAILURE); + } + break; + case '?': + return (EXIT_FAILURE); + break; + default: + action = ACTION_PRINT_PREFIX; + break; + } + } + + argc -= optind; + argv += optind; + + if (argc == 0 && + action != ACTION_LOAD && + action != ACTION_CLEARSTATS && + action != ACTION_PRINT_PREFIX) { + fprintf(stderr, "Expected key after options\n"); + return (EXIT_FAILURE); + } + + no_retries = 0; + + while ((err = cmap_initialize_map(&handle, map)) == CS_ERR_TRY_AGAIN && no_retries++ < MAX_TRY_AGAIN) { + sleep(1); + } + + if (err != CS_OK) { + fprintf (stderr, "Failed to initialize the cmap API. Error %s\n", cs_strerror(err)); + exit (EXIT_FAILURE); + } + + switch (action) { + case ACTION_PRINT_PREFIX: + if (argc == 0) { + count_of_no_result = print_iter(handle, NULL); + } else { + for (i = 0; i < argc; i++) { + count_of_no_result += print_iter(handle, argv[i]); + } + } + + if (count_of_no_result > 0 && count_of_no_result >= argc) { + return (EXIT_FAILURE); + } + break; + case ACTION_GET: + for (i = 0; i < argc; i++) { + err = cmap_get(handle, argv[i], NULL, &value_len, &type); + if (err == CS_OK) { + print_key(handle, argv[i], value_len, NULL, type); + } else { + fprintf(stderr, "Can't get key %s. Error %s\n", argv[i], cs_strerror(err)); + return (EXIT_FAILURE); + } + } + break; + case ACTION_DELETE: + for (i = 0; i < argc; i++) { + err = cmap_delete(handle, argv[i]); + if (err != CS_OK) { + fprintf(stderr, "Can't delete key %s. Error %s\n", argv[i], cs_strerror(err)); + return (EXIT_FAILURE); + } + } + break; + case ACTION_DELETE_PREFIX: + for (i = 0; i < argc; i++) { + delete_with_prefix(handle, argv[i]); + } + break; + case ACTION_LOAD: + read_in_config_file(handle, settings_file); + break; + case ACTION_TRACK: + for (i = 0; i < argc; i++) { + add_track(handle, argv[i], track_prefix); + } + track_changes(handle); + break; + case ACTION_SET: + if (argc < 3) { + fprintf(stderr, "At least 3 parameters are expected for set\n"); + return (EXIT_FAILURE); + } + + set_key(handle, argv[0], argv[1], argv[2]); + break; + case ACTION_CLEARSTATS: + clear_stats(handle, clear_opt); + break; + + } + + err = cmap_finalize(handle); + if (err != CS_OK) { + fprintf (stderr, "Failed to finalize the cmap API. Error %s\n", cs_strerror(err)); + exit (EXIT_FAILURE); + } + + return (0); +} |