OptsVisitor: introduce basic list modes
We're going to need more state while processing a list of repeated options. This change eliminates "repeated_opts_first" and adds a new state variable: list_mode repeated_opts repeated_opts_first -------------- ------------- ------------------- LM_NONE NULL false LM_STARTED non-NULL true LM_IN_PROGRESS non-NULL false Additionally, it is documented that lookup_scalar() and processed(), both called by opts_type_XXX(), are invalid in LM_STARTED -- generated qapi code calls opts_next_list() to allocate the very first link before trying to parse a scalar into it. List mode restrictions are expressed in positive / inclusive form. Signed-off-by: Laszlo Ersek <lersek@redhat.com> Tested-by: Wanlong Gao <gaowanlong@cn.fujitsu.com> Signed-off-by: Luiz Capitulino <lcapitulino@redhat.com>
This commit is contained in:
parent
4a44d85e28
commit
d957043412
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* Options Visitor
|
* Options Visitor
|
||||||
*
|
*
|
||||||
* Copyright Red Hat, Inc. 2012
|
* Copyright Red Hat, Inc. 2012, 2013
|
||||||
*
|
*
|
||||||
* Author: Laszlo Ersek <lersek@redhat.com>
|
* Author: Laszlo Ersek <lersek@redhat.com>
|
||||||
*
|
*
|
||||||
@ -18,6 +18,15 @@
|
|||||||
#include "qapi/visitor-impl.h"
|
#include "qapi/visitor-impl.h"
|
||||||
|
|
||||||
|
|
||||||
|
enum ListMode
|
||||||
|
{
|
||||||
|
LM_NONE, /* not traversing a list of repeated options */
|
||||||
|
LM_STARTED, /* opts_start_list() succeeded */
|
||||||
|
LM_IN_PROGRESS /* opts_next_list() has been called */
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef enum ListMode ListMode;
|
||||||
|
|
||||||
struct OptsVisitor
|
struct OptsVisitor
|
||||||
{
|
{
|
||||||
Visitor visitor;
|
Visitor visitor;
|
||||||
@ -35,8 +44,8 @@ struct OptsVisitor
|
|||||||
/* The list currently being traversed with opts_start_list() /
|
/* The list currently being traversed with opts_start_list() /
|
||||||
* opts_next_list(). The list must have a struct element type in the
|
* opts_next_list(). The list must have a struct element type in the
|
||||||
* schema, with a single mandatory scalar member. */
|
* schema, with a single mandatory scalar member. */
|
||||||
|
ListMode list_mode;
|
||||||
GQueue *repeated_opts;
|
GQueue *repeated_opts;
|
||||||
bool repeated_opts_first;
|
|
||||||
|
|
||||||
/* If "opts_root->id" is set, reinstantiate it as a fake QemuOpt for
|
/* If "opts_root->id" is set, reinstantiate it as a fake QemuOpt for
|
||||||
* uniformity. Only its "name" and "str" fields are set. "fake_id_opt" does
|
* uniformity. Only its "name" and "str" fields are set. "fake_id_opt" does
|
||||||
@ -156,9 +165,11 @@ opts_start_list(Visitor *v, const char *name, Error **errp)
|
|||||||
OptsVisitor *ov = DO_UPCAST(OptsVisitor, visitor, v);
|
OptsVisitor *ov = DO_UPCAST(OptsVisitor, visitor, v);
|
||||||
|
|
||||||
/* we can't traverse a list in a list */
|
/* we can't traverse a list in a list */
|
||||||
assert(ov->repeated_opts == NULL);
|
assert(ov->list_mode == LM_NONE);
|
||||||
ov->repeated_opts = lookup_distinct(ov, name, errp);
|
ov->repeated_opts = lookup_distinct(ov, name, errp);
|
||||||
ov->repeated_opts_first = (ov->repeated_opts != NULL);
|
if (ov->repeated_opts != NULL) {
|
||||||
|
ov->list_mode = LM_STARTED;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -168,10 +179,13 @@ opts_next_list(Visitor *v, GenericList **list, Error **errp)
|
|||||||
OptsVisitor *ov = DO_UPCAST(OptsVisitor, visitor, v);
|
OptsVisitor *ov = DO_UPCAST(OptsVisitor, visitor, v);
|
||||||
GenericList **link;
|
GenericList **link;
|
||||||
|
|
||||||
if (ov->repeated_opts_first) {
|
switch (ov->list_mode) {
|
||||||
ov->repeated_opts_first = false;
|
case LM_STARTED:
|
||||||
|
ov->list_mode = LM_IN_PROGRESS;
|
||||||
link = list;
|
link = list;
|
||||||
} else {
|
break;
|
||||||
|
|
||||||
|
case LM_IN_PROGRESS: {
|
||||||
const QemuOpt *opt;
|
const QemuOpt *opt;
|
||||||
|
|
||||||
opt = g_queue_pop_head(ov->repeated_opts);
|
opt = g_queue_pop_head(ov->repeated_opts);
|
||||||
@ -180,6 +194,11 @@ opts_next_list(Visitor *v, GenericList **list, Error **errp)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
link = &(*list)->next;
|
link = &(*list)->next;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
abort();
|
||||||
}
|
}
|
||||||
|
|
||||||
*link = g_malloc0(sizeof **link);
|
*link = g_malloc0(sizeof **link);
|
||||||
@ -192,14 +211,16 @@ opts_end_list(Visitor *v, Error **errp)
|
|||||||
{
|
{
|
||||||
OptsVisitor *ov = DO_UPCAST(OptsVisitor, visitor, v);
|
OptsVisitor *ov = DO_UPCAST(OptsVisitor, visitor, v);
|
||||||
|
|
||||||
|
assert(ov->list_mode == LM_STARTED || ov->list_mode == LM_IN_PROGRESS);
|
||||||
ov->repeated_opts = NULL;
|
ov->repeated_opts = NULL;
|
||||||
|
ov->list_mode = LM_NONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static const QemuOpt *
|
static const QemuOpt *
|
||||||
lookup_scalar(const OptsVisitor *ov, const char *name, Error **errp)
|
lookup_scalar(const OptsVisitor *ov, const char *name, Error **errp)
|
||||||
{
|
{
|
||||||
if (ov->repeated_opts == NULL) {
|
if (ov->list_mode == LM_NONE) {
|
||||||
GQueue *list;
|
GQueue *list;
|
||||||
|
|
||||||
/* the last occurrence of any QemuOpt takes effect when queried by name
|
/* the last occurrence of any QemuOpt takes effect when queried by name
|
||||||
@ -207,6 +228,7 @@ lookup_scalar(const OptsVisitor *ov, const char *name, Error **errp)
|
|||||||
list = lookup_distinct(ov, name, errp);
|
list = lookup_distinct(ov, name, errp);
|
||||||
return list ? g_queue_peek_tail(list) : NULL;
|
return list ? g_queue_peek_tail(list) : NULL;
|
||||||
}
|
}
|
||||||
|
assert(ov->list_mode == LM_IN_PROGRESS);
|
||||||
return g_queue_peek_head(ov->repeated_opts);
|
return g_queue_peek_head(ov->repeated_opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -214,9 +236,12 @@ lookup_scalar(const OptsVisitor *ov, const char *name, Error **errp)
|
|||||||
static void
|
static void
|
||||||
processed(OptsVisitor *ov, const char *name)
|
processed(OptsVisitor *ov, const char *name)
|
||||||
{
|
{
|
||||||
if (ov->repeated_opts == NULL) {
|
if (ov->list_mode == LM_NONE) {
|
||||||
g_hash_table_remove(ov->unprocessed_opts, name);
|
g_hash_table_remove(ov->unprocessed_opts, name);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
assert(ov->list_mode == LM_IN_PROGRESS);
|
||||||
|
/* do nothing */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -365,7 +390,7 @@ opts_start_optional(Visitor *v, bool *present, const char *name,
|
|||||||
OptsVisitor *ov = DO_UPCAST(OptsVisitor, visitor, v);
|
OptsVisitor *ov = DO_UPCAST(OptsVisitor, visitor, v);
|
||||||
|
|
||||||
/* we only support a single mandatory scalar field in a list node */
|
/* we only support a single mandatory scalar field in a list node */
|
||||||
assert(ov->repeated_opts == NULL);
|
assert(ov->list_mode == LM_NONE);
|
||||||
*present = (lookup_distinct(ov, name, NULL) != NULL);
|
*present = (lookup_distinct(ov, name, NULL) != NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user