summaryrefslogtreecommitdiffstats
path: root/src/bltNsUtil.C
blob: e1bd9f6e75eec22a00b33b98c1ed82304577d5d2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
/*
 * Smithsonian Astrophysical Observatory, Cambridge, MA, USA
 * This code has been modified under the terms listed below and is made
 * available under the same terms.
 */

/*
 *	Copyright 1997-2008 George A Howlett.
 *
 *	Permission is hereby granted, free of charge, to any person obtaining
 *	a copy of this software and associated documentation files (the
 *	"Software"), to deal in the Software without restriction, including
 *	without limitation the rights to use, copy, modify, merge, publish,
 *	distribute, sublicense, and/or sell copies of the Software, and to
 *	permit persons to whom the Software is furnished to do so, subject to
 *	the following conditions:
 *
 *	The above copyright notice and this permission notice shall be
 *	included in all copies or substantial portions of the Software.
 *
 *	THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 *	EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 *	MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 *	NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
 *	LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
 *	OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
 *	WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */

#ifdef __CYGWIN__
extern "C" {
#include <tclInt.h>
}
#else
#include <tclInt.h>
#endif

#include "bltNsUtil.h"

using namespace Blt;

Tcl_Namespace* Blt::GetCommandNamespace(Tcl_Command cmdToken)
{
  Command* cmdPtr = (Command*)cmdToken;
  return (Tcl_Namespace *)cmdPtr->nsPtr;
}

int Blt::ParseObjectName(Tcl_Interp* interp, const char *path, 
			Blt_ObjectName *namePtr, unsigned int flags)
{
  namePtr->nsPtr = NULL;
  namePtr->name = NULL;
  char* colon = NULL;

  /* Find the last namespace separator in the qualified name. */
  char* last = (char *)(path + strlen(path));
  while (--last > path) {
    if ((*last == ':') && (*(last - 1) == ':')) {
      last++;		/* just after the last "::" */
      colon = last - 2;
      break;
    }
  }
  if (colon == NULL) {
    namePtr->name = path;
    if ((flags & BLT_NO_DEFAULT_NS) == 0) {
      namePtr->nsPtr = Tcl_GetCurrentNamespace(interp);
    }
    return 1;		/* No namespace designated in name. */
  }

  /* Separate the namespace and the object name. */
  *colon = '\0';
  if (path[0] == '\0') {
    namePtr->nsPtr = Tcl_GetGlobalNamespace(interp);
  } else {
    namePtr->nsPtr = Tcl_FindNamespace(interp, (char *)path, NULL, 
				       (flags & BLT_NO_ERROR_MSG) ? 0 : TCL_LEAVE_ERR_MSG);
  }
  /* Repair the string. */    *colon = ':';

  if (namePtr->nsPtr == NULL) {
    return 0;		/* Namespace doesn't exist. */
  }
  namePtr->name =last;
  return 1;
}

char* Blt::MakeQualifiedName(Blt_ObjectName *namePtr, Tcl_DString *resultPtr)
{
  Tcl_DStringInit(resultPtr);
  if ((namePtr->nsPtr->fullName[0] != ':') || 
      (namePtr->nsPtr->fullName[1] != ':') ||
      (namePtr->nsPtr->fullName[2] != '\0')) {
    Tcl_DStringAppend(resultPtr, namePtr->nsPtr->fullName, -1);
  }
  Tcl_DStringAppend(resultPtr, "::", -1);
  Tcl_DStringAppend(resultPtr, (char *)namePtr->name, -1);
  return Tcl_DStringValue(resultPtr);
}

static Tcl_Namespace* NamespaceOfVariable(Var *varPtr)
{
  if (varPtr->flags & VAR_IN_HASHTABLE) {
    VarInHash *vhashPtr = (VarInHash *)varPtr;
    TclVarHashTable *vtablePtr;

    vtablePtr = (TclVarHashTable *)vhashPtr->entry.tablePtr;
    return (Tcl_Namespace*)(vtablePtr->nsPtr);
  }
  return NULL;
}

Tcl_Namespace* Blt::GetVariableNamespace(Tcl_Interp* interp, const char *path)
{
  Blt_ObjectName objName;
  if (!ParseObjectName(interp, path, &objName, BLT_NO_DEFAULT_NS))
    return NULL;

  if (objName.nsPtr == NULL) {
    Var*varPtr = (Var*)Tcl_FindNamespaceVar(interp, (char *)path, 
					    (Tcl_Namespace *)NULL, 
					    TCL_GLOBAL_ONLY);
    if (varPtr)
      return NamespaceOfVariable(varPtr);

  }
  return objName.nsPtr;    
}