int nft_mxml_num_parse(mxml_node_t *tree, const char *node_name, uint32_t mxml_flags, int base, void *number, enum nft_type type);
const char *nft_mxml_str_parse(mxml_node_t *tree, const char *node_name, uint32_t mxml_flags);
int nft_mxml_family_parse(mxml_node_t *tree, const char *node_name, uint32_t mxml_flags);
-struct nft_set_elem *nft_mxml_set_elem_parse(mxml_node_t *node);
+
+struct nft_set_elem;
+int nft_mxml_set_elem_parse(mxml_node_t *node, struct nft_set_elem *e);
#endif
#ifdef JSON_PARSING
return family;
}
-
-struct nft_set_elem *nft_mxml_set_elem_parse(mxml_node_t *node)
-{
- mxml_node_t *save;
- char *set_elem_str;
- struct nft_set_elem *elem;
-
- if (node == NULL)
- goto einval;
-
- if (strcmp(node->value.opaque, "set_elem") != 0)
- goto einval;
-
- elem = nft_set_elem_alloc();
- if (elem == NULL)
- goto enomem;
-
- /* This is a hack for mxml to print just the current node */
- save = node->next;
- node->next = NULL;
-
- set_elem_str = mxmlSaveAllocString(node, MXML_NO_CALLBACK);
- node->next = save;
-
- if (set_elem_str == NULL) {
- xfree(elem);
- goto enomem;
- }
-
- if (nft_set_elem_parse(elem, NFT_SET_PARSE_XML,
- set_elem_str) != 0) {
- xfree(set_elem_str);
- xfree(elem);
- return NULL;
- }
-
- xfree(set_elem_str);
-
- return elem;
-einval:
- errno = EINVAL;
- return NULL;
-enomem:
- errno = ENOMEM;
- return NULL;
-}
#endif
s->flags |= (1 << NFT_SET_ATTR_FLAGS);
-
if (nft_mxml_num_parse(tree, "key_type", MXML_DESCEND_FIRST,
BASE_DEC, &s->key_type, NFT_TYPE_U32) != 0)
goto err;
s->flags |= (1 << NFT_SET_ATTR_DATA_LEN);
- /* Iterate over each <set_elem> */
for (node = mxmlFindElement(tree, tree, "set_elem", NULL,
NULL, MXML_DESCEND);
node != NULL;
node = mxmlFindElement(node, tree, "set_elem", NULL,
NULL, MXML_DESCEND)) {
- elem = nft_mxml_set_elem_parse(node);
+ elem = nft_set_elem_alloc();
if (elem == NULL)
goto err;
+ if (nft_mxml_set_elem_parse(node, elem) < 0)
+ goto err;
+
list_add_tail(&elem->head, &s->element_list);
}
}
EXPORT_SYMBOL(nft_set_elems_nlmsg_parse);
-static int nft_set_elem_xml_parse(struct nft_set_elem *e, char *xml)
-{
#ifdef XML_PARSING
- mxml_node_t *tree;
+int nft_mxml_set_elem_parse(mxml_node_t *tree, struct nft_set_elem *e)
+{
mxml_node_t *node;
int set_elem_data;
- tree = mxmlLoadString(NULL, xml, MXML_OPAQUE_CALLBACK);
- if (tree == NULL) {
- errno = EINVAL;
- return -1;
- }
-
- if (strcmp(tree->value.opaque, "set_elem") != 0) {
- errno = EINVAL;
- goto err;
- }
-
if (nft_mxml_num_parse(tree, "flags", MXML_DESCEND_FIRST,
BASE_DEC, &e->set_elem_flags,
NFT_TYPE_U32) != 0)
- goto err;
+ return -1;
e->flags |= (1 << NFT_SET_ELEM_ATTR_FLAGS);
- if (nft_mxml_data_reg_parse(tree, "key",
- &e->key) != DATA_VALUE)
- goto err;
+ if (nft_mxml_data_reg_parse(tree, "key", &e->key) != DATA_VALUE)
+ return -1;
e->flags |= (1 << NFT_SET_ELEM_ATTR_KEY);
e->flags |= (1 << NFT_SET_ELEM_ATTR_CHAIN);
break;
default:
- goto err;
+ return -1;
}
}
-
- mxmlDelete(tree);
return 0;
+}
+#endif
+static int nft_set_elem_xml_parse(struct nft_set_elem *e, char *xml)
+{
+#ifdef XML_PARSING
+ mxml_node_t *tree;
+ int ret;
+
+ tree = mxmlLoadString(NULL, xml, MXML_OPAQUE_CALLBACK);
+ if (tree == NULL) {
+ errno = EINVAL;
+ return -1;
+ }
+
+ if (strcmp(tree->value.opaque, "set_elem") != 0) {
+ errno = EINVAL;
+ goto err;
+ }
+
+ ret = nft_mxml_set_elem_parse(tree, e);
+ mxmlDelete(tree);
+ return ret;
err:
mxmlDelete(tree);
return -1;