summaryrefslogtreecommitdiffhomepage
path: root/src/variable.c
diff options
context:
space:
mode:
authorYukihiro "Matz" Matsumoto <[email protected]>2018-09-20 13:50:16 +0900
committerYukihiro "Matz" Matsumoto <[email protected]>2018-09-20 13:50:16 +0900
commitd192a52edb8c222c3238f883aec08e6c80751a61 (patch)
treece0e6b0aba5b33d8367a8331deec139108f95014 /src/variable.c
parent3a9caad8ebe63dfe2b8583c72fc5e275f13e25c3 (diff)
parent5e6cd86f6eedea213fbeb5f1e7a5aa0ab303036a (diff)
downloadmruby-d192a52edb8c222c3238f883aec08e6c80751a61.tar.gz
mruby-d192a52edb8c222c3238f883aec08e6c80751a61.zip
Merge branch 'assign-anon-class-name' of https://github.com/dearblue/mruby into dearblue-assign-anon-class-name
Diffstat (limited to 'src/variable.c')
-rw-r--r--src/variable.c53
1 files changed, 50 insertions, 3 deletions
diff --git a/src/variable.c b/src/variable.c
index 506b4b63e..01077a34e 100644
--- a/src/variable.c
+++ b/src/variable.c
@@ -344,6 +344,10 @@ mrb_iv_get(mrb_state *mrb, mrb_value obj, mrb_sym sym)
return mrb_nil_value();
}
+#ifdef MRB_IMPROVE_META_PROGRAMMING
+static inline void assign_class_name(mrb_state *mrb, struct RObject *obj, mrb_sym sym, mrb_value v);
+#endif
+
MRB_API void
mrb_obj_iv_set(mrb_state *mrb, struct RObject *obj, mrb_sym sym, mrb_value v)
{
@@ -352,6 +356,9 @@ mrb_obj_iv_set(mrb_state *mrb, struct RObject *obj, mrb_sym sym, mrb_value v)
if (MRB_FROZEN_P(obj)) {
mrb_raisef(mrb, E_FROZEN_ERROR, "can't modify frozen %S", mrb_obj_value(obj));
}
+#ifdef MRB_IMPROVE_META_PROGRAMMING
+ assign_class_name(mrb, obj, sym, v);
+#endif
if (!obj->iv) {
obj->iv = iv_new(mrb);
}
@@ -360,6 +367,40 @@ mrb_obj_iv_set(mrb_state *mrb, struct RObject *obj, mrb_sym sym, mrb_value v)
mrb_write_barrier(mrb, (struct RBasic*)obj);
}
+#ifdef MRB_IMPROVE_META_PROGRAMMING
+static inline mrb_bool
+is_namespace(enum mrb_vtype tt)
+{
+ return tt == MRB_TT_CLASS || tt == MRB_TT_MODULE ? TRUE : FALSE;
+}
+
+static inline void
+assign_class_name(mrb_state *mrb, struct RObject *obj, mrb_sym sym, mrb_value v)
+{
+ if (is_namespace(obj->tt) && is_namespace(mrb_type(v))) {
+ struct RObject *c = mrb_obj_ptr(v);
+ if (obj != c && ISUPPER(mrb_sym2name(mrb, sym)[0])) {
+ mrb_sym id_classname = mrb_intern_lit(mrb, "__classname__");
+ mrb_value o = mrb_obj_iv_get(mrb, c, id_classname);
+
+ if (mrb_nil_p(o)) {
+ mrb_sym id_outer = mrb_intern_lit(mrb, "__outer__");
+ o = mrb_obj_iv_get(mrb, c, id_outer);
+
+ if (mrb_nil_p(o)) {
+ if ((struct RClass *)obj == mrb->object_class) {
+ mrb_obj_iv_set(mrb, c, id_classname, mrb_symbol_value(sym));
+ }
+ else {
+ mrb_obj_iv_set(mrb, c, id_outer, mrb_obj_value(obj));
+ }
+ }
+ }
+ }
+ }
+}
+#endif
+
MRB_API void
mrb_iv_set(mrb_state *mrb, mrb_value obj, mrb_sym sym, mrb_value v)
{
@@ -1069,8 +1110,14 @@ mrb_class_find_path(mrb_state *mrb, struct RClass *c)
str = mrb_sym2name_len(mrb, name, &len);
mrb_str_cat(mrb, path, str, len);
- iv_del(mrb, c->iv, mrb_intern_lit(mrb, "__outer__"), NULL);
- iv_put(mrb, c->iv, mrb_intern_lit(mrb, "__classname__"), path);
- mrb_field_write_barrier_value(mrb, (struct RBasic*)c, path);
+#ifdef MRB_IMPROVE_META_PROGRAMMING
+ if (RSTRING_PTR(path)[0] != '#') {
+#endif
+ iv_del(mrb, c->iv, mrb_intern_lit(mrb, "__outer__"), NULL);
+ iv_put(mrb, c->iv, mrb_intern_lit(mrb, "__classname__"), path);
+ mrb_field_write_barrier_value(mrb, (struct RBasic*)c, path);
+#ifdef MRB_IMPROVE_META_PROGRAMMING
+ }
+#endif
return path;
}