openscad/src/context.cc

238 lines
7.4 KiB
C++

/*
* OpenSCAD (www.openscad.org)
* Copyright (C) 2009-2011 Clifford Wolf <clifford@clifford.at> and
* Marius Kintel <marius@kintel.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* As a special exception, you have permission to link this program
* with the CGAL library and distribute executables, as long as you
* follow the requirements of the GNU GPL in regard to all of the
* software in the executable aside from CGAL.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
#include "context.h"
#include "evalcontext.h"
#include "expression.h"
#include "function.h"
#include "module.h"
#include "builtin.h"
#include "printutils.h"
#include <boost/foreach.hpp>
#include <boost/filesystem.hpp>
namespace fs = boost::filesystem;
#include "boosty.h"
// $children is not a config_variable. config_variables have dynamic scope,
// meaning they are passed down the call chain implicitly.
// $children is simply misnamed and shouldn't have included the '$'.
static bool is_config_variable(const std::string &name) {
return name[0] == '$' && name != "$children";
}
/*!
Initializes this context. Optionally initializes a context for an
external library. Note that if parent is null, a new stack will be
created, and all children will share the root parent's stack.
*/
Context::Context(const Context *parent)
: parent(parent)
{
if (parent) {
assert(parent->ctx_stack && "Parent context stack was null!");
this->ctx_stack = parent->ctx_stack;
this->document_path = parent->document_path;
}
else {
this->ctx_stack = new Stack;
}
this->ctx_stack->push_back(this);
}
Context::~Context()
{
assert(this->ctx_stack && "Context stack was null at destruction!");
this->ctx_stack->pop_back();
if (!parent) delete this->ctx_stack;
}
/*!
Initialize context from a module argument list and a evaluation context
which may pass variables which will be preferred over default values.
*/
void Context::setVariables(const AssignmentList &args,
const EvalContext *evalctx)
{
BOOST_FOREACH(const Assignment &arg, args) {
set_variable(arg.first, arg.second ? arg.second->evaluate(this->parent) : ValuePtr::undefined);
}
if (evalctx) {
size_t posarg = 0;
for (size_t i=0; i<evalctx->numArgs(); i++) {
const std::string &name = evalctx->getArgName(i);
ValuePtr val = evalctx->getArgValue(i);
if (name.empty()) {
if (posarg < args.size()) this->set_variable(args[posarg++].first, val);
} else {
this->set_variable(name, val);
}
}
}
}
void Context::set_variable(const std::string &name, const ValuePtr &value)
{
if (is_config_variable(name)) this->config_variables[name] = value;
else this->variables[name] = value;
}
void Context::set_variable(const std::string &name, const Value &value)
{
set_variable(name, ValuePtr(value));
}
void Context::set_constant(const std::string &name, const ValuePtr &value)
{
if (this->constants.find(name) != this->constants.end()) {
PRINTB("WARNING: Attempt to modify constant '%s'.", name);
}
else {
this->constants[name] = value;
}
}
void Context::set_constant(const std::string &name, const Value &value)
{
set_constant(name, ValuePtr(value));
}
void Context::apply_variables(const Context &other)
{
for (ValueMap::const_iterator it = other.variables.begin();it != other.variables.end();it++) {
set_variable((*it).first, (*it).second);
}
}
ValuePtr Context::lookup_variable(const std::string &name, bool silent) const
{
if (!this->ctx_stack) {
PRINT("ERROR: Context had null stack in lookup_variable()!!");
return ValuePtr::undefined;
}
if (is_config_variable(name)) {
for (int i = this->ctx_stack->size()-1; i >= 0; i--) {
const ValueMap &confvars = ctx_stack->at(i)->config_variables;
if (confvars.find(name) != confvars.end())
return confvars.find(name)->second;
}
return ValuePtr::undefined;
}
if (!this->parent && this->constants.find(name) != this->constants.end())
return this->constants.find(name)->second;
if (this->variables.find(name) != this->variables.end())
return this->variables.find(name)->second;
if (this->parent)
return this->parent->lookup_variable(name, silent);
if (!silent)
PRINTB("WARNING: Ignoring unknown variable '%s'.", name);
return ValuePtr::undefined;
}
bool Context::has_local_variable(const std::string &name) const
{
if (is_config_variable(name))
return config_variables.find(name) != config_variables.end();
if (!parent && constants.find(name) != constants.end())
return true;
return variables.find(name) != variables.end();
}
/**
* This is separated because PRINTB uses quite a lot of stack space
* and the methods using it evaluate_function() and instantiate_module()
* are called often when recursive functions or modules are evaluated.
*
* @param what what is ignored
* @param name name of the ignored object
*/
static void print_ignore_warning(const char *what, const char *name)
{
PRINTB("WARNING: Ignoring unknown %s '%s'.", what % name);
}
ValuePtr Context::evaluate_function(const std::string &name, const EvalContext *evalctx) const
{
if (this->parent) return this->parent->evaluate_function(name, evalctx);
print_ignore_warning("function", name.c_str());
return ValuePtr::undefined;
}
AbstractNode *Context::instantiate_module(const ModuleInstantiation &inst, EvalContext *evalctx) const
{
if (this->parent) return this->parent->instantiate_module(inst, evalctx);
print_ignore_warning("module", inst.name().c_str());
return NULL;
}
/*!
Returns the absolute path to the given filename, unless it's empty.
*/
std::string Context::getAbsolutePath(const std::string &filename) const
{
if (!filename.empty() && !boosty::is_absolute(fs::path(filename))) {
return boosty::absolute(fs::path(this->document_path) / filename).string();
}
else {
return filename;
}
}
#ifdef DEBUG
std::string Context::dump(const AbstractModule *mod, const ModuleInstantiation *inst)
{
std::stringstream s;
if (inst)
s << boost::format("ModuleContext %p (%p) for %s inst (%p)") % this % this->parent % inst->name() % inst;
else
s << boost::format("Context: %p (%p)") % this % this->parent;
s << boost::format(" document path: %s") % this->document_path;
if (mod) {
const Module *m = dynamic_cast<const Module*>(mod);
if (m) {
s << " module args:";
BOOST_FOREACH(const Assignment &arg, m->definition_arguments) {
s << boost::format(" %s = %s") % arg.first % variables[arg.first];
}
}
}
typedef std::pair<std::string, ValuePtr> ValueMapType;
s << " vars:";
BOOST_FOREACH(const ValueMapType &v, constants) {
s << boost::format(" %s = %s") % v.first % v.second;
}
BOOST_FOREACH(const ValueMapType &v, variables) {
s << boost::format(" %s = %s") % v.first % v.second;
}
BOOST_FOREACH(const ValueMapType &v, config_variables) {
s << boost::format(" %s = %s") % v.first % v.second;
}
return s.str();
}
#endif