2 SuperCollider real time audio synthesis system
3 Copyright (c) 2002 James McCartney. All rights reserved.
4 http://www.audiosynth.com
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program; if not, write to the Free Software
18 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21 #include "PyrKernel.h"
22 #include "PyrKernelProto.h"
23 #include "PyrPrimitive.h"
24 #include "PyrPrimitiveProto.h"
25 #include "PyrSymbol.h"
26 #include "PyrListPrim.h"
27 #include "SC_InlineUnaryOp.h"
28 #include "SC_InlineBinaryOp.h"
29 #include "PyrSignal.h"
30 #include "PyrMessage.h"
37 int objectPerform(VMGlobals
*g
, int numArgsPushed
);
39 int ivxIdentDict_array
, ivxIdentDict_size
, ivxIdentDict_parent
, ivxIdentDict_proto
, ivxIdentDict_know
;
41 int class_array_index
, class_array_maxsubclassindex
;
42 int class_identdict_index
, class_identdict_maxsubclassindex
;
44 PyrClass
*class_identdict
;
45 PyrSymbol
*s_proto
, *s_parent
;
46 PyrSymbol
*s_delta
, *s_dur
, *s_stretch
;
48 #define HASHSYMBOL(sym) (sym >> 5)
50 #define ISKINDOF(obj, lo, hi) (\
51 objClassIndex = slotRawInt(&obj->classptr->classIndex), \
52 objClassIndex >= lo && objClassIndex <= hi)
55 int prArrayMultiChanExpand(struct VMGlobals
*g
, int numArgsPushed
)
57 PyrSlot
*a
, *slot
, *slots1
, *slots2
, *slots3
, *slots4
;
58 PyrObject
*obj1
, *obj2
, *obj3
, *obj4
;
59 int i
, j
, size
, len
, maxlen
;
62 obj1
= slotRawObject(a
);
66 for (j
=0; j
<size
; ++j
) {
69 if (slotRawObject(slot
)->classptr
== class_array
) {
70 len
= slotRawObject(slot
)->size
;
71 maxlen
= len
> maxlen
? len
: maxlen
;
72 } else if (isKindOf(slotRawObject(slot
), class_sequenceable_collection
) && (slotRawObject(slot
)->classptr
!= class_string
)) {
73 return errFailed
; // this primitive only handles Arrays.
78 obj2
= newPyrArray(g
->gc
, maxlen
, 0, true);
81 for (i
=0; i
<maxlen
; ++i
) {
82 obj3
= newPyrArray(g
->gc
, size
, 0, false);
84 SetObject(slots2
+ i
, obj3
);
87 for (j
=0; j
<size
; ++j
) {
90 if (slotRawObject(slot
)->classptr
== class_array
&& slotRawObject(slot
)->size
> 0) {
91 obj4
= slotRawObject(slot
);
93 slotCopy(&slots3
[j
],&slots4
[i
% obj4
->size
]);
95 slotCopy(&slots3
[j
],slot
);
98 slotCopy(&slots3
[j
],slot
);
109 int arrayAtIdentityHash(PyrObject
*array
, PyrSlot
*key
)
111 PyrSlot
*slots
, *test
;
112 unsigned int i
, start
, end
, hash
, maxHash
;
114 hash
= calcHash(key
);
115 maxHash
= array
->size
;
116 start
= hash
% maxHash
;
118 slots
= array
->slots
;
119 for (i
=start
; i
<end
; i
++) {
121 if (IsNil(test
) || SlotEq(test
, key
))
125 for (i
=0; i
<=end
; i
++) {
127 if (IsNil(test
) || SlotEq(test
, key
))
134 int prArray_AtIdentityHash(struct VMGlobals
*g
, int numArgsPushed
);
135 int prArray_AtIdentityHash(struct VMGlobals
*g
, int numArgsPushed
)
141 a
= g
->sp
- 1; // array
144 array
= slotRawObject(a
);
145 if(array
->size
== 0) return errFailed
;
146 index
= arrayAtIdentityHash(array
, b
);
152 int arrayAtIdentityHashInPairs(PyrObject
*array
, PyrSlot
*key
)
154 PyrSlot
*slots
, *test
;
155 unsigned int i
, start
, end
, hash
, maxHash
;
157 hash
= calcHash(key
);
158 maxHash
= array
->size
>> 1;
159 start
= (hash
% maxHash
) << 1;
161 slots
= array
->slots
;
162 for (i
=start
; i
<end
; i
+=2) {
164 if (IsNil(test
) || SlotEq(test
, key
))
168 for (i
=0; i
<=end
; i
+=2) {
170 if (IsNil(test
) || SlotEq(test
, key
))
177 int arrayAtIdentityHashInPairsWithHash(PyrObject
*array
, PyrSlot
*key
, int hash
)
179 PyrSlot
*slots
, *test
;
180 unsigned int i
, start
, end
, maxHash
;
182 maxHash
= array
->size
>> 1;
183 start
= (hash
% maxHash
) << 1;
185 slots
= array
->slots
;
186 for (i
=start
; i
<end
; i
+=2) {
188 if (IsNil(test
) || SlotEq(test
, key
))
192 for (i
=0; i
<=end
; i
+=2) {
194 if (IsNil(test
) || SlotEq(test
, key
))
201 int identDictPut(struct VMGlobals
*g
, PyrObject
*dict
, PyrSlot
*key
, PyrSlot
*value
);
202 int identDictPut(struct VMGlobals
*g
, PyrObject
*dict
, PyrSlot
*key
, PyrSlot
*value
)
204 PyrSlot
*slot
, *newslot
;
208 bool knows
= IsTrue(dict
->slots
+ ivxIdentDict_know
);
209 if (knows
&& IsSym(key
)) {
210 if (slotRawSymbol(key
) == s_parent
) {
211 slotCopy(&dict
->slots
[ivxIdentDict_parent
],value
);
212 g
->gc
->GCWrite(dict
, value
);
215 if (slotRawSymbol(key
) == s_proto
) {
216 slotCopy(&dict
->slots
[ivxIdentDict_proto
],value
);
217 g
->gc
->GCWrite(dict
, value
);
221 array
= slotRawObject(&dict
->slots
[ivxIdentDict_array
]);
222 if (!isKindOf((PyrObject
*)array
, class_array
)) return errFailed
;
224 index
= arrayAtIdentityHashInPairs(array
, key
);
225 slot
= array
->slots
+ index
;
226 slotCopy(&slot
[1],value
);
227 g
->gc
->GCWrite(array
, value
);
230 g
->gc
->GCWrite(array
, key
);
231 size
= slotRawInt(&dict
->slots
[ivxIdentDict_size
]) + 1;
232 SetRaw(&dict
->slots
[ivxIdentDict_size
], size
);
233 if (array
->size
< size
*3) {
235 newarray
= newPyrArray(g
->gc
, size
*3, 0, false);
236 newarray
->size
= ARRAYMAXINDEXSIZE(newarray
);
237 nilSlots(newarray
->slots
, newarray
->size
);
239 for (i
=0; i
<array
->size
; i
+=2, slot
+=2) {
241 index
= arrayAtIdentityHashInPairs(newarray
, slot
);
242 newslot
= newarray
->slots
+ index
;
243 slotCopy(&newslot
[0],&slot
[0]);
244 slotCopy(&newslot
[1],&slot
[1]);
247 SetRaw(&dict
->slots
[ivxIdentDict_array
], newarray
);
248 g
->gc
->GCWrite(dict
, newarray
);
254 int prIdentDict_Put(struct VMGlobals
*g
, int numArgsPushed
);
255 int prIdentDict_Put(struct VMGlobals
*g
, int numArgsPushed
)
259 a
= g
->sp
- 2; // dict
260 b
= g
->sp
- 1; // key
262 if(IsNil(b
)) return errWrongType
;
263 if(IsNil(c
)) return errFailed
; // will call removeAt
264 return identDictPut(g
, slotRawObject(a
), b
, c
);
267 int prIdentDict_PutGet(struct VMGlobals
*g
, int numArgsPushed
);
268 int prIdentDict_PutGet(struct VMGlobals
*g
, int numArgsPushed
)
270 PyrSlot
*a
, *b
, *c
, *d
, *slot
, *newslot
;
275 a
= g
->sp
- 2; // dict
276 b
= g
->sp
- 1; // key
278 d
= ++g
->sp
; // push the stack to save the receiver
281 dict
= slotRawObject(d
);
282 array
= slotRawObject(&dict
->slots
[ivxIdentDict_array
]);
283 if (!isKindOf((PyrObject
*)array
, class_array
)) {
289 index
= arrayAtIdentityHashInPairs(array
, b
);
290 slot
= array
->slots
+ index
;
291 slotCopy(a
,&slot
[1]);
292 slotCopy(&slot
[1],c
);
293 g
->gc
->GCWrite(array
, c
);
296 g
->gc
->GCWrite(array
, b
);
297 size
= slotRawInt(&dict
->slots
[ivxIdentDict_size
]) + 1;
298 SetRaw(&dict
->slots
[ivxIdentDict_size
], size
);
299 if (array
->size
< size
*3) {
301 newarray
= newPyrArray(g
->gc
, size
*3, 0, true);
302 newarray
->size
= ARRAYMAXINDEXSIZE(newarray
);
303 nilSlots(newarray
->slots
, newarray
->size
);
305 for (i
=0; i
<array
->size
; i
+=2, slot
+=2) {
307 index
= arrayAtIdentityHashInPairs(newarray
, slot
);
308 newslot
= newarray
->slots
+ index
;
309 slotCopy(&newslot
[0],&slot
[0]);
310 slotCopy(&newslot
[1],&slot
[1]);
313 SetRaw(&dict
->slots
[ivxIdentDict_array
], newarray
);
314 g
->gc
->GCWrite(dict
, newarray
);
323 int prArray_AtIdentityHashInPairs(struct VMGlobals
*g
, int numArgsPushed
);
324 int prArray_AtIdentityHashInPairs(struct VMGlobals
*g
, int numArgsPushed
)
329 a
= g
->sp
- 1; // array
332 if(slotRawObject(a
)->size
< 2) return errFailed
;
333 i
= arrayAtIdentityHashInPairs(slotRawObject(a
), b
);
339 bool identDict_lookupNonNil(PyrObject
*dict
, PyrSlot
*key
, int hash
, PyrSlot
*result
);
340 bool identDict_lookupNonNil(PyrObject
*dict
, PyrSlot
*key
, int hash
, PyrSlot
*result
)
343 PyrSlot
*dictslots
= dict
->slots
;
344 PyrSlot
*arraySlot
= dictslots
+ ivxIdentDict_array
;
346 if (isKindOfSlot(arraySlot
, class_array
)) {
347 PyrObject
*array
= slotRawObject(arraySlot
);
349 int index
= arrayAtIdentityHashInPairsWithHash(array
, key
, hash
);
350 if (SlotEq(key
, array
->slots
+ index
)) {
351 slotCopy(result
,&array
->slots
[index
+ 1]);
356 PyrClass
*identDictClass
= s_identitydictionary
->u
.classobj
;
357 PyrSlot
*parentSlot
= dictslots
+ ivxIdentDict_parent
;
358 PyrSlot
* protoSlot
= dictslots
+ ivxIdentDict_proto
;
359 if (isKindOfSlot(parentSlot
, identDictClass
)) {
360 if (isKindOfSlot(protoSlot
, identDictClass
)) {
362 if (identDict_lookupNonNil(slotRawObject(protoSlot
), key
, hash
, result
)) return true;
365 dict
= slotRawObject(parentSlot
);
366 goto again
; // tail call
368 if (isKindOfSlot(protoSlot
, identDictClass
)) {
369 dict
= slotRawObject(protoSlot
);
370 goto again
; // tail call
376 bool identDict_lookup(PyrObject
*dict
, PyrSlot
*key
, int hash
, PyrSlot
*result
);
377 bool identDict_lookup(PyrObject
*dict
, PyrSlot
*key
, int hash
, PyrSlot
*result
)
380 PyrSlot
*dictslots
= dict
->slots
;
381 PyrSlot
*arraySlot
= dictslots
+ ivxIdentDict_array
;
383 if (isKindOfSlot(arraySlot
, class_array
)) {
384 PyrObject
*array
= slotRawObject(arraySlot
);
386 int index
= arrayAtIdentityHashInPairsWithHash(array
, key
, hash
);
387 if (SlotEq(key
, array
->slots
+ index
)) {
388 slotCopy(result
,&array
->slots
[index
+ 1]);
393 PyrClass
*identDictClass
= s_identitydictionary
->u
.classobj
;
394 PyrSlot
*parentSlot
= dictslots
+ ivxIdentDict_parent
;
395 PyrSlot
* protoSlot
= dictslots
+ ivxIdentDict_proto
;
396 if (isKindOfSlot(parentSlot
, identDictClass
)) {
397 if (isKindOfSlot(protoSlot
, identDictClass
)) {
399 if (identDict_lookup(slotRawObject(protoSlot
), key
, hash
, result
)) return true;
402 dict
= slotRawObject(parentSlot
);
403 goto again
; // tail call
405 if (isKindOfSlot(protoSlot
, identDictClass
)) {
406 dict
= slotRawObject(protoSlot
);
407 goto again
; // tail call
414 int prIdentDict_At(struct VMGlobals
*g
, int numArgsPushed
);
415 int prIdentDict_At(struct VMGlobals
*g
, int numArgsPushed
)
417 PyrSlot
* a
= g
->sp
- 1; // dict
418 PyrSlot
* key
= g
->sp
; // key
419 PyrObject
*dict
= slotRawObject(a
);
421 bool knows
= IsTrue(dict
->slots
+ ivxIdentDict_know
);
422 if (knows
&& IsSym(key
)) {
423 if (slotRawSymbol(key
) == s_parent
) {
424 slotCopy(a
,&dict
->slots
[ivxIdentDict_parent
]);
427 if (slotRawSymbol(key
) == s_proto
) {
428 slotCopy(a
,&dict
->slots
[ivxIdentDict_proto
]);
433 identDict_lookup(dict
, key
, calcHash(key
), a
);
437 int prSymbol_envirGet(struct VMGlobals
*g
, int numArgsPushed
);
438 int prSymbol_envirGet(struct VMGlobals
*g
, int numArgsPushed
)
445 PyrSlot
* currentEnvironmentSlot
= &g
->classvars
->slots
[1];
446 PyrObject
*dict
= slotRawObject(currentEnvironmentSlot
);
448 if (!IsObj(currentEnvironmentSlot
)) return errFailed
;
450 if (!ISKINDOF(dict
, class_identdict_index
, class_identdict_maxsubclassindex
)) return errFailed
;
452 identDict_lookup(dict
, a
, calcHash(a
), &result
);
459 int prSymbol_envirPut(struct VMGlobals
*g
, int numArgsPushed
);
460 int prSymbol_envirPut(struct VMGlobals
*g
, int numArgsPushed
)
465 a
= g
->sp
- 1; // key
468 PyrSlot
* currentEnvironmentSlot
= &g
->classvars
->slots
[1];
469 PyrObject
*dict
= slotRawObject(currentEnvironmentSlot
);
471 if (!IsObj(currentEnvironmentSlot
)) return errFailed
;
473 if (!ISKINDOF(dict
, class_identdict_index
, class_identdict_maxsubclassindex
)) return errFailed
;
475 int err
= identDictPut(g
, dict
, a
, b
);
484 int prEvent_Delta(struct VMGlobals
*g
, int numArgsPushed
);
485 int prEvent_Delta(struct VMGlobals
*g
, int numArgsPushed
)
487 PyrSlot
*a
, key
, dur
, stretch
, delta
;
488 double fdur
, fstretch
;
493 SetSymbol(&key
, s_delta
);
494 identDict_lookup(slotRawObject(a
), &key
, calcHash(&key
), &delta
);
496 if (NotNil(&delta
)) {
499 SetSymbol(&key
, s_dur
);
500 identDict_lookup(slotRawObject(a
), &key
, calcHash(&key
), &dur
);
502 err
= slotDoubleVal(&dur
, &fdur
);
504 if (NotNil(&dur
)) return err
;
509 SetSymbol(&key
, s_stretch
);
510 identDict_lookup(slotRawObject(a
), &key
, calcHash(&key
), &stretch
);
512 err
= slotDoubleVal(&stretch
, &fstretch
);
514 if (NotNil(&stretch
)) return err
;
519 SetFloat(a
, fdur
* fstretch
);
525 void PriorityQueueAdd(struct VMGlobals
*g
, PyrObject
* queueobj
, PyrSlot
* item
, double time
);
526 void PriorityQueueAdd(struct VMGlobals
*g
, PyrObject
* queueobj
, PyrSlot
* item
, double time
)
528 PyrObject
*schedq
, *newschedq
;
531 PyrSlot
*schedqSlot
= queueobj
->slots
;
532 if (!IsObj(schedqSlot
)) {
534 schedq
= newPyrArray(g
->gc
, size
, 0, true);
536 SetInt(schedq
->slots
+ 0, 0); // stability count
537 SetObject(schedqSlot
, schedq
);
538 g
->gc
->GCWrite(queueobj
, schedq
);
540 schedq
= slotRawObject(schedqSlot
);
541 maxsize
= ARRAYMAXINDEXSIZE(schedq
);
543 if (size
+3 > maxsize
) {
545 newschedq
= newPyrArray(g
->gc
, maxsize
*2, 0, true);
546 newschedq
->size
= size
;
548 slotCopy(newschedq
->slots
, schedq
->slots
, size
);
549 assert(IsInt(newschedq
->slots
));
551 SetObject(schedqSlot
, newschedq
);
552 g
->gc
->GCWrite(queueobj
, newschedq
);
558 addheap(g
, schedq
, time
, item
);
561 int prPriorityQueueAdd(struct VMGlobals
*g
, int numArgsPushed
);
562 int prPriorityQueueAdd(struct VMGlobals
*g
, int numArgsPushed
)
565 PyrSlot
*a
= g
->sp
- 2; // priority queue
566 PyrSlot
*b
= g
->sp
- 1; // time
567 PyrSlot
*c
= g
->sp
; // item
570 int err
= slotDoubleVal(b
, &time
);
571 if (err
) return errNone
; // nil is OK, nothing gets added
573 PriorityQueueAdd(g
, slotRawObject(a
), c
, time
);
578 void PriorityQueuePop(VMGlobals
*g
, PyrObject
*queueobj
, PyrSlot
*result
);
579 void PriorityQueuePop(VMGlobals
*g
, PyrObject
*queueobj
, PyrSlot
*result
)
581 PyrSlot
*schedqSlot
= queueobj
->slots
;
583 if (IsObj(schedqSlot
)) {
584 PyrObject
*schedq
= slotRawObject(schedqSlot
);
586 if (!getheap(g
, schedq
, &time
, result
)) {
594 void PriorityQueueTop(PyrObject
*queueobj
, PyrSlot
*result
);
595 void PriorityQueueTop(PyrObject
*queueobj
, PyrSlot
*result
)
597 PyrSlot
*schedqSlot
= queueobj
->slots
;
599 if (IsObj(schedqSlot
)) {
600 PyrObject
*schedq
= slotRawObject(schedqSlot
);
601 if (schedq
->size
> 1) {
602 slotCopy(result
,&schedq
->slots
[1]);
611 void PriorityQueueClear(PyrObject
*queueobj
);
612 void PriorityQueueClear(PyrObject
*queueobj
)
614 PyrSlot
*schedqSlot
= queueobj
->slots
;
616 if (IsObj(schedqSlot
)) {
617 PyrObject
*schedq
= slotRawObject(schedqSlot
);
618 SetInt(schedq
->slots
, 0); // stability count
623 bool PriorityQueueEmpty(PyrObject
*queueobj
);
624 bool PriorityQueueEmpty(PyrObject
*queueobj
)
626 PyrSlot
*schedqSlot
= queueobj
->slots
;
628 if (IsObj(schedqSlot
)) {
629 PyrObject
*schedq
= slotRawObject(schedqSlot
);
630 if (schedq
->size
> 1) {
637 int prPriorityQueuePop(struct VMGlobals
*g
, int numArgsPushed
);
638 int prPriorityQueuePop(struct VMGlobals
*g
, int numArgsPushed
)
640 PyrSlot
* a
= g
->sp
; // priority queue
642 PriorityQueuePop(g
, slotRawObject(a
), a
);
646 int prPriorityQueueTop(struct VMGlobals
*g
, int numArgsPushed
);
647 int prPriorityQueueTop(struct VMGlobals
*g
, int numArgsPushed
)
649 PyrSlot
* a
= g
->sp
; // priority queue
651 PriorityQueueTop(slotRawObject(a
), a
);
655 int prPriorityQueueClear(struct VMGlobals
*g
, int numArgsPushed
);
656 int prPriorityQueueClear(struct VMGlobals
*g
, int numArgsPushed
)
658 PyrSlot
* a
= g
->sp
; // priority queue
660 PriorityQueueClear(slotRawObject(a
));
664 int prPriorityQueueEmpty(struct VMGlobals
*g
, int numArgsPushed
);
665 int prPriorityQueueEmpty(struct VMGlobals
*g
, int numArgsPushed
)
669 a
= g
->sp
; // priority queue
671 if (PriorityQueueEmpty(slotRawObject(a
))) {
679 void PriorityQueuePostpone(PyrObject
* queueobj
, double time
);
680 void PriorityQueuePostpone(PyrObject
* queueobj
, double time
)
682 PyrSlot
*schedqSlot
= queueobj
->slots
;
684 if (IsObj(schedqSlot
)) {
685 PyrObject
*schedq
= slotRawObject(schedqSlot
);
686 PyrSlot
* slots
= schedq
->slots
;
687 for (int i
=1; i
< schedq
->size
; i
+=3) {
688 SetRaw(&slots
[i
], slotRawFloat(&slots
[i
]) + time
);
693 int prPriorityQueuePostpone(struct VMGlobals
*g
, int numArgsPushed
);
694 int prPriorityQueuePostpone(struct VMGlobals
*g
, int numArgsPushed
)
696 PyrSlot
*a
= g
->sp
- 1; // priority queue
697 PyrSlot
*b
= g
->sp
; // time
700 int err
= slotDoubleVal(b
, &time
);
703 PyrObject
*queueobj
= slotRawObject(a
);
704 PriorityQueuePostpone(queueobj
, time
);
709 void initListPrimitives();
710 void initListPrimitives()
714 base
= nextPrimitiveIndex();
716 definePrimitive(base
, index
++, "_Array_AtIdentityHash", prArray_AtIdentityHash
, 2, 0);
717 definePrimitive(base
, index
++, "_Array_AtIdentityHashInPairs", prArray_AtIdentityHashInPairs
, 2, 0);
718 definePrimitive(base
, index
++, "_IdentDict_Put", prIdentDict_Put
, 3, 0);
719 definePrimitive(base
, index
++, "_IdentDict_PutGet", prIdentDict_PutGet
, 3, 0);
720 definePrimitive(base
, index
++, "_IdentDict_At", prIdentDict_At
, 2, 0);
721 definePrimitive(base
, index
++, "_Symbol_envirGet", prSymbol_envirGet
, 1, 0);
722 definePrimitive(base
, index
++, "_Symbol_envirPut", prSymbol_envirPut
, 2, 0);
723 definePrimitive(base
, index
++, "_ArrayMultiChannelExpand", prArrayMultiChanExpand
, 1, 0);
725 definePrimitive(base
, index
++, "_PriorityQueueAdd", prPriorityQueueAdd
, 3, 0);
726 definePrimitive(base
, index
++, "_PriorityQueuePop", prPriorityQueuePop
, 1, 0);
727 definePrimitive(base
, index
++, "_PriorityQueueTop", prPriorityQueueTop
, 1, 0);
728 definePrimitive(base
, index
++, "_PriorityQueueClear", prPriorityQueueClear
, 1, 0);
729 definePrimitive(base
, index
++, "_PriorityQueueEmpty", prPriorityQueueEmpty
, 1, 0);
730 definePrimitive(base
, index
++, "_PriorityQueuePostpone", prPriorityQueuePostpone
, 2, 0);
732 definePrimitive(base
, index
++, "_Event_Delta", prEvent_Delta
, 1, 0);
740 ivxIdentDict_array
= instVarOffset("IdentityDictionary", "array");
741 ivxIdentDict_size
= instVarOffset("IdentityDictionary", "size");
742 ivxIdentDict_parent
= instVarOffset("IdentityDictionary", "parent");
743 ivxIdentDict_proto
= instVarOffset("IdentityDictionary", "proto");
744 ivxIdentDict_know
= instVarOffset("IdentityDictionary", "know");
746 sym
= getsym("IdentityDictionary");
747 class_identdict
= sym
? sym
->u
.classobj
: NULL
;
748 class_identdict_index
= slotRawInt(&class_identdict
->classIndex
);
749 class_identdict_maxsubclassindex
= slotRawInt(&class_identdict
->maxSubclassIndex
);
751 class_array_index
= slotRawInt(&class_array
->classIndex
);
752 class_array_maxsubclassindex
= slotRawInt(&class_array
->maxSubclassIndex
);
754 s_parent
= getsym("parent");
755 s_proto
= getsym("proto");
756 s_delta
= getsym("delta");
757 s_dur
= getsym("dur");
758 s_stretch
= getsym("stretch");