mirror of
https://github.com/rkd77/elinks.git
synced 2025-01-03 14:57:44 -05:00
Make walk_dom_nodes() use a stack context; reduces the DOM stack state size
This commit is contained in:
parent
a4831fef2d
commit
9e7dfb1ddf
@ -289,18 +289,71 @@ search_dom_stack(struct dom_stack *stack, enum dom_node_type type,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
struct dom_stack_walk_state {
|
||||||
|
/* Used for recording which node list are currently being 'decended'
|
||||||
|
* into. E.g. whether we are iterating all child elements or attributes
|
||||||
|
* of an element. */
|
||||||
|
struct dom_node_list *list;
|
||||||
|
/* The index (in the list above) which are currently being handled. */
|
||||||
|
size_t index;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct dom_stack_context_info dom_stack_walk_context_info = {
|
||||||
|
/* Object size: */ sizeof(struct dom_stack_walk_state),
|
||||||
|
/* Push: */
|
||||||
|
{
|
||||||
|
/* */ NULL,
|
||||||
|
/* DOM_NODE_ELEMENT */ NULL,
|
||||||
|
/* DOM_NODE_ATTRIBUTE */ NULL,
|
||||||
|
/* DOM_NODE_TEXT */ NULL,
|
||||||
|
/* DOM_NODE_CDATA_SECTION */ NULL,
|
||||||
|
/* DOM_NODE_ENTITY_REFERENCE */ NULL,
|
||||||
|
/* DOM_NODE_ENTITY */ NULL,
|
||||||
|
/* DOM_NODE_PROC_INSTRUCTION */ NULL,
|
||||||
|
/* DOM_NODE_COMMENT */ NULL,
|
||||||
|
/* DOM_NODE_DOCUMENT */ NULL,
|
||||||
|
/* DOM_NODE_DOCUMENT_TYPE */ NULL,
|
||||||
|
/* DOM_NODE_DOCUMENT_FRAGMENT */ NULL,
|
||||||
|
/* DOM_NODE_NOTATION */ NULL,
|
||||||
|
},
|
||||||
|
/* Pop: */
|
||||||
|
{
|
||||||
|
/* */ NULL,
|
||||||
|
/* DOM_NODE_ELEMENT */ NULL,
|
||||||
|
/* DOM_NODE_ATTRIBUTE */ NULL,
|
||||||
|
/* DOM_NODE_TEXT */ NULL,
|
||||||
|
/* DOM_NODE_CDATA_SECTION */ NULL,
|
||||||
|
/* DOM_NODE_ENTITY_REFERENCE */ NULL,
|
||||||
|
/* DOM_NODE_ENTITY */ NULL,
|
||||||
|
/* DOM_NODE_PROC_INSTRUCTION */ NULL,
|
||||||
|
/* DOM_NODE_COMMENT */ NULL,
|
||||||
|
/* DOM_NODE_DOCUMENT */ NULL,
|
||||||
|
/* DOM_NODE_DOCUMENT_TYPE */ NULL,
|
||||||
|
/* DOM_NODE_DOCUMENT_FRAGMENT */ NULL,
|
||||||
|
/* DOM_NODE_NOTATION */ NULL,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/* FIXME: Instead of walking all nodes in the tree only visit those which are
|
/* FIXME: Instead of walking all nodes in the tree only visit those which are
|
||||||
* of actual interest to the contexts on the stack. */
|
* of actual interest to the contexts on the stack. */
|
||||||
void
|
void
|
||||||
walk_dom_nodes(struct dom_stack *stack, struct dom_node *root)
|
walk_dom_nodes(struct dom_stack *stack, struct dom_node *root)
|
||||||
{
|
{
|
||||||
|
struct dom_stack_context *context;
|
||||||
|
|
||||||
assert(root && stack);
|
assert(root && stack);
|
||||||
|
|
||||||
|
context = add_dom_stack_context(stack, NULL, &dom_stack_walk_context_info);
|
||||||
|
if (!context)
|
||||||
|
return;
|
||||||
|
|
||||||
push_dom_node(stack, root);
|
push_dom_node(stack, root);
|
||||||
|
|
||||||
while (!dom_stack_is_empty(stack)) {
|
while (!dom_stack_is_empty(stack)) {
|
||||||
struct dom_stack_state *state = get_dom_stack_top(stack);
|
struct dom_stack_state *state = get_dom_stack_top(stack);
|
||||||
struct dom_node_list *list = state->list;
|
struct dom_stack_walk_state *wstate = get_dom_stack_state_data(context, state);
|
||||||
|
struct dom_node_list *list = wstate->list;
|
||||||
struct dom_node *node = state->node;
|
struct dom_node *node = state->node;
|
||||||
|
|
||||||
switch (node->type) {
|
switch (node->type) {
|
||||||
@ -312,7 +365,7 @@ walk_dom_nodes(struct dom_stack *stack, struct dom_node *root)
|
|||||||
if (!list) list = node->data.element.map;
|
if (!list) list = node->data.element.map;
|
||||||
|
|
||||||
if (list == node->data.element.children) break;
|
if (list == node->data.element.children) break;
|
||||||
if (is_dom_node_list_member(list, state->index)
|
if (is_dom_node_list_member(list, wstate->index)
|
||||||
&& list == node->data.element.map)
|
&& list == node->data.element.map)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -327,7 +380,7 @@ walk_dom_nodes(struct dom_stack *stack, struct dom_node *root)
|
|||||||
if (!list) list = node->data.document_type.entities;
|
if (!list) list = node->data.document_type.entities;
|
||||||
|
|
||||||
if (list == node->data.document_type.notations) break;
|
if (list == node->data.document_type.notations) break;
|
||||||
if (is_dom_node_list_member(list, state->index)
|
if (is_dom_node_list_member(list, wstate->index)
|
||||||
&& list == node->data.document_type.entities)
|
&& list == node->data.document_type.entities)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -347,14 +400,14 @@ walk_dom_nodes(struct dom_stack *stack, struct dom_node *root)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Reset list state if it is a new list */
|
/* Reset list state if it is a new list */
|
||||||
if (list != state->list) {
|
if (list != wstate->list) {
|
||||||
state->list = list;
|
wstate->list = list;
|
||||||
state->index = 0;
|
wstate->index = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If we have next child node */
|
/* If we have next child node */
|
||||||
if (is_dom_node_list_member(list, state->index)) {
|
if (is_dom_node_list_member(list, wstate->index)) {
|
||||||
struct dom_node *child = list->entries[state->index++];
|
struct dom_node *child = list->entries[wstate->index++];
|
||||||
|
|
||||||
if (push_dom_node(stack, child))
|
if (push_dom_node(stack, child))
|
||||||
continue;
|
continue;
|
||||||
@ -362,6 +415,8 @@ walk_dom_nodes(struct dom_stack *stack, struct dom_node *root)
|
|||||||
|
|
||||||
pop_dom_node(stack);
|
pop_dom_node(stack);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
done_dom_stack_context(stack, context);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -14,13 +14,6 @@ typedef void (*dom_stack_callback_T)(struct dom_stack *, struct dom_node *, void
|
|||||||
struct dom_stack_state {
|
struct dom_stack_state {
|
||||||
struct dom_node *node;
|
struct dom_node *node;
|
||||||
|
|
||||||
/* Used for recording which node list are currently being 'decended'
|
|
||||||
* into. E.g. whether we are iterating all child elements or attributes
|
|
||||||
* of an element. */
|
|
||||||
struct dom_node_list *list;
|
|
||||||
/* The index (in the list above) which are currently being handled. */
|
|
||||||
size_t index;
|
|
||||||
|
|
||||||
/* The depth of the state in the stack. This is amongst other things
|
/* The depth of the state in the stack. This is amongst other things
|
||||||
* used to get the state object data. */
|
* used to get the state object data. */
|
||||||
unsigned int depth;
|
unsigned int depth;
|
||||||
|
Loading…
Reference in New Issue
Block a user