RAPL

Artifact [2953abc805]
Login

Artifact [2953abc805]

Artifact 2953abc805001f8b749fcc746603458dd9d98c7d:


/*====================================================
 * rapl_register_cmd.js "A Tcl like language implementation in Javascript named WebRAPL 
 * (Web Rapid Application Programming Language)"
 *
 * registration of core built in commands for RAPL
 *
 * Released under the same terms as Tcl itself.
 * (BSD license found at <http://www.tcl.tk/software/tcltk/license.html>)
 *
 * Based on Picol by Salvatore Sanfilippo (<http://antirez.com/page/picol>)
 * (c) Stéphane Arnold 2007
 * Richard Suchenwirth 2007: cleanup, additions
 * Arnulf Wiedemann    2011: a lot of additions for Tcl 8.6 commands
 */

RP.add("rapl-pkg-core", function(R, name) {

function PkgCore(interp) {
  R.log('constructor called', '2.life', 'PkgCore', true);
  // kweight
  var pkg_core = this;
  pkg_core.interp = interp;
  var constructor = pkg_core.constructor;
  PkgCore.superclass.constructor.apply(pkg_core, arguments);

  R.Base.pkg_core_oid++;
  pkg_core.oid = R.Base.pkg_core_oid;

  //---------------------------------- Commands in alphabetical order
  /* ==================== command append ===================================== */
  interp.registerCommand("::append", function (interp, args) {
    args.shift();
    this.requireMinArgc(args, 1, "append", "");
    var var_name = args.shift().toString();
    var create_it = 1;
    /* FIXME !!! need handling for array names here */
    var var_obj = interp.global_ns_ptr.lookupVariableEx(interp, var_name, 0, "", 1, 1);
    if (var_obj.isUndefined()) {
      var_obj.setValue("");
    }
    var str = interp.getVarValue(var_name).toString();
    for (var i = 0; i < args.length; i++) {
      str += args[i].toString();
    }
//    interp.setVar(var_name, str);
    var_obj.setValue(str, null);
    return str;
  });

  /* ==================== command break ===================================== */
  interp.registerCommand("::break", function (interp, args) {
    interp.code = this.BREAK;
    return;
  });


  /* ==================== command catch ===================================== */
  interp.registerCommand("::catch", function (interp, args) {
    args.shift();
    this.requireMinArgc(args, 1, "catch", "");
    var body = args.shift();
    var var_obj = null;
    if (args.length > 0) {
        var result_var_name = args.shift();
        var_obj = interp.global_ns_ptr.lookupVariableEx(interp, result_var_name, 0, "", 1, 1);
    }
    try {
      var res = this.eval(body);
      if (var_obj != null) {
        var_obj.setValue(res.toString());
      }
    } catch (e) {
      if (var_obj != null) {
        var_obj.setValue(e);
      }
      return 1;
    }
    return 0;
  });

  /* ==================== command continue ===================================== */
  interp.registerCommand("::continue", function (interp, args) {
    interp.code = this.CONTINUE;
    return;
  });

  /* ==================== command clock format ===================================== */
  interp.registerSubCommand("::clock", "format", function (interp, args) {
    var now = new Date();
    now.setTime(args[1]);
    return now.toString();
  });

  /* ==================== command clock scan ===================================== */
  interp.registerSubCommand("::clock", "scan", function (interp, args) {
    return Date.parse(args[1]);
   });

  /* ==================== command clock seconds ===================================== */
  interp.registerSubCommand("::clock", "seconds", function (interp, args) {
    return (new Date()).valueOf();
  });

  /* ==================== command dom ===================================== */
  if(typeof(jQuery) != 'undefined') {
    interp.registerCommand("::dom", function (interp, args) {
      var selector = args[1].toString();
      var fn = args[2].toString();
      args = args.slice(3);
      for (var i in args) {
        args[i] = args[i].toString();
      }
      var q = $(selector);
      q[fn].apply(q,args);
      return "dom  " + selector;
    });
  }

  /* ==================== command eval ===================================== */
  interp.registerCommand("::eval",function (interp, args) {
    this.requireMinArgc(args, 2, "eval", "");
    for (var i = 1; i < args.length; i++) {
      args[i] = args[i].toString();
    }
    if (args.length == 2) {
      var code = args[1];
    } else {
      var code = args.slice(1).join(" ");
    }
    return interp.eval_statement.eval(code);
  });

  var sqrt = "";
  sqrt = Math.sqrt; // "publish" other Math.* functions as needed

  /* ==================== command expr ===================================== */
  interp.registerCommand("::expr", function (interp, args) {
    var expr_result_ptr = new Array();
    var retcode;

    if (args.length == 2) {
      retcode = interp.eval_expression.evalExpression(args[1], expr_result_ptr);
    } else {
      if (args.length > 2) {
        var obj_ptr;

        obj_ptr = interp.default_obj.concatObj(args.length - 1, args.splice(1));
        obj_ptr.incrRefCount();
        retcode = interp.eval_expression.evalExpression(obj_ptr, expr_result_ptr);
        obj_ptr.decrRefCount();
      } else {
        interp.wrongNumArgs(1, arg, "expression ?...?");
        return interp.ERROR;
      }
    }
    if (retcode != interp.OK) {
      return retcode;
    }
    expr_result_ptr = expr_result_ptr[0];
    interp.setResult(expr_result_ptr);
    expr_result_ptr.decrRefCount();
    return interp.OK;
  });

  /* ==================== command file join ===================================== */
  interp.registerSubCommand("::file", "join", function (interp, args) {
    if (args.length < 2) {
      interp.wrongNumArgs(1, args, "file_name_part ?file_name_part ...?");
      return interp.ERROR;
    }
    var path = args[1];
    for (var i = 2; i < args.length; i++) {
      path += "/"+args[i].toString();
    }
    interp.setResultString(path);
    return interp.OK;
  });

  /* ==================== command for ===================================== */
  interp.registerCommand("::for", function (interp, args) {
    this.requireExactArgc(args, 5, "for", "");
    interp.eval_statement.eval(args[1].toString());
    if(interp.code != this.OK) return;
    var cond = "::set _ "+args[2].toString();
    var step = args[3].toString();
    var body = args[4].toString();
    interp.inLoop = true;
    interp.code = this.OK;
    while (true) {
      test = interp.statement_parser.objectify(interp.eval_statement.eval(cond));
      if (!test.toBoolean()) break;
      interp.eval_statement.eval(body);
      var ic = interp.code; // tested after step command
      interp.eval_statement.eval(step);
      if(ic == this.BREAK) break;
      if(ic == this.CONTINUE) continue;
    }
    interp.inLoop = false;
    if(interp.code == this.BREAK || interp.code == this.CONTINUE)
      interp.code = this.OK;
    return "";
  });

  /* ==================== command foreach ===================================== */
  interp.registerCommand("::foreach", function (interp, args) {
    return interp.foreachMapHelper(interp, args, 0);
  });

  /* ==================== command format ===================================== */
  interp.registerCommand("::format", function (interp, args) {
    this.requireMinArgc(args, 2, "format", "");
    throw "format not yet implemented";
  });

  /* ==================== command gets ===================================== */
  interp.registerCommand("::gets", function (interp, args) {
    this.requireArgcRange(args, 2, 3, "gets", "");
    var reply = prompt(args[1],"");
    if(args[2] != null) {
      interp.setVar(args[2],interp.statement_parser.objectify(reply));
      return reply.length;
    } else return reply;
  });

  /* ==================== command if ===================================== */
  interp.registerCommand("::if", function (interp, args) {
    var idx = 1;
    var bool = new Array();
    var retcode;
    var current = 1;
    var falsebody = 0;
    var err = false;
    var argc = args.length;

    if (args.length >= 3) {
      while (1) {
        /* Far not enough arguments given! */
        if (current >= argc) {
          err = true;
          break;
        }
        if ((retval = interp.eval_expression.getBoolFromExpr(args[current++], bool)) != interp.OK) {
          return retval;
	}
	bool = bool[0];
        /* There lacks something, isn't it? */
        if (current >= argc) {
          err = true;
	  break;
        }
        if (args[current] == "then") {
          current++;
	}
        /* Tsk tsk, no then-clause? */
        if (current >= argc) {
          err = true;
	  break;
	}
        if (bool) {
          return interp.eval_statement.evalObj(args[current]);
	}
        /* Ok: no else-clause follows */
        if (++current >= argc) {
          interp.setResult(intp.string_obj_type.newEmptyStringObj());
          return interp.OK;
        }
        falsebody = current++;
        if (args[falsebody] == "else") {
          /* IIICKS - else-clause isn't last cmd? */
          if (current != argc - 1) {
            err = true;
	    break
	  }
          return interp.eval_statement.evalObj(args[current]);
        } else {
          if (args[falsebody] == "elseif") {
            /* Ok: elseif follows meaning all the stuff
             * again (how boring...) 
	     */
            continue;
	  } else {
            /* OOPS - else-clause is not last cmd? */
            if (falsebody != argc - 1) {
              err = true;
	      break;
	    }
            return interp.eval_statement.evalObj(args[falsebody]);
	  }
	}
      }
      if (!err) {
        return interp.OK;
      }
    }
    interp.wrongNumArgs(1, args, "condition ?then? trueBody ?elseif ...? ?else? falseBody");
    return interp.ERROR;
  });

  /* ==================== command incr ===================================== */
  interp.registerCommand("::incr", function (interp, args) {
    var wide_value;
    var increment = 1;
    var int_obj_ptr;

    if (args.length != 2 && args.length != 3) {
      interp.wrongNumArgs(1, args, "varName ?increment?");
      return interp.ERROR;
    }
    if (args.length == 3) {
      increment = new Array()
      if (interp.int_obj_type.getWide(args[2], increment) != interp.OK) {
        return interp.ERROR;
      }
      increment = increment[0];
    }
    int_obj_ptr = interp.variable_obj_type.getVariable(args[1], interp.FUNCTION_FLAGS_UNSHARED);
    if (!int_obj_ptr) {
      /* Set missing variable to 0 */
      wide_value = 0;
    } else {
      wide_value = new Array();
      if (interp.int_obj_type.getWide(int_obj_ptr, wide_value) != interp.OK) {
        return interp.ERROR;
      }
    }
    if (!int_obj_ptr || int_obj_ptr.isShared()) {
      int_obj_ptr = interp.int_obj_type.newIntObj(parseInt(wide_value, 10) + parseInt(increment, 10));
      if (interp.variable_obj_type.setVariable(args[1], int_obj_ptr) != interp.OK) {
        int_obj_ptr.freeNewObj();
        return interp.ERROR;
      }
    } else {
      /* Can do it the quick way */
      int_obj_ptr.invalidateStringRep();
      int_obj_ptr.wideValue(parseInt(wide_value, 10) + parseInt(increment));
      /* The following step is required in order to invalidate the
       * string repr of "FOO" if the var name is on the form of "FOO(IDX)" 
       */
      if (args[1].obj_type != interp.variable_obj_type) {
        /* Note that this can't fail since GetVariable already succeeded */
        interp.variable_obj_type.setVariable(args[1], int_obj_ptr);
      }
    }
    interp.setResult(int_obj_ptr);
    return interp.OK;
  });

  /* ==================== command interp alias ===================================== */
  interp.registerSubCommand("::interp", "alias", function (interp, args) {
    args.shift();
    this.requireMinArgc(args, 2, "interp alias", "");
    var src_path = args.shift().toString();
    var src_cmd = args.shift().toString();
    if (args.length == 0) {
      /* return the target_cmd */
      if (src_path != "") {
        throw "interp alias interp other current interp not yet implemented, src_path must be empty!";
      }
      if (src_path == "") {
        src_path = interp;
      }
      var result_str = "";
      for (var i = 0; i < interp.aliases.length; i++) {
        var my_alias = interp.aliases[i];
	if (my_alias.src_path == src_path) {
          if (my_alias.src_cmd == src_cmd) {
            result_str = my_alias.target_cmd;
	    break;
	  }
	}
      }
      return result_str;
    }
    if (args.length == 1) {
      /* seems to be the unset, so the argument has to be empty */
      var arg3 = args.shift();
      if (src_path != "") {
        throw "interp alias interp other current interp not yet implemented, src_path must be empty!";
      }
      if (src_path == "") {
        src_path = interp;
      }
      var my_aliases = new Array();;
      var found = false;
      for (var i = 0; i < interp.aliases.length; i++) {
        var my_alias = interp.aliases[i];
	if ((my_alias.src_path != src_path) || (my_alias.src_cmd != src_cmd)) {
          my_aliases.push(my_alias);
	} else {
          found = true;
	}
      }
      if (found) {
        interp.aliases = my_aliases;
      } else {
	if (src_path == interp) {
          src_path = "{}";
	} else {
          throw "multiple interps not yet implemented!"
	}
        throw "could not unset alias: \""+src_path+" "+src_cmd+"\" does not exist";
      }
      return;
    }
    if (args.length < 2) {
      throw "usage: interp alias srcPath srcCmd targetPath targetCmd ?arg ...?";
    }
    var target_path = args.shift().toString();
    var target_cmd = args.shift().toString();
    if ((src_path != "") || (target_path != "")) {
      throw "interp alias interp other current interp not yet implemented, src_path and target_path must be empty!";
    }
    if (src_path == "") {
      src_path = interp;
    }
    if (target_path == "") {
      target_path = interp;
    }
    interp.aliases.push(new R.InterpAlias(src_path, src_cmd, target_path, target_cmd, args));
  });

  /* ==================== command interp aliases ===================================== */
  interp.registerSubCommand("::interp", "aliases", function (interp, args) {
    args.shift();
    this.requireArgcRange(args, 0, 1, "interp aliases", "");
    var src_path = null;
    if (args.length > 0) {
      src_path = args.shift().toString();
      if (src_path != "") {
        throw "interp alias interp other current interp not yet implemented, src_path must be empty!";
      }
      if (src_path == "") {
        src_path = interp;
      }
    }
    var result_str = "";
    var sep = "";
    for (var i = 0; i < interp.aliases.length; i++) {
      var my_alias = interp.aliases[i];
      if ((src_path == null) || (my_alias.src_path == src_path)) {
        result_str += sep+my_alias.src_cmd;
	sep = " ";
      }
    }
    return result_str;
  });

  /* ==================== command join ===================================== */
  interp.registerCommand("::join", function (interp, args) {
    var join_str;
    var join_str_len;
    var i;
    var list_len;
    var res_obj_ptr;

    if (args.length != 2 && args.length != 3) {
      interp.wrongNumArgs(1, args, "list ?joinString?");
      return interp.ERROR;
    }
    /* Init */
    if (args.length == 2) {
      join_str = " ";
      join_str_len = 1;
    } else {
      join_str = args[2].getString();
      join_str_len = args[2].getStringLength();
    }
    list_len = interp.list_obj_type.listLength(args[1]);
    res_obj_ptr = interp.string_obj_type.newStringObj(null, 0);
    /* Split */
    for (i = 0; i < list_len; i++) {
      var obj_ptr = new Array();

      obj_ptr[0] = null;
      interp.list_obj_type.listIndex(args[1], i, obj_ptr, interp.FUNCTION_FLAGS_NONE);
      obj_ptr = obj_ptr[0];
      interp.string_obj_type.appendObj(res_obj_ptr, obj_ptr);
      if (i + 1 != list_len) {
        interp.string_obj_type.appendString(res_obj_ptr, join_str, join_str_len);
      }
    }
    interp.setResult(res_obj_ptr);
    return interp.OK;
  });

  /* ==================== command jsdebug ===================================== */
  interp.registerCommand("::jsdebug", function (interp, args) {
    var val = 1;
    var type = args[1].toString();
    if (args.length > 1) {
      val = parseInt(args[2].getString());
    }
    switch(type) {
    case "eval_subst_debug":
print("evsubst!"+typeof interp.eval_statement.eval_subst_debug+"!"+val+"!");
      interp.eval_statement.eval_subst_debug = val;
      break;
    case "eval_ipol_debug":
print("evipol!"+typeof interp.eval_statement.eval_ipol_debug+"!");
      interp.eval_statement.eval_ipol_debug = val;
      break;
    case "eval_cmd_debug":
      interp.eval_statement.eval2_cmd_debug = val;
      break;
    case "expr_debug":
      interp.eval_statement.expr_debug = val;
      break;
    case "expr_parser_debug":
      interp.eval_statement.expr_parser_debug = val;
      break;
    case "class_command_debug":
      this.class_command_debug = 1;
      break;
    default:
      throw "bad type for js_debug: \""+type+"\"";
    }
print("jsdebug set: "+type+"!"+val+"!");
    return interp.OK;
  });

  /* ==================== command jseval ===================================== */
  interp.registerCommand("::jseval", function (interp, args) {
    args.shift();
    var result = eval(args[0].toString());
    return result;
  });

  /* ==================== command puts ===================================== */
  interp.registerCommand("::puts", function (interp, args) {
    if (args.length != 2 && args.length != 3) {
       interp.wrongNumArgs(1, args, "?-nonewline? string");
        return interp.ERROR;
    }
    if (args.length == 3) {
      if (args[1] != "-nonewline") {
        interp.setResultString("The second argument must be -nonewline", -1);
        return interp.ERROR;
      } else {
        print(args[2]);
      }
    } else {
      print(args[1]);
    }
    return interp.OK;
  });

  /* ==================== command proc ===================================== */
  interp.registerCommand("::proc", function (interp, args) {
    var full_name;
    var ns_ptr = new Array();
    var alt_ns_ptr = new Array();
    var cxt_ns_ptr = new Array();
    var proc_name = new Array();

    if (args.length != 4 && args.length != 5) {
      interp.wrongNumArgs(1, args, "name arglist ?statics? body");
      return interp.ERROR;
    }
    /*
     * Determine the namespace where the procedure should reside. Unless the
     * command name includes namespace qualifiers, this will be the current
     * namespace.
     */
    full_name = args[1].getString();
    interp.variable_obj_type.getNamespaceForQualName(full_name, null, 0, ns_ptr, alt_ns_ptr, cxt_ns_ptr, proc_name);
    ns_ptr = ns_ptr[0];
    alt_ns_ptr = alt_ns_ptr[0];
    cxt_ns_ptr = cxt_ns_ptr[0];
    proc_name = proc_name[0];
    if (ns_ptr == null) {
      interp.setResultString("can't create procedure \""+full_name+"\": unknown namespace");
      interp.setErrorCode(["TCL", "VALUE", "COMMAND"]);
      return interp.ERROR;
    }
    if ((ns_ptr != interp.global_ns_ptr) && (proc_name != null) && (proc_name.charAt(0) == ':')) {
      interp.setResultString("can't create procedure \""+proc_name+"\" in non-global namespace with name starting with \":\"");
      interp.setErrorCode(["TCL", "VALUE", "COMMAND"]);
      return interp.ERROR;
    }
    if (args.length == 4) {
      return interp.command_obj_type.createProcedure(args[1], args[2], null, args[3], ns_ptr);
    } else {
      return interp.command_obj_type.createProcedure(args[1], args[2], args[3], args[4], ns_ptr);
    }
  });

  /* ==================== command regexp ===================================== */
  interp.registerCommand("::regexp", function (interp, args) {
    var regcomp_flags = 0;
    var regexp;
    var i = 1;
    var source_str;
    var match;

    if (args.length < 3) {
      interp.wrongNumArgs(1, args, "?switches? exp string ?matchVar? ?subMatchVar subMatchVar ...?");
      return interp.ERROR;
    }
    regexp = interp.regexp_obj_type.setFromAny(args[i], regcomp_flags);
    if (regexp == null) {
      return interp.ERROR;
    }
    pattern = args[i];
    source_str = args[i + 1].getString();
    i++;
    i++;
    match = regexp.exec(source_str);
    if (match == null) {
      interp.setResultBool(interp.false_obj);
      return interp.OK;
    }
    var j = 0;
    for (; i < match.length; i++) {
      if (args.length > i) {
        interp.variable_obj_type.setVariable(args[i], interp.string_obj_type.newStringObj(match[j], -1));
	j++;
      }
    }
    interp.setResultBool(interp.true_obj);
    return interp.OK;

  });

  /* ==================== command regsub ===================================== */
  interp.registerCommand("::regsub", function (interp, args) {
    this.requireMinArgc(args, 2, "regsub", "");
    throw "regsub not yet implemented";
  });

  /* ==================== command rename ===================================== */
  interp.registerCommand("::rename", function (interp, args) {
    this.requireExactArgc(args, 3, "rename", "");
    interp.renameCommand(args[1], args[2]);
  });

  /* ==================== command return ===================================== */
  interp.registerCommand("::return", function (interp, args) {
    var core_cmd = interp.core_pkg;
    var errorcode = null;
    var errorinfo = null;
    var level = 1;
    var return_code = interp.OK;
    var options = null;
    var stack_trace_obj = null;
    var error_code_obj = null;
    var result = new Array();
    var r = "";
    var option_flags = new Array();
    var options = new Array();
    var error_msg = new Array();
    var processed_args = new Array();

    interp.eval_statement.code = interp.RETURN;
    if (args.length == 1) {
      /* seems to be a return value only */
      r = args.shift();
    }
    if (interp.getCallOptions(core_cmd.return_options, args.slice(1), "return", option_flags, options, error_msg, processed_args) != interp.OK) {
       interp.setResultString(error_msg[0]);
       return interp.ERROR;
    }
    processed_args = processed_args[0];
    option_flags = option_flags[0];
    options = options[0];
    if (args.length - processed_args > 2) {
      interp.setResultString("usage: return ?option value ...? ?result?");
      return interp.ERROR;
    }
    /* take care of command as args[0] */
    processed_args++;
    if (option_flags & core_cmd.return_options.RETURN_OPT_ERRORCODE) {
      error_code_obj = options["-errorcode"] ;
      throw "return -errorcode not yet implemented"
    }
    if (option_flags & core_cmd.return_options.RETURN_OPT_CODE) {
      code = options["-code"];
      if (interp.return_code_obj_type.getReturnCode(code, result) == interp.ERROR) {
        return interp.ERROR;
      }
      return_code = result[0];
    }
    if (option_flags & core_cmd.return_options.RETURN_OPT_ERRORINFO) {
      error_info_obj = options["-errorinfo"] ;
    }
    if (option_flags & core_cmd.return_options.RETURN_OPT_LEVEL) {
      var level_obj = options["-level"];

      level = new Array();
      if (interp.int_obj_type.getLong(level_obj, level) != interp.OK || level[0] < 0) {
        interp.setResultFormatted("bad level \"%#s\"", level_obj);
        return interp.ERROR;
      }
    }
    if (option_flags & core_cmd.return_options.RETURN_OPT_OPTIONS) {
      options = options["-options"];
      interp.setResultString("return -options not yet implemented");
      return interp.ERROR;
    }
//    if (processed_args != args.length - 1 && processed_args != args.length) {
//      interp.wrongNumArgs(1, args, "?-code code? ?-errorinfo stacktrace? ?-level level? ?result?");
//      return interp.ERROR;
//    }
    /* If a stack trace is supplied and code is error, set the stack trace */
    if (stack_trace_obj && return_code == interp.ERROR) {
      interp.setStackTrace(stack_trace_obj);
    }
    /* If an error code list is supplied, set the global $errorCode */
    if (error_code_obj && return_code == interp.ERROR) {
      interp.variable_obj_type.setGlobalVariableStr("errorCode", error_code_obj);
    }
    interp.return_code = return_code;
    interp.return_level = level;
    if (processed_args == args.length - 1) {
      interp.setResult(args[processed_args]);
    }
    return interp.RETURN;
  });

  /* ==================== command scan ===================================== */
  interp.registerCommand("::scan", function (interp, args) {
    this.requireMinArgc(args, 2, "scan", "");
    throw "scan not yet implemented";
  });

  /* ==================== command set ===================================== */
  interp.registerCommand("::set", function (interp, args) {
    if ((args.length != 2) && (args.length != 3)) {
      interp.wrongNumArgs(interp, 1, args, "varName ?newValue?");
      return interp.ERROR;
    }
    if (args.length == 2) {
      var obj_ptr;

      obj_ptr = interp.variable_obj_type.getVariable(args[1], interp.FUNCTION_FLAGS_LEAVE_ERR_MSG);
      if (obj_ptr == null) {
        return interp.ERROR;
      }
      interp.setResult(obj_ptr);
      return interp.OK;
    }
    /* args.length == 3 */
    if (interp.variable_obj_type.setVariable(args[1], args[2]) != interp.OK) {
       return interp.ERROR;
    }
    interp.setResult(args[2]);
    return interp.OK;
  });

  /* ==================== command source ===================================== */
  interp.registerCommand("::source", function (interp, args) {
    if (args.length != 2) {
      interp.wrongNumArgs(1, args, 1, "filename");
    }
    var source_file = args[1];
    return interp.source(interp, source_file);
  });

  /* ==================== command switch ===================================== */
  interp.registerCommand("::switch", function (interp, args) {
    var usage = '?-switch ...? string {?pattern body ...? ?default body?}';
    if (args.length < 3) {
      interp.wrongNumArgs(1, args, usage);
      return interp.ERROR;
    }
    var body_start_idx = 1;
    var have_exact_opt = 0;
    var have_glob_opt = 0;
    var have_regexp_opt = 0;
    var have_command_opt = 0;
    var have_nocase_opt = 0;
    var have_matchvar_opt = 0;
    var have_indexvar_opt = 0;
    var match_var = null;
    var index_var = null;
    var switch_string;
    var body;
    var obj;
    var text;
    var parser;
    var pattern;
    var token;
    var match_body = null;
    var have_match;
    var default_body = null;
    var re = new RegExp("^[-].*");

    for (body_start_idx = 1; body_start_idx < args.length; body_start_idx++) {
      my_arg = args[body_start_idx].toString();
      if (re.test(my_arg)) {
        if (my_arg == "-exact") {
          if (have_command_opt != 0) {
            interp.setResultFormatted('bad option "-exact": -command option already found');
	    return interp.ERROR;
	  }
          if (have_exact_opt != 0) {
            interp.setResultFormatted('bad option "-exact": -exact option already found');
	    return interp.ERROR;
          }
          if (have_glob_opt != 0) {
            interp.setResultFormatted('bad option "-exact": -glob option already found');
	    return interp.ERROR;
          }
          if (have_regexp_opt != 0) {
            interp.setResultFormatted('bad option "-exact": -regexp option already found');
	    return interp.ERROR;
          }
          have_exact_opt = 1;
        }
        if (my_arg == "-glob") {
          if (have_command_opt != 0) {
            interp.setResultFormatted('bad option "-glob": -command option already found');
	    return interp.ERROR;
          }
          if (have_exact_opt != 0) {
            interp.setResultFormatted('bad option "-glob": -exact option already found');
	    return interp.ERROR;
          }
          if (have_regexp_opt != 0) {
            interp.setResultFormatted('bad option "-glob": -regexp option already found');
	    return interp.ERROR;
          }
          if (have_glob_opt != 0) {
            interp.setResultFormatted('bad option "-glob": -glob option already found');
	    return interp.ERROR;
          }
          have_glob_opt = 1;
        }
        if (my_arg == "-regexp") {
          if (have_command_opt != 0) {
            interp.setResultFormatted('bad option "-regexp": -command option already found');
	    return interp.ERROR;
          }
          if (have_regexp_opt != 0) {
            interp.setResultFormatted('bad option "-regexp": -regexp option already found');
	    return interp.ERROR;
          }
          if (have_exact_opt != 0) {
            interp.setResultFormatted('bad option "-regexp": -exact option already found');
	    return interp.ERROR;
          }
          if (have_glob_opt != 0) {
            interp.setResultFormatted('bad option "-regexp": -glob option already found');
	    return interp.ERROR;
          }
          have_regexp_opt = 1;
        }
        if (my_arg == "-command") {
          if (have_command_opt != 0) {
            interp.setResultFormatted('bad option "-command": -command option already found');
	    return interp.ERROR;
          }
          if (have_regexp_opt != 0) {
            interp.setResultFormatted('bad option "-command": -regexp option already found');
	    return interp.ERROR;
          }
          if (have_exact_opt != 0) {
            interp.setResultFormatted('bad option "-command": -exact option already found');
	    return interp.ERROR;
          }
          if (have_glob_opt != 0) {
            interp.setResultFormatted('bad option "-command": -glob option already found');
	    return interp.ERROR;
          }
          have_command_opt = 1;
        }
        if (my_arg == "-nocase") {
          if (have_nocase_opt != 0) {
            interp.setResultFormatted('bad option "-nocase": -nocase option already found');
	    return interp.ERROR;
          }
          have_nocase_opt = 1;
        }
        if (my_arg == "-matchvar") {
          if (have_matchvar_opt != 0) {
            interp.setResultFormatted('bad option "-matchvar": -matchvar option already found');
	    return interp.ERROR;
          }
          if (have_regexp_opt == 0) {
            interp.setResultFormatted('-matchvar option requires -regexp option');
	    return interp.ERROR;
          }
          have_matchvar_opt = 1;
          if (body_start_idx > args.length-1) {
            interp.wrongNumArgs(1, args, usage);
	    return interp.ERROR;
          }
          body_start_idx++;
          matchvar = args[body_start_idx].toString();
        }
        if (my_arg == "-indexvar") {
          if (have_indexvar_opt != 0) {
            interp.setResultFormatted('bad option "-indexvar": -indexvar option already found');
	    return interp.ERROR;
          }
          if (have_regexp_opt == 0) {
            interp.setResultFormatted('-indexvar option requires -regexp option');
	    return interp.ERROR;
          }
          have_indexvar_opt = 1;
          if (body_start_idx > args.length-1) {
            interp.wrongNumArgs(1, args, usage);
	    return interp.ERROR;
          }
          body_start_idx++;
          indexvar = args[body_start_idx].toString();
        }
        if (my_arg == "--") {
          body_start_idx++;
          break;
        }
      } else {
        break;
      }
    }
    if (body_start_idx == args.length - 1) {
      interp.wrongNumArgs(1, args, usage);
      return interp.ERROR;
    }
    if ((have_exact_opt == 0) && (have_glob_opt == 0) && (have_regexp_opt == 0)) {
      have_exact_opt = 1;
    }
    /* if we have one argument left, it must be a list, otherwise we have the
     * list elements in the args array
     */
    var case_list;
    var i;
    var script = 0;
    var str_obj = args[body_start_idx++];
    var pat_count = args.length - body_start_idx;
    if (pat_count == 1) {
      var vector = new Array();

      pat_count = new Array();
      interp.list_obj_type.listGetElements(args[args.length-1], pat_count, vector);
      pat_count = pat_count[0];
      case_list = vector[0];
    } else {
      case_list = args.slice(body_start_idx);
    }
    if (pat_count == 0 || pat_count % 2 != 0) {
      interp.wrongNumArgs(1, args, usage);
      return interp.ERROR;
    }
    have_match = 0;
    for (i = 0; script == 0 && i < pat_count; i += 2) {
      var pat_obj = case_list[i];

      if (pat_obj.getString() != "default" || i < (pat_count - 2)) {
        if (have_exact_opt) {
          if (str_obj.getString() == pat_obj.getString()) {
            script = case_list[i + 1];
	  }
	}
        if (have_glob_opt) {
          if (pat_obj.stringMatchObj(str_obj, 0)) {
            script = case_list[i + 1];
	  }
	}
        if (have_regexp_opt) {
          command = interp.string_obj_type.newStringObj("regexp", -1);
          have_command_opt = 1;
	}
	var subst_key_obj_ptr = new Array();
        if (interp.script_obj_type.substObj(str_obj, subst_key_obj_ptr, interp.FUNCTION_FLAGS_NONE) != interp.OK) {
	  return interp.ERROR;
	}
	subst_key_obj_ptr = subst_key_obj_ptr[0];
        if (have_command_opt) {
          
          var rc = interp.eval_statement.commandMatchObj(command, pat_obj, subst_key_obj_ptr, 0);
          /* After the execution of a command we need to
           * make sure to reconvert the object into a list
           * again. Only for the single-list style [switch]. */
          if (args.length - i == 1) { 
            var vector = new Array();
	    var count = new Array();

            interp.list_obj_type.listGetElements(args[opt], count, vector);
	    pat_count = count[0];
            case_list = vector[0];
          }
          /* command is here already decref'd */
          if (rc < 0) {
//            return -rc;
            continue;
          }
          if (rc > 0) {
            script = case_list[i + 1];
	  }
        }
      } else {
        script = case_list[i + 1];
      }
    }
    for (; i < pat_count && script.getString() == "-"; i += 2) {
      script = case_list[i + 1];
    }
    if (script && script.getString() == "-") {
        interp.setResultFormatted("no body specified for pattern \"%#s\"", case_list[i - 2]);
        return interp.ERROR;
    }
    interp.setEmptyResult();
    if (script) {
      return interp.eval_statement.evalObj(script);
    }
    return interp.OK;
  });

  /* ==================== sec_msec ===================================== */
  function sec_msec () {
    var t = new Date();
    return t.getSeconds()*1000 + t.getMilliseconds();
  }

  /* ==================== command time ===================================== */
  interp.registerCommand("::time", function (interp, args) {
    if ((args.length != 2) && (args.length != 3)) {
      interp.wrongNumArgs(interp, 1, args, "command ?count?");
      return interp.ERROR;
    }
    var code = args[1];
    if (args.length > 2) {
      var n = args[2];
    } else {
      var n = 1;
    }
    var t0 = sec_msec();
//var xx = interp.string_obj_type.newStringObj("10", -1);
//var yy = interp.string_obj_type.newStringObj("a", -1);
    for(var i = 0; i < n; i++) {
      interp.eval_statement.evalObj(code);
//        interp.setVar("a", xx);
//        interp.variable_obj_type.setVariable(yy, xx);
    }
//print("TT!"+((sec_msec()-t0)*1000/n + " microseconds per iteration")+"!");
    interp.setResult(interp.string_obj_type.newStringObj((sec_msec()-t0)*1000/n + " microseconds per iteration", -1));
    return interp.OK;
  });

  /* ==================== command uplevel ===================================== */
  interp.registerCommand("::uplevel", function (interp, args) {
    if (args.length >= 2) {
      var retcode;
      var saved_call_frame;
      var target_call_frame;
      var obj_ptr;
      var str;
      var argc = args.length;
      var i = 1;

      /* Save the old call_frame pointer */
      saved_call_frame = interp.frame_ptr;
      /* Lookup the target frame pointer */
      str = args[1].getString();
      if ((str.charAt(0) >= '0' && str.charAt(0) <= '9') || strcharAt(0) == '#') {
        target_call_frame = interp.getCallFrameByLevel(args[i]);
        argc--;
	i++;
      } else {
        target_call_frame = interp.getCallFrameByLevel(null);
      }
      if (target_call_frame == null) {
        return interp.ERROR;
      }
      if (argc < 2) {
        i--;
        interp.wrongNumArgs(1, args, "?level? command ?arg ...?");
        return interp.ERROR;
      }
      /* Eval the code in the target call_frame. */
      interp.frame_ptr = target_call_frame;
      if (argc == 2) {
        retcode = interp.eval_statement.evalObj(args[i]);
      } else {
        obj_ptr = interp.default_obj.concatObj(argc - 1, args.slice(i));
        obj_ptr.incrRefCount();
        retcode = interp.eval_statement.evalObj(obj_ptr);
        obj_ptr.decrRefCount();
      }
      interp.frame_ptr = saved_call_frame;
      return retcode;
    } else {
      interp.wrongNumArgs(1, args, "?level? command ?arg ...?");
      return interp.ERROR;
    }
  });

  /* ==================== command upvar ===================================== */
  interp.registerCommand("::upvar", function (interp, args) {
    var i;
    var target_call_frame;
    var idx = 1;

    /* Lookup the target frame pointer */
    if (args.length > 3 && (args.length % 2 == 0)) {
      target_call_frame = interp.getCallFrameByLevel(args[idx]);
      idx++;
    } else {
      target_call_frame = interp.getCallFrameByLevel(null);
    }
    if (target_call_frame == null) {
      return interp.ERROR;
    }
    /* Check for arity */
    if (idx < args.length - 2) {
      interp.wrongNumArgs(1, args, "?level? otherVar localVar ?otherVar localVar ...?");
      return interp.ERROR;
    }
    /* Now... for every other/local couple: */
    for (i = idx; i < args.length; i += 2) {
      if (interp.variable_obj_type.setVariableLink(args[i + 1], args[i], target_call_frame) != interp.OK) {
        return interp.ERROR;
      }
    }
    return interp.OK;
  });

  /* ==================== command unknown ===================================== */
  interp.registerCommand("::unknown", function (interp, args) {
    interp.setResultString("no such command: \""+args[1]+"\"");
    return interp.ERROR;
  });

  /* ==================== command unset ===================================== */
  interp.registerCommand("::unset", function (interp, args) {
    this.requireExactArgc(args, 2, "unset", "");
    interp.setVar(args[1], null);
  });

  /* ==================== command while ===================================== */
  interp.registerCommand("::while", function (interp, args) {
    var out = false;

    if (args.length != 3) {
      interp.wrongNumArgs(1, args, "condition body");
      return interp.ERROR;
    }
    /* The general purpose implementation of while starts here */
    while (1) {
      var bool = new Array();
      var ret_code;

      if ((ret_code = interp.eval_expression.getBoolFromExpr(args[1], bool)) != interp.OK) {
        return ret_code;
      }
      bool = bool[0];
      if (!bool) {
        break;
      }
      if ((ret_code = interp.eval_statement.evalObj(args[2])) != interp.OK) {
        switch (ret_code) {
        case interp.BREAK:
          out = true;
          break;
        case interp.CONTINUE:
          continue;
          break;
        default:
          return ret_code;
        }
      }
      if (out) {
        break;
      }
    }
    interp.setEmptyResult();
    return interp.OK;
  });

  /* ==================== command variable ===================================== */
  interp.registerCommand("::variable", function (interp, args) {
    var var_name;
    var tail;
    var i;
    var j;
    var cp;
    var var_ptr;
    var array_ptr = new Array();
    var var_value_ptr;
    var i;
    var result;
    var var_name_ptr;
    var tail_ptr;

    if (args.length < 2) {
      interp.wrongNumArgs(interp, 1, args, "variable ?value variable value ...?");
      return interp.ERROR;
    }
    for (i = 1 ; i < args.length ; i += 2) {
      /*
       * Look up each variable in the current namespace context, creating it
       * if necessary.
       */
      var_name_ptr = args[i];
      var_name = var_name_ptr.getString();
      var_ptr = interp.variable_obj_type.objLookupVarEx(var_name_ptr, null,
        (interp.NAMESPACE_ONLY | interp.FUNCTION_FLAGS_LEAVE_ERR_MSG), "define",
        /*createPart1*/ 1, /*createPart2*/ 0, array_ptr);
      if (array_ptr[0] != null) {
        /*
         * Variable cannot be an element in an array. If arrayPtr is
         * non-NULL, it is, so throw up an error and return.
         */
        interp.objVarErrMsg(var_name_ptr, null, "define", is_array_element, -1);
        interp.setErrorCode(["TCL", "UPVAR", "LOCAL_ELEMENT"]);
        return interp.ERROR;
      }
      if (var_ptr == null) {
        return interp.ERROR;
      }
      /*
       * Mark the variable as a namespace variable and increment its
       * reference count so that it will persist until its namespace is
       * destroyed or until the variable is unset.
       */
      interp.variable_obj_type.setVarNamespaceVar(var_ptr);
      /*
       * If a value was specified, set the variable to that value.
       * Otherwise, if the variable is new, leave it undefined. (If the
       * variable already exists and no value was specified, leave its value
       * unchanged; just create the local link if we're in a RAPL procedure).
       */
      if (i + 1 < args.length) {	/* A value was specified. */
        var_value_ptr = interp.variable_obj_type.ptrSetVar(var_ptr, array_ptr, var_name_ptr, null, args[i + 1],
            interp.NAMESPACE_LOOKUP_NAMESPACE_ONLY | interp.FUNCTION_FLAGS_LEAVE_ERR_MSG, -1);
        if (var_value_ptr == null) {
          return interp.ERROR;
        }
      }
      /*
       * If we are executing inside a Rapl procedure, create a local variable
       * linked to the new namespace variable "var_name".
       */
      if (interp.var_frame_ptr.hasLocalVars()) {
        /*
         * var_name might have a scope qualifier, but the name for the
         * local "link" variable must be the simple name at the tail.
         *
         * Locate tail in one pass: drop any prefix after two *or more*
         * consecutive ":" characters).
         */
        tail = var_name;
        i = 0;
        j = i;
        for (; i < var_name.length; i++) {
          if (tail.charAt(i) == ':') {
	    j = i;
            while (tail.charAt(j) == ':') {
	      j++;
            }
          }
        }
        tail = tail.substring(j);
        /*
         * Create a local link "tail" to the variable "var_name" in the
         * current namespace.
         */
        if (tail == var_name) {
          tail_ptr = var_name_ptr;
        } else {
	  tail_ptr = interp.string_obj_type.newStringObj(tail, -1);
	  tail_ptr.incrRefCount();
        }
        result = interp.variable_obj_type.objMakeUpvar(null, var_name_ptr, /*other_p2*/ null,
	      /*other_flags*/ interp.NAMESPACE_LOOKUP_NAMESPACE_ONLY,
	      /*my_name*/ tail_ptr, /*my_flags*/ 0, -1);
        if (tail != var_name) {
	  tail_ptr.decrRefCount();
        }
        if (result != interp.OK) {
          return result;
        }
      }
    }
    return interp.OK;
  });

  /* ==================== command native ===================================== */
  // native cmdname {function(interp, args) {...}}
  interp.registerCommand("::native", function (interp, args) {
    this.requireExactArgc(args, 3, "native", "");
    var cmd = args[1].toList();
    var func = eval(args[2].toString());
//print("in: "+args[2].toString()+", func: "+ func);
    if (cmd.length == 1) {
      interp.registerCommand(cmd[0].toString(), func);
      return;
    }
    base = cmd[0].toString();
    cmd.shift();
    interp.registerSubCommand(base, cmd.join(" "), eval(args[2].toString()));
    return;
  });

  /* ==================== command showLevelInfo ===================================== */
  interp.registerCommand("::showLevelInfo", function (interp, args) {

    args.shift();
    this.requireMinArgc(args, 1, "showLevelInfo", "");
    var txt = "";
    if (args.length > 0) {
      txt = args.shift();
    }
    for (var i = interp.eval_statement.level-1; i >= 0; i--) {
      print("LEVEL_INFO!"+i+"!"+txt+"!"+interp.level_infos[i]+"!");
    }
  });

  R.log('constructor end', '2.life', 'PkgCore', true);
}

var RETURN_OPT_CODE = 1;
var RETURN_OPT_ERRORCODE = 2;
var RETURN_OPT_ERRORINFO = 4;
var RETURN_OPT_LEVEL = 8;
var RETURN_OPT_OPTIONS = 16;

R.extend(PkgCore, R.Token, {
  my_name: "PkgCore",

  return_options: {
    RETURN_OPT_CODE: 1,
    "1": "-code",
    "-code": {    
      opt_name: "-code",
      opt_code: RETURN_OPT_CODE,
      opt_val: 1,
    },
    RETURN_OPT_ERRORCODE: 2,
    "2": "-errorcode",
    "-errorcode": {    
      opt_name: "-errorcode",
      opt_code: RETURN_OPT_ERRORCODE,
      opt_val: 1,
    },
    RETURN_OPT_ERRORINFO: 4,
    "4": "-errorinfo",
    "-errorinfo": {    
      opt_name: "-errorinfo",
      opt_code: RETURN_OPT_ERRORINFO,
      opt_val: 1,
    },
    RETURN_OPT_LEVEL: 8,
    "8": "-level",
    "-level": {    
      opt_name: "-level",
      opt_code: RETURN_OPT_LEVEL,
      opt_val: 1,
    },
    RETURN_OPT_OPTIONS: 16,
    "16": "-options",
    "-options": {    
      opt_name: "-options",
      opt_code: RETURN_OPT_OPTIONS,
      opt_val: 1,
    },
  },

});

PkgCore.prototype.constructor = PkgCore;

R.PkgCore = PkgCore;

}, "0.0.1", {});