blob: 977f7b00e2ed33764596f3807b4ebefd305a59e5 [file] [log] [blame]
Marc-André Lemburg36619082001-01-17 19:11:13 +00001from test_support import verify, verbose, TestFailed
Neil Schemenauer88c761a2001-07-12 13:25:53 +00002import sys
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +00003import gc
4
Guido van Rossumc907bd82001-10-02 19:49:47 +00005def expect(actual, expected, name):
6 if actual != expected:
7 raise TestFailed, "test_%s: actual %d, expected %d" % (
8 name, actual, expected)
9
Guido van Rossum048eb752001-10-02 21:24:57 +000010def expect_nonzero(actual, name):
11 if actual == 0:
12 raise TestFailed, "test_%s: unexpected zero" % name
Guido van Rossumc907bd82001-10-02 19:49:47 +000013
Neil Schemenauerfaae2662000-09-22 15:26:20 +000014def run_test(name, thunk):
15 if verbose:
16 print "testing %s..." % name,
Guido van Rossumc907bd82001-10-02 19:49:47 +000017 thunk()
18 if verbose:
19 print "ok"
Neil Schemenauerfaae2662000-09-22 15:26:20 +000020
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000021def test_list():
22 l = []
23 l.append(l)
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000024 gc.collect()
25 del l
Guido van Rossumc907bd82001-10-02 19:49:47 +000026 expect(gc.collect(), 1, "list")
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000027
28def test_dict():
29 d = {}
30 d[1] = d
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000031 gc.collect()
32 del d
Guido van Rossumc907bd82001-10-02 19:49:47 +000033 expect(gc.collect(), 1, "dict")
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000034
35def test_tuple():
Neil Schemenauera53cf792000-09-15 22:32:29 +000036 # since tuples are immutable we close the loop with a list
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000037 l = []
38 t = (l,)
39 l.append(t)
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000040 gc.collect()
41 del t
42 del l
Guido van Rossumc907bd82001-10-02 19:49:47 +000043 expect(gc.collect(), 2, "tuple")
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000044
45def test_class():
46 class A:
47 pass
48 A.a = A
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000049 gc.collect()
50 del A
Guido van Rossum048eb752001-10-02 21:24:57 +000051 expect_nonzero(gc.collect(), "class")
52
Tim Peters1ce150c2001-10-15 22:49:27 +000053def test_newstyleclass():
Guido van Rossum048eb752001-10-02 21:24:57 +000054 class A(object):
Tim Peters1ce150c2001-10-15 22:49:27 +000055 pass
Guido van Rossum048eb752001-10-02 21:24:57 +000056 gc.collect()
57 del A
58 expect_nonzero(gc.collect(), "staticclass")
59
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000060def test_instance():
61 class A:
62 pass
63 a = A()
64 a.a = a
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000065 gc.collect()
66 del a
Guido van Rossum048eb752001-10-02 21:24:57 +000067 expect_nonzero(gc.collect(), "instance")
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000068
Guido van Rossum9475a232001-10-05 20:51:39 +000069def test_newinstance():
70 class A(object):
71 pass
72 a = A()
73 a.a = a
74 gc.collect()
75 del a
76 expect_nonzero(gc.collect(), "newinstance")
77 class B(list):
78 pass
79 class C(B, A):
80 pass
81 a = C()
82 a.a = a
83 gc.collect()
84 del a
85 expect_nonzero(gc.collect(), "newinstance(2)")
Guido van Rossum05e01ee2002-06-12 14:38:04 +000086 del B, C
87 expect_nonzero(gc.collect(), "newinstance(3)")
88 A.a = A()
89 del A
90 expect_nonzero(gc.collect(), "newinstance(4)")
91 expect(gc.collect(), 0, "newinstance(5)")
Guido van Rossum9475a232001-10-05 20:51:39 +000092
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000093def test_method():
Neil Schemenauera53cf792000-09-15 22:32:29 +000094 # Tricky: self.__init__ is a bound method, it references the instance.
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +000095 class A:
96 def __init__(self):
97 self.init = self.__init__
98 a = A()
99 gc.collect()
100 del a
Guido van Rossum048eb752001-10-02 21:24:57 +0000101 expect_nonzero(gc.collect(), "method")
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +0000102
103def test_finalizer():
Neil Schemenauera53cf792000-09-15 22:32:29 +0000104 # A() is uncollectable if it is part of a cycle, make sure it shows up
105 # in gc.garbage.
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +0000106 class A:
107 def __del__(self): pass
108 class B:
109 pass
110 a = A()
111 a.a = a
112 id_a = id(a)
113 b = B()
114 b.b = b
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +0000115 gc.collect()
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +0000116 del a
117 del b
Guido van Rossum048eb752001-10-02 21:24:57 +0000118 expect_nonzero(gc.collect(), "finalizer")
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000119 for obj in gc.garbage:
120 if id(obj) == id_a:
121 del obj.a
122 break
123 else:
Guido van Rossumc907bd82001-10-02 19:49:47 +0000124 raise TestFailed, "didn't find obj in garbage (finalizer)"
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000125 gc.garbage.remove(obj)
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +0000126
127def test_function():
Neil Schemenauera53cf792000-09-15 22:32:29 +0000128 # Tricky: f -> d -> f, code should call d.clear() after the exec to
129 # break the cycle.
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +0000130 d = {}
131 exec("def f(): pass\n") in d
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +0000132 gc.collect()
133 del d
Guido van Rossumc907bd82001-10-02 19:49:47 +0000134 expect(gc.collect(), 2, "function")
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000135
Neil Schemenauer88c761a2001-07-12 13:25:53 +0000136def test_frame():
137 def f():
138 frame = sys._getframe()
139 gc.collect()
140 f()
Guido van Rossumc907bd82001-10-02 19:49:47 +0000141 expect(gc.collect(), 1, "frame")
Neil Schemenauer88c761a2001-07-12 13:25:53 +0000142
143
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000144def test_saveall():
145 # Verify that cyclic garbage like lists show up in gc.garbage if the
146 # SAVEALL option is enabled.
147 debug = gc.get_debug()
148 gc.set_debug(debug | gc.DEBUG_SAVEALL)
149 l = []
150 l.append(l)
151 id_l = id(l)
152 del l
153 gc.collect()
154 try:
155 for obj in gc.garbage:
156 if id(obj) == id_l:
157 del obj[:]
158 break
159 else:
Guido van Rossumc907bd82001-10-02 19:49:47 +0000160 raise TestFailed, "didn't find obj in garbage (saveall)"
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000161 gc.garbage.remove(obj)
162 finally:
163 gc.set_debug(debug)
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +0000164
Neil Schemenauera53cf792000-09-15 22:32:29 +0000165def test_del():
166 # __del__ methods can trigger collection, make this to happen
167 thresholds = gc.get_threshold()
168 gc.enable()
169 gc.set_threshold(1)
170
Fred Drake004d5e62000-10-23 17:22:08 +0000171 class A:
172 def __del__(self):
173 dir(self)
Neil Schemenauera53cf792000-09-15 22:32:29 +0000174 a = A()
175 del a
176
177 gc.disable()
178 apply(gc.set_threshold, thresholds)
Fred Drake004d5e62000-10-23 17:22:08 +0000179
Tim Petersd2225592002-03-28 21:08:30 +0000180class Ouch:
181 n = 0
182 def __del__(self):
183 Ouch.n = Ouch.n + 1
Tim Petersc62b95e2002-07-11 19:07:45 +0000184 if Ouch.n % 17 == 0:
Tim Petersd2225592002-03-28 21:08:30 +0000185 gc.collect()
186
187def test_trashcan():
188 # "trashcan" is a hack to prevent stack overflow when deallocating
189 # very deeply nested tuples etc. It works in part by abusing the
190 # type pointer and refcount fields, and that can yield horrible
191 # problems when gc tries to traverse the structures.
192 # If this test fails (as it does in 2.0, 2.1 and 2.2), it will
193 # most likely die via segfault.
194
Tim Petersc62b95e2002-07-11 19:07:45 +0000195 # Note: In 2.3 the possibility for compiling without cyclic gc was
196 # removed, and that in turn allows the trashcan mechanism to work
197 # via much simpler means (e.g., it never abuses the type pointer or
198 # refcount fields anymore). Since it's much less likely to cause a
199 # problem now, the various constants in this expensive (we force a lot
200 # of full collections) test are cut back from the 2.2 version.
Tim Petersd2225592002-03-28 21:08:30 +0000201 gc.enable()
Tim Petersc62b95e2002-07-11 19:07:45 +0000202 N = 150
203 for count in range(2):
Tim Petersd2225592002-03-28 21:08:30 +0000204 t = []
205 for i in range(N):
206 t = [t, Ouch()]
207 u = []
208 for i in range(N):
209 u = [u, Ouch()]
210 v = {}
211 for i in range(N):
212 v = {1: v, 2: Ouch()}
213 gc.disable()
Jeremy Hyltonc5007aa2000-06-30 05:02:53 +0000214
215def test_all():
Guido van Rossumc907bd82001-10-02 19:49:47 +0000216 gc.collect() # Delete 2nd generation garbage
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000217 run_test("lists", test_list)
218 run_test("dicts", test_dict)
219 run_test("tuples", test_tuple)
220 run_test("classes", test_class)
Tim Peters1ce150c2001-10-15 22:49:27 +0000221 run_test("new style classes", test_newstyleclass)
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000222 run_test("instances", test_instance)
Guido van Rossum9475a232001-10-05 20:51:39 +0000223 run_test("new instances", test_newinstance)
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000224 run_test("methods", test_method)
225 run_test("functions", test_function)
Neil Schemenauer88c761a2001-07-12 13:25:53 +0000226 run_test("frames", test_frame)
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000227 run_test("finalizers", test_finalizer)
228 run_test("__del__", test_del)
229 run_test("saveall", test_saveall)
Tim Petersd2225592002-03-28 21:08:30 +0000230 run_test("trashcan", test_trashcan)
Vladimir Marangozovf9d20c32000-08-06 22:45:31 +0000231
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000232def test():
233 if verbose:
234 print "disabling automatic collection"
Vladimir Marangozovf9d20c32000-08-06 22:45:31 +0000235 enabled = gc.isenabled()
236 gc.disable()
Guido van Rossum3fcd4522002-06-13 11:53:12 +0000237 verify(not gc.isenabled())
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000238 debug = gc.get_debug()
239 gc.set_debug(debug & ~gc.DEBUG_LEAK) # this test is supposed to leak
Vladimir Marangozovf9d20c32000-08-06 22:45:31 +0000240
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000241 try:
242 test_all()
243 finally:
244 gc.set_debug(debug)
245 # test gc.enable() even if GC is disabled by default
246 if verbose:
247 print "restoring automatic collection"
248 # make sure to always test gc.enable()
249 gc.enable()
Marc-André Lemburg36619082001-01-17 19:11:13 +0000250 verify(gc.isenabled())
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000251 if not enabled:
252 gc.disable()
Vladimir Marangozovf9d20c32000-08-06 22:45:31 +0000253
254
Neil Schemenauerfaae2662000-09-22 15:26:20 +0000255test()