summaryrefslogtreecommitdiff
path: root/jim-namespace.c
diff options
context:
space:
mode:
authorSteve Bennett <steveb@workware.net.au>2011-12-01 17:25:36 +1000
committerSteve Bennett <steveb@workware.net.au>2011-12-12 13:44:16 +1000
commit7f383c6726fd71c23d622753152faf749124ca22 (patch)
tree32cf6285c78d54e4931d0558e895c0d8b077ce17 /jim-namespace.c
parent1f0d4b7361480fd029dbf5b5462d3a6a0068e5d0 (diff)
Add support for lightweight namespaces
See README.namespaces Signed-off-by: Steve Bennett <steveb@workware.net.au>
Diffstat (limited to 'jim-namespace.c')
-rw-r--r--jim-namespace.c333
1 files changed, 333 insertions, 0 deletions
diff --git a/jim-namespace.c b/jim-namespace.c
new file mode 100644
index 0000000..9b606c7
--- /dev/null
+++ b/jim-namespace.c
@@ -0,0 +1,333 @@
+/*
+ * Support for namespaces in jim
+ *
+ * (c) 2011 Steve Bennett <steveb@workware.net.au>
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following
+ * disclaimer in the documentation and/or other materials
+ * provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE JIM TCL PROJECT ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
+ * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
+ * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
+ * JIM TCL PROJECT OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
+ * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
+ * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
+ * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ *
+ * The views and conclusions contained in the software and documentation
+ * are those of the authors and should not be interpreted as representing
+ * official policies, either expressed or implied, of the Jim Tcl Project.
+ *
+ * Based on code originally from Tcl 6.7:
+ *
+ * Copyright 1987-1991 Regents of the University of California
+ * Permission to use, copy, modify, and distribute this
+ * software and its documentation for any purpose and without
+ * fee is hereby granted, provided that the above copyright
+ * notice appear in all copies. The University of California
+ * makes no representations about the suitability of this
+ * software for any purpose. It is provided "as is" without
+ * express or implied warranty.
+ */
+
+#include <limits.h>
+#include <stdlib.h>
+#include <string.h>
+#include <stdio.h>
+#include <assert.h>
+
+#include "jim.h"
+#include "jimautoconf.h"
+#include "jim-subcmd.h"
+
+/* -----------------------------------------------------------------------------
+ * Namespace support
+ * ---------------------------------------------------------------------------*/
+
+/**
+ * nsObj is a canonical namespace name (.e.g. "" for root, "abc" for ::abc)
+ *
+ * The given name is appended to the namespace name to produce a complete canonical name.
+ *
+ * e.g. "" "abc" => abc
+ * "" "::abc" => abc
+ * "" "abc::def" => abc::def
+ * "abc" "def" => abc::def
+ * "abc" "::def" => def
+ *
+ */
+Jim_Obj *JimCanonicalNamespace(Jim_Interp *interp, Jim_Obj *nsObj, Jim_Obj *nameObj)
+{
+ Jim_Obj *objPtr;
+ const char *name = Jim_String(nameObj);
+ assert(nameObj->refCount != 0);
+ assert(nsObj->refCount != 0);
+ if (name[0] == ':' && name[1] == ':') {
+ /* Absolute namespace */
+ while (*++name == ':') {
+ }
+ return Jim_NewStringObj(interp, name, -1);
+ }
+ if (Jim_Length(nsObj) == 0) {
+ /* Relative to the global namespace */
+ return nameObj;
+ }
+ /* Relative to non-global namespace */
+ objPtr = Jim_DuplicateObj(interp, nsObj);
+ Jim_AppendString(interp, objPtr, "::", 2);
+ Jim_AppendObj(interp, objPtr, nameObj);
+ return objPtr;
+}
+
+int Jim_CreateNamespaceVariable(Jim_Interp *interp, Jim_Obj *varNameObj, Jim_Obj *targetNameObj)
+{
+ int rc;
+ Jim_IncrRefCount(varNameObj);
+
+ /* push non-namespace vars if in namespace eval? */
+ rc = Jim_SetVariableLink(interp, varNameObj, targetNameObj, interp->topFramePtr);
+
+ Jim_DecrRefCount(interp, varNameObj);
+
+ return rc;
+}
+
+/**
+ * Returns the parent of the given namespace.
+ *
+ * ::bob::tom => ::bob
+ * bob::tom => bob
+ * ::bob => ::
+ * bob => ""
+ * :: => ""
+ * "" => ""
+ */
+Jim_Obj *Jim_NamespaceQualifiers(Jim_Interp *interp, Jim_Obj *ns)
+{
+ const char *name = Jim_String(ns);
+ const char *pt = strrchr(name, ':');
+ if (pt && pt != name && pt[-1] == ':') {
+ return Jim_NewStringObj(interp, name, pt - name - 1);
+ }
+ else {
+ return interp->emptyObj;
+ }
+}
+
+Jim_Obj *Jim_NamespaceTail(Jim_Interp *interp, Jim_Obj *ns)
+{
+ const char *name = Jim_String(ns);
+ const char *pt = strrchr(name, ':');
+ if (pt && pt != name && pt[-1] == ':') {
+ return Jim_NewStringObj(interp, pt + 1, -1);
+ }
+ else {
+ return ns;
+ }
+}
+
+static Jim_Obj *JimNamespaceCurrent(Jim_Interp *interp)
+{
+ Jim_Obj *objPtr = Jim_NewStringObj(interp, "::", 2);
+ Jim_AppendObj(interp, objPtr, interp->framePtr->nsObj);
+ return objPtr;
+}
+
+static int JimVariableCmd(Jim_Interp *interp, int argc, Jim_Obj *const *argv)
+{
+ int retcode = JIM_OK;
+
+ if (argc > 3) {
+ Jim_WrongNumArgs(interp, 1, argv, "name ?value?");
+ return JIM_ERR;
+ }
+ if (argc > 1) {
+ Jim_Obj *targetNameObj;
+ Jim_Obj *localNameObj;
+
+#if 0
+ /* XXX should we give an error on dict sugar syntax? */
+ if (JimValidName(interp, "variable", argv[1]) != JIM_OK) {
+ return JIM_ERR;
+ }
+#endif
+
+ targetNameObj = JimCanonicalNamespace(interp, interp->framePtr->nsObj, argv[1]);
+
+ localNameObj = Jim_NamespaceTail(interp, argv[1]);
+ Jim_IncrRefCount(localNameObj);
+ if (interp->framePtr->level != 0 || Jim_Length(interp->framePtr->nsObj) != 0) {
+ Jim_CreateNamespaceVariable(interp, localNameObj, targetNameObj);
+ }
+
+ /* Set the variable via the local name */
+ if (argc > 2) {
+ retcode = Jim_SetVariable(interp, localNameObj, argv[2]);
+ }
+ Jim_DecrRefCount(interp, localNameObj);
+ }
+ return retcode;
+}
+
+/* XXX: Temporary */
+static int Jim_EvalEnsemble(Jim_Interp *interp, const char *basecmd, const char *subcmd, int argc, Jim_Obj *const *argv)
+{
+ Jim_Obj *prefixObj = Jim_NewStringObj(interp, basecmd, -1);
+
+ Jim_AppendString(interp, prefixObj, " ", 1);
+ Jim_AppendString(interp, prefixObj, subcmd, -1);
+
+ return Jim_EvalObjPrefix(interp, prefixObj, argc, argv);
+}
+
+static int JimNamespaceCmd(Jim_Interp *interp, int argc, Jim_Obj *const *argv)
+{
+ Jim_Obj *nsObj;
+ Jim_Obj *objPtr;
+ int option;
+ static const char * const options[] = {
+ "eval", "current", "canonical", "qualifiers", "parent", "tail", "delete",
+ "origin", "code", "inscope", "import", "export",
+ "which", "upvar", NULL
+ };
+ enum
+ {
+ OPT_EVAL, OPT_CURRENT, OPT_CANONICAL, OPT_QUALIFIERS, OPT_PARENT, OPT_TAIL, OPT_DELETE,
+ OPT_ORIGIN, OPT_CODE, OPT_INSCOPE, OPT_IMPORT, OPT_EXPORT,
+ OPT_WHICH, OPT_UPVAR,
+ };
+
+ if (argc < 2) {
+ Jim_WrongNumArgs(interp, 1, argv, "subcommand ?arg ...?");
+ return JIM_ERR;
+ }
+
+ if (Jim_GetEnum(interp, argv[1], options, &option, "subcommand", JIM_ERRMSG | JIM_ENUM_ABBREV) != JIM_OK) {
+ return JIM_ERR;
+ }
+
+ switch (option) {
+ case OPT_EVAL:
+ if (argc < 4) {
+ Jim_WrongNumArgs(interp, 2, argv, "name arg ?arg...?");
+ return JIM_ERR;
+ }
+ if (argc == 4) {
+ objPtr = argv[3];
+ }
+ else {
+ objPtr = Jim_ConcatObj(interp, argc - 3, argv + 3);
+ }
+
+ nsObj = JimCanonicalNamespace(interp, interp->framePtr->nsObj, argv[2]);
+ return Jim_EvalNamespace(interp, objPtr, nsObj);
+
+ case OPT_CURRENT:
+ if (argc != 2) {
+ Jim_WrongNumArgs(interp, 2, argv, "");
+ return JIM_ERR;
+ }
+ Jim_SetResult(interp, JimNamespaceCurrent(interp));
+ return JIM_OK;
+
+ case OPT_CANONICAL:
+ if (argc > 4) {
+ Jim_WrongNumArgs(interp, 2, argv, "?current? ?name?");
+ return JIM_ERR;
+ }
+ if (argc == 2) {
+ Jim_SetResult(interp, interp->framePtr->nsObj);
+ }
+ else if (argc == 3) {
+ Jim_SetResult(interp, JimCanonicalNamespace(interp, interp->framePtr->nsObj, argv[2]));
+ }
+ else {
+ Jim_SetResult(interp, JimCanonicalNamespace(interp, argv[2], argv[3]));
+ }
+ return JIM_OK;
+
+ case OPT_QUALIFIERS:
+ if (argc != 3) {
+ Jim_WrongNumArgs(interp, 2, argv, "string");
+ return JIM_ERR;
+ }
+ Jim_SetResult(interp, Jim_NamespaceQualifiers(interp, argv[2]));
+ return JIM_OK;
+
+ case OPT_EXPORT:
+ return JIM_OK;
+
+ case OPT_TAIL:
+ if (argc != 3) {
+ Jim_WrongNumArgs(interp, 2, argv, "string");
+ return JIM_ERR;
+ }
+ Jim_SetResult(interp, Jim_NamespaceTail(interp, argv[2]));
+ return JIM_OK;
+
+ case OPT_PARENT:
+ if (argc != 2 && argc != 3) {
+ Jim_WrongNumArgs(interp, 2, argv, "?name?");
+ return JIM_ERR;
+ }
+ else {
+ const char *name;
+
+ if (argc == 3) {
+ objPtr = argv[2];
+ }
+ else {
+ objPtr = interp->framePtr->nsObj;
+ }
+ if (Jim_Length(objPtr) == 0 || Jim_CompareStringImmediate(interp, objPtr, "::")) {
+ return JIM_OK;
+ }
+ objPtr = Jim_NamespaceQualifiers(interp, objPtr);
+
+ name = Jim_String(objPtr);
+
+ if (name[0] != ':' || name[1] != ':') {
+ /* Make it fully scoped */
+ Jim_SetResultString(interp, "::", 2);
+ Jim_AppendObj(interp, Jim_GetResult(interp), objPtr);
+ Jim_IncrRefCount(objPtr);
+ Jim_DecrRefCount(interp, objPtr);
+ }
+ else {
+ Jim_SetResult(interp, objPtr);
+ }
+ }
+ return JIM_OK;
+ }
+
+ /* Implemented as a Tcl helper proc.
+ * Note that calling a proc will change the current namespace,
+ * so helper procs must call [uplevel namespace canon] to get the callers
+ * namespace.
+ */
+ return Jim_EvalEnsemble(interp, "namespace", options[option], argc - 2, argv + 2);
+}
+
+int Jim_namespaceInit(Jim_Interp *interp)
+{
+ if (Jim_PackageProvide(interp, "namespace", "1.0", JIM_ERRMSG))
+ return JIM_ERR;
+
+ Jim_CreateCommand(interp, "namespace", JimNamespaceCmd, NULL, NULL);
+ Jim_CreateCommand(interp, "variable", JimVariableCmd, NULL, NULL);
+ return JIM_OK;
+}
+