/* * virtypedparam.c: utility functions for dealing with virTypedParameters * * Copyright (C) 2011-2014 Red Hat, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see * . * */ #include #include "virtypedparam.h" #include #include "viralloc.h" #include "virutil.h" #include "virerror.h" #include "virstring.h" #define VIR_FROM_THIS VIR_FROM_NONE VIR_ENUM_IMPL(virTypedParameter, VIR_TYPED_PARAM_LAST, "unknown", "int", "uint", "llong", "ullong", "double", "boolean", "string", ); /* When editing this file, ensure that public exported functions * (those in libvirt_public.syms) either trigger no errors, or else * reset error on entrance and call virDispatchError() on exit; while * internal utility functions (those in libvirt_private.syms) may * report errors that the caller will dispatch. */ static int virTypedParamsSortName(const void *left, const void *right) { const virTypedParameter *param_left = left, *param_right = right; return strcmp(param_left->field, param_right->field); } /* Validate that PARAMS contains only recognized parameter names with * correct types, and with no duplicates except for parameters * specified with VIR_TYPED_PARAM_MULTIPLE flag in type. * Pass in as many name/type pairs as appropriate, and pass NULL to end * the list of accepted parameters. Return 0 on success, -1 on failure * with error message already issued. */ int virTypedParamsValidate(virTypedParameterPtr params, int nparams, ...) { va_list ap; int ret = -1; size_t i, j; const char *name, *last_name = NULL; int type; size_t nkeys = 0, nkeysalloc = 0; virTypedParameterPtr sorted = NULL, keys = NULL; va_start(ap, nparams); if (VIR_ALLOC_N(sorted, nparams) < 0) goto cleanup; /* Here we intentionally don't copy values */ memcpy(sorted, params, sizeof(*params) * nparams); qsort(sorted, nparams, sizeof(*sorted), virTypedParamsSortName); name = va_arg(ap, const char *); while (name) { type = va_arg(ap, int); if (VIR_RESIZE_N(keys, nkeysalloc, nkeys, 1) < 0) goto cleanup; if (virStrcpyStatic(keys[nkeys].field, name) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Field name '%s' too long"), name); goto cleanup; } keys[nkeys].type = type & ~VIR_TYPED_PARAM_MULTIPLE; /* Value is not used anyway */ keys[nkeys].value.i = type & VIR_TYPED_PARAM_MULTIPLE; nkeys++; name = va_arg(ap, const char *); } qsort(keys, nkeys, sizeof(*keys), virTypedParamsSortName); for (i = 0, j = 0; i < nparams && j < nkeys;) { if (STRNEQ(sorted[i].field, keys[j].field)) { j++; } else { if (STREQ_NULLABLE(last_name, sorted[i].field) && !(keys[j].value.i & VIR_TYPED_PARAM_MULTIPLE)) { virReportError(VIR_ERR_INVALID_ARG, _("parameter '%s' occurs multiple times"), sorted[i].field); goto cleanup; } if (sorted[i].type != keys[j].type) { const char *badtype; badtype = virTypedParameterTypeToString(sorted[i].type); if (!badtype) badtype = virTypedParameterTypeToString(0); virReportError(VIR_ERR_INVALID_ARG, _("invalid type '%s' for parameter '%s', " "expected '%s'"), badtype, sorted[i].field, virTypedParameterTypeToString(keys[j].type)); goto cleanup; } last_name = sorted[i].field; i++; } } if (j == nkeys && i != nparams) { virReportError(VIR_ERR_ARGUMENT_UNSUPPORTED, _("parameter '%s' not supported"), sorted[i].field); goto cleanup; } ret = 0; cleanup: va_end(ap); VIR_FREE(sorted); VIR_FREE(keys); return ret; } /* Check if params contains only specified parameter names. Return true if * only specified names are present in params, false if params contains any * unspecified parameter name. */ bool virTypedParamsCheck(virTypedParameterPtr params, int nparams, const char **names, int nnames) { size_t i, j; for (i = 0; i < nparams; i++) { bool found = false; for (j = 0; j < nnames; j++) { if (STREQ(params[i].field, names[j])) { found = true; break; } } if (!found) return false; } return true; } char * virTypedParameterToString(virTypedParameterPtr param) { char *value = NULL; switch (param->type) { case VIR_TYPED_PARAM_INT: ignore_value(virAsprintf(&value, "%d", param->value.i)); break; case VIR_TYPED_PARAM_UINT: ignore_value(virAsprintf(&value, "%u", param->value.ui)); break; case VIR_TYPED_PARAM_LLONG: ignore_value(virAsprintf(&value, "%lld", param->value.l)); break; case VIR_TYPED_PARAM_ULLONG: ignore_value(virAsprintf(&value, "%llu", param->value.ul)); break; case VIR_TYPED_PARAM_DOUBLE: ignore_value(virAsprintf(&value, "%g", param->value.d)); break; case VIR_TYPED_PARAM_BOOLEAN: ignore_value(virAsprintf(&value, "%d", param->value.b)); break; case VIR_TYPED_PARAM_STRING: ignore_value(VIR_STRDUP(value, param->value.s)); break; default: virReportError(VIR_ERR_INTERNAL_ERROR, _("unexpected type %d for field %s"), param->type, param->field); } return value; } /* Assign name, type, and the appropriately typed arg to param; in the * case of a string, the caller is assumed to have malloc'd a string, * or can pass NULL to have this function malloc an empty string. * Return 0 on success, -1 after an error message on failure. */ int virTypedParameterAssign(virTypedParameterPtr param, const char *name, int type, ...) { va_list ap; int ret = -1; va_start(ap, type); if (virStrcpyStatic(param->field, name) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Field name '%s' too long"), name); goto cleanup; } param->type = type; switch (type) { case VIR_TYPED_PARAM_INT: param->value.i = va_arg(ap, int); break; case VIR_TYPED_PARAM_UINT: param->value.ui = va_arg(ap, unsigned int); break; case VIR_TYPED_PARAM_LLONG: param->value.l = va_arg(ap, long long int); break; case VIR_TYPED_PARAM_ULLONG: param->value.ul = va_arg(ap, unsigned long long int); break; case VIR_TYPED_PARAM_DOUBLE: param->value.d = va_arg(ap, double); break; case VIR_TYPED_PARAM_BOOLEAN: param->value.b = !!va_arg(ap, int); break; case VIR_TYPED_PARAM_STRING: param->value.s = va_arg(ap, char *); if (!param->value.s && VIR_STRDUP(param->value.s, "") < 0) goto cleanup; break; default: virReportError(VIR_ERR_INTERNAL_ERROR, _("unexpected type %d for field %s"), type, name); goto cleanup; } ret = 0; cleanup: va_end(ap); return ret; } /* Assign name, type, and convert the argument from a const string. * In case of a string, the string is copied. * Return 0 on success, -1 after an error message on failure. */ int virTypedParameterAssignFromStr(virTypedParameterPtr param, const char *name, int type, const char *val) { int ret = -1; if (!val) { virReportError(VIR_ERR_INVALID_ARG, _("NULL value for field '%s'"), name); goto cleanup; } if (virStrcpyStatic(param->field, name) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Field name '%s' too long"), name); goto cleanup; } param->type = type; switch (type) { case VIR_TYPED_PARAM_INT: if (virStrToLong_i(val, NULL, 10, ¶m->value.i) < 0) { virReportError(VIR_ERR_INVALID_ARG, _("Invalid value for field '%s': expected int"), name); goto cleanup; } break; case VIR_TYPED_PARAM_UINT: if (virStrToLong_ui(val, NULL, 10, ¶m->value.ui) < 0) { virReportError(VIR_ERR_INVALID_ARG, _("Invalid value for field '%s': " "expected unsigned int"), name); goto cleanup; } break; case VIR_TYPED_PARAM_LLONG: if (virStrToLong_ll(val, NULL, 10, ¶m->value.l) < 0) { virReportError(VIR_ERR_INVALID_ARG, _("Invalid value for field '%s': " "expected long long"), name); goto cleanup; } break; case VIR_TYPED_PARAM_ULLONG: if (virStrToLong_ull(val, NULL, 10, ¶m->value.ul) < 0) { virReportError(VIR_ERR_INVALID_ARG, _("Invalid value for field '%s': " "expected unsigned long long"), name); goto cleanup; } break; case VIR_TYPED_PARAM_DOUBLE: if (virStrToDouble(val, NULL, ¶m->value.d) < 0) { virReportError(VIR_ERR_INVALID_ARG, _("Invalid value for field '%s': " "expected double"), name); goto cleanup; } break; case VIR_TYPED_PARAM_BOOLEAN: if (STRCASEEQ(val, "true") || STREQ(val, "1")) { param->value.b = true; } else if (STRCASEEQ(val, "false") || STREQ(val, "0")) { param->value.b = false; } else { virReportError(VIR_ERR_INVALID_ARG, _("Invalid boolean value for field '%s'"), name); goto cleanup; } break; case VIR_TYPED_PARAM_STRING: if (VIR_STRDUP(param->value.s, val) < 0) goto cleanup; break; default: virReportError(VIR_ERR_INTERNAL_ERROR, _("unexpected type %d for field %s"), type, name); goto cleanup; } ret = 0; cleanup: return ret; } /** * virTypedParamsReplaceString: * @params: pointer to the array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to set * @value: the value to store into the parameter * * Sets new value @value to parameter called @name with char * type. If the * parameter does not exist yet in @params, it is automatically created and * @naprams is incremented by one. Otherwise current value of the parameter * is freed on success. The function creates its own copy of @value string, * which needs to be freed using virTypedParamsFree or virTypedParamsClear. * * Returns 0 on success, -1 on error. */ int virTypedParamsReplaceString(virTypedParameterPtr *params, int *nparams, const char *name, const char *value) { char *str = NULL; char *old = NULL; size_t n = *nparams; virTypedParameterPtr param; param = virTypedParamsGet(*params, n, name); if (param) { if (param->type != VIR_TYPED_PARAM_STRING) { virReportError(VIR_ERR_INVALID_ARG, _("Parameter '%s' is not a string"), param->field); goto error; } old = param->value.s; } else { if (VIR_EXPAND_N(*params, n, 1) < 0) goto error; param = *params + n - 1; } if (VIR_STRDUP(str, value) < 0) goto error; if (virTypedParameterAssign(param, name, VIR_TYPED_PARAM_STRING, str) < 0) { param->value.s = old; VIR_FREE(str); goto error; } VIR_FREE(old); *nparams = n; return 0; error: return -1; } int virTypedParamsCopy(virTypedParameterPtr *dst, virTypedParameterPtr src, int nparams) { size_t i; *dst = NULL; if (!src || nparams <= 0) return 0; if (VIR_ALLOC_N(*dst, nparams) < 0) return -1; for (i = 0; i < nparams; i++) { ignore_value(virStrcpyStatic((*dst)[i].field, src[i].field)); (*dst)[i].type = src[i].type; if (src[i].type == VIR_TYPED_PARAM_STRING) { if (VIR_STRDUP((*dst)[i].value.s, src[i].value.s) < 0) { virTypedParamsFree(*dst, i - 1); *dst = NULL; return -1; } } else { (*dst)[i].value = src[i].value; } } return 0; } /* The following APIs are public and their signature may never change. */ /** * virTypedParamsGet: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * * Finds typed parameter called @name. * * Returns pointer to the parameter or NULL if it does not exist in @params. * This function does not raise an error, even when returning NULL. */ virTypedParameterPtr virTypedParamsGet(virTypedParameterPtr params, int nparams, const char *name) { size_t i; /* No need to reset errors, since this function doesn't report any. */ if (!params || !name) return NULL; for (i = 0; i < nparams; i++) { if (STREQ(params[i].field, name)) return params + i; } return NULL; } /** * virTypedParamsFilter: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * @ret: pointer to the returned array * * Filters @params retaining only the parameters named @name in the * resulting array @ret. Caller should free the @ret array but not * the items since they are pointing to the @params elements. * * Returns amount of elements in @ret on success, -1 on error. */ int virTypedParamsFilter(virTypedParameterPtr params, int nparams, const char *name, virTypedParameterPtr **ret) { size_t i, n = 0; virCheckNonNullArgGoto(params, error); virCheckNonNullArgGoto(name, error); virCheckNonNullArgGoto(ret, error); if (VIR_ALLOC_N(*ret, nparams) < 0) goto error; for (i = 0; i < nparams; i++) { if (STREQ(params[i].field, name)) { (*ret)[n] = ¶ms[i]; n++; } } return n; error: return -1; } #define VIR_TYPED_PARAM_CHECK_TYPE(check_type) \ do { if (param->type != check_type) { \ virReportError(VIR_ERR_INVALID_ARG, \ _("Invalid type '%s' requested for parameter '%s', " \ "actual type is '%s'"), \ virTypedParameterTypeToString(check_type), \ name, \ virTypedParameterTypeToString(param->type)); \ virDispatchError(NULL); \ return -1; \ } } while (0) /** * virTypedParamsGetInt: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * @value: where to store the parameter's value * * Finds typed parameter called @name and store its int value in @value. The * function fails with VIR_ERR_INVALID_ARG error if the parameter does not * have the expected type. By passing NULL as @value, the function may be * used to check presence and type of the parameter. * * Returns 1 on success, 0 when the parameter does not exist in @params, or * -1 on error. */ int virTypedParamsGetInt(virTypedParameterPtr params, int nparams, const char *name, int *value) { virTypedParameterPtr param; virResetLastError(); if (!(param = virTypedParamsGet(params, nparams, name))) return 0; VIR_TYPED_PARAM_CHECK_TYPE(VIR_TYPED_PARAM_INT); if (value) *value = param->value.i; return 1; } /** * virTypedParamsGetUInt: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * @value: where to store the parameter's value * * Finds typed parameter called @name and store its unsigned int value in * @value. The function fails with VIR_ERR_INVALID_ARG error if the parameter * does not have the expected type. By passing NULL as @value, the function * may be used to check presence and type of the parameter. * * Returns 1 on success, 0 when the parameter does not exist in @params, or * -1 on error. */ int virTypedParamsGetUInt(virTypedParameterPtr params, int nparams, const char *name, unsigned int *value) { virTypedParameterPtr param; virResetLastError(); if (!(param = virTypedParamsGet(params, nparams, name))) return 0; VIR_TYPED_PARAM_CHECK_TYPE(VIR_TYPED_PARAM_UINT); if (value) *value = param->value.ui; return 1; } /** * virTypedParamsGetLLong: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * @value: where to store the parameter's value * * Finds typed parameter called @name and store its long long int value in * @value. The function fails with VIR_ERR_INVALID_ARG error if the parameter * does not have the expected type. By passing NULL as @value, the function * may be used to check presence and type of the parameter. * * Returns 1 on success, 0 when the parameter does not exist in @params, or * -1 on error. */ int virTypedParamsGetLLong(virTypedParameterPtr params, int nparams, const char *name, long long *value) { virTypedParameterPtr param; virResetLastError(); if (!(param = virTypedParamsGet(params, nparams, name))) return 0; VIR_TYPED_PARAM_CHECK_TYPE(VIR_TYPED_PARAM_LLONG); if (value) *value = param->value.l; return 1; } /** * virTypedParamsGetULLong: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * @value: where to store the parameter's value * * Finds typed parameter called @name and store its unsigned long long int * value in @value. The function fails with VIR_ERR_INVALID_ARG error if the * parameter does not have the expected type. By passing NULL as @value, the * function may be used to check presence and type of the parameter. * * Returns 1 on success, 0 when the parameter does not exist in @params, or * -1 on error. */ int virTypedParamsGetULLong(virTypedParameterPtr params, int nparams, const char *name, unsigned long long *value) { virTypedParameterPtr param; virResetLastError(); if (!(param = virTypedParamsGet(params, nparams, name))) return 0; VIR_TYPED_PARAM_CHECK_TYPE(VIR_TYPED_PARAM_ULLONG); if (value) *value = param->value.ul; return 1; } /** * virTypedParamsGetDouble: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * @value: where to store the parameter's value * * Finds typed parameter called @name and store its double value in @value. * The function fails with VIR_ERR_INVALID_ARG error if the parameter does not * have the expected type. By passing NULL as @value, the function may be used * to check presence and type of the parameter. * * Returns 1 on success, 0 when the parameter does not exist in @params, or * -1 on error. */ int virTypedParamsGetDouble(virTypedParameterPtr params, int nparams, const char *name, double *value) { virTypedParameterPtr param; virResetLastError(); if (!(param = virTypedParamsGet(params, nparams, name))) return 0; VIR_TYPED_PARAM_CHECK_TYPE(VIR_TYPED_PARAM_DOUBLE); if (value) *value = param->value.d; return 1; } /** * virTypedParamsGetBoolean: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * @value: where to store the parameter's value * * Finds typed parameter called @name and store its boolean value in @value. * The function fails with VIR_ERR_INVALID_ARG error if the parameter does not * have the expected type. By passing NULL as @value, the function may be used * to check presence and type of the parameter. * * Returns 1 on success, 0 when the parameter does not exist in @params, or * -1 on error. */ int virTypedParamsGetBoolean(virTypedParameterPtr params, int nparams, const char *name, int *value) { virTypedParameterPtr param; virResetLastError(); if (!(param = virTypedParamsGet(params, nparams, name))) return 0; VIR_TYPED_PARAM_CHECK_TYPE(VIR_TYPED_PARAM_BOOLEAN); if (value) *value = !!param->value.b; return 1; } /** * virTypedParamsGetString: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * @value: where to store the parameter's value * * Finds typed parameter called @name and store its char * value in @value. * The function does not create a copy of the string and the caller must not * free the string @value points to. The function fails with * VIR_ERR_INVALID_ARG error if the parameter does not have the expected type. * By passing NULL as @value, the function may be used to check presence and * type of the parameter. * * Returns 1 on success, 0 when the parameter does not exist in @params, or * -1 on error. */ int virTypedParamsGetString(virTypedParameterPtr params, int nparams, const char *name, const char **value) { virTypedParameterPtr param; virResetLastError(); if (!(param = virTypedParamsGet(params, nparams, name))) return 0; VIR_TYPED_PARAM_CHECK_TYPE(VIR_TYPED_PARAM_STRING); if (value) *value = param->value.s; return 1; } /** * virTypedParamsGetStringList: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @name: name of the parameter to find * @values: array of returned values * * Finds all parameters with desired @name within @params and * store their values into @values. The @values array is self * allocated and its length is stored into @picked. When no * longer needed, caller should free the returned array, but not * the items since they are taken from @params array. * * Returns amount of strings in @values array on success, * -1 otherwise. */ int virTypedParamsGetStringList(virTypedParameterPtr params, int nparams, const char *name, const char ***values) { size_t i, n; int nfiltered; virTypedParameterPtr *filtered = NULL; virResetLastError(); virCheckNonNullArgGoto(values, error); *values = NULL; nfiltered = virTypedParamsFilter(params, nparams, name, &filtered); if (nfiltered < 0) goto error; if (nfiltered && VIR_ALLOC_N(*values, nfiltered) < 0) goto error; for (n = 0, i = 0; i < nfiltered; i++) { if (filtered[i]->type == VIR_TYPED_PARAM_STRING) (*values)[n++] = filtered[i]->value.s; } VIR_FREE(filtered); return n; error: if (values) VIR_FREE(*values); VIR_FREE(filtered); virDispatchError(NULL); return -1; } /** * virTypedParamsAddInt: * @params: pointer to the array of typed parameters * @nparams: number of parameters in the @params array * @maxparams: maximum number of parameters that can be stored in @params * array without allocating more memory * @name: name of the parameter to find * @value: the value to store into the new parameter * * Adds new parameter called @name with int type and sets its value to @value. * If @params array points to NULL or to a space that is not large enough to * accommodate the new parameter (@maxparams < @nparams + 1), the function * allocates more space for it and updates @maxparams. On success, @nparams * is incremented by one. The function fails with VIR_ERR_INVALID_ARG error * if the parameter already exists in @params. * * Returns 0 on success, -1 on error. */ int virTypedParamsAddInt(virTypedParameterPtr *params, int *nparams, int *maxparams, const char *name, int value) { size_t max = *maxparams; size_t n = *nparams; virResetLastError(); if (VIR_RESIZE_N(*params, max, n, 1) < 0) goto error; *maxparams = max; if (virTypedParameterAssign(*params + n, name, VIR_TYPED_PARAM_INT, value) < 0) goto error; *nparams += 1; return 0; error: virDispatchError(NULL); return -1; } /** * virTypedParamsAddUInt: * @params: pointer to the array of typed parameters * @nparams: number of parameters in the @params array * @maxparams: maximum number of parameters that can be stored in @params * array without allocating more memory * @name: name of the parameter to find * @value: the value to store into the new parameter * * Adds new parameter called @name with unsigned int type and sets its value * to @value. If @params array points to NULL or to a space that is not large * enough to accommodate the new parameter (@maxparams < @nparams + 1), the * function allocates more space for it and updates @maxparams. On success, * @nparams is incremented by one. The function fails with VIR_ERR_INVALID_ARG * error if the parameter already exists in @params. * * Returns 0 on success, -1 on error. */ int virTypedParamsAddUInt(virTypedParameterPtr *params, int *nparams, int *maxparams, const char *name, unsigned int value) { size_t max = *maxparams; size_t n = *nparams; virResetLastError(); if (VIR_RESIZE_N(*params, max, n, 1) < 0) goto error; *maxparams = max; if (virTypedParameterAssign(*params + n, name, VIR_TYPED_PARAM_UINT, value) < 0) goto error; *nparams += 1; return 0; error: virDispatchError(NULL); return -1; } /** * virTypedParamsAddLLong: * @params: pointer to the array of typed parameters * @nparams: number of parameters in the @params array * @maxparams: maximum number of parameters that can be stored in @params * array without allocating more memory * @name: name of the parameter to find * @value: the value to store into the new parameter * * Adds new parameter called @name with long long int type and sets its value * to @value. If @params array points to NULL or to a space that is not large * enough to accommodate the new parameter (@maxparams < @nparams + 1), the * function allocates more space for it and updates @maxparams. On success, * @nparams is incremented by one. The function fails with VIR_ERR_INVALID_ARG * error if the parameter already exists in @params. * * Returns 0 on success, -1 on error. */ int virTypedParamsAddLLong(virTypedParameterPtr *params, int *nparams, int *maxparams, const char *name, long long value) { size_t max = *maxparams; size_t n = *nparams; virResetLastError(); if (VIR_RESIZE_N(*params, max, n, 1) < 0) goto error; *maxparams = max; if (virTypedParameterAssign(*params + n, name, VIR_TYPED_PARAM_LLONG, value) < 0) goto error; *nparams += 1; return 0; error: virDispatchError(NULL); return -1; } /** * virTypedParamsAddULLong: * @params: pointer to the array of typed parameters * @nparams: number of parameters in the @params array * @maxparams: maximum number of parameters that can be stored in @params * array without allocating more memory * @name: name of the parameter to find * @value: the value to store into the new parameter * * Adds new parameter called @name with unsigned long long type and sets its * value to @value. If @params array points to NULL or to a space that is not * large enough to accommodate the new parameter (@maxparams < @nparams + 1), * the function allocates more space for it and updates @maxparams. On success, * @nparams is incremented by one. The function fails with VIR_ERR_INVALID_ARG * error if the parameter already exists in @params. * * Returns 0 on success, -1 on error. */ int virTypedParamsAddULLong(virTypedParameterPtr *params, int *nparams, int *maxparams, const char *name, unsigned long long value) { size_t max = *maxparams; size_t n = *nparams; virResetLastError(); if (VIR_RESIZE_N(*params, max, n, 1) < 0) goto error; *maxparams = max; if (virTypedParameterAssign(*params + n, name, VIR_TYPED_PARAM_ULLONG, value) < 0) goto error; *nparams += 1; return 0; error: virDispatchError(NULL); return -1; } /** * virTypedParamsAddDouble: * @params: pointer to the array of typed parameters * @nparams: number of parameters in the @params array * @maxparams: maximum number of parameters that can be stored in @params * array without allocating more memory * @name: name of the parameter to find * @value: the value to store into the new parameter * * Adds new parameter called @name with double type and sets its value to * @value. If @params array points to NULL or to a space that is not large * enough to accommodate the new parameter (@maxparams < @nparams + 1), the * function allocates more space for it and updates @maxparams. On success, * @nparams is incremented by one. The function fails with VIR_ERR_INVALID_ARG * error if the parameter already exists in @params. * * Returns 0 on success, -1 on error. */ int virTypedParamsAddDouble(virTypedParameterPtr *params, int *nparams, int *maxparams, const char *name, double value) { size_t max = *maxparams; size_t n = *nparams; virResetLastError(); if (VIR_RESIZE_N(*params, max, n, 1) < 0) goto error; *maxparams = max; if (virTypedParameterAssign(*params + n, name, VIR_TYPED_PARAM_DOUBLE, value) < 0) goto error; *nparams += 1; return 0; error: virDispatchError(NULL); return -1; } /** * virTypedParamsAddBoolean: * @params: pointer to the array of typed parameters * @nparams: number of parameters in the @params array * @maxparams: maximum number of parameters that can be stored in @params * array without allocating more memory * @name: name of the parameter to find * @value: the value to store into the new parameter * * Adds new parameter called @name with boolean type and sets its value to * @value. If @params array points to NULL or to a space that is not large * enough to accommodate the new parameter (@maxparams < @nparams + 1), the * function allocates more space for it and updates @maxparams. On success, * @nparams is incremented by one. The function fails with VIR_ERR_INVALID_ARG * error if the parameter already exists in @params. * * Returns 0 on success, -1 on error. */ int virTypedParamsAddBoolean(virTypedParameterPtr *params, int *nparams, int *maxparams, const char *name, int value) { size_t max = *maxparams; size_t n = *nparams; virResetLastError(); if (VIR_RESIZE_N(*params, max, n, 1) < 0) goto error; *maxparams = max; if (virTypedParameterAssign(*params + n, name, VIR_TYPED_PARAM_BOOLEAN, value) < 0) goto error; *nparams += 1; return 0; error: virDispatchError(NULL); return -1; } /** * virTypedParamsAddString: * @params: pointer to the array of typed parameters * @nparams: number of parameters in the @params array * @maxparams: maximum number of parameters that can be stored in @params * array without allocating more memory * @name: name of the parameter to find * @value: the value to store into the new parameter * * Adds new parameter called @name with char * type and sets its value to * @value. The function creates its own copy of @value string, which needs to * be freed using virTypedParamsFree or virTypedParamsClear. If @params array * points to NULL or to a space that is not large enough to accommodate the * new parameter (@maxparams < @nparams + 1), the function allocates more * space for it and updates @maxparams. On success, @nparams is incremented * by one. The function fails with VIR_ERR_INVALID_ARG error if the parameter * already exists in @params. * * Returns 0 on success, -1 on error. */ int virTypedParamsAddString(virTypedParameterPtr *params, int *nparams, int *maxparams, const char *name, const char *value) { char *str = NULL; size_t max = *maxparams; size_t n = *nparams; virResetLastError(); if (VIR_RESIZE_N(*params, max, n, 1) < 0) goto error; *maxparams = max; if (VIR_STRDUP(str, value) < 0) goto error; if (virTypedParameterAssign(*params + n, name, VIR_TYPED_PARAM_STRING, str) < 0) { VIR_FREE(str); goto error; } *nparams += 1; return 0; error: virDispatchError(NULL); return -1; } /** * virTypedParamsAddStringList: * @params: array of typed parameters * @nparams: number of parameters in the @params array * @maxparams: maximum number of parameters that can be stored in @params * array without allocating more memory * @name: name of the parameter to store values to * @values: the values to store into the new parameters * * Packs NULL-terminated list of strings @values into @params under the * key @name. * * Returns 0 on success, -1 on error. */ int virTypedParamsAddStringList(virTypedParameterPtr *params, int *nparams, int *maxparams, const char *name, const char **values) { size_t i; int rv = -1; if (!values) return 0; for (i = 0; values[i]; i++) { if ((rv = virTypedParamsAddString(params, nparams, maxparams, name, values[i])) < 0) break; } return rv; } /** * virTypedParamsAddFromString: * @params: pointer to the array of typed parameters * @nparams: number of parameters in the @params array * @maxparams: maximum number of parameters that can be stored in @params * array without allocating more memory * @name: name of the parameter to find * @type: type of the parameter * @value: the value to store into the new parameter encoded as a string * * Adds new parameter called @name with the requested @type and parses its * value from the @value string. If the requested type is string, the function * creates its own copy of the @value string, which needs to be freed using * virTypedParamsFree or virTypedParamsClear. If @params array points to NULL * or to a space that is not large enough to accommodate the new parameter * (@maxparams < @nparams + 1), the function allocates more space for it and * updates @maxparams. On success, @nparams is incremented by one. The * function fails with VIR_ERR_INVALID_ARG error if the parameter already * exists in @params. * * Returns 0 on success, -1 on error. */ int virTypedParamsAddFromString(virTypedParameterPtr *params, int *nparams, int *maxparams, const char *name, int type, const char *value) { size_t max = *maxparams; size_t n = *nparams; virResetLastError(); if (VIR_RESIZE_N(*params, max, n, 1) < 0) goto error; *maxparams = max; if (virTypedParameterAssignFromStr(*params + n, name, type, value) < 0) goto error; *nparams += 1; return 0; error: virDispatchError(NULL); return -1; } /** * virTypedParamsClear: * @params: the array of typed parameters * @nparams: number of parameters in the @params array * * Frees all memory used by string parameters. The memory occupied by @params * is not freed; use virTypedParamsFree if you want it to be freed too. * * Returns nothing. */ void virTypedParamsClear(virTypedParameterPtr params, int nparams) { size_t i; if (!params) return; for (i = 0; i < nparams; i++) { if (params[i].type == VIR_TYPED_PARAM_STRING) VIR_FREE(params[i].value.s); } } /** * virTypedParamsFree: * @params: the array of typed parameters * @nparams: number of parameters in the @params array * * Frees all memory used by string parameters and the memory occupied by * @params. * * Returns nothing. */ void virTypedParamsFree(virTypedParameterPtr params, int nparams) { virTypedParamsClear(params, nparams); VIR_FREE(params); } /** * virTypedParamsRemoteFree: * @remote_params_val: array of typed parameters as specified by * (remote|admin)_protocol.h * @remote_params_len: number of parameters in @remote_params_val * * Frees memory used by string representations of parameter identificators, * memory used by string values of parameters and the memory occupied by * @remote_params_val itself. * * Returns nothing. */ void virTypedParamsRemoteFree(virTypedParameterRemotePtr remote_params_val, unsigned int remote_params_len) { size_t i; if (!remote_params_val) return; for (i = 0; i < remote_params_len; i++) { VIR_FREE(remote_params_val[i].field); if (remote_params_val[i].value.type == VIR_TYPED_PARAM_STRING) VIR_FREE(remote_params_val[i].value.remote_typed_param_value.s); } VIR_FREE(remote_params_val); } /** * virTypedParamsDeserialize: * @remote_params: protocol data to be deserialized (obtained from remote side) * @remote_params_len: number of parameters returned in @remote_params * @limit: user specified maximum limit to @remote_params_len * @params: pointer which will hold the deserialized @remote_params data * @nparams: number of entries in @params * * This function will attempt to deserialize protocol-encoded data obtained * from remote side. Two modes of operation are supported, depending on the * caller's design: * 1) Older APIs do not rely on deserializer allocating memory for @params, * thus calling the deserializer twice, once to find out the actual number of * parameters for @params to hold, followed by an allocation of @params and * a second call to the deserializer to actually retrieve the data. * 2) Newer APIs rely completely on the deserializer to allocate the right * amount of memory for @params to hold all the data obtained in * @remote_params. * * If used with model 1, two checks are performed, first one comparing the user * specified limit to the actual size of remote data and the second one * verifying the user allocated amount of memory is indeed capable of holding * remote data @remote_params. * With model 2, only the first check against @limit is performed. * * Returns 0 on success or -1 in case of an error. */ int virTypedParamsDeserialize(virTypedParameterRemotePtr remote_params, unsigned int remote_params_len, int limit, virTypedParameterPtr *params, int *nparams) { size_t i = 0; int rv = -1; bool userAllocated = *params != NULL; if (limit && remote_params_len > limit) { virReportError(VIR_ERR_RPC, _("too many parameters '%u' for limit '%d'"), remote_params_len, limit); goto cleanup; } if (userAllocated) { /* Check the length of the returned list carefully. */ if (remote_params_len > *nparams) { virReportError(VIR_ERR_RPC, _("too many parameters '%u' for nparams '%d'"), remote_params_len, *nparams); goto cleanup; } } else { if (VIR_ALLOC_N(*params, remote_params_len) < 0) goto cleanup; } *nparams = remote_params_len; /* Deserialize the result. */ for (i = 0; i < remote_params_len; ++i) { virTypedParameterPtr param = *params + i; virTypedParameterRemotePtr remote_param = remote_params + i; if (virStrcpyStatic(param->field, remote_param->field) < 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("parameter %s too big for destination"), remote_param->field); goto cleanup; } param->type = remote_param->value.type; switch (param->type) { case VIR_TYPED_PARAM_INT: param->value.i = remote_param->value.remote_typed_param_value.i; break; case VIR_TYPED_PARAM_UINT: param->value.ui = remote_param->value.remote_typed_param_value.ui; break; case VIR_TYPED_PARAM_LLONG: param->value.l = remote_param->value.remote_typed_param_value.l; break; case VIR_TYPED_PARAM_ULLONG: param->value.ul = remote_param->value.remote_typed_param_value.ul; break; case VIR_TYPED_PARAM_DOUBLE: param->value.d = remote_param->value.remote_typed_param_value.d; break; case VIR_TYPED_PARAM_BOOLEAN: param->value.b = remote_param->value.remote_typed_param_value.b; break; case VIR_TYPED_PARAM_STRING: if (VIR_STRDUP(param->value.s, remote_param->value.remote_typed_param_value.s) < 0) goto cleanup; break; default: virReportError(VIR_ERR_RPC, _("unknown parameter type: %d"), param->type); goto cleanup; } } rv = 0; cleanup: if (rv < 0) { if (userAllocated) { virTypedParamsClear(*params, i); } else { virTypedParamsFree(*params, i); *params = NULL; *nparams = 0; } } return rv; } /** * virTypedParamsSerialize: * @params: array of parameters to be serialized and later sent to remote side * @nparams: number of elements in @params * @limit: user specified maximum limit to @remote_params_len * @remote_params_val: protocol independent remote representation of @params * @remote_params_len: the final number of elements in @remote_params_val * @flags: bitwise-OR of virTypedParameterFlags * * This method serializes typed parameters provided by @params into * @remote_params_val which is the representation actually being sent. * It also checks, if the @limit imposed by RPC on the maximum number of * parameters is not exceeded. * * Server side using this method also filters out any string parameters that * must not be returned to older clients and handles possibly sparse arrays * returned by some APIs. * * Returns 0 on success, -1 on error. */ int virTypedParamsSerialize(virTypedParameterPtr params, int nparams, int limit, virTypedParameterRemotePtr *remote_params_val, unsigned int *remote_params_len, unsigned int flags) { size_t i; size_t j; int rv = -1; virTypedParameterRemotePtr params_val = NULL; int params_len = nparams; if (nparams > limit) { virReportError(VIR_ERR_RPC, _("too many parameters '%d' for limit '%d'"), nparams, limit); goto cleanup; } if (VIR_ALLOC_N(params_val, nparams) < 0) goto cleanup; for (i = 0, j = 0; i < nparams; ++i) { virTypedParameterPtr param = params + i; virTypedParameterRemotePtr val = params_val + j; /* NOTE: Following snippet is relevant to server only, because * virDomainGetCPUStats can return a sparse array; also, we can't pass * back strings to older clients. */ if (!param->type || (!(flags & VIR_TYPED_PARAM_STRING_OKAY) && param->type == VIR_TYPED_PARAM_STRING)) { --params_len; continue; } /* This will be either freed by virNetServerDispatchCall or call(), * depending on the calling side, i.e. server or client */ if (VIR_STRDUP(val->field, param->field) < 0) goto cleanup; val->value.type = param->type; switch (param->type) { case VIR_TYPED_PARAM_INT: val->value.remote_typed_param_value.i = param->value.i; break; case VIR_TYPED_PARAM_UINT: val->value.remote_typed_param_value.ui = param->value.ui; break; case VIR_TYPED_PARAM_LLONG: val->value.remote_typed_param_value.l = param->value.l; break; case VIR_TYPED_PARAM_ULLONG: val->value.remote_typed_param_value.ul = param->value.ul; break; case VIR_TYPED_PARAM_DOUBLE: val->value.remote_typed_param_value.d = param->value.d; break; case VIR_TYPED_PARAM_BOOLEAN: val->value.remote_typed_param_value.b = param->value.b; break; case VIR_TYPED_PARAM_STRING: if (VIR_STRDUP(val->value.remote_typed_param_value.s, param->value.s) < 0) goto cleanup; break; default: virReportError(VIR_ERR_RPC, _("unknown parameter type: %d"), param->type); goto cleanup; } j++; } *remote_params_val = params_val; *remote_params_len = params_len; params_val = NULL; rv = 0; cleanup: virTypedParamsRemoteFree(params_val, nparams); return rv; }