2 * base code for various Tcl extensions
3 * Copyright 2006-2012 Ian Jackson
5 * This program is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU General Public License as
7 * published by the Free Software Foundation; either version 2 of the
8 * License, or (at your option) any later version.
10 * This program is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this library; if not, see <http://www.gnu.org/licenses/>.
19 #include "chiark-tcl-base.h"
24 const IdDataSpec *idds;
31 IdDataAssocData *assoc;
35 static void assoc_del(ClientData assoc_cd, Tcl_Interp *ip) {
36 IdDataAssocData *assoc;
41 for (ix=0, p=assoc->a; ix<assoc->n; ix++, p++) {
44 assert(*(int*)v == ix);
46 assoc->idds->destroyitem(ip,v);
53 static void setobjdataid(Tcl_Interp *interp, Tcl_Obj *o,
54 int ix, const IdDataSpec *idds) {
56 IdDataAssocData *assoc;
58 assoc= Tcl_GetAssocData(interp, (char*)idds->assockey, 0);
60 assoc= TALLOC(sizeof(*assoc));
64 Tcl_SetAssocData(interp, (char*)idds->assockey, assoc_del, assoc);
67 dv= TALLOC(sizeof(*dv));
72 o->typePtr= &cht_tabledataid_nearlytype;
73 o->internalRep.otherValuePtr= dv;
76 int cht_tabledataid_parse(Tcl_Interp *ip, Tcl_Obj *o, const IdDataSpec *idds) {
80 IdDataAssocData *assoc;
83 if (o->typePtr != &cht_tabledataid_nearlytype) goto convert;
85 dv= o->internalRep.otherValuePtr;
86 if (dv->interp != ip) goto convert;
88 if (dv->assoc->idds != idds) goto convert;
93 l= strlen(idds->valprefix);
94 str= Tcl_GetStringFromObj(o,0);
95 if (memcmp(str,idds->valprefix,l))
96 return cht_staticerr(ip,"bad id (wrong prefix)",0);
98 errno=0; ul=strtoul(str+l,&ep,10);
99 if (errno || *ep) return cht_staticerr(ip,"bad id number",0);
100 if (ul > INT_MAX) return cht_staticerr(ip,"out of range id number",0);
103 setobjdataid(ip,o,ul,idds);
107 int cht_pat_iddata(Tcl_Interp *ip, Tcl_Obj *o, void **rv, const IdDataSpec *idds) {
110 IdDataAssocData *assoc;
113 rc= cht_tabledataid_parse(ip,o,idds);
116 dv= o->internalRep.otherValuePtr;
120 if (ix >= assoc->n || !(r= assoc->a[ix]))
121 return cht_staticerr(ip,"id not in use",0);
123 assert(*(int*)r == ix);
129 Tcl_Obj *cht_ret_iddata(Tcl_Interp *ip, void *val, const IdDataSpec *idds) {
130 /* Command procedure implementation may set val->ix,
131 * ie *(int*)val, to -1, to mean it's a new struct. Otherwise
132 * it had better be an old one !
136 IdDataAssocData *assoc;
140 setobjdataid(ip,o,0,idds);
141 dv= o->internalRep.otherValuePtr;
146 for (ix=0; ix<assoc->n && assoc->a[ix]; ix++);
148 assert(assoc->n < INT_MAX/4);
151 assoc->a= TREALLOC(assoc->a, assoc->n*sizeof(*assoc->a));
152 while (ix<assoc->n) assoc->a[ix++]=0;
158 assert(val == assoc->a[ix]);
161 Tcl_InvalidateStringRep(o);
165 void cht_tabledataid_disposing(Tcl_Interp *ip, void *val, const IdDataSpec *idds) {
166 IdDataAssocData *assoc;
172 assoc= Tcl_GetAssocData(ip, (char*)idds->assockey, 0);
173 assert(assoc->a[ix] == val);
178 static int tabledataid_nt_sfa(Tcl_Interp *ip, Tcl_Obj *o) {
182 static void tabledataid_nt_free(Tcl_Obj *o) {
183 TFREE(o->internalRep.otherValuePtr);
184 o->internalRep.otherValuePtr= 0;
187 static void tabledataid_nt_dup(Tcl_Obj *src, Tcl_Obj *dup) {
188 IdDataValue *sv, *dv;
190 sv= src->internalRep.otherValuePtr;
191 dv= TALLOC(sizeof(*dv));
193 dup->typePtr= &cht_tabledataid_nearlytype;
194 dup->internalRep.otherValuePtr= dv;
197 static void tabledataid_nt_ustr(Tcl_Obj *o) {
198 const IdDataValue *dv;
199 const IdDataAssocData *assoc;
200 const IdDataSpec *idds;
203 dv= o->internalRep.otherValuePtr;
207 snprintf(buf,sizeof(buf), "%d", dv->ix);
208 cht_obj_updatestr_vstringls(o,
209 idds->valprefix, strlen(idds->valprefix),
214 Tcl_ObjType cht_tabledataid_nearlytype = {
216 tabledataid_nt_free, tabledataid_nt_dup,
217 tabledataid_nt_ustr, tabledataid_nt_sfa