mirror of
				https://git.proxmox.com/git/mirror_corosync
				synced 2025-11-04 08:24:07 +00:00 
			
		
		
		
	and kill the last gcc warnings Reviewed-by: Steven Dake <sdake@redhat.com> Signed-off-by: Fabio M. Di Nitto <fdinitto@redhat.com>
		
			
				
	
	
		
			746 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			746 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2011 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 <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_PRINT_ALL,
 | 
						|
	ACTION_PRINT_PREFIX,
 | 
						|
	ACTION_TRACK
 | 
						|
};
 | 
						|
 | 
						|
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;
 | 
						|
 | 
						|
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] [-adghsTt] [params...]\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("Delete key:\n");
 | 
						|
	printf("    corosync-cmapctl -d key_name...\n");
 | 
						|
	printf("\n");
 | 
						|
	printf("Get key:\n");
 | 
						|
	printf("    corosync-cmapctl [-b] -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 prefix key_name:\n");
 | 
						|
	printf("    corosync-cmapctl [-b] -t key_name\n");
 | 
						|
	printf("\n");
 | 
						|
	printf("Track changes on keys with key_name name:\n");
 | 
						|
	printf("    corosync-cmapctl [-b] -T key_name\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;
 | 
						|
 | 
						|
		if (err == CS_ERR_TRY_AGAIN) {
 | 
						|
			sleep(1);
 | 
						|
			no_retries++;
 | 
						|
		}
 | 
						|
 | 
						|
		if (no_retries > MAX_TRY_AGAIN) {
 | 
						|
			end_loop = 1;
 | 
						|
		}
 | 
						|
	};
 | 
						|
 | 
						|
	if (err != CS_OK) {
 | 
						|
		fprintf(stderr, "Can't get value of %s. Error %s\n", key_name, cs_strerror(err));
 | 
						|
 | 
						|
		return ;
 | 
						|
	}
 | 
						|
 | 
						|
	printf("%s (", key_name);
 | 
						|
 | 
						|
	switch (type) {
 | 
						|
	case CMAP_VALUETYPE_INT8:
 | 
						|
		printf("%s) = %"PRId8, "i8", i8);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_UINT8:
 | 
						|
		printf("%s) = %"PRIu8, "u8", u8);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_INT16:
 | 
						|
		printf("%s) = %"PRId16, "i16", i16);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_UINT16:
 | 
						|
		printf("%s) = %"PRIu16, "u16", u16);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_INT32:
 | 
						|
		printf("%s) = %"PRId32, "i32", i32);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_UINT32:
 | 
						|
		printf("%s) = %"PRIu32, "u32", u32);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_INT64:
 | 
						|
		printf("%s) = %"PRId64, "i64", i64);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_UINT64:
 | 
						|
		printf("%s) = %"PRIu64, "u64", u64);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_FLOAT:
 | 
						|
		printf("%s) = %f", "flt", flt);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_DOUBLE:
 | 
						|
		printf("%s) = %lf", "dbl", dbl);
 | 
						|
		break;
 | 
						|
	case CMAP_VALUETYPE_STRING:
 | 
						|
		printf("%s) = %s", "str", 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 void 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;
 | 
						|
 | 
						|
	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) {
 | 
						|
		print_key(handle, key_name, value_len, NULL, type);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
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);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
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;
 | 
						|
	int track_prefix;
 | 
						|
	int no_retries;
 | 
						|
 | 
						|
	action = ACTION_PRINT_PREFIX;
 | 
						|
	track_prefix = 1;
 | 
						|
 | 
						|
	while ((c = getopt(argc, argv, "hgsdtTb")) != -1) {
 | 
						|
		switch (c) {
 | 
						|
		case 'h':
 | 
						|
			return print_help();
 | 
						|
			break;
 | 
						|
		case 'b':
 | 
						|
			show_binary++;
 | 
						|
			break;
 | 
						|
		case 'g':
 | 
						|
			action = ACTION_GET;
 | 
						|
			break;
 | 
						|
		case 's':
 | 
						|
			action = ACTION_SET;
 | 
						|
			break;
 | 
						|
		case 'd':
 | 
						|
			action = ACTION_DELETE;
 | 
						|
			break;
 | 
						|
		case 't':
 | 
						|
			action = ACTION_TRACK;
 | 
						|
			break;
 | 
						|
		case 'T':
 | 
						|
			action = ACTION_TRACK;
 | 
						|
			track_prefix = 0;
 | 
						|
			break;
 | 
						|
		case '?':
 | 
						|
			return (EXIT_FAILURE);
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			action = ACTION_PRINT_PREFIX;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (argc == 1 || (argc == 2 && show_binary)) {
 | 
						|
		action = ACTION_PRINT_ALL;
 | 
						|
	}
 | 
						|
 | 
						|
	argc -= optind;
 | 
						|
	argv += optind;
 | 
						|
 | 
						|
	if (argc == 0 && action != ACTION_PRINT_ALL) {
 | 
						|
		fprintf(stderr, "Expected key after options\n");
 | 
						|
		return (EXIT_FAILURE);
 | 
						|
	}
 | 
						|
 | 
						|
	no_retries = 0;
 | 
						|
	while ((err = cmap_initialize(&handle)) == 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_ALL:
 | 
						|
		print_iter(handle, NULL);
 | 
						|
		break;
 | 
						|
	case ACTION_PRINT_PREFIX:
 | 
						|
		for (i = 0; i < argc; i++) {
 | 
						|
			print_iter(handle, argv[i]);
 | 
						|
		}
 | 
						|
		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));
 | 
						|
			}
 | 
						|
		}
 | 
						|
		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));
 | 
						|
			}
 | 
						|
		}
 | 
						|
		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;
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	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);
 | 
						|
}
 |