#include <msp/core/raii.h>
#include <msp/strings/format.h>
#include "optimize.h"
+#include "reflect.h"
using namespace std;
namespace GL {
namespace SL {
+ConstantSpecializer::ConstantSpecializer():
+ values(0)
+{ }
+
+void ConstantSpecializer::apply(Stage &stage, const map<string, int> &v)
+{
+ values = &v;
+ stage.content.visit(*this);
+}
+
+void ConstantSpecializer::visit(VariableDeclaration &var)
+{
+ bool specializable = false;
+ if(var.layout)
+ {
+ vector<Layout::Qualifier> &qualifiers = var.layout->qualifiers;
+ for(vector<Layout::Qualifier>::iterator i=qualifiers.begin(); (!specializable && i!=qualifiers.end()); ++i)
+ if(i->name=="constant_id")
+ {
+ specializable = true;
+ qualifiers.erase(i);
+ }
+
+ if(qualifiers.empty())
+ var.layout = 0;
+ }
+
+ if(specializable)
+ {
+ map<string, int>::const_iterator i = values->find(var.name);
+ if(i!=values->end())
+ {
+ RefPtr<Literal> literal = new Literal;
+ if(var.type=="bool")
+ {
+ literal->token = (i->second ? "true" : "false");
+ literal->value = static_cast<bool>(i->second);
+ }
+ else if(var.type=="int")
+ {
+ literal->token = lexical_cast<string>(i->second);
+ literal->value = i->second;
+ }
+ var.init_expression = literal;
+ }
+ }
+}
+
+
InlineableFunctionLocator::InlineableFunctionLocator():
current_function(0),
return_count(0)
InlineContentInjector::InlineContentInjector():
source_func(0),
- pass(DEPENDS)
+ pass(REFERENCED)
{ }
const string &InlineContentInjector::apply(Stage &stage, FunctionDeclaration &target_func, Block &tgt_blk, const NodeList<Statement>::iterator &ins_pt, FunctionCall &call)
{
source_func = call.declaration->definition;
- // Collect all declarations the inlined function depends on.
- pass = DEPENDS;
- source_func->visit(*this);
-
/* Populate referenced_names from the target function so we can rename
variables from the inlined function that would conflict. */
pass = REFERENCED;
pass = INLINE;
staging_block.parent = &tgt_blk;
staging_block.variables.clear();
- remap_prefix = source_func->name;
std::vector<RefPtr<VariableDeclaration> > params;
params.reserve(source_func->parameters.size());
SetForScope<Pass> set_pass(pass, RENAME);
var->visit(*this);
- staging_block.body.push_back(0);
- staging_block.body.back() = var;
+ staging_block.body.push_back_nocopy(var);
params.push_back(var);
}
SetForScope<Pass> set_pass(pass, RENAME);
r_inlined_statement->visit(*this);
- staging_block.body.push_back(0);
- staging_block.body.back() = r_inlined_statement;
+ staging_block.body.push_back_nocopy(r_inlined_statement);
}
/* Now collect names from the staging block. Local variables that would
global identifiers used by the source function. */
pass = RENAME;
staging_block.parent = source_func->body.parent;
- remap_prefix = target_func.name;
target_func.visit(*this);
// Put the argument expressions in place after all renaming has been done.
tgt_blk.body.splice(ins_pt, staging_block.body);
- NodeReorderer().apply(stage, target_func, dependencies);
+ NodeReorderer().apply(stage, target_func, DependencyCollector().apply(*source_func));
return r_result_name;
}
if(i!=staging_block.variables.end())
var.name = i->second->name;
}
- else if(pass==DEPENDS && var.declaration)
- {
- dependencies.insert(var.declaration);
- var.declaration->visit(*this);
- }
else if(pass==REFERENCED)
referenced_names.insert(var.name);
}
void InlineContentInjector::visit(InterfaceBlockReference &iface)
{
- if(pass==DEPENDS && iface.declaration)
- {
- dependencies.insert(iface.declaration);
- iface.declaration->visit(*this);
- }
- else if(pass==REFERENCED)
+ if(pass==REFERENCED)
referenced_names.insert(iface.name);
}
void InlineContentInjector::visit(FunctionCall &call)
{
- if(pass==DEPENDS && call.declaration)
- dependencies.insert(call.declaration);
- else if(pass==REFERENCED)
+ if(pass==REFERENCED)
referenced_names.insert(call.name);
TraversingVisitor::visit(call);
}
if(pass==RENAME)
{
+ /* Check against conflicts with the other context as well as variables
+ already renamed here. */
+ bool conflict = (staging_block.variables.count(var.name) || referenced_names.count(var.name));
staging_block.variables[var.name] = &var;
- if(referenced_names.count(var.name))
+ if(conflict)
{
- string mapped_name = get_unused_variable_name(staging_block, var.name, remap_prefix);
+ string mapped_name = get_unused_variable_name(staging_block, var.name);
if(mapped_name!=var.name)
{
staging_block.variables[mapped_name] = &var;
}
}
}
- else if(pass==DEPENDS && var.type_declaration)
- {
- dependencies.insert(var.type_declaration);
- var.type_declaration->visit(*this);
- }
else if(pass==REFERENCED)
referenced_names.insert(var.type);
}
if(pass==INLINE && ret.expression)
{
// Create a new variable to hold the return value of the inlined function.
- r_result_name = get_unused_variable_name(staging_block, "_return", source_func->name);
+ r_result_name = get_unused_variable_name(staging_block, "_return");
RefPtr<VariableDeclaration> var = new VariableDeclaration;
var->source = ret.source;
var->line = ret.line;
void FunctionInliner::visit(FunctionCall &call)
{
+ for(NodeArray<Expression>::iterator i=call.arguments.begin(); (!r_inlined_here && i!=call.arguments.end()); ++i)
+ visit(*i);
+
if(r_inlined_here)
return;
- for(NodeArray<Expression>::iterator i=call.arguments.begin(); i!=call.arguments.end(); ++i)
- visit(*i);
-
FunctionDeclaration *def = call.declaration;
if(def)
def = def->definition;
// This will later get removed by UnusedVariableRemover.
if(result_name.empty())
- result_name = "msp_unused_from_inline";
+ result_name = "_msp_unused_from_inline";
RefPtr<VariableReference> ref = new VariableReference;
ref->name = result_name;
}
-ExpressionInliner::ExpressionInfo::ExpressionInfo():
- expression(0),
- assign_scope(0),
- inline_point(0),
- trivial(false),
- available(true)
-{ }
-
-
ExpressionInliner::ExpressionInliner():
r_ref_info(0),
r_any_inlined(false),
/* Mark variables as output if they're used by the next stage or the
graphics API. */
if(interface_block)
- var_info.output = (interface_block->interface=="out" && (interface_block->linked_block || !interface_block->name.compare(0, 3, "gl_")));
+ var_info.output = (interface_block->interface=="out" && (interface_block->linked_block || !interface_block->block_name.compare(0, 3, "gl_")));
else
var_info.output = (var.interface=="out" && (stage->type==Stage::FRAGMENT || var.linked_declaration || !var.name.compare(0, 3, "gl_")));
else
{
VariableInfo &var_info = variables[&iface];
- var_info.output = (iface.interface=="out" && (iface.linked_block || !iface.name.compare(0, 3, "gl_")));
+ var_info.output = (iface.interface=="out" && (iface.linked_block || !iface.block_name.compare(0, 3, "gl_")));
}
}