/*
 *  slicontrol.cc
 *
 *  This file is part of NEST.
 *
 *  Copyright (C) 2004 The NEST Initiative
 *
 *  NEST 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.
 *
 *  NEST 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 NEST.  If not, see <http://www.gnu.org/licenses/>.
 *
 */

/*
    SLI's control structures
*/

#include "config.h"

#include <time.h>
#include <sys/time.h>  // required to fix header dependencies in OS X, HEP
#include <sys/times.h>
#include <sys/resource.h>
#include <unistd.h>
// sstream has functions std::?stringstream
// strstream has functions std::?strstream
// HEP 2002-10-06
#ifdef HAVE_SSTREAM
#include <sstream>
#else
#include <strstream>
#endif

#include "sliconfig.h"
#include "slicontrol.h"
#include "namedatum.h"
#include "booldatum.h"
#include "arraydatum.h"
#include "stringdatum.h"
#include "integerdatum.h"
#include "doubledatum.h"
#include "scanner.h"
#include "parser.h"
#include "iostreamdatum.h"
#include "dictstack.h"
#include "functiondatum.h"
#include "processes.h"

#include "sliexceptions.h"


/*BeginDocumentation
Name: backtrace_on - enable stack backtrace on error.
Synopsis: backtrace_on -> -
Description:

This functions enables a human readable backtrace of the execution
stack. This is useful to locate where precisely an error occured. Note
that this function also disables the interpreter's tail recursion
optimization and will therefore impose a small performance
penalty. The command backtrace_off disables the stack backtrace and
re-enables tail recursion optimization.

Example:

SeeAlso:
backtrace_off
*/
void Backtrace_onFunction::execute(SLIInterpreter *i) const
{
  i->backtrace_on();
  i->EStack.pop();
}

/*BeginDocumentation
Name: backtrace_off - Disable the stack backtrace on error.
Synopsis: backtrace_off -> -

Description:
This functions disables the backtrace of the execution
stack and re-enables tail recursion optimization.

SeeAlso: backtrace_on

*/
void Backtrace_offFunction::execute(SLIInterpreter *i) const
{
  i->backtrace_off();
  i->EStack.pop();
}

void OStackdumpFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop(); // never forget me!!

    i->OStack.dump(std::cout);
}
void EStackdumpFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop(); // never forget me!!

    i->EStack.dump(std::cout);
}
/*BeginDocumentation
Name: loop - repeatedly execute a procedure
Synopsis: proc loop -
Description:
 loop takes a procedure object an executes it repeatedly.
 Since there is no direct termination condition, the loop
 has to be terminated by calling exit.
 If the procedure has to be evaluated for a certain number of
 times, consider the use of repeat or for.
 If some container should be iterated, consider forall or Map
SeeAlso: exit, repeat, for, forall, forallindexed, Map
*/
void LoopFunction::execute(SLIInterpreter *i) const
{
    if (i->OStack.load() ==0 )
      {
	i->raiseerror(i->StackUnderflowError);
	return;
      }
    if(dynamic_cast<ProcedureDatum *>(i->OStack.top().datum())==0)
      {
	i->raiseerror(i->ArgumentTypeError);
	return;
      }

    i->EStack.pop();
    i->EStack.push_by_ref(i->baselookup(i->mark_name));
    i->EStack.push_move(i->OStack.top());
    i->EStack.push_by_pointer(new IntegerDatum(0));
    i->EStack.push_by_ref(i->baselookup(i->iloop_name));
    i->inc_call_depth();
    i->OStack.pop();
}

/*BeginDocumentation
Name: exit - exit a loop construct
Description: exit can be used to leave loop structures
             like loop, repeat, for, forall, Map, etc.
             in a clean way.
Remarks: This command does not exit the SLI interpreter! Use quit instead.
*/
void ExitFunction::execute(SLIInterpreter *i) const
{

    static Token mark = i->baselookup(i->mark_name);

    size_t n=1;
    size_t l=i->EStack.load();
    while( ( l > n ) && !( i->EStack.pick(n++) == mark) ) ;
    if(n>= l)
    {
      i->raiseerror("EStackUnderflow");
      return;
    }
    i->dec_call_depth();
    i->EStack.pop(n);
}

/*BeginDocumentation
Name: if - conditionaly execute a procedure
Synopsis:
  boolean {procedure} if -> -
  boolean   anytoken  if -> -
Description: if executes the supplied token if the boolean
             is true. The supplied token usually is a procedure.

	     Alternatives: Function if_ (undocumented)
	     -> behaviour and synopsis are the same, except that no
	     warnings or error messages are thrown.

Examples:  1 0 gt { (1 > 0) = } if -> (1 > 0)

SeeAlso: ifelse
*/
void IfFunction::execute(SLIInterpreter *i) const
{
        // OStack: bool proc
        //          1    0
    static Token true_token(new BoolDatum(true));

    if((i->OStack.load()>=2))
    {
	i->EStack.pop();
	if(i->OStack.pick(1) == true_token )
	{
	    if(i->step_mode())
	    {
		std::cerr << "if:"
			  << " Executing true branch."
			  << std::endl;
	    }
	    i->EStack.push_move(i->OStack.top());
	}
	i->OStack.pop(2);
    }
    else throw StackUnderflow(2,i->OStack.load());
}

/*BeginDocumentation
Name: ifelse - conditionaly execute a procedure
Synopsis:
  boolean  {proc1}  {proc2}   ifelse -> -
  boolean anytoken1 anytoken1 ifelse -> -
Description:
  ifelse executes anytoken1 if the boolean is true, and anytoken2
  otherwise.
  The supplied tokens usually are procedures.

  Alternatives: Function ifelse_ (undocumented)
  -> behaviour and synopsis are the same, except that no
  warnings or error messages are thrown.

Examples:
     0 1 gt { (1 > 0) = } { (1 <= 0) =} ifelse -> (0 <= 1)

SeeAlso: if
*/
void IfelseFunction::execute(SLIInterpreter *i) const
{
        // OStack: bool tproc fproc
        //          2    1      0
    static Token true_token(new BoolDatum(true));

    if(i->OStack.load()<3 )
	throw StackUnderflow(3,i->OStack.load());

    i->EStack.pop();

    if(i->OStack.pick(2) == true_token )
    {
      if(i->step_mode())
      {
	std::cerr << "ifelse:"
		  << " Executing true branch."
		  << std::endl;
      }
      i->EStack.push_move(i->OStack.pick(1));
    }
    else
    {
      if(i->step_mode())
      {
	std::cerr << "ifelse:"
		  << " Executing false branch."
		  << std::endl;
      }
      i->EStack.push_move(i->OStack.pick(0));
    }

    i->OStack.pop(3);

}

/*BeginDocumentation
Name: repeat - execute a procedure n times
Synopsis: n proc repeat
Description:
 repeat executes the supplied procedure n times.
 The loop can be left prematurely using exit.

Note: The interation counter is not available
 to the procedure. If this is desired, use for instead.

Examples:
SLI ] 3 { (Hello world) = } repeat
Hello world
Hello world
Hello world
SLI ]

SeeAlso: for, loop, exit
*/
void RepeatFunction::execute(SLIInterpreter *i) const
{

        // level  1  0
        // stack: n proc repeat
    if(i->OStack.load()>=2)
    {
	i->EStack.pop();

	ProcedureDatum *proc=
	    dynamic_cast<ProcedureDatum *>(i->OStack.top().datum());
	if(proc !=NULL)
	{
	    IntegerDatum *id= dynamic_cast<IntegerDatum*>(i->OStack.pick(1).datum());
	    if(id==0)
		throw ArgumentType(1);

	    i->EStack.push_by_ref(i->baselookup(i->mark_name));
	    i->EStack.push_move(i->OStack.pick(1));
	    i->EStack.push_move(i->OStack.pick(0));
	    i->EStack.push_by_pointer(new IntegerDatum(proc->size()));
	    i->EStack.push_by_ref(i->baselookup(i->irepeat_name));
	    i->inc_call_depth();
	    i->OStack.pop(2);
	}
	else throw ArgumentType(0);
    }
    else throw StackUnderflow(2,i->OStack.load());
}

/*BeginDocumentation
Name: stopped - returns true if execution was stopped by stop
Synopsis:
  xobj stopped -> true;  if the object was aborted with stop
               -> false; otherwise.
Description:
  stopped is part of a pair of commands which implement the
  PostScript exception mechanism.
  stopped is applied to an executable object, most probably a
  procedure, and returns true if this object raised a stop
  signal and false otherwise.
  This is accomplished by first pushing an internal name which
  resolves to false and then executing the object.
  stop will pop the stack down to this name and return true. If
  stop is not called, the name is eventually executed and thus the
  return value false.

  Note that when the executable object was stopped by a call
  to raiseerror, the name of the routine that caused the
  error has can be found on the operand stack (see raiseerror).

  The stop/stopped pair is used to implement SLI's error handling
  capabilities.

Notes: stop, stopped is PostScript compatible
References:   The Red Book, sec. 3.10
SeeAlso: stop, raiseerror
*/
void StoppedFunction::execute(SLIInterpreter *i) const
{
    if(i->OStack.load()==0)
	throw StackUnderflow(1,i->OStack.load());

  i->EStack.pop();
  i->EStack.push_by_pointer(new NameDatum(i->istopped_name));
  i->EStack.push_move(i->OStack.top());
  i->OStack.pop();
}

/*BeginDocumentation
Name: stop - raise a stop signal
Synopsis:
  stop -> -

Desctiption:
  stop causes the execution stack to be popped until an
  enclosing stopped context is found. Effectively, the
  stopped/stop combination equals the catch/throw pair of
  C++.
  stop/stopped is used to implement SLI's error handling
  capabilities.
Notes: stop, stopped is PostScript compatible.
References: The Red Book, sec. 3.10
SeeAlso: stopped, raiseerror
*/
void StopFunction::execute(SLIInterpreter *i) const
{

    size_t l=i->EStack.load();
    NameDatum istopped(i->istopped_name);

    bool found=false;
    size_t n=1;

    while( ( l > n ) &&  !(found))
        found=i->EStack.pick(n++).contains(istopped);

    if(i->catch_errors() && ! found)
      i->debug_mode_on();

    if(i->get_debug_mode() || i->show_backtrace())
    {
      if(i->show_backtrace() || ! found)
	i->stack_backtrace(l-1);

      std::cerr << "In stop: An error or stop was raised."
		<<" Unrolling stack by " << n << " levels."
		<< std::endl;
      if(!found)
	{
	  std::cerr << "No 'stopped' context found." << std::endl
		    << "Stack unrolling will erase the execution stack." << std::endl
		    << "Entering debug mode. Type '?' for help." << std::endl;
	}
      
      if(i->get_debug_mode())
	{
	  char c=i->debug_commandline(i->EStack.top());
	  if(c=='i') // in interactive mode, we leave the stack as it is.
	    return;
	}
    }

    if(found)
      i->OStack.push(true);
    else
      {
	i->message(30,"stop",
		   "No stopped context was found! \n");
	i->EStack.clear();
	return;
      }

    i->EStack.pop(n);
}

/*BeginDocumentation
Name: closeinput - Close current input file.
FirstVersion: 25 Jul 2005, Gewaltig
*/

void CloseinputFunction::execute(SLIInterpreter *i) const
{
  size_t l=i->EStack.load();

  bool found=false;
  size_t n=1;

  while( ( l > n ) &&  !(found))
    found = i->EStack.pick(n++)->isoftype(SLIInterpreter::XIstreamtype);


  if(i->catch_errors() || ! found)
    i->debug_mode_on();

  if(i->get_debug_mode() || i->show_backtrace())
  {
    if(i->show_backtrace() || ! found)
      i->stack_backtrace(n);

    std::cerr << "In closeinput: Termination of input file requested."
        <<" Unrolling stack by " << n << " levels."
        << std::endl;
    if(!found)
    {
      std::cerr << "In closeinput: No active input file was found." << std::endl
          << "Stack unrolling will erase the execution stack." << std::endl
          << "Entering debug mode. Type '?' for help." << std::endl;
    }

    if(i->get_debug_mode())
    {
      char c=i->debug_commandline(i->EStack.top());
      if(c=='i') // in interactive mode, we leave the stack as it is.
        return;
    }
  }

  if(!found)
  {
    i->message(30,"closeinput",
               "No active input file was found. \n  Restarting...");
    i->EStack.clear();
    i->EStack.push(i->baselookup(Name("start")));
    return;
  }

  i->EStack.pop(n);

}

/* BeginDocumentation
Name: currentname -  returns the most recently resolved name
Synopsis:
 currentname -> name true
             -> false
Description:
 currentname returns the most recently resolved name whose contents
 is still under execution.
 currentname is useful for error handling purposes where a procedure
 has to know the name with which it has been called.
Example:

 /divide
 {
    0 eq
    {
      currentname /DivisionByZero raiseerror
    } if
    div
} def

Note:
 This function is not foolproof, as it will fail for bound procedures.

 currentname evaluates certain debugging information on the execution
 stack. Note that the SLI interpreter can be compiled not to generate
 this information. In this case currentname fails, i.e. it will always
 return false
*/
void CurrentnameFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    size_t n=0; // skip my own name
    size_t l=i->EStack.load();

        // top level %%lookup must belong to currentname, so
        // remove it and the name.
    if(i->EStack.top()==i->baselookup(i->ilookup_name))
    {
        assert(l>2);
        n+=2;
    }

    bool found=false;

    while( ( l > n ) && !found)
        found= i->EStack.pick(n++)==i->baselookup(i->ilookup_name);

    if(found)
    {
        i->OStack.push(i->EStack.pick(n));
        i->OStack.push(true);
    }
    else
        i->EStack.push(false);

}

void IparsestdinFunction::execute(SLIInterpreter *i) const
{
    Token t;

    i->parse->readToken(std::cin, t);
    if(t.contains(i->parse->scan()->EndSymbol))
        i->EStack.pop();
    else
        i->EStack.push_move(t);
}

/*BeginDocumentation
Name: parsestdin - Read and execute tokens from standard input
Description: parsestdin repeatedly reads and executes SLI commands from
the standard input stream (cin) until an end-of-file symbol is excountered
or the command exit is executed.

*/
void ParsestdinFunction::execute(SLIInterpreter *i) const
{
    Token t;

    i->parse->readToken(std::cin, t);
    if(t.contains(i->parse->scan()->EndSymbol))
        i->EStack.pop();
    else
      {
        i->EStack.pop();
        i->EStack.push_move(t);
      }
}

void IparseFunction::execute(SLIInterpreter *i) const
{

        // Estack: handle  iparse
        // pick      1         0

    XIstreamDatum *is =
	dynamic_cast<XIstreamDatum *>(i->EStack.pick(1).datum());
    assert(is != NULL);	assert(is->valid());

    Token t;
    if(i->parse->readToken(**is, t))
    {
      if(t.contains(i->parse->scan()->EndSymbol))
        i->EStack.pop(2);
      else
        i->EStack.push_move(t);
    }
    else
    {
      i->EStack.swap(); i->EStack.pop(); // remove stream-token
      i->raiseerror("SyntaxError");
      return;
    }
}

void DefFunction::execute(SLIInterpreter *i) const
{
        // Def should also check the "writeable" Flag of the
        // name!
    if(i->OStack.load()<2)
	throw StackUnderflow(2,i->OStack.load());

    LiteralDatum *nd=dynamic_cast<LiteralDatum *>(i->OStack.pick(1).datum());
    if(nd == NULL)
	throw ArgumentType(1);

    // if(nd->writeable())
    // {
        i->def_move(*nd,i->OStack.top());
        i->OStack.pop(2);
        i->EStack.pop();
    // }
    // else   // Name was write Protected: OStack is left untouched!
    // {
    //     Name myname(i->getcurrentname());
    //     i->EStack.pop();
    //     i->raiseerror(myname,i->WriteProtectedError);
    // }

}

/*BeginDocumentation
Name: Set - Define an association between a name and an object in the current dictionary
Synopsis: 
  obj literal   Set -> -
  [... [obj_1 ...] ... obj_n] [... [literal_1 ...] ... literal_n] Set -> -
Description: 
 In the first form Set is identical to def, except for the reversed parameters and
 creates or modifies an entry for the literal in the current dictionary. The new value 
 assigned to the literal is obj.
 In the second form multiple simultaneous assignments are made to the literals contained in
 the second. The nesting of this array is arbitrary, indicated in the synopsis by the 
 inner brackets, and the same object are taken from the identical positions in first array. 


Examples: 
 {1 2 add} /myproc Set
 [4 3] [/x /y] Set 
 [[4 3 7] [-9 1]] [[/a /b /c] [/x /y]] Set 
 [[4 9] 3] [/x /y] Set 
 [[4 9] 3] [[/a /b] /y] Set 

SeeAlso: def, undef, begin, end
*/
void SetFunction::execute(SLIInterpreter *i) const
{
    if(i->OStack.load()<2)
	throw StackUnderflow(2,i->OStack.load());

  LiteralDatum *nd=dynamic_cast<LiteralDatum *>(i->OStack.top().datum());
  if(nd ==0)
      throw ArgumentType(0);

  // if(nd->writeable())
  //   {
      i->def_move(*nd,i->OStack.pick(1));
      i->OStack.pop(2);
      i->EStack.pop();
  //   }
  // else   // Name was write Protected: OStack is left untouched!
  //   {
  //     Name myname(i->getcurrentname());
  //     i->EStack.pop();
  //     i->raiseerror(myname,i->WriteProtectedError);
  //   }

}

/*BeginDocumentation
Name: load - Search for a key in each dictionary on the dictionary stack.
Synopsis: /name load -> obj
Description: Load tries to find an association for /name in each dictionary
 on the dictionary stack, starting with the current (top) dictionary.
 If an association is found, load pushes the associated value on the
 stack. If no association is found, an UndefineName error is raised.
SeeAlso: lookup, def
*/
void LoadFunction::execute(SLIInterpreter *i) const
{
    i->assert_stack_load(1);

    LiteralDatum *name= dynamic_cast<LiteralDatum *>(i->OStack.top().datum());
    if(name == NULL)
    {
      i->raiseerror(i->ArgumentTypeError);
      return;
    }

    Token contents= i->lookup(*name);
    if(contents.datum()!=NULL)
    {
        i->OStack.pop();
        i->OStack.push_move(contents);
        i->EStack.pop();
    }
    else
    {
        Name myname(i->getcurrentname());
        i->EStack.pop();

        i->raiseerror(myname, i->UndefinedNameError);
    }
}

/*BeginDocumentation
Name: lookup -  Search for a key in each dictionay on the dictionary stack.
Synopsis: /name lookup -> obj true
                       -> false
Description: lookup tries to find an association for /name in each dictionary
 on the dictionary stack, starting with the current (top) dictionary.
 If an association is found, lookup pushes the associated value on the
 stack followed by the boolean true.
 If no association is found, false is pushed.
SeeAlso: load, def
*/
void LookupFunction::execute(SLIInterpreter *i) const
{

  if(i->OStack.load() <1)
    {
      i->raiseerror(i->StackUnderflowError);
      return;
    }

  LiteralDatum *name= dynamic_cast<LiteralDatum *>(i->OStack.top().datum());
  if(name == 0)
    {
      i->raiseerror(i->ArgumentTypeError);
      return;
    }

  i->EStack.pop();

  Token contents= i->lookup(*name);
  i->OStack.pop();
  if(contents.datum()!=NULL)
  {
    i->OStack.push_move(contents);
    i->OStack.push(true);
  }
  else
    i->OStack.push(false);
}



/*BeginDocumentation
Name: for - execute a procedure for a sequence of numbers
Synopsis: n1 s n2 proc for -> -
Description:
   for repeatedly evaluates the supplied procedure for all
   values from n1 to n2 in steps of s. In each iteration
   proc is called with the current iteration counter as
   argument.
   The loop can be quit prematurely by calling exit.
   If the value of the iteration counter is not needed,
   use repeat instead.
Examples:
SLI ] 1 1 10 {=} for
1
2
3
4
5
6
7
8
9
10
SLI ]
SeeAlso: repeat, exit, loop
*/
void ForFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    ProcedureDatum *proc=
        dynamic_cast<ProcedureDatum *>(i->OStack.top().datum());
    assert(proc !=NULL);

    i->EStack.push_by_ref(i->baselookup(i->mark_name));
    i->EStack.push_move(i->OStack.pick(2));      // increment
    i->EStack.push_move(i->OStack.pick(1));      // limit
    i->EStack.push_move(i->OStack.pick(3));      // initial as counter
    i->EStack.push_move(i->OStack.pick(0));      // procedure
    i->EStack.push_by_pointer(new IntegerDatum(proc->size()));
    i->EStack.push_by_ref(i->baselookup(i->ifor_name)); // %for
    i->inc_call_depth();
    i->OStack.pop(4);
}

/*
BeginDocumentation

   Name: forall - Call a procedure for each element of a list/string/dictionary

   Synopsis:
     [v1 ... vn] {f}                   forall ->  f(v1) ... f(vn)
     (c1...cn)   {f}                   forall ->  f(c1) ... f(cn)
     <</key1 val1 ... /keyn valn>> {f} forall ->  f(/key1 val1) ... f(/keyn valn)

   Parameters:
     [v1,...,vn]    - list of n arbitrary objects
     (c1...cn)      - arbitrary string
     <</keyi vali>> - arbitrary dictionary
     {f}            - function which can operate on the elements of the
                      array, or on key/value pairs of the dictionary.
                      f is not required to return a specific number
		      of values.

   Description:
     Arrays:
       For each element of the input array, forall calls f.
       forall is similar to Map, however, it does not construct a
       new list from the return values.

     Dictionaries:
       For each key/value pair of the dictionary, forall calls f.
       Order on the operand stack will be: key, value. (I.e. value on top.)
 
       *Note: The dictionary contents are copied before operation.
        This can be a potentially expensive operation.*

     Loops can be nested. The loop can be quit prematurely with exit.

   Examples:
     [1 2 3 4 5]  {=} forall -> - % Print all values of the list
     [1 2 3 4 5]  {} forall  -> 1 2 3 4 5
     (abc) {=} forall -> prints 97 98 99 on separate lines
     <</a 1 /b 2>> {== ==} forall -> prints 1 /a 2 /b on separate lines

   Author:
     Marc-Oliver Gewaltig, Ruediger Kupper (dictionary variant)

   References: The Red Book

   SeeAlso: Map, MapAt, MapIndexed, Table, forallindexed, NestList, FoldList, Fold, exit

*/

/******************************/
/* forall_di                  */
/*  wrapper around forall_a   */
/*  implemented in SLI        */
/*  see typeinit.sli          */
/******************************/

/******************************/
/* forall_a                   */
/*  call: obj proc forall     */
/*  pick   1    0             */
/******************************/
void Forall_aFunction::execute(SLIInterpreter *i) const
{
    static Token mark(i->baselookup(i->mark_name));
    static Token forall(i->baselookup(i->iforallarray_name));

    ProcedureDatum *proc=
        static_cast<ProcedureDatum *>(i->OStack.top().datum());
    
    i->EStack.pop();
    i->EStack.push_by_ref(mark);
    i->EStack.push_move(i->OStack.pick(1));        // push object
    i->EStack.push_by_pointer(new IntegerDatum(0));          // push array counter
    i->EStack.push_by_ref(i->OStack.pick(0));       // push procedure
    i->EStack.push_by_pointer(new IntegerDatum(proc->size()));          // push procedure counter
    i->EStack.push_by_ref(forall);
    i->OStack.pop(2);
    i->inc_call_depth();
}


/******************************/
/* forall_iter                */
/*  call: obj proc forall     */
/*  pick   1    0             */
/******************************/
void Forall_iterFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    ProcedureDatum *proc=
        dynamic_cast<ProcedureDatum *>(i->OStack.top().datum());
    assert(proc !=NULL);

    i->EStack.push(i->baselookup(i->mark_name));
    i->EStack.push_move(i->OStack.pick(1));        // push iterator
    i->EStack.push_move(i->OStack.pick(0));        // push procedure

    i->EStack.push(i->baselookup(i->iforalliter_name));
    i->OStack.pop(2);
    i->inc_call_depth();
}


/*
BeginDocumentation

   Name: forallindexed - Call a procedure for each element of a list/string

   Synopsis:
     [v1,...,vn] {f} forallindexed ->  f(0,v1),...,f(n-1,vn)

   Parameters:
     [v1,...,vn] - list of n arbitrary objects or string
     {f}         - function which can operate on the elements of the
                   array. f is not required to return a specific number
		   of values.

   Description:
     For each element of the input array, forallindexed calls f with
     two arguments, the element and its index within the list/string.
     forallindexed is similar to forall, only that the index of the
     element is also passed to the function f.

     Alternatives: Functions forallindexed_a for a list, forallindexed_s
     for a string (both undocumented) -> behaviour and synopsis are the same.

   Examples:

   [(a) (b)]  {} forallindexed -> (a) 0 (b) 1

   Author:
    Marc-Oliver Gewaltig

   References: The Red Book

   SeeAlso: Map, MapIndexed, Table, forall, NestList, FoldList

*/

/******************************/
/* forallindexed_a            */
/*  call: obj proc forall     */
/*  pick   1    0             */
/******************************/
void Forallindexed_aFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    ProcedureDatum *proc=
        dynamic_cast<ProcedureDatum *>(i->OStack.top().datum());
    assert(proc !=NULL);

    i->EStack.push(i->baselookup(i->mark_name));
    i->EStack.push_move(i->OStack.pick(1));        // push object

    ArrayDatum  *ad= dynamic_cast<ArrayDatum *>(i->EStack.top().datum());
    assert(ad !=NULL);

    i->EStack.push(ad->size()); // push limit
    i->EStack.push(0);          // push initial counter
    i->EStack.push_move(i->OStack.pick(0));       // push procedure

    i->EStack.push(i->baselookup(i->iforallindexedarray_name));
    i->inc_call_depth();
    i->OStack.pop(2);
}

/******************************/
/* forallindexed_s            */
/*  call: obj proc forall     */
/*  pick   1    0             */
/******************************/
void Forallindexed_sFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    ProcedureDatum *proc=
        dynamic_cast<ProcedureDatum *>(i->OStack.top().datum());
    assert(proc !=NULL);

    i->EStack.push(i->baselookup(i->mark_name));
    i->EStack.push_move(i->OStack.pick(1));        // push object

    StringDatum  *sd= dynamic_cast<StringDatum *>(i->EStack.top().datum());
    assert(sd !=NULL);

    i->EStack.push(sd->size()); // push limit
    i->EStack.push(0);          // push initial counter
    i->EStack.push_move(i->OStack.pick(0));       // push procedure

    i->EStack.push(i->baselookup(i->iforallindexedstring_name));
    i->inc_call_depth();
    i->OStack.pop(2);
}

/******************************/
/* forall_s                   */
/*  call: obj proc forall     */
/*  pick   1    0             */
/******************************/
void Forall_sFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    ProcedureDatum *proc=
        dynamic_cast<ProcedureDatum *>(i->OStack.top().datum());
    assert(proc !=NULL);

    i->EStack.push(i->baselookup(i->mark_name));
    i->EStack.push_move(i->OStack.pick(1));        // push object

    StringDatum  *sd= dynamic_cast<StringDatum *>(i->EStack.top().datum());
    assert(sd !=NULL);

    i->EStack.push(new IntegerDatum(sd->size())); // push limit
    i->EStack.push(new IntegerDatum(0));          // push initial counter
    i->EStack.push_move(i->OStack.pick(0));       // push procedure

    i->EStack.push(i->baselookup(i->iforallstring_name));
    i->inc_call_depth();
    i->OStack.pop(2);
}

/* BeginDocumentation
 Name: raiseerror - raise an error to the system
 Synopsis:
 /command /error raiserror -> /command (side-effects see below!)

 Description:
   raiseerror is a SLI interface to the interpreter's error
   handling mechanism (see The Red Book for details). If an error
   is raised, the following actions are performed:
   * the value of errordict /newerror is set to true
   * the value of errordict /commandname is set to the name of the
     command which raised the error
   * the name of the command which raised the error is pushed on
     the operand stack.
   * If the value of errordict /recorstack is true,
     the state of the interpreter is saved:
     - the operand stack is copied to errordict /ostack
     - the execution stack is copied to errordict /estack
     - the dictionary stack is copied to errordict /dstack
   * stop is called. Stop then tries to find an enclosing stopped
     context and calls the associated function.

   This mechanism is explained in detail in The PostScript Reference Manual.

   Please note that when raiserror is called, the state of the stacks
   should be restored to its initial state.

 Examples:
   /save_sqrt
   {
      0 gt % is the argument positive ?
      {
        sqrt
      }
      { % else, issue an error
        /save_sqrt /PositiveValueExpected raiseerror
      } ifelse
   } def

 Bugs: lets wait...
 Author: Gewaltig
 Remarks: not part of PostScript, but conform to the mechanism
 References: See the Red Book for PostScript's error handling facilities
 SeeAlso: raiseagain, stop, stopped, errordict
*/

void RaiseerrorFunction::execute(SLIInterpreter *i) const
{
        // pick :  2     1
        // call : /cmd /err raiseerror

    i->EStack.pop();
    Token err;
    Token cmd;

    i->OStack.pop_move(err);
    i->OStack.pop_move(cmd);

    Name *errorname= dynamic_cast<Name *>(err.datum());
    Name *cmdname  = dynamic_cast<Name *>(cmd.datum());
    if(errorname == NULL || cmdname == NULL)
    {
      i->message(SLIInterpreter::M_ERROR, "raiseerror","Usage: /command /errorname raiserror");
      i->raiseerror("ArgumentType");
      return;
    }

    i->raiseerror(*cmdname, *errorname);
}

/* BeginDocumentation
 Name: print_error - print an error based on the errordict
 Synopsis:
 /command print_error -> --

 Description:
   print_error prints a message describing the content of
   the error dictionary. Please see the example below for
   information about how to use this function. 
   Note: The errordict parameters command and message are
   reset after a call to print_error. The errorname is not
   reset, so every call to print_error will use the same
   errorname until this parameter is redefined.

 Examples:
   errordict /errorname /MyError put_d
   errordict /command /my_function put_d
   errordict /message (Something went wrong.) put_d
   /my_function print_error

 Bugs:
 Author:
 Remarks:
 References:
 SeeAlso: handleerror, raiseerror, raiseagain, stop, stopped, errordict
*/

void PrinterrorFunction::execute(SLIInterpreter *i) const
{
  // The name of the command that raised the error should be 
  // placed at the top of the OStack.
  i->assert_stack_load(1);

  // Call print_error function.
  i->print_error(i->OStack.top());

  i->OStack.pop();
  i->EStack.pop();
}

/* BeginDocumentation
 Name: raiseagain - re-raise the last error
 Synopsis:  raiseagain

 Description:
   raiseagain re-raises a previously raised error. This is useful
   if an error handler cannot cope with a particular error (e.g. a signal)
   and wants to pass it to an upper level handler. Thus, nestet error handlers
   are possible.

 Bugs: lets wait...
 Author: Gewaltig
 Remarks: not part of PostScript
 References: See the Red Book for PostScript's error handling facilities
 SeeAlso: raiseerror, stop, stopped, errordict
*/

void RaiseagainFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    i->raiseagain();
}

/*BeginDocumentation
Name: cycles - return the number of elapsed interpreter cycles
Synopsis: cycles -> n
*/
void CyclesFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    Token cycles(new IntegerDatum(i->cycles()));

    i->OStack.push(cycles);
}

void CodeAccessedFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    Token c(new IntegerDatum(i->code_accessed));

    i->OStack.push(c);
}

void CodeExecutedFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    Token c(new IntegerDatum(i->code_executed));

    i->OStack.push(c);
}






/*BeginDocumentation
Name: quit - leave the SLI interpreter, optionally return exit code

Synopsis:
           quit   -> -
  exitcode quit_i -> -

Description:
This function leaves the SLI interpreter, returning an exit code to
the calling shell. The first variant (quit) returns the value
EXIT_SUCCESS (usually 0). The second variant (quit_i) returns the
value given by the integer argument. All data which has not been saved
is lost.

Common exit codes and their reasons can be found in the dictionary
statusdict/exitcodes.

Parameters:
exitcode (integertype): exit code to return to the shell. If this
parameter is not given, statusdict/exitcode is used instead.

Examples:
quit      % quit SLI with exit code statusdict/exitcode.
23 quit_i % quit SLI with exit code 23.

Author: unknown, JM Eppler, R Kupper

FirstVersion: long time ago

Remarks:
This command should not be mixed up with exit.

Availability: SLI-2.0

References:
POSIX Programmer's Manual.

SeeAlso: exit
*/
void QuitFunction::execute(SLIInterpreter *i) const
{
  i->EStack.clear();
}

/*BeginDocumentation
Name: exec - execute an object
Synopsis: any exec -> -
Description: exec tries to execute the object by moving it to
the execution stack.
Examples: {1 2 add} exec -> 3
SeeAlso: cvx
*/
void ExecFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
    i->EStack.push_move(i->OStack.top());
    i->OStack.pop();
}

/*BeginDocumentation
Name: typeinfo - return the type of an object
Synopsis: any type -> any literal
Description: typeinfo returns a literal name, which represents
 the type of the argument. The argument is left on the stack.
SeeAlso: typestack, type
*/
void TypeinfoFunction::execute(SLIInterpreter *i) const
{
    i->assert_stack_load(1);  

    i->EStack.pop();
    Token tn(new LiteralDatum(i->OStack.top()->gettypename()));

    i->OStack.push_move(tn);
}

void SwitchFunction::execute(SLIInterpreter *i) const
{
        // mark obj1 obj2 ... objn switch
        // Executes obj1 to obj2. If one object executes
        // exit, the execution of all other objects is
        // terminated.
    Name myname(i->getcurrentname());
    i->EStack.pop();

    Token mark_token(i->baselookup(i->mark_name));

    i->EStack.push(mark_token); // no push_move since token will
                                // be needed.
    i->EStack.push(i->baselookup(i->ipop_name));

    unsigned long depth = i->OStack.load();
    unsigned long pos = 0;
    
    if(depth == 0)
      throw TypeMismatch("At least 1 argument.", "Nothing.");

    bool found =  (i->OStack.pick(pos) == mark_token);

    while( (pos < depth) && ! found )
    {
        i->EStack.push_move(i->OStack.pick(pos));
        found =  (i->OStack.pick(++pos) == mark_token);
    }

    if(found)
        i->OStack.pop(pos+1);
    else
      i->raiseerror(myname ,Name("UnmatchedMark"));
}

void SwitchdefaultFunction::execute(SLIInterpreter *i) const
{
  // mark obj1 obj2 ... objn defobj switchdefault
  // checks if n=0.
  // if so, executes defobj.
  // else pops defobj and calls switch to execute obj1..objn
  // If one object executes exit, the execution of all other
  // objects is terminated.

  Name myname(i->getcurrentname());        // needed for raiseerror
  i->EStack.pop();                         // no multiple calls !!!

  Token mark_token(i->baselookup(i->mark_name));

  i->EStack.push(mark_token);
  i->EStack.push(i->baselookup(i->ipop_name));

  unsigned long depth = i->OStack.load();
  unsigned long pos = 0;

  if(depth == 0)
    throw TypeMismatch("At least 1 argument.", "Nothing.");

  if (depth>1 && i->OStack.pick(1) !=  mark_token  // default action
      && i->OStack.pick(0) != mark_token)          // is not the only one
    i->OStack.pop();                               // thus pop it!

  bool found =  (i->OStack.pick(pos) == mark_token);


    while( (pos < depth) && ! found )
    {
        i->EStack.push_move(i->OStack.pick(pos));
        found =  (i->OStack.pick(++pos) == mark_token);
    }

    if(found)
        i->OStack.pop(pos+1);
    else
        i->raiseerror(myname ,Name("UnmatchedMark"));
}

void CaseFunction::execute(SLIInterpreter *i) const
{
        // true  obj case -> obj
        // false obj case -> -
        // case is used in combinaion with the switch statement


    if(i->OStack.pick(1) == i->baselookup(i->true_name))
    {
       i->OStack.swap();
       i->OStack.pop();
       i->EStack.pop();
    }
    else if(i->OStack.pick(1) == i->baselookup(i->false_name))
    {
        i->OStack.pop(2);
        i->EStack.pop();
    }
    else
    {
        Name myname(i->getcurrentname());
        i->raiseerror(myname, i->ArgumentTypeError);
    }
}

/*BeginDocumentation
Name: counttomark - count number of objects on the stack from top to marker
Synopsis: mark obj1 ... objn counttomark -> mark obj1 ... objn n
SeeAlso: count
*/
void CounttomarkFunction::execute(SLIInterpreter *i) const
{
    unsigned long depth = i->OStack.load();
    unsigned long pos   = 0;
    Token mark_token(i->baselookup(i->mark_name));

    bool found = false;

    while( (pos < depth) && !found)
    {
        found = (i->OStack.pick(pos) == mark_token);
        ++pos;
    }

    if(found)
    {
        Token it(new IntegerDatum(pos-1));
        i->OStack.push_move(it);
        i->EStack.pop();
    }
    else
    {
        Name myname(i->getcurrentname());
        i->EStack.pop();
        i->raiseerror(myname,Name("UnmatchedMark"));
    }
}

/* BeginDocumentation
 Name: pclocks - returns POSIX clocks for real, user, system time
 Synopsis:  pclocks -> [rclocks uclocks sclocks cuclocks csclocks]
 Description:
 Calls the POSIX times() function to obtain real, user,
 and system time clock counts, as well as user and system time clock
 counts for all children.  Real time clocks have arbitrary origin,
 i.e., only differences are meaningful.
 See man times(3) for details.

 Note: results for user and system time may not be reliable if more
 than one thread is used.

 Author: Hans Ekkehard Plesser
 FirstVersion: 2003-07-29, based on Ptimesfunction
 References: man 2 times
 SeeAlso: pclockspersec, ptimes, realtime, usertime, systemtime, tic, toc
*/

void PclocksFunction::execute(SLIInterpreter *i) const
{
  struct tms foo;
  const clock_t realtime = times(&foo);

  if ( realtime == (clock_t)(-1) )
  {
    i->message(SLIInterpreter::M_ERROR, "PclocksFunction",
	     "System function times() returned error!");
    i->raiseerror(Processes::systemerror(i));
    return;
  }

  Token rtime((long)realtime);
  Token utime((long)foo.tms_utime);
  Token stime((long)foo.tms_stime);
  Token cutime((long)foo.tms_cutime);
  Token cstime((long)foo.tms_cstime);

  ArrayDatum result;
  result.push_back(rtime);
  result.push_back(utime);
  result.push_back(stime);
  result.push_back(cutime);
  result.push_back(cstime);

  i->EStack.pop();
  i->OStack.push(result);
}

/* BeginDocumentation
Name: pclockspersec - POSIX clock ticks per second
Synopsis: pclockspersec -> clockticks
Description:
pclockspersec i an integer variable containing the number of
POSIX clock ticks per second.
Author: Hans Ekkehard Plesser
FirstVersion: 2003-07-29
Remarks: Replaces clockspersecond.
References: man 2 times
SeeAlso: pclocks, ptimes, realtime, usertime, systemtime, tic, toc
*/
void PclockspersecFunction::execute(SLIInterpreter *i) const
{
  const long cps = sysconf(_SC_CLK_TCK);

  if ( cps <= 0 )
  {
    i->message(SLIInterpreter::M_ERROR, "PclockspersecFunction",
	     "This system does not support sysconf(_SC_CLK_TCK)!");
    i->raiseerror("FunctionUnsupported");
    return;
  }

  Token result(cps);
  i->EStack.pop();
  i->OStack.push(result);
}

/* BeginDocumentation
 Name: pgetrusage - Get resource consumption information
 Synopsis:  pgetrusage - selfinfo childinfo
 Description:
 Calls the POSIX getrusage() function to obtain information on
 memory consumption, context switches, I/O operatin count, etc,
 for both the main process and its children.  Information is
 returned in dictionaries.

 Author: Hans Ekkehard Plesser
 FirstVersion: 2003-07-29
 Remarks: At least under Linux, child processes return 0 for all
 entries, while the main process seems to produce meaningfull data
 only for minflt and majflt, i.e., page reclaims and faults.
 References: man 2 getrusage
 SeeAlso: pclockspersec, ptimes, realtime, usertime, systemtime, tic, toc
*/

void PgetrusageFunction::execute(SLIInterpreter *i) const
{
  DictionaryDatum self;
  DictionaryDatum children;

  if ( !getinfo_(RUSAGE_SELF, self) )
  {
    i->message(SLIInterpreter::M_ERROR, "PgetrusageFunction",
	     "System function getrusage() returned error for self!");
    i->raiseerror(Processes::systemerror(i));
    return;
  }

  if ( !getinfo_(RUSAGE_CHILDREN, children) )
  {
    i->message(SLIInterpreter::M_ERROR, "PgetrusageFunction",
	     "System function getrusage() returned error for children!");
    i->raiseerror(Processes::systemerror(i));
    return;
  }

  i->EStack.pop();
  i->OStack.push(self);
  i->OStack.push(children);

}

bool PgetrusageFunction::getinfo_(int who, DictionaryDatum& dict) const
{
  struct rusage data;

  if ( getrusage(who, &data) != 0 )
    return false;

  dict = new Dictionary;
  assert(dict.valid());

  (*dict)["maxrss"] =   data.ru_maxrss;     /* maximum resident set size */
  (*dict)["ixrss"] =    data.ru_ixrss;      /* integral shared memory size */
  (*dict)["idrss"] =    data.ru_idrss;      /* integral unshared data size */
  (*dict)["isrss"] =    data.ru_isrss;      /* integral unshared stack size */
  (*dict)["minflt"] =   data.ru_minflt;     /* page reclaims */
  (*dict)["majflt"] =   data.ru_majflt;     /* page faults */
  (*dict)["nswap"] =    data.ru_nswap;      /* swaps */
  (*dict)["inblock"] =  data.ru_inblock;    /* block input operations */
  (*dict)["oublock"] =  data.ru_oublock;    /* block output operations */
  (*dict)["msgsnd"] =   data.ru_msgsnd;     /* messages sent */
  (*dict)["msgrcv"] =   data.ru_msgrcv;     /* messages received */
  (*dict)["nsignals"] = data.ru_nsignals;   /* signals received */
  (*dict)["nvcsw"] =    data.ru_nvcsw;      /* voluntary context switches */
  (*dict)["nivcsw"] =   data.ru_nivcsw;     /* involuntary context switches */

  return true;
}


/* BeginDocumentation
 Name: time - return wall clock time in s since 1.1.1970 0
 Synopsis:  time -> int
 Description: calls time() and returns seconds since 1.1.1970,
 00:00:00 UTC.  This is mainly meant as a tool to generate random seeds.
 Author: Hans E. Plesser
 FirstVersion: 2001-10-03
 SeeAlso: clock, usertime, tic, toc, sleep
*/

void TimeFunction::execute(SLIInterpreter *i) const
{
  long secs = time(0);
  Token tmp( new IntegerDatum(secs));
  i->EStack.pop();
  i->OStack.push_move(tmp);
}

/*BeginDocumentation
Name: sleep_i - suspends proces for n seconds
Synopsis:  n sleep_i -> -
Description:
Calls the POSIX select() function.
SeeAlso: clock, usertime, tic, toc
*/

void Sleep_iFunction::execute(SLIInterpreter *i) const
{ 
  i->assert_stack_load(1);

  const long sec    =  static_cast<long>(i->OStack.pick(0));  
  const long usec   = 0;
  struct timeval tv = { sec, usec };

  if (sec>0)
    select( 0, 0, 0, 0, &tv );
  
  i->OStack.pop();
  i->EStack.pop(); 
}

/*BeginDocumentation
Name: sleep_i - suspends proces for x seconds
Synopsis:  x sleep_d -> -
Description:
Calls the POSIX select() function.
SeeAlso: clock, usertime, tic, toc
*/

void Sleep_dFunction::execute(SLIInterpreter *i) const
{ 
  i->assert_stack_load(1);

  const long sec    = 0;
  const long usec   = 
    static_cast<long>(static_cast<double>(i->OStack.pick(0))*1000000.);

  struct timeval tv = { sec, usec };

  if (usec>0)
    select( 0, 0, 0, 0, &tv );

  i->OStack.pop();
  i->EStack.pop(); 
}


/*BeginDocumentation
Name: token_s - read a token from a string
Synopsis:  string token_s -> post any true
                            false
References: The Red Book
SeeAlso: token
*/

void  Token_sFunction::execute(SLIInterpreter *i) const
{
  i->EStack.pop();
  assert(i->OStack.load()>0);

  StringDatum *sd= dynamic_cast<StringDatum *>(i->OStack.top().datum());
  assert(sd != NULL);
#ifdef HAVE_SSTREAM
  std::istringstream in(sd->c_str());
#else
  std::istrstream in(sd->c_str());
#endif

  Token t;
  i->parse->clear_context(); // this clears the previously parsed strings.
  i->parse->readToken(in, t);
  if(t.contains(i->parse->scan()->EndSymbol))
  {
    i->OStack.pop();
    i->OStack.push(false);
  }
  else
  {
    std::string s;
    char c;

    i->OStack.push_move(t);
    while(in.get(c))
      s+=c;
    *sd = s;  // this is correct, since sd points to the stringdatum on the
             // ostack.
    i->OStack.push(true);
  }
}
/*BeginDocumentation
Name: token_is - read a token from an input stream
Synopsis:  istream token_is -> istream any true
                              istream false
References: The Red Book
SeeAlso: token
*/

void  Token_isFunction::execute(SLIInterpreter *i) const
{
  i->assert_stack_load(1);

  i->EStack.pop();

  IstreamDatum *sd= dynamic_cast<IstreamDatum *>(i->OStack.top().datum());

  if(sd == NULL)
    throw TypeMismatch("istream", "something else");

  Token t;
  i->parse->readToken(**sd, t);
  if(t.contains(i->parse->scan()->EndSymbol))
  {
    i->OStack.push(false);
  }
  else
  {
    i->OStack.push_move(t);
    i->OStack.push(true);
  }
}

/*BeginDocumentation
Name: symbol_s - read a symbol from a string
Synopsis:  string symbol_s -> post any true
                              false
SeeAlso: token
*/

void  Symbol_sFunction::execute(SLIInterpreter *i) const
{
  i->EStack.pop();
  assert(i->OStack.load()>0);

  StringDatum *sd= dynamic_cast<StringDatum *>(i->OStack.top().datum());
  assert(sd != NULL);
#ifdef HAVE_SSTREAM
  std::istringstream in(sd->c_str());
#else
  std::istrstream in(sd->c_str());
#endif

  Token t;
  i->parse->clear_context(); // this clears the previously parsed strings.
  i->parse->readSymbol(in, t);
  if(t.contains(i->parse->scan()->EndSymbol))
  {
    i->OStack.pop();
    i->OStack.push(false);
  }
  else
  {
    std::string s;
    char c;

    i->OStack.push_move(t);
    while(in.get(c))
      s+=c;
    *sd = s;  // this is correct, since sd points to the stringdatum on the
             // ostack.
    i->OStack.push(true);
  }
}





/* BeginDocumentation
 Name: setguard - limit the number of interpreter cycles
 Synopsis:  n setguard -> --
 Description: This command forces the interpreter to stop after
   it has performed n cycles. setguard is useful for testing programs
   with long running loops.

 Parameters: n : an integer argument greater than zero
 Examples:
 Bugs:
 Author: Gewaltig
 FirstVersion: ?
 Remarks: not part of PostScript
 References:
 SeeAlso: removeguard
*/

void SetGuardFunction::execute(SLIInterpreter *i) const
{
  i->assert_stack_load(1);
  IntegerDatum *count= dynamic_cast<IntegerDatum *>(i->OStack.top().datum());
  assert(count != NULL);
  i->setcycleguard(count->get());
  i->OStack.pop();
  i->EStack.pop();
}


/* BeginDocumentation
 Name: removeguard - removes the limit on the number of interpreter cycles
 Synopsis:  removeguard -> --
 Description: This command removes the restriction on the number
   of possible interpreter cycles, imposed by setguard.

 Parameters: none
 Examples:
 Bugs:
 Author: Gewaltig
 FirstVersion: ?
 Remarks: not part of PostScript
 References:
 SeeAlso: setguard
*/
void RemoveGuardFunction::execute(SLIInterpreter *i) const
{
  i->removecycleguard();
  i->EStack.pop();
}


/*
BeginDocumentation
Name: debugon - Start SLI level debugger.

Description:
The SLI Debug mode allows you to debug SLI programs by tracing
the execution of the individual commands.

In debug mode, each SLI instruction is stepped individually. After
each instruction, the user can see the next command, the contents of
the stack, the entire procedure being executed.

It is also possible to set a breakpoint, so that variables and stacks
in the procedure context can be examined and modified.


The debug mode is controled by the debug level. Whenever execution
enters a procedure or loop, the debug level is increased.
If the procedure is left, the debug level is decreased.
The user can specify a max debug level, beyond which the debug mode
is suspended.
This way, the user can skip over functions which are beyond a certain
level.

The following commands are available:

  next        - Trace (execute) next command.
  continue    - Continue this level without debugging
  step        - Step over deeper levels.
  list        - list current procedure or loop.
  where       - show backtrace of execution stack.
  stack       - show operand stack.
  estack      - show execution stack.
  edit        - enter interactive mode.
  stop        - raise an exception.
  help        - display this list.
  quit        - quit debug mode.
  show next   - show next command.
  show stack  - show operand stack.
  show backtrace- same as 'where'.
  show estack - show execution stack.
  toggle stack     - toggle stack display.
  toggle catch     - toggle debug on error.
  toggle backtrace - toggle stack backtrace on error.
  toggle tailrecursion - toggle tail-recursion optimisation.

Note: This mode is still experimental.
SeeAlso: debugoff, debug
*/
void DebugOnFunction::execute(SLIInterpreter *i) const
{
  std::cerr << "Starting debug mode." << std::endl;
  i->debug_options();
  i->debug_mode_on();
  i->set_max_call_depth(i->get_call_depth()+5);
  i->EStack.pop();
}

/*
BeginDocumentation
Name: debugoff - Stop SLI level debugging mode.
Description:
debugoff is used to quit the debugging mode at a specific position in the code.
Example:

In this example, the parameter assignments as well as the
calculation will be dine in debug mode.

/myproc
{
   << >> begin
   debugon % Enter debugging
   /a Set  % store first parameter
   /b Set  % store second parameter
   a b add a b mul add
   debugoff
   end
} def

Note: This mode is still experimental.
SeeAlso: debugon, debug
*/
void DebugOffFunction::execute(SLIInterpreter *i) const
{
  i->debug_mode_off();
  i->EStack.pop();
}

/*BeginDocumentation
Name: debug - execute an object in debug mode.
Synopsis: any debug -> -
Description: debug tries to execute the object by moving it to
the execution stack.
Examples: {1 2 add} debug -> 3
SeeAlso: exec, debugon, debugoff
*/
void DebugFunction::execute(SLIInterpreter *i) const
{
    if(i->OStack.load()==0)
    {
      i->raiseerror(i->StackUnderflowError);
      return;
    }

    i->EStack.pop();
    i->EStack.push(new NameDatum("debugoff"));
    i->EStack.push_move(i->OStack.top());
    i->EStack.push(new NameDatum("debugon"));
    i->OStack.pop();
}

void SetVerbosityFunction::execute(SLIInterpreter *i) const
{
  assert(i->OStack.load()>0);
  IntegerDatum *count= dynamic_cast<IntegerDatum *>(i->OStack.top().datum());
  assert(count != NULL);
  i->verbosity(count->get());
  i->OStack.pop();
  i->EStack.pop();
}

/*BeginDocumentation
Name: verbosity - return the current verbosity level for interpreter messages
Synopsis: verbosity -> n
SeeAlso: setverbosity, message
*/

void VerbosityFunction::execute(SLIInterpreter *i) const
{
    Token tmp( new IntegerDatum(i->verbosity()));
    i->EStack.pop();
    i->OStack.push_move(tmp);
}

void StartFunction::execute(SLIInterpreter *i) const
{
    i->EStack.clear();
    i->message(SLIInterpreter::M_ERROR, "Start", "Something went wrong "
      "during initialization of NEST or one of its modules. Probably "
      "there is a bug in the startup scripts. Please send the output "
      "of NEST to the nest_user@nest-initiative.org mailing list to help "
      "us to diagnose the problem. You can try to find the bug by "
      "re-starting NEST with the option: --debug");
}

void MessageFunction::execute(SLIInterpreter *i) const
{
        // call : level (from) (message) message

  assert(i->OStack.load() >= 3);

  IntegerDatum *lev= dynamic_cast<IntegerDatum *>(i->OStack.pick(2).datum());
  assert(lev != NULL);
  StringDatum  *frm= dynamic_cast<StringDatum  *>(i->OStack.pick(1).datum());
  assert(frm != NULL);
  StringDatum  *msg= dynamic_cast<StringDatum  *>(i->OStack.pick(0).datum());
  assert(msg != NULL);

  i->message(lev->get(),frm->c_str(),msg->c_str());
  i->OStack.pop(3);
  i->EStack.pop();

}

/*BeginDocumentation
Name: noop - no operation function
Description: This function does nothing. It is used for benchmark purposes.
*/

void NoopFunction::execute(SLIInterpreter *i) const
{
    i->EStack.pop();
}


const SetGuardFunction setguardfunction;
const RemoveGuardFunction removeguardfunction;


const Backtrace_onFunction      backtrace_onfunction;
const Backtrace_offFunction    backtrace_offfunction;
const OStackdumpFunction        ostackdumpfunction;
const EStackdumpFunction        estackdumpfunction;
const LoopFunction             loopfunction;
const ExitFunction             exitfunction;
const QuitFunction             quitfunction;
const IfFunction               iffunction;
const IfelseFunction           ifelsefunction;
const RepeatFunction           repeatfunction;
const CloseinputFunction       closeinputfunction;
const StoppedFunction          stoppedfunction;
const StopFunction             stopfunction;
const CurrentnameFunction      currentnamefunction;
const IparsestdinFunction      iparsestdinfunction;
const ParsestdinFunction       parsestdinfunction;
const IparseFunction           iparsefunction;
const DefFunction              deffunction;
const SetFunction              setfunction;
const LoadFunction             loadfunction;
const LookupFunction           lookupfunction;

const ForFunction              forfunction;
const Forall_aFunction         forall_afunction;
const Forall_iterFunction      forall_iterfunction;
const Forallindexed_aFunction  forallindexed_afunction;
const Forallindexed_sFunction  forallindexed_sfunction;
const Forall_sFunction         forall_sfunction;
const RaiseerrorFunction       raiseerrorfunction;
const PrinterrorFunction       printerrorfunction;
const RaiseagainFunction       raiseagainfunction;

const CyclesFunction           cyclesfunction;
const CodeAccessedFunction     codeaccessedfunction;
const CodeExecutedFunction     codeexecutedfunction;
const ExecFunction             execfunction;
const TypeinfoFunction         typeinfofunction;
const SwitchFunction           switchfunction;
const SwitchdefaultFunction    switchdefaultfunction;
const CaseFunction             casefunction;
const CounttomarkFunction      counttomarkfunction;
const PclocksFunction          pclocksfunction;
const PclockspersecFunction    pclockspersecfunction;
const PgetrusageFunction       pgetrusagefunction;
const TimeFunction             timefunction;
const Sleep_dFunction          sleep_dfunction;
const Sleep_iFunction          sleep_ifunction;

const Token_sFunction          token_sfunction;
const Token_isFunction         token_isfunction;

const Symbol_sFunction         symbol_sfunction;

const SetVerbosityFunction setverbosityfunction;
const VerbosityFunction    verbosityfunction;
const MessageFunction      messagefunction;
const NoopFunction         noopfunction;
const StartFunction        startfunction;
const DebugOnFunction      debugonfunction;
const DebugOffFunction     debugofffunction;
const DebugFunction        debugfunction;

/*BeginDocumentation
Name: mark - puts a mark on the stack

Description: A mark is a token which is lying on the stack and
can be found by the user. The mark is used :
1) by using it as command mark
2) by using it as [ when creating an array
3) by using it as << when creating a dict

Examples: [ 1 2 add ] -> [3]
mark 1 2 add ] -> [3]

Author: docu by Marc Oliver Gewaltig and Sirko Straube
SeeAlso: counttomark, arraystore, switch, case
 */

void  init_slicontrol(SLIInterpreter *i)
{
  // Define the built-in symbols
  i->def(i->true_name,  BoolDatum(true));
  i->def(i->false_name, BoolDatum(false));
  i->def(i->mark_name,  LiteralDatum(i->mark_name));
  i->def(Name("<<"),    LiteralDatum(i->mark_name));
#ifdef PS_ARRAYS
  i->def(Name("["),    LiteralDatum(i->mark_name));
#endif
  i->def(i->istopped_name, BoolDatum(false));

  i->def(i->newerror_name,    BoolDatum(false));
  i->def(i->recordstacks_name,BoolDatum(false));

  i->createcommand("backtrace_on",&backtrace_onfunction);
  i->createcommand("backtrace_off",&backtrace_offfunction);
  i->createcommand("estackdump",  &estackdumpfunction);
  i->createcommand("ostackdump",  &ostackdumpfunction);
  i->createcommand("loop",  &loopfunction);
  i->createcommand("exit",  &exitfunction);
  i->createcommand("quit",  &quitfunction);
  i->createcommand("if",    &iffunction);
  i->createcommand("ifelse",&ifelsefunction);
  i->createcommand("repeat",&repeatfunction);
  i->createcommand("closeinput",&closeinputfunction);
  i->createcommand("stop",&stopfunction);
  i->createcommand("stopped",&stoppedfunction);
  i->createcommand("currentname",&currentnamefunction);
  i->createcommand("parsestdin", &parsestdinfunction);
  i->createcommand(":parsestdin",&iparsestdinfunction);
  i->createcommand("::parse",     &iparsefunction);
  i->createcommand("start",     &startfunction);
  i->createcommand("def",&deffunction);
  i->createcommand("Set",&setfunction);
  i->createcommand("load",&loadfunction);
  i->createcommand("lookup",&lookupfunction);
  i->createcommand("for",&forfunction);
  i->createcommand("forall_a",&forall_afunction);
  i->createcommand("forall_iter",&forall_iterfunction);
  i->createcommand("forallindexed_a",&forallindexed_afunction);
  i->createcommand("forallindexed_s",&forallindexed_sfunction);
  i->createcommand("forall_s",&forall_sfunction);
  i->createcommand("raiseerror",&raiseerrorfunction);
  i->createcommand("print_error",&printerrorfunction);
  i->createcommand("raiseagain",&raiseagainfunction);
  i->createcommand("cycles",&cyclesfunction);
  i->createcommand("code_accessed",&codeaccessedfunction);
  i->createcommand("code_executed",&codeexecutedfunction);
  i->createcommand("exec",&execfunction);
  i->createcommand("typeinfo",&typeinfofunction);
  i->createcommand("switch",&switchfunction);
  i->createcommand("switchdefault",&switchdefaultfunction);
  i->createcommand("case",  &casefunction);
  i->createcommand("counttomark",  &counttomarkfunction);
  i->createcommand("pclocks",         &pclocksfunction);
  i->createcommand("pclockspersec",         &pclockspersecfunction);
  i->createcommand("pgetrusage",         &pgetrusagefunction);
  i->createcommand("time",         &timefunction);
  i->createcommand("sleep_d",         &sleep_dfunction);
  i->createcommand("sleep_i",         &sleep_ifunction);

  i->createcommand("token_s",     &token_sfunction);
  i->createcommand("token_is",     &token_isfunction);

  i->createcommand("symbol_s",     &symbol_sfunction);

  i->createcommand("setguard", &setguardfunction);
  i->createcommand("removeguard", &removeguardfunction);
  i->createcommand("setverbosity_i", &setverbosityfunction);
  i->createcommand("verbosity", &verbosityfunction);
  i->createcommand("message_", &messagefunction);
  i->createcommand("noop", &noopfunction);
  i->createcommand("debug", &debugfunction);
  i->createcommand("debugon", &debugonfunction);
  i->createcommand("debugoff", &debugofffunction);
}