string-input-visitor.c 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. /*
  2. * String parsing visitor
  3. *
  4. * Copyright Red Hat, Inc. 2012
  5. *
  6. * Author: Paolo Bonzini <pbonzini@redhat.com>
  7. *
  8. * This work is licensed under the terms of the GNU LGPL, version 2.1 or later.
  9. * See the COPYING.LIB file in the top-level directory.
  10. *
  11. */
  12. #include "qemu-common.h"
  13. #include "qapi/string-input-visitor.h"
  14. #include "qapi/visitor-impl.h"
  15. #include "qapi/qmp/qerror.h"
  16. #include "qemu/option.h"
  17. struct StringInputVisitor
  18. {
  19. Visitor visitor;
  20. const char *string;
  21. };
  22. static void parse_type_int(Visitor *v, int64_t *obj, const char *name,
  23. Error **errp)
  24. {
  25. StringInputVisitor *siv = DO_UPCAST(StringInputVisitor, visitor, v);
  26. char *endp = (char *) siv->string;
  27. long long val;
  28. errno = 0;
  29. if (siv->string) {
  30. val = strtoll(siv->string, &endp, 0);
  31. }
  32. if (!siv->string || errno || endp == siv->string || *endp) {
  33. error_set(errp, QERR_INVALID_PARAMETER_TYPE, name ? name : "null",
  34. "integer");
  35. return;
  36. }
  37. *obj = val;
  38. }
  39. static void parse_type_size(Visitor *v, uint64_t *obj, const char *name,
  40. Error **errp)
  41. {
  42. StringInputVisitor *siv = DO_UPCAST(StringInputVisitor, visitor, v);
  43. Error *err = NULL;
  44. uint64_t val;
  45. if (siv->string) {
  46. parse_option_size(name, siv->string, &val, &err);
  47. } else {
  48. error_set(errp, QERR_INVALID_PARAMETER_TYPE, name ? name : "null",
  49. "size");
  50. return;
  51. }
  52. if (err) {
  53. error_propagate(errp, err);
  54. return;
  55. }
  56. *obj = val;
  57. }
  58. static void parse_type_bool(Visitor *v, bool *obj, const char *name,
  59. Error **errp)
  60. {
  61. StringInputVisitor *siv = DO_UPCAST(StringInputVisitor, visitor, v);
  62. if (siv->string) {
  63. if (!strcasecmp(siv->string, "on") ||
  64. !strcasecmp(siv->string, "yes") ||
  65. !strcasecmp(siv->string, "true")) {
  66. *obj = true;
  67. return;
  68. }
  69. if (!strcasecmp(siv->string, "off") ||
  70. !strcasecmp(siv->string, "no") ||
  71. !strcasecmp(siv->string, "false")) {
  72. *obj = false;
  73. return;
  74. }
  75. }
  76. error_set(errp, QERR_INVALID_PARAMETER_TYPE, name ? name : "null",
  77. "boolean");
  78. }
  79. static void parse_type_str(Visitor *v, char **obj, const char *name,
  80. Error **errp)
  81. {
  82. StringInputVisitor *siv = DO_UPCAST(StringInputVisitor, visitor, v);
  83. if (siv->string) {
  84. *obj = g_strdup(siv->string);
  85. } else {
  86. error_set(errp, QERR_INVALID_PARAMETER_TYPE, name ? name : "null",
  87. "string");
  88. }
  89. }
  90. static void parse_type_number(Visitor *v, double *obj, const char *name,
  91. Error **errp)
  92. {
  93. StringInputVisitor *siv = DO_UPCAST(StringInputVisitor, visitor, v);
  94. char *endp = (char *) siv->string;
  95. double val;
  96. errno = 0;
  97. if (siv->string) {
  98. val = strtod(siv->string, &endp);
  99. }
  100. if (!siv->string || errno || endp == siv->string || *endp) {
  101. error_set(errp, QERR_INVALID_PARAMETER_TYPE, name ? name : "null",
  102. "number");
  103. return;
  104. }
  105. *obj = val;
  106. }
  107. static void parse_start_optional(Visitor *v, bool *present,
  108. const char *name, Error **errp)
  109. {
  110. StringInputVisitor *siv = DO_UPCAST(StringInputVisitor, visitor, v);
  111. if (!siv->string) {
  112. *present = false;
  113. return;
  114. }
  115. *present = true;
  116. }
  117. Visitor *string_input_get_visitor(StringInputVisitor *v)
  118. {
  119. return &v->visitor;
  120. }
  121. void string_input_visitor_cleanup(StringInputVisitor *v)
  122. {
  123. g_free(v);
  124. }
  125. StringInputVisitor *string_input_visitor_new(const char *str)
  126. {
  127. StringInputVisitor *v;
  128. v = g_malloc0(sizeof(*v));
  129. v->visitor.type_enum = input_type_enum;
  130. v->visitor.type_int = parse_type_int;
  131. v->visitor.type_size = parse_type_size;
  132. v->visitor.type_bool = parse_type_bool;
  133. v->visitor.type_str = parse_type_str;
  134. v->visitor.type_number = parse_type_number;
  135. v->visitor.start_optional = parse_start_optional;
  136. v->string = str;
  137. return v;
  138. }