dynfunc.t

documentation

#charset "us-ascii"

/*
 *   Copyright (c) 2001, 2006 Michael J. Roberts
 *   
 *   This file is part of TADS 3.
 *   
 *   This module defines classes related to the DynamicFunc intrinsic
 *   class.  You should include this source file in your build if you're
 *   using the DynamicFunc class.  
 */

#include <tads.h>
#include <dynfunc.h>


/* ------------------------------------------------------------------------ */
/*
 *   Compiler: This object provides a simplified interface to the dynamic
 *   compiler.  The methods here can be used instead of manually creating
 *   DynamicFunc instances.
 *   
 *   The main advantage of using this object to compile code is that it
 *   automatically provides access to the global symbol table that was used
 *   to compile the current program, for use in dynamic code.  Without the
 *   global symbol table, dynamic code won't have access to object names,
 *   property names, function names, and so on.  That doesn't stop you from
 *   compiling code that only depends upon its own function parameters and
 *   local variables, but for most purposes the global symbols are useful
 *   to have around.
 *   
 *   Note that including this object in a project will automatically save
 *   the global symbol table in the compiled .t3 file.  This increases the
 *   size of the .t3 file, as well as memory usage during execution.  If
 *   you're concerned about minimizing the .t3 file size or the run-time
 *   memory footprint, *and* you don't need global symbols for dynamic code
 *   (or you don't use the dynamic compiler at all), you can save some
 *   space by omitting this whole module from the build.  
 */
Compiler: PreinitObject
    /*
     *   Compile an expression or function.  'str' is a string giving the
     *   code to compile.  This can be a simple value expression, such as
     *   'Me.location' or 'new BigNumber(12345).sqrt()'.  Or, it can be a
     *   complete unnamed function definition, using this syntax:
     *   
     *.    'function(x, y, z) { ...body of function... }'
     *   
     *   The body of the function can contain any executable code that you
     *   could write in a regular function in static code: if, while,
     *   switch, return, etc.
     *   
     *   The return value is a DynamicFunc containing the compiled
     *   expression or function.  You call it by using the return value as
     *   though it were a function:
     *   
     *.    local f = Compiler.compile('Me.location');
     *.    local loc = f();  
     *   
     *   If the source string was just an expression, it acts like a
     *   function that takes zero arguments, and returns the computed value
     *   of the expression.  The expression is evaluated anew each time you
     *   invoke it, so you'll get the "live" value of an expression that
     *   refers to object properties or other external data.  In the
     *   example above, we'd get the current value of Me.location every
     *   time we call f().
     *   
     *   The source string is actually compiled immediately when you call
     *   this function.  This means it's checked for errors, such as syntax
     *   errors and unknown symbol names.  If the code contains any errors,
     *   this method throws a CompilerException describing the problem.
     */
    compile(str, locals?)
    {
        /* compile the string, using our saved global symbol table */
        return new DynamicFunc(str, symtab_, locals, macros_);
    }

    /*
     *   Compile a dynamic function string, and add it to the global symbol
     *   table as a function with the given name.  This effectively creates
     *   a new named function that you can call from other dynamic code
     *   objects. 
     */
    defineFunc(name, str, locals?)
    {
        /* 
         *   compile the string, and add it to our symbol table under the
         *   given function name 
         */
        symtab_[name] = new DynamicFunc(str, symtab_, locals, macros_);
    }

    /*
     *   Evaluate an expression.  'str' is a string giving code to compile.
     *   In most cases, this is simply a simple value expression, although
     *   it's also acceptable to use the 'function()' syntax to create a
     *   function that takes no arguments.
     *   
     *   This method compiles the source string and immediately calls the
     *   resulting compiled code.  The return value is the value returned
     *   from the compiled code itself.  This method thus provides a quick
     *   way to evaluate an expression.
     *   
     *   If the string contains any syntax errors or other compilation
     *   errors, the method throws a CompilerException.  In addition, it's
     *   possible for the compiled code to throw exceptions of its own;
     *   this method doesn't catch those, leaving it up to the caller to
     *   handle them.
     *   
     *   If you expect to evaluate the same expression repeatedly, you're
     *   better off using compile() to get the compiled representation of
     *   the expression, and then call that compiled code each time the
     *   value is needed.  That's more efficient than using eval() each
     *   time, since eval() to recompile the expression on every call,
     *   which is a fairly complex process.  
     */
    eval(str, locals?)
    {
        /* 
         *   compile the string, call the resulting function, and return
         *   the result from the function 
         */
        return (new DynamicFunc(str, symtab_, locals, macros_))();
    }

    /*
     *   During preinit, save a reference to the program's global symbol
     *   table in a property of self.  The VM always makes the global
     *   symbols available during preinit, but by default it discards the
     *   table after that because most programs don't need it.  That means
     *   that the symbols aren't available by default during normal
     *   execution.  However, saving a reference here prevents the garbage
     *   collector from discarding the table when preinit finishes, which
     *   forces it to be saved in the final .t3 file and thus makes it
     *   available permanently.  
     */
    execute()
    {
        /* save the global symbol table */
        symtab_ = t3GetGlobalSymbols(T3GlobalSymbols);
        macros_ = t3GetGlobalSymbols(T3PreprocMacros);
    }

    /* a saved reference to the global symbol table */
    symtab_ = nil

    /* a saved referenced to the preprocessor macro table */
    macros_ = nil
;
    

/* ------------------------------------------------------------------------ */
/*
 *   Compiler Exception.  'new DynamicFunc()' throws an exception of this
 *   class if an error occurs compiling the source code of the new object.
 */
class CompilerException: Exception
    construct(msg) { errmsg_ = msg; }
    displayException()
    {
        if (errmsg_ != nil)
            "<<errmsg_>>";
        else
            "Source code compilation error";
    }

    /* the error message from the compiler */
    errmsg_ = nil
;

/* export the compiler exception for use by the intrinsic class */
export CompilerException 'DynamicFunc.CompilerException';

TADS 3 Library Manual
Generated on 5/16/2013 from TADS version 3.1.3