source: EcnlProtoTool/trunk/mruby-2.1.1/src/variable.c@ 439

Last change on this file since 439 was 439, checked in by coas-nagasima, 4 years ago

mrubyを2.1.1に更新

  • Property svn:eol-style set to native
  • Property svn:mime-type set to text/x-csrc;charset=UTF-8
File size: 23.3 KB
Line 
1/*
2** variable.c - mruby variables
3**
4** See Copyright Notice in mruby.h
5*/
6
7#include <mruby.h>
8#include <mruby/array.h>
9#include <mruby/class.h>
10#include <mruby/proc.h>
11#include <mruby/string.h>
12#include <mruby/variable.h>
13
14#ifndef MRB_IV_SEGMENT_SIZE
15#define MRB_IV_SEGMENT_SIZE 4
16#endif
17
18typedef struct segment {
19 mrb_sym key[MRB_IV_SEGMENT_SIZE];
20 mrb_value val[MRB_IV_SEGMENT_SIZE];
21 struct segment *next;
22} segment;
23
24/* Instance variable table structure */
25typedef struct iv_tbl {
26 segment *rootseg;
27 size_t size;
28 size_t last_len;
29} iv_tbl;
30
31/* Creates the instance variable table. */
32static iv_tbl*
33iv_new(mrb_state *mrb)
34{
35 iv_tbl *t;
36
37 t = (iv_tbl*)mrb_malloc(mrb, sizeof(iv_tbl));
38 t->size = 0;
39 t->rootseg = NULL;
40 t->last_len = 0;
41
42 return t;
43}
44
45/* Set the value for the symbol in the instance variable table. */
46static void
47iv_put(mrb_state *mrb, iv_tbl *t, mrb_sym sym, mrb_value val)
48{
49 segment *seg;
50 segment *prev = NULL;
51 segment *matched_seg = NULL;
52 size_t matched_idx = 0;
53 size_t i;
54
55 if (t == NULL) return;
56 seg = t->rootseg;
57 while (seg) {
58 for (i=0; i<MRB_IV_SEGMENT_SIZE; i++) {
59 mrb_sym key = seg->key[i];
60 /* Found room in last segment after last_len */
61 if (!seg->next && i >= t->last_len) {
62 seg->key[i] = sym;
63 seg->val[i] = val;
64 t->last_len = i+1;
65 t->size++;
66 return;
67 }
68 if (!matched_seg && key == 0) {
69 matched_seg = seg;
70 matched_idx = i;
71 }
72 else if (key == sym) {
73 seg->val[i] = val;
74 return;
75 }
76 }
77 prev = seg;
78 seg = seg->next;
79 }
80
81 /* Not found */
82 if (matched_seg) {
83 matched_seg->key[matched_idx] = sym;
84 matched_seg->val[matched_idx] = val;
85 t->size++;
86 return;
87 }
88
89 seg = (segment*)mrb_malloc(mrb, sizeof(segment));
90 seg->next = NULL;
91 seg->key[0] = sym;
92 seg->val[0] = val;
93 t->last_len = 1;
94 t->size++;
95 if (prev) {
96 prev->next = seg;
97 }
98 else {
99 t->rootseg = seg;
100 }
101}
102
103/* Get a value for a symbol from the instance variable table. */
104static mrb_bool
105iv_get(mrb_state *mrb, iv_tbl *t, mrb_sym sym, mrb_value *vp)
106{
107 segment *seg;
108 size_t i;
109
110 if (t == NULL) return FALSE;
111 seg = t->rootseg;
112 while (seg) {
113 for (i=0; i<MRB_IV_SEGMENT_SIZE; i++) {
114 mrb_sym key = seg->key[i];
115
116 if (!seg->next && i >= t->last_len) {
117 return FALSE;
118 }
119 if (key == sym) {
120 if (vp) *vp = seg->val[i];
121 return TRUE;
122 }
123 }
124 seg = seg->next;
125 }
126 return FALSE;
127}
128
129/* Deletes the value for the symbol from the instance variable table. */
130static mrb_bool
131iv_del(mrb_state *mrb, iv_tbl *t, mrb_sym sym, mrb_value *vp)
132{
133 segment *seg;
134 size_t i;
135
136 if (t == NULL) return FALSE;
137 seg = t->rootseg;
138 while (seg) {
139 for (i=0; i<MRB_IV_SEGMENT_SIZE; i++) {
140 mrb_sym key = seg->key[i];
141
142 if (!seg->next && i >= t->last_len) {
143 return FALSE;
144 }
145 if (key == sym) {
146 t->size--;
147 seg->key[i] = 0;
148 if (vp) *vp = seg->val[i];
149 return TRUE;
150 }
151 }
152 seg = seg->next;
153 }
154 return FALSE;
155}
156
157/* Iterates over the instance variable table. */
158static void
159iv_foreach(mrb_state *mrb, iv_tbl *t, mrb_iv_foreach_func *func, void *p)
160{
161 segment *seg;
162 size_t i;
163
164 if (t == NULL) return;
165 seg = t->rootseg;
166 while (seg) {
167 for (i=0; i<MRB_IV_SEGMENT_SIZE; i++) {
168 mrb_sym key = seg->key[i];
169
170 /* no value in last segment after last_len */
171 if (!seg->next && i >= t->last_len) {
172 return;
173 }
174 if (key != 0) {
175 if ((*func)(mrb, key, seg->val[i], p) != 0) {
176 return;
177 }
178 }
179 }
180 seg = seg->next;
181 }
182 return;
183}
184
185/* Get the size of the instance variable table. */
186static size_t
187iv_size(mrb_state *mrb, iv_tbl *t)
188{
189 segment *seg;
190 size_t size = 0;
191
192 if (t == NULL) return 0;
193 if (t->size > 0) return t->size;
194 seg = t->rootseg;
195 while (seg) {
196 if (seg->next == NULL) {
197 size += t->last_len;
198 return size;
199 }
200 seg = seg->next;
201 size += MRB_IV_SEGMENT_SIZE;
202 }
203 /* empty iv_tbl */
204 return 0;
205}
206
207/* Copy the instance variable table. */
208static iv_tbl*
209iv_copy(mrb_state *mrb, iv_tbl *t)
210{
211 segment *seg;
212 iv_tbl *t2;
213
214 size_t i;
215
216 seg = t->rootseg;
217 t2 = iv_new(mrb);
218
219 while (seg != NULL) {
220 for (i=0; i<MRB_IV_SEGMENT_SIZE; i++) {
221 mrb_sym key = seg->key[i];
222 mrb_value val = seg->val[i];
223
224 if ((seg->next == NULL) && (i >= t->last_len)) {
225 return t2;
226 }
227 iv_put(mrb, t2, key, val);
228 }
229 seg = seg->next;
230 }
231 return t2;
232}
233
234/* Free memory of the instance variable table. */
235static void
236iv_free(mrb_state *mrb, iv_tbl *t)
237{
238 segment *seg;
239
240 seg = t->rootseg;
241 while (seg) {
242 segment *p = seg;
243 seg = seg->next;
244 mrb_free(mrb, p);
245 }
246 mrb_free(mrb, t);
247}
248
249static int
250iv_mark_i(mrb_state *mrb, mrb_sym sym, mrb_value v, void *p)
251{
252 mrb_gc_mark_value(mrb, v);
253 return 0;
254}
255
256static void
257mark_tbl(mrb_state *mrb, iv_tbl *t)
258{
259 iv_foreach(mrb, t, iv_mark_i, 0);
260}
261
262void
263mrb_gc_mark_gv(mrb_state *mrb)
264{
265 mark_tbl(mrb, mrb->globals);
266}
267
268void
269mrb_gc_free_gv(mrb_state *mrb)
270{
271 if (mrb->globals)
272 iv_free(mrb, mrb->globals);
273}
274
275void
276mrb_gc_mark_iv(mrb_state *mrb, struct RObject *obj)
277{
278 mark_tbl(mrb, obj->iv);
279}
280
281size_t
282mrb_gc_mark_iv_size(mrb_state *mrb, struct RObject *obj)
283{
284 return iv_size(mrb, obj->iv);
285}
286
287void
288mrb_gc_free_iv(mrb_state *mrb, struct RObject *obj)
289{
290 if (obj->iv) {
291 iv_free(mrb, obj->iv);
292 }
293}
294
295mrb_value
296mrb_vm_special_get(mrb_state *mrb, mrb_sym i)
297{
298 return mrb_fixnum_value(0);
299}
300
301void
302mrb_vm_special_set(mrb_state *mrb, mrb_sym i, mrb_value v)
303{
304}
305
306static mrb_bool
307obj_iv_p(mrb_value obj)
308{
309 switch (mrb_type(obj)) {
310 case MRB_TT_OBJECT:
311 case MRB_TT_CLASS:
312 case MRB_TT_MODULE:
313 case MRB_TT_SCLASS:
314 case MRB_TT_HASH:
315 case MRB_TT_DATA:
316 case MRB_TT_EXCEPTION:
317 return TRUE;
318 default:
319 return FALSE;
320 }
321}
322
323MRB_API mrb_value
324mrb_obj_iv_get(mrb_state *mrb, struct RObject *obj, mrb_sym sym)
325{
326 mrb_value v;
327
328 if (obj->iv && iv_get(mrb, obj->iv, sym, &v))
329 return v;
330 return mrb_nil_value();
331}
332
333MRB_API mrb_value
334mrb_iv_get(mrb_state *mrb, mrb_value obj, mrb_sym sym)
335{
336 if (obj_iv_p(obj)) {
337 return mrb_obj_iv_get(mrb, mrb_obj_ptr(obj), sym);
338 }
339 return mrb_nil_value();
340}
341
342static inline void assign_class_name(mrb_state *mrb, struct RObject *obj, mrb_sym sym, mrb_value v);
343
344void
345mrb_obj_iv_set_force(mrb_state *mrb, struct RObject *obj, mrb_sym sym, mrb_value v)
346{
347 assign_class_name(mrb, obj, sym, v);
348 if (!obj->iv) {
349 obj->iv = iv_new(mrb);
350 }
351 iv_put(mrb, obj->iv, sym, v);
352 mrb_write_barrier(mrb, (struct RBasic*)obj);
353}
354
355MRB_API void
356mrb_obj_iv_set(mrb_state *mrb, struct RObject *obj, mrb_sym sym, mrb_value v)
357{
358 mrb_check_frozen(mrb, obj);
359 mrb_obj_iv_set_force(mrb, obj, sym, v);
360}
361
362/* Iterates over the instance variable table. */
363MRB_API void
364mrb_iv_foreach(mrb_state *mrb, mrb_value obj, mrb_iv_foreach_func *func, void *p)
365{
366 if (!obj_iv_p(obj)) return;
367 iv_foreach(mrb, mrb_obj_ptr(obj)->iv, func, p);
368}
369
370static inline mrb_bool
371namespace_p(enum mrb_vtype tt)
372{
373 return tt == MRB_TT_CLASS || tt == MRB_TT_MODULE ? TRUE : FALSE;
374}
375
376static inline void
377assign_class_name(mrb_state *mrb, struct RObject *obj, mrb_sym sym, mrb_value v)
378{
379 if (namespace_p(obj->tt) && namespace_p(mrb_type(v))) {
380 struct RObject *c = mrb_obj_ptr(v);
381 if (obj != c && ISUPPER(mrb_sym_name_len(mrb, sym, NULL)[0])) {
382 mrb_sym id_classname = mrb_intern_lit(mrb, "__classname__");
383 mrb_value o = mrb_obj_iv_get(mrb, c, id_classname);
384
385 if (mrb_nil_p(o)) {
386 mrb_sym id_outer = mrb_intern_lit(mrb, "__outer__");
387 o = mrb_obj_iv_get(mrb, c, id_outer);
388
389 if (mrb_nil_p(o)) {
390 if ((struct RClass *)obj == mrb->object_class) {
391 mrb_obj_iv_set_force(mrb, c, id_classname, mrb_symbol_value(sym));
392 }
393 else {
394 mrb_obj_iv_set_force(mrb, c, id_outer, mrb_obj_value(obj));
395 }
396 }
397 }
398 }
399 }
400}
401
402MRB_API void
403mrb_iv_set(mrb_state *mrb, mrb_value obj, mrb_sym sym, mrb_value v)
404{
405 if (obj_iv_p(obj)) {
406 mrb_obj_iv_set(mrb, mrb_obj_ptr(obj), sym, v);
407 }
408 else {
409 mrb_raise(mrb, E_ARGUMENT_ERROR, "cannot set instance variable");
410 }
411}
412
413MRB_API mrb_bool
414mrb_obj_iv_defined(mrb_state *mrb, struct RObject *obj, mrb_sym sym)
415{
416 iv_tbl *t;
417
418 t = obj->iv;
419 if (t) {
420 return iv_get(mrb, t, sym, NULL);
421 }
422 return FALSE;
423}
424
425MRB_API mrb_bool
426mrb_iv_defined(mrb_state *mrb, mrb_value obj, mrb_sym sym)
427{
428 if (!obj_iv_p(obj)) return FALSE;
429 return mrb_obj_iv_defined(mrb, mrb_obj_ptr(obj), sym);
430}
431
432MRB_API mrb_bool
433mrb_iv_name_sym_p(mrb_state *mrb, mrb_sym iv_name)
434{
435 const char *s;
436 mrb_int len;
437
438 s = mrb_sym_name_len(mrb, iv_name, &len);
439 if (len < 2) return FALSE;
440 if (s[0] != '@') return FALSE;
441 if (ISDIGIT(s[1])) return FALSE;
442 return mrb_ident_p(s+1, len-1);
443}
444
445MRB_API void
446mrb_iv_name_sym_check(mrb_state *mrb, mrb_sym iv_name)
447{
448 if (!mrb_iv_name_sym_p(mrb, iv_name)) {
449 mrb_name_error(mrb, iv_name, "'%n' is not allowed as an instance variable name", iv_name);
450 }
451}
452
453MRB_API void
454mrb_iv_copy(mrb_state *mrb, mrb_value dest, mrb_value src)
455{
456 struct RObject *d = mrb_obj_ptr(dest);
457 struct RObject *s = mrb_obj_ptr(src);
458
459 if (d->iv) {
460 iv_free(mrb, d->iv);
461 d->iv = 0;
462 }
463 if (s->iv) {
464 mrb_write_barrier(mrb, (struct RBasic*)d);
465 d->iv = iv_copy(mrb, s->iv);
466 }
467}
468
469static int
470inspect_i(mrb_state *mrb, mrb_sym sym, mrb_value v, void *p)
471{
472 mrb_value str = *(mrb_value*)p;
473 const char *s;
474 mrb_int len;
475 mrb_value ins;
476 char *sp = RSTRING_PTR(str);
477
478 /* need not to show internal data */
479 if (sp[0] == '-') { /* first element */
480 sp[0] = '#';
481 mrb_str_cat_lit(mrb, str, " ");
482 }
483 else {
484 mrb_str_cat_lit(mrb, str, ", ");
485 }
486 s = mrb_sym_name_len(mrb, sym, &len);
487 mrb_str_cat(mrb, str, s, len);
488 mrb_str_cat_lit(mrb, str, "=");
489 if (mrb_object_p(v)) {
490 ins = mrb_any_to_s(mrb, v);
491 }
492 else {
493 ins = mrb_inspect(mrb, v);
494 }
495 mrb_str_cat_str(mrb, str, ins);
496 return 0;
497}
498
499mrb_value
500mrb_obj_iv_inspect(mrb_state *mrb, struct RObject *obj)
501{
502 iv_tbl *t = obj->iv;
503 size_t len = iv_size(mrb, t);
504
505 if (len > 0) {
506 const char *cn = mrb_obj_classname(mrb, mrb_obj_value(obj));
507 mrb_value str = mrb_str_new_capa(mrb, 30);
508
509 mrb_str_cat_lit(mrb, str, "-<");
510 mrb_str_cat_cstr(mrb, str, cn);
511 mrb_str_cat_lit(mrb, str, ":");
512 mrb_str_cat_str(mrb, str, mrb_ptr_to_str(mrb, obj));
513
514 iv_foreach(mrb, t, inspect_i, &str);
515 mrb_str_cat_lit(mrb, str, ">");
516 return str;
517 }
518 return mrb_any_to_s(mrb, mrb_obj_value(obj));
519}
520
521MRB_API mrb_value
522mrb_iv_remove(mrb_state *mrb, mrb_value obj, mrb_sym sym)
523{
524 if (obj_iv_p(obj)) {
525 iv_tbl *t = mrb_obj_ptr(obj)->iv;
526 mrb_value val;
527
528 mrb_check_frozen(mrb, mrb_obj_ptr(obj));
529 if (iv_del(mrb, t, sym, &val)) {
530 return val;
531 }
532 }
533 return mrb_undef_value();
534}
535
536static int
537iv_i(mrb_state *mrb, mrb_sym sym, mrb_value v, void *p)
538{
539 mrb_value ary;
540 const char* s;
541 mrb_int len;
542
543 ary = *(mrb_value*)p;
544 s = mrb_sym_name_len(mrb, sym, &len);
545 if (len > 1 && s[0] == '@' && s[1] != '@') {
546 mrb_ary_push(mrb, ary, mrb_symbol_value(sym));
547 }
548 return 0;
549}
550
551/* 15.3.1.3.23 */
552/*
553 * call-seq:
554 * obj.instance_variables -> array
555 *
556 * Returns an array of instance variable names for the receiver. Note
557 * that simply defining an accessor does not create the corresponding
558 * instance variable.
559 *
560 * class Fred
561 * attr_accessor :a1
562 * def initialize
563 * @iv = 3
564 * end
565 * end
566 * Fred.new.instance_variables #=> [:@iv]
567 */
568mrb_value
569mrb_obj_instance_variables(mrb_state *mrb, mrb_value self)
570{
571 mrb_value ary;
572
573 ary = mrb_ary_new(mrb);
574 if (obj_iv_p(self)) {
575 iv_foreach(mrb, mrb_obj_ptr(self)->iv, iv_i, &ary);
576 }
577 return ary;
578}
579
580static int
581cv_i(mrb_state *mrb, mrb_sym sym, mrb_value v, void *p)
582{
583 mrb_value ary;
584 const char* s;
585 mrb_int len;
586
587 ary = *(mrb_value*)p;
588 s = mrb_sym_name_len(mrb, sym, &len);
589 if (len > 2 && s[0] == '@' && s[1] == '@') {
590 mrb_ary_push(mrb, ary, mrb_symbol_value(sym));
591 }
592 return 0;
593}
594
595/* 15.2.2.4.19 */
596/*
597 * call-seq:
598 * mod.class_variables(inherit=true) -> array
599 *
600 * Returns an array of the names of class variables in <i>mod</i>.
601 *
602 * class One
603 * @@var1 = 1
604 * end
605 * class Two < One
606 * @@var2 = 2
607 * end
608 * One.class_variables #=> [:@@var1]
609 * Two.class_variables #=> [:@@var2]
610 */
611mrb_value
612mrb_mod_class_variables(mrb_state *mrb, mrb_value mod)
613{
614 mrb_value ary;
615 struct RClass *c;
616 mrb_bool inherit = TRUE;
617
618 mrb_get_args(mrb, "|b", &inherit);
619 ary = mrb_ary_new(mrb);
620 c = mrb_class_ptr(mod);
621 while (c) {
622 iv_foreach(mrb, c->iv, cv_i, &ary);
623 if (!inherit) break;
624 c = c->super;
625 }
626 return ary;
627}
628
629mrb_value
630mrb_mod_cv_get(mrb_state *mrb, struct RClass *c, mrb_sym sym)
631{
632 struct RClass * cls = c;
633 mrb_value v;
634 int given = FALSE;
635
636 while (c) {
637 if (c->iv && iv_get(mrb, c->iv, sym, &v)) {
638 given = TRUE;
639 }
640 c = c->super;
641 }
642 if (given) return v;
643 if (cls && cls->tt == MRB_TT_SCLASS) {
644 mrb_value klass;
645
646 klass = mrb_obj_iv_get(mrb, (struct RObject *)cls,
647 mrb_intern_lit(mrb, "__attached__"));
648 c = mrb_class_ptr(klass);
649 if (c->tt == MRB_TT_CLASS || c->tt == MRB_TT_MODULE) {
650 given = FALSE;
651 while (c) {
652 if (c->iv && iv_get(mrb, c->iv, sym, &v)) {
653 given = TRUE;
654 }
655 c = c->super;
656 }
657 if (given) return v;
658 }
659 }
660 mrb_name_error(mrb, sym, "uninitialized class variable %n in %C", sym, cls);
661 /* not reached */
662 return mrb_nil_value();
663}
664
665MRB_API mrb_value
666mrb_cv_get(mrb_state *mrb, mrb_value mod, mrb_sym sym)
667{
668 return mrb_mod_cv_get(mrb, mrb_class_ptr(mod), sym);
669}
670
671MRB_API void
672mrb_mod_cv_set(mrb_state *mrb, struct RClass *c, mrb_sym sym, mrb_value v)
673{
674 struct RClass * cls = c;
675
676 while (c) {
677 iv_tbl *t = c->iv;
678
679 if (iv_get(mrb, t, sym, NULL)) {
680 mrb_check_frozen(mrb, c);
681 iv_put(mrb, t, sym, v);
682 mrb_write_barrier(mrb, (struct RBasic*)c);
683 return;
684 }
685 c = c->super;
686 }
687
688 if (cls && cls->tt == MRB_TT_SCLASS) {
689 mrb_value klass;
690
691 klass = mrb_obj_iv_get(mrb, (struct RObject*)cls,
692 mrb_intern_lit(mrb, "__attached__"));
693 switch (mrb_type(klass)) {
694 case MRB_TT_CLASS:
695 case MRB_TT_MODULE:
696 case MRB_TT_SCLASS:
697 c = mrb_class_ptr(klass);
698 break;
699 default:
700 c = cls;
701 break;
702 }
703 }
704 else{
705 c = cls;
706 }
707
708 mrb_check_frozen(mrb, c);
709 if (!c->iv) {
710 c->iv = iv_new(mrb);
711 }
712
713 iv_put(mrb, c->iv, sym, v);
714 mrb_write_barrier(mrb, (struct RBasic*)c);
715}
716
717MRB_API void
718mrb_cv_set(mrb_state *mrb, mrb_value mod, mrb_sym sym, mrb_value v)
719{
720 mrb_mod_cv_set(mrb, mrb_class_ptr(mod), sym, v);
721}
722
723mrb_bool
724mrb_mod_cv_defined(mrb_state *mrb, struct RClass * c, mrb_sym sym)
725{
726 while (c) {
727 iv_tbl *t = c->iv;
728 if (iv_get(mrb, t, sym, NULL)) return TRUE;
729 c = c->super;
730 }
731
732 return FALSE;
733}
734
735MRB_API mrb_bool
736mrb_cv_defined(mrb_state *mrb, mrb_value mod, mrb_sym sym)
737{
738 return mrb_mod_cv_defined(mrb, mrb_class_ptr(mod), sym);
739}
740
741mrb_value
742mrb_vm_cv_get(mrb_state *mrb, mrb_sym sym)
743{
744 struct RClass *c;
745
746 struct RProc *p = mrb->c->ci->proc;
747
748 for (;;) {
749 c = MRB_PROC_TARGET_CLASS(p);
750 if (c->tt != MRB_TT_SCLASS) break;
751 p = p->upper;
752 }
753 return mrb_mod_cv_get(mrb, c, sym);
754}
755
756void
757mrb_vm_cv_set(mrb_state *mrb, mrb_sym sym, mrb_value v)
758{
759 struct RClass *c;
760 struct RProc *p = mrb->c->ci->proc;
761
762 for (;;) {
763 c = MRB_PROC_TARGET_CLASS(p);
764 if (c->tt != MRB_TT_SCLASS) break;
765 p = p->upper;
766 }
767 mrb_mod_cv_set(mrb, c, sym, v);
768}
769
770static void
771mod_const_check(mrb_state *mrb, mrb_value mod)
772{
773 switch (mrb_type(mod)) {
774 case MRB_TT_CLASS:
775 case MRB_TT_MODULE:
776 case MRB_TT_SCLASS:
777 break;
778 default:
779 mrb_raise(mrb, E_TYPE_ERROR, "constant look-up for non class/module");
780 break;
781 }
782}
783
784static mrb_value
785const_get(mrb_state *mrb, struct RClass *base, mrb_sym sym)
786{
787 struct RClass *c = base;
788 mrb_value v;
789 mrb_bool retry = FALSE;
790 mrb_value name;
791
792L_RETRY:
793 while (c) {
794 if (c->iv) {
795 if (iv_get(mrb, c->iv, sym, &v))
796 return v;
797 }
798 c = c->super;
799 }
800 if (!retry && base->tt == MRB_TT_MODULE) {
801 c = mrb->object_class;
802 retry = TRUE;
803 goto L_RETRY;
804 }
805 name = mrb_symbol_value(sym);
806 return mrb_funcall_argv(mrb, mrb_obj_value(base), mrb_intern_lit(mrb, "const_missing"), 1, &name);
807}
808
809MRB_API mrb_value
810mrb_const_get(mrb_state *mrb, mrb_value mod, mrb_sym sym)
811{
812 mod_const_check(mrb, mod);
813 return const_get(mrb, mrb_class_ptr(mod), sym);
814}
815
816mrb_value
817mrb_vm_const_get(mrb_state *mrb, mrb_sym sym)
818{
819 struct RClass *c;
820 struct RClass *c2;
821 mrb_value v;
822 struct RProc *proc;
823
824 c = MRB_PROC_TARGET_CLASS(mrb->c->ci->proc);
825 if (iv_get(mrb, c->iv, sym, &v)) {
826 return v;
827 }
828 c2 = c;
829 while (c2 && c2->tt == MRB_TT_SCLASS) {
830 mrb_value klass;
831
832 if (!iv_get(mrb, c2->iv, mrb_intern_lit(mrb, "__attached__"), &klass)) {
833 c2 = NULL;
834 break;
835 }
836 c2 = mrb_class_ptr(klass);
837 }
838 if (c2 && (c2->tt == MRB_TT_CLASS || c2->tt == MRB_TT_MODULE)) c = c2;
839 mrb_assert(!MRB_PROC_CFUNC_P(mrb->c->ci->proc));
840 proc = mrb->c->ci->proc;
841 while (proc) {
842 c2 = MRB_PROC_TARGET_CLASS(proc);
843 if (c2 && iv_get(mrb, c2->iv, sym, &v)) {
844 return v;
845 }
846 proc = proc->upper;
847 }
848 return const_get(mrb, c, sym);
849}
850
851MRB_API void
852mrb_const_set(mrb_state *mrb, mrb_value mod, mrb_sym sym, mrb_value v)
853{
854 mod_const_check(mrb, mod);
855 if (mrb_type(v) == MRB_TT_CLASS || mrb_type(v) == MRB_TT_MODULE) {
856 mrb_class_name_class(mrb, mrb_class_ptr(mod), mrb_class_ptr(v), sym);
857 }
858 mrb_iv_set(mrb, mod, sym, v);
859}
860
861void
862mrb_vm_const_set(mrb_state *mrb, mrb_sym sym, mrb_value v)
863{
864 struct RClass *c;
865
866 c = MRB_PROC_TARGET_CLASS(mrb->c->ci->proc);
867 mrb_obj_iv_set(mrb, (struct RObject*)c, sym, v);
868}
869
870MRB_API void
871mrb_const_remove(mrb_state *mrb, mrb_value mod, mrb_sym sym)
872{
873 mod_const_check(mrb, mod);
874 mrb_iv_remove(mrb, mod, sym);
875}
876
877MRB_API void
878mrb_define_const(mrb_state *mrb, struct RClass *mod, const char *name, mrb_value v)
879{
880 mrb_obj_iv_set(mrb, (struct RObject*)mod, mrb_intern_cstr(mrb, name), v);
881}
882
883MRB_API void
884mrb_define_global_const(mrb_state *mrb, const char *name, mrb_value val)
885{
886 mrb_define_const(mrb, mrb->object_class, name, val);
887}
888
889static int
890const_i(mrb_state *mrb, mrb_sym sym, mrb_value v, void *p)
891{
892 mrb_value ary;
893 const char* s;
894 mrb_int len;
895
896 ary = *(mrb_value*)p;
897 s = mrb_sym_name_len(mrb, sym, &len);
898 if (len >= 1 && ISUPPER(s[0])) {
899 mrb_int i, alen = RARRAY_LEN(ary);
900
901 for (i=0; i<alen; i++) {
902 if (mrb_symbol(RARRAY_PTR(ary)[i]) == sym)
903 break;
904 }
905 if (i==alen) {
906 mrb_ary_push(mrb, ary, mrb_symbol_value(sym));
907 }
908 }
909 return 0;
910}
911
912/* 15.2.2.4.24 */
913/*
914 * call-seq:
915 * mod.constants -> array
916 *
917 * Returns an array of all names of contants defined in the receiver.
918 */
919mrb_value
920mrb_mod_constants(mrb_state *mrb, mrb_value mod)
921{
922 mrb_value ary;
923 mrb_bool inherit = TRUE;
924 struct RClass *c = mrb_class_ptr(mod);
925
926 mrb_get_args(mrb, "|b", &inherit);
927 ary = mrb_ary_new(mrb);
928 while (c) {
929 iv_foreach(mrb, c->iv, const_i, &ary);
930 if (!inherit) break;
931 c = c->super;
932 if (c == mrb->object_class) break;
933 }
934 return ary;
935}
936
937MRB_API mrb_value
938mrb_gv_get(mrb_state *mrb, mrb_sym sym)
939{
940 mrb_value v;
941
942 if (iv_get(mrb, mrb->globals, sym, &v))
943 return v;
944 return mrb_nil_value();
945}
946
947MRB_API void
948mrb_gv_set(mrb_state *mrb, mrb_sym sym, mrb_value v)
949{
950 iv_tbl *t;
951
952 if (!mrb->globals) {
953 mrb->globals = iv_new(mrb);
954 }
955 t = mrb->globals;
956 iv_put(mrb, t, sym, v);
957}
958
959MRB_API void
960mrb_gv_remove(mrb_state *mrb, mrb_sym sym)
961{
962 iv_del(mrb, mrb->globals, sym, NULL);
963}
964
965static int
966gv_i(mrb_state *mrb, mrb_sym sym, mrb_value v, void *p)
967{
968 mrb_value ary;
969
970 ary = *(mrb_value*)p;
971 mrb_ary_push(mrb, ary, mrb_symbol_value(sym));
972 return 0;
973}
974
975/* 15.3.1.2.4 */
976/* 15.3.1.3.14 */
977/*
978 * call-seq:
979 * global_variables -> array
980 *
981 * Returns an array of the names of global variables.
982 *
983 * global_variables.grep /std/ #=> [:$stdin, :$stdout, :$stderr]
984 */
985mrb_value
986mrb_f_global_variables(mrb_state *mrb, mrb_value self)
987{
988 iv_tbl *t = mrb->globals;
989 mrb_value ary = mrb_ary_new(mrb);
990
991 iv_foreach(mrb, t, gv_i, &ary);
992 return ary;
993}
994
995static mrb_bool
996mrb_const_defined_0(mrb_state *mrb, mrb_value mod, mrb_sym id, mrb_bool exclude, mrb_bool recurse)
997{
998 struct RClass *klass = mrb_class_ptr(mod);
999 struct RClass *tmp;
1000 mrb_bool mod_retry = FALSE;
1001
1002 tmp = klass;
1003retry:
1004 while (tmp) {
1005 if (iv_get(mrb, tmp->iv, id, NULL)) {
1006 return TRUE;
1007 }
1008 if (!recurse && (klass != mrb->object_class)) break;
1009 tmp = tmp->super;
1010 }
1011 if (!exclude && !mod_retry && (klass->tt == MRB_TT_MODULE)) {
1012 mod_retry = TRUE;
1013 tmp = mrb->object_class;
1014 goto retry;
1015 }
1016 return FALSE;
1017}
1018
1019MRB_API mrb_bool
1020mrb_const_defined(mrb_state *mrb, mrb_value mod, mrb_sym id)
1021{
1022 return mrb_const_defined_0(mrb, mod, id, TRUE, TRUE);
1023}
1024
1025MRB_API mrb_bool
1026mrb_const_defined_at(mrb_state *mrb, mrb_value mod, mrb_sym id)
1027{
1028 return mrb_const_defined_0(mrb, mod, id, TRUE, FALSE);
1029}
1030
1031MRB_API mrb_value
1032mrb_attr_get(mrb_state *mrb, mrb_value obj, mrb_sym id)
1033{
1034 return mrb_iv_get(mrb, obj, id);
1035}
1036
1037struct csym_arg {
1038 struct RClass *c;
1039 mrb_sym sym;
1040};
1041
1042static int
1043csym_i(mrb_state *mrb, mrb_sym sym, mrb_value v, void *p)
1044{
1045 struct csym_arg *a = (struct csym_arg*)p;
1046 struct RClass *c = a->c;
1047
1048 if (mrb_type(v) == c->tt && mrb_class_ptr(v) == c) {
1049 a->sym = sym;
1050 return 1; /* stop iteration */
1051 }
1052 return 0;
1053}
1054
1055static mrb_sym
1056find_class_sym(mrb_state *mrb, struct RClass *outer, struct RClass *c)
1057{
1058 struct csym_arg arg;
1059
1060 if (!outer) return 0;
1061 if (outer == c) return 0;
1062 arg.c = c;
1063 arg.sym = 0;
1064 iv_foreach(mrb, outer->iv, csym_i, &arg);
1065 return arg.sym;
1066}
1067
1068static struct RClass*
1069outer_class(mrb_state *mrb, struct RClass *c)
1070{
1071 mrb_value ov;
1072
1073 ov = mrb_obj_iv_get(mrb, (struct RObject*)c, mrb_intern_lit(mrb, "__outer__"));
1074 if (mrb_nil_p(ov)) return NULL;
1075 switch (mrb_type(ov)) {
1076 case MRB_TT_CLASS:
1077 case MRB_TT_MODULE:
1078 return mrb_class_ptr(ov);
1079 default:
1080 break;
1081 }
1082 return NULL;
1083}
1084
1085static mrb_bool
1086detect_outer_loop(mrb_state *mrb, struct RClass *c)
1087{
1088 struct RClass *t = c; /* tortoise */
1089 struct RClass *h = c; /* hare */
1090
1091 for (;;) {
1092 if (h == NULL) return FALSE;
1093 h = outer_class(mrb, h);
1094 if (h == NULL) return FALSE;
1095 h = outer_class(mrb, h);
1096 t = outer_class(mrb, t);
1097 if (t == h) return TRUE;
1098 }
1099}
1100
1101mrb_value
1102mrb_class_find_path(mrb_state *mrb, struct RClass *c)
1103{
1104 struct RClass *outer;
1105 mrb_value path;
1106 mrb_sym name;
1107 const char *str;
1108 mrb_int len;
1109
1110 if (detect_outer_loop(mrb, c)) return mrb_nil_value();
1111 outer = outer_class(mrb, c);
1112 if (outer == NULL) return mrb_nil_value();
1113 name = find_class_sym(mrb, outer, c);
1114 if (name == 0) return mrb_nil_value();
1115 str = mrb_class_name(mrb, outer);
1116 path = mrb_str_new_capa(mrb, 40);
1117 mrb_str_cat_cstr(mrb, path, str);
1118 mrb_str_cat_cstr(mrb, path, "::");
1119
1120 str = mrb_sym_name_len(mrb, name, &len);
1121 mrb_str_cat(mrb, path, str, len);
1122 if (RSTRING_PTR(path)[0] != '#') {
1123 iv_del(mrb, c->iv, mrb_intern_lit(mrb, "__outer__"), NULL);
1124 iv_put(mrb, c->iv, mrb_intern_lit(mrb, "__classname__"), path);
1125 mrb_field_write_barrier_value(mrb, (struct RBasic*)c, path);
1126 path = mrb_str_dup(mrb, path);
1127 }
1128 return path;
1129}
1130
1131#define identchar(c) (ISALNUM(c) || (c) == '_' || !ISASCII(c))
1132
1133mrb_bool
1134mrb_ident_p(const char *s, mrb_int len)
1135{
1136 mrb_int i;
1137
1138 for (i = 0; i < len; i++) {
1139 if (!identchar(s[i])) return FALSE;
1140 }
1141 return TRUE;
1142}
Note: See TracBrowser for help on using the repository browser.