summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJeremy Evans <code@jeremyevans.net>2021-02-09 15:39:34 -0800
committerJeremy Evans <code@jeremyevans.net>2021-02-11 07:45:52 -0800
commit2d877327e12f499ac1a0d7096f4314cc92ef228e (patch)
treee20d0748b139036111a0d9c0c6f1cbb2606576fd
parentb83ad02997dfa910c08dd2d900f24468c15db264 (diff)
Make a cyclic prepend not modify ancestors for the receiver
Check for cyclic prepend before making any changes. This requires scanning the module ancestor chain twice, but in general modules do not have large numbers of ancestors.
Notes
Notes: Merged: https://github.com/ruby/ruby/pull/4165
-rw-r--r--class.c14
-rw-r--r--test/ruby/test_module.rb7
2 files changed, 19 insertions, 2 deletions
diff --git a/class.c b/class.c
index f97956c1d3..12a67d16bc 100644
--- a/class.c
+++ b/class.c
@@ -1024,6 +1024,18 @@ include_modules_at(const VALUE klass, VALUE c, VALUE module, int search_super)
struct rb_id_table *const klass_m_tbl = RCLASS_M_TBL(klass_origin);
VALUE original_klass = klass;
+ if (klass_m_tbl) {
+ VALUE original_module = module;
+
+ while (module) {
+ if (klass_m_tbl == RCLASS_M_TBL(module))
+ return -1;
+ module = RCLASS_SUPER(module);
+ }
+
+ module = original_module;
+ }
+
while (module) {
int c_seen = FALSE;
int superclass_seen = FALSE;
@@ -1032,8 +1044,6 @@ include_modules_at(const VALUE klass, VALUE c, VALUE module, int search_super)
if (klass == c) {
c_seen = TRUE;
}
- if (klass_m_tbl && klass_m_tbl == RCLASS_M_TBL(module))
- return -1;
if (klass_origin != c || search_super) {
/* ignore if the module included already in superclasses for include,
* ignore if the module included before origin class for prepend
diff --git a/test/ruby/test_module.rb b/test/ruby/test_module.rb
index 6b53b7d5f4..0a5597fd6c 100644
--- a/test/ruby/test_module.rb
+++ b/test/ruby/test_module.rb
@@ -478,6 +478,13 @@ class TestModule < Test::Unit::TestCase
assert_raise(ArgumentError) { Module.new { include } }
end
+ def test_prepend_self
+ m = Module.new
+ assert_equal([m], m.ancestors)
+ m.prepend(m) rescue nil
+ assert_equal([m], m.ancestors)
+ end
+
def test_prepend_works_with_duped_classes
m = Module.new
a = Class.new do