summaryrefslogtreecommitdiffstats
path: root/tools/corosync-cmapctl.c
diff options
context:
space:
mode:
Diffstat (limited to 'tools/corosync-cmapctl.c')
-rw-r--r--tools/corosync-cmapctl.c996
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);
+}