glsl: move to compiler/
Signed-off-by: Emil Velikov <emil.velikov@collabora.com> Acked-by: Matt Turner <mattst88@gmail.com> Acked-by: Jose Fonseca <jfonseca@vmware.com>
This commit is contained in:

committed by
Emil Velikov

parent
a39a8fbbaa
commit
eb63640c1d
209
src/compiler/glsl/ir_hierarchical_visitor.h
Normal file
209
src/compiler/glsl/ir_hierarchical_visitor.h
Normal file
@@ -0,0 +1,209 @@
|
||||
/* -*- c++ -*- */
|
||||
/*
|
||||
* Copyright © 2010 Intel Corporation
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a
|
||||
* copy of this software and associated documentation files (the "Software"),
|
||||
* to deal in the Software without restriction, including without limitation
|
||||
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
* and/or sell copies of the Software, and to permit persons to whom the
|
||||
* Software is furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice (including the next
|
||||
* paragraph) shall be included in all copies or substantial portions of the
|
||||
* Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
* DEALINGS IN THE SOFTWARE.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
#ifndef IR_HIERARCHICAL_VISITOR_H
|
||||
#define IR_HIERARCHICAL_VISITOR_H
|
||||
|
||||
/**
|
||||
* Enumeration values returned by visit methods to guide processing
|
||||
*/
|
||||
enum ir_visitor_status {
|
||||
visit_continue, /**< Continue visiting as normal. */
|
||||
visit_continue_with_parent, /**< Don't visit siblings, continue w/parent. */
|
||||
visit_stop /**< Stop visiting immediately. */
|
||||
};
|
||||
|
||||
|
||||
#ifdef __cplusplus
|
||||
/**
|
||||
* Base class of hierarchical visitors of IR instruction trees
|
||||
*
|
||||
* Hierarchical visitors differ from traditional visitors in a couple of
|
||||
* important ways. Rather than having a single \c visit method for each
|
||||
* subclass in the composite, there are three kinds of visit methods.
|
||||
* Leaf-node classes have a traditional \c visit method. Internal-node
|
||||
* classes have a \c visit_enter method, which is invoked just before
|
||||
* processing child nodes, and a \c visit_leave method which is invoked just
|
||||
* after processing child nodes.
|
||||
*
|
||||
* In addition, each visit method and the \c accept methods in the composite
|
||||
* have a return value which guides the navigation. Any of the visit methods
|
||||
* can choose to continue visiting the tree as normal (by returning \c
|
||||
* visit_continue), terminate visiting any further nodes immediately (by
|
||||
* returning \c visit_stop), or stop visiting sibling nodes (by returning \c
|
||||
* visit_continue_with_parent).
|
||||
*
|
||||
* These two changes combine to allow nagivation of children to be implemented
|
||||
* in the composite's \c accept method. The \c accept method for a leaf-node
|
||||
* class will simply call the \c visit method, as usual, and pass its return
|
||||
* value on. The \c accept method for internal-node classes will call the \c
|
||||
* visit_enter method, call the \c accept method of each child node, and,
|
||||
* finally, call the \c visit_leave method. If any of these return a value
|
||||
* other that \c visit_continue, the correct action must be taken.
|
||||
*
|
||||
* The final benefit is that the hierarchical visitor base class need not be
|
||||
* abstract. Default implementations of every \c visit, \c visit_enter, and
|
||||
* \c visit_leave method can be provided. By default each of these methods
|
||||
* simply returns \c visit_continue. This allows a significant reduction in
|
||||
* derived class code.
|
||||
*
|
||||
* For more information about hierarchical visitors, see:
|
||||
*
|
||||
* http://c2.com/cgi/wiki?HierarchicalVisitorPattern
|
||||
* http://c2.com/cgi/wiki?HierarchicalVisitorDiscussion
|
||||
*/
|
||||
|
||||
class ir_hierarchical_visitor {
|
||||
public:
|
||||
ir_hierarchical_visitor();
|
||||
|
||||
/**
|
||||
* \name Visit methods for leaf-node classes
|
||||
*/
|
||||
/*@{*/
|
||||
virtual ir_visitor_status visit(class ir_rvalue *);
|
||||
virtual ir_visitor_status visit(class ir_variable *);
|
||||
virtual ir_visitor_status visit(class ir_constant *);
|
||||
virtual ir_visitor_status visit(class ir_loop_jump *);
|
||||
virtual ir_visitor_status visit(class ir_barrier *);
|
||||
|
||||
/**
|
||||
* ir_dereference_variable isn't technically a leaf, but it is treated as a
|
||||
* leaf here for a couple reasons. By not automatically visiting the one
|
||||
* child ir_variable node from the ir_dereference_variable, ir_variable
|
||||
* nodes can always be handled as variable declarations. Code that used
|
||||
* non-hierarchical visitors had to set an "in a dereference" flag to
|
||||
* determine how to handle an ir_variable. By forcing the visitor to
|
||||
* handle the ir_variable within the ir_dereference_variable visitor, this
|
||||
* kludge can be avoided.
|
||||
*
|
||||
* In addition, I can envision no use for having separate enter and leave
|
||||
* methods. Anything that could be done in the enter and leave methods
|
||||
* that couldn't just be done in the visit method.
|
||||
*/
|
||||
virtual ir_visitor_status visit(class ir_dereference_variable *);
|
||||
/*@}*/
|
||||
|
||||
/**
|
||||
* \name Visit methods for internal-node classes
|
||||
*/
|
||||
/*@{*/
|
||||
virtual ir_visitor_status visit_enter(class ir_loop *);
|
||||
virtual ir_visitor_status visit_leave(class ir_loop *);
|
||||
virtual ir_visitor_status visit_enter(class ir_function_signature *);
|
||||
virtual ir_visitor_status visit_leave(class ir_function_signature *);
|
||||
virtual ir_visitor_status visit_enter(class ir_function *);
|
||||
virtual ir_visitor_status visit_leave(class ir_function *);
|
||||
virtual ir_visitor_status visit_enter(class ir_expression *);
|
||||
virtual ir_visitor_status visit_leave(class ir_expression *);
|
||||
virtual ir_visitor_status visit_enter(class ir_texture *);
|
||||
virtual ir_visitor_status visit_leave(class ir_texture *);
|
||||
virtual ir_visitor_status visit_enter(class ir_swizzle *);
|
||||
virtual ir_visitor_status visit_leave(class ir_swizzle *);
|
||||
virtual ir_visitor_status visit_enter(class ir_dereference_array *);
|
||||
virtual ir_visitor_status visit_leave(class ir_dereference_array *);
|
||||
virtual ir_visitor_status visit_enter(class ir_dereference_record *);
|
||||
virtual ir_visitor_status visit_leave(class ir_dereference_record *);
|
||||
virtual ir_visitor_status visit_enter(class ir_assignment *);
|
||||
virtual ir_visitor_status visit_leave(class ir_assignment *);
|
||||
virtual ir_visitor_status visit_enter(class ir_call *);
|
||||
virtual ir_visitor_status visit_leave(class ir_call *);
|
||||
virtual ir_visitor_status visit_enter(class ir_return *);
|
||||
virtual ir_visitor_status visit_leave(class ir_return *);
|
||||
virtual ir_visitor_status visit_enter(class ir_discard *);
|
||||
virtual ir_visitor_status visit_leave(class ir_discard *);
|
||||
virtual ir_visitor_status visit_enter(class ir_if *);
|
||||
virtual ir_visitor_status visit_leave(class ir_if *);
|
||||
virtual ir_visitor_status visit_enter(class ir_emit_vertex *);
|
||||
virtual ir_visitor_status visit_leave(class ir_emit_vertex *);
|
||||
virtual ir_visitor_status visit_enter(class ir_end_primitive *);
|
||||
virtual ir_visitor_status visit_leave(class ir_end_primitive *);
|
||||
/*@}*/
|
||||
|
||||
|
||||
/**
|
||||
* Utility function to process a linked list of instructions with a visitor
|
||||
*/
|
||||
void run(struct exec_list *instructions);
|
||||
|
||||
/* Some visitors may need to insert new variable declarations and
|
||||
* assignments for portions of a subtree, which means they need a
|
||||
* pointer to the current instruction in the stream, not just their
|
||||
* node in the tree rooted at that instruction.
|
||||
*
|
||||
* This is implemented by visit_list_elements -- if the visitor is
|
||||
* not called by it, nothing good will happen.
|
||||
*/
|
||||
class ir_instruction *base_ir;
|
||||
|
||||
/**
|
||||
* Callback function that is invoked on entry to each node visited.
|
||||
*
|
||||
* \warning
|
||||
* Visitor classes derived from \c ir_hierarchical_visitor \b may \b not
|
||||
* invoke this function. This can be used, for example, to cause the
|
||||
* callback to be invoked on every node type except one.
|
||||
*/
|
||||
void (*callback_enter)(class ir_instruction *ir, void *data);
|
||||
|
||||
/**
|
||||
* Callback function that is invoked on exit of each node visited.
|
||||
*
|
||||
* \warning
|
||||
* Visitor classes derived from \c ir_hierarchical_visitor \b may \b not
|
||||
* invoke this function. This can be used, for example, to cause the
|
||||
* callback to be invoked on every node type except one.
|
||||
*/
|
||||
void (*callback_leave)(class ir_instruction *ir, void *data);
|
||||
|
||||
/**
|
||||
* Extra data parameter passed to the per-node callback_enter function
|
||||
*/
|
||||
void *data_enter;
|
||||
|
||||
/**
|
||||
* Extra data parameter passed to the per-node callback_leave function
|
||||
*/
|
||||
void *data_leave;
|
||||
|
||||
/**
|
||||
* Currently in the LHS of an assignment?
|
||||
*
|
||||
* This is set and cleared by the \c ir_assignment::accept method.
|
||||
*/
|
||||
bool in_assignee;
|
||||
};
|
||||
|
||||
void visit_tree(ir_instruction *ir,
|
||||
void (*callback_enter)(class ir_instruction *ir, void *data),
|
||||
void *data_enter,
|
||||
void (*callback_leave)(class ir_instruction *ir, void *data) = NULL,
|
||||
void *data_leave = NULL);
|
||||
|
||||
ir_visitor_status visit_list_elements(ir_hierarchical_visitor *v, exec_list *l,
|
||||
bool statement_list = true);
|
||||
#endif /* __cplusplus */
|
||||
|
||||
#endif /* IR_HIERARCHICAL_VISITOR_H */
|
Reference in New Issue
Block a user