1import ctypes
2import gc
3import unittest
4
5from clang.cindex import AvailabilityKind
6from clang.cindex import CursorKind
7from clang.cindex import TemplateArgumentKind
8from clang.cindex import TranslationUnit
9from clang.cindex import TypeKind
10from .util import get_cursor
11from .util import get_cursors
12from .util import get_tu
13
14
15kInput = """\
16struct s0 {
17  int a;
18  int b;
19};
20
21struct s1;
22
23void f0(int a0, int a1) {
24  int l0, l1;
25
26  if (a0)
27    return;
28
29  for (;;) {
30    break;
31  }
32}
33"""
34
35kParentTest = """\
36        class C {
37            void f();
38        }
39
40        void C::f() { }
41    """
42
43kTemplateArgTest = """\
44        template <int kInt, typename T, bool kBool>
45        void foo();
46
47        template<>
48        void foo<-7, float, true>();
49    """
50
51class TestCursor(unittest.TestCase):
52    def test_get_children(self):
53        tu = get_tu(kInput)
54
55        it = tu.cursor.get_children()
56        tu_nodes = list(it)
57
58        self.assertEqual(len(tu_nodes), 3)
59        for cursor in tu_nodes:
60            self.assertIsNotNone(cursor.translation_unit)
61
62        self.assertNotEqual(tu_nodes[0], tu_nodes[1])
63        self.assertEqual(tu_nodes[0].kind, CursorKind.STRUCT_DECL)
64        self.assertEqual(tu_nodes[0].spelling, 's0')
65        self.assertEqual(tu_nodes[0].is_definition(), True)
66        self.assertEqual(tu_nodes[0].location.file.name, 't.c')
67        self.assertEqual(tu_nodes[0].location.line, 1)
68        self.assertEqual(tu_nodes[0].location.column, 8)
69        self.assertGreater(tu_nodes[0].hash, 0)
70        self.assertIsNotNone(tu_nodes[0].translation_unit)
71
72        s0_nodes = list(tu_nodes[0].get_children())
73        self.assertEqual(len(s0_nodes), 2)
74        self.assertEqual(s0_nodes[0].kind, CursorKind.FIELD_DECL)
75        self.assertEqual(s0_nodes[0].spelling, 'a')
76        self.assertEqual(s0_nodes[0].type.kind, TypeKind.INT)
77        self.assertEqual(s0_nodes[1].kind, CursorKind.FIELD_DECL)
78        self.assertEqual(s0_nodes[1].spelling, 'b')
79        self.assertEqual(s0_nodes[1].type.kind, TypeKind.INT)
80
81        self.assertEqual(tu_nodes[1].kind, CursorKind.STRUCT_DECL)
82        self.assertEqual(tu_nodes[1].spelling, 's1')
83        self.assertEqual(tu_nodes[1].displayname, 's1')
84        self.assertEqual(tu_nodes[1].is_definition(), False)
85
86        self.assertEqual(tu_nodes[2].kind, CursorKind.FUNCTION_DECL)
87        self.assertEqual(tu_nodes[2].spelling, 'f0')
88        self.assertEqual(tu_nodes[2].displayname, 'f0(int, int)')
89        self.assertEqual(tu_nodes[2].is_definition(), True)
90
91    def test_references(self):
92        """Ensure that references to TranslationUnit are kept."""
93        tu = get_tu('int x;')
94        cursors = list(tu.cursor.get_children())
95        self.assertGreater(len(cursors), 0)
96
97        cursor = cursors[0]
98        self.assertIsInstance(cursor.translation_unit, TranslationUnit)
99
100        # Delete reference to TU and perform a full GC.
101        del tu
102        gc.collect()
103        self.assertIsInstance(cursor.translation_unit, TranslationUnit)
104
105        # If the TU was destroyed, this should cause a segfault.
106        parent = cursor.semantic_parent
107
108    def test_canonical(self):
109        source = 'struct X; struct X; struct X { int member; };'
110        tu = get_tu(source)
111
112        cursors = []
113        for cursor in tu.cursor.get_children():
114            if cursor.spelling == 'X':
115                cursors.append(cursor)
116
117        self.assertEqual(len(cursors), 3)
118        self.assertEqual(cursors[1].canonical, cursors[2].canonical)
119
120    def test_is_const_method(self):
121        """Ensure Cursor.is_const_method works."""
122        source = 'class X { void foo() const; void bar(); };'
123        tu = get_tu(source, lang='cpp')
124
125        cls = get_cursor(tu, 'X')
126        foo = get_cursor(tu, 'foo')
127        bar = get_cursor(tu, 'bar')
128        self.assertIsNotNone(cls)
129        self.assertIsNotNone(foo)
130        self.assertIsNotNone(bar)
131
132        self.assertTrue(foo.is_const_method())
133        self.assertFalse(bar.is_const_method())
134
135    def test_is_converting_constructor(self):
136        """Ensure Cursor.is_converting_constructor works."""
137        source = 'class X { explicit X(int); X(double); X(); };'
138        tu = get_tu(source, lang='cpp')
139
140        xs = get_cursors(tu, 'X')
141
142        self.assertEqual(len(xs), 4)
143        self.assertEqual(xs[0].kind, CursorKind.CLASS_DECL)
144        cs = xs[1:]
145        self.assertEqual(cs[0].kind, CursorKind.CONSTRUCTOR)
146        self.assertEqual(cs[1].kind, CursorKind.CONSTRUCTOR)
147        self.assertEqual(cs[2].kind, CursorKind.CONSTRUCTOR)
148
149        self.assertFalse(cs[0].is_converting_constructor())
150        self.assertTrue(cs[1].is_converting_constructor())
151        self.assertFalse(cs[2].is_converting_constructor())
152
153
154    def test_is_copy_constructor(self):
155        """Ensure Cursor.is_copy_constructor works."""
156        source = 'class X { X(); X(const X&); X(X&&); };'
157        tu = get_tu(source, lang='cpp')
158
159        xs = get_cursors(tu, 'X')
160        self.assertEqual(xs[0].kind, CursorKind.CLASS_DECL)
161        cs = xs[1:]
162        self.assertEqual(cs[0].kind, CursorKind.CONSTRUCTOR)
163        self.assertEqual(cs[1].kind, CursorKind.CONSTRUCTOR)
164        self.assertEqual(cs[2].kind, CursorKind.CONSTRUCTOR)
165
166        self.assertFalse(cs[0].is_copy_constructor())
167        self.assertTrue(cs[1].is_copy_constructor())
168        self.assertFalse(cs[2].is_copy_constructor())
169
170    def test_is_default_constructor(self):
171        """Ensure Cursor.is_default_constructor works."""
172        source = 'class X { X(); X(int); };'
173        tu = get_tu(source, lang='cpp')
174
175        xs = get_cursors(tu, 'X')
176        self.assertEqual(xs[0].kind, CursorKind.CLASS_DECL)
177        cs = xs[1:]
178        self.assertEqual(cs[0].kind, CursorKind.CONSTRUCTOR)
179        self.assertEqual(cs[1].kind, CursorKind.CONSTRUCTOR)
180
181        self.assertTrue(cs[0].is_default_constructor())
182        self.assertFalse(cs[1].is_default_constructor())
183
184    def test_is_move_constructor(self):
185        """Ensure Cursor.is_move_constructor works."""
186        source = 'class X { X(); X(const X&); X(X&&); };'
187        tu = get_tu(source, lang='cpp')
188
189        xs = get_cursors(tu, 'X')
190        self.assertEqual(xs[0].kind, CursorKind.CLASS_DECL)
191        cs = xs[1:]
192        self.assertEqual(cs[0].kind, CursorKind.CONSTRUCTOR)
193        self.assertEqual(cs[1].kind, CursorKind.CONSTRUCTOR)
194        self.assertEqual(cs[2].kind, CursorKind.CONSTRUCTOR)
195
196        self.assertFalse(cs[0].is_move_constructor())
197        self.assertFalse(cs[1].is_move_constructor())
198        self.assertTrue(cs[2].is_move_constructor())
199
200    def test_is_default_method(self):
201        """Ensure Cursor.is_default_method works."""
202        source = 'class X { X() = default; }; class Y { Y(); };'
203        tu = get_tu(source, lang='cpp')
204
205        xs = get_cursors(tu, 'X')
206        ys = get_cursors(tu, 'Y')
207
208        self.assertEqual(len(xs), 2)
209        self.assertEqual(len(ys), 2)
210
211        xc = xs[1]
212        yc = ys[1]
213
214        self.assertTrue(xc.is_default_method())
215        self.assertFalse(yc.is_default_method())
216
217    def test_is_mutable_field(self):
218        """Ensure Cursor.is_mutable_field works."""
219        source = 'class X { int x_; mutable int y_; };'
220        tu = get_tu(source, lang='cpp')
221
222        cls = get_cursor(tu, 'X')
223        x_ = get_cursor(tu, 'x_')
224        y_ = get_cursor(tu, 'y_')
225        self.assertIsNotNone(cls)
226        self.assertIsNotNone(x_)
227        self.assertIsNotNone(y_)
228
229        self.assertFalse(x_.is_mutable_field())
230        self.assertTrue(y_.is_mutable_field())
231
232    def test_is_static_method(self):
233        """Ensure Cursor.is_static_method works."""
234
235        source = 'class X { static void foo(); void bar(); };'
236        tu = get_tu(source, lang='cpp')
237
238        cls = get_cursor(tu, 'X')
239        foo = get_cursor(tu, 'foo')
240        bar = get_cursor(tu, 'bar')
241        self.assertIsNotNone(cls)
242        self.assertIsNotNone(foo)
243        self.assertIsNotNone(bar)
244
245        self.assertTrue(foo.is_static_method())
246        self.assertFalse(bar.is_static_method())
247
248    def test_is_pure_virtual_method(self):
249        """Ensure Cursor.is_pure_virtual_method works."""
250        source = 'class X { virtual void foo() = 0; virtual void bar(); };'
251        tu = get_tu(source, lang='cpp')
252
253        cls = get_cursor(tu, 'X')
254        foo = get_cursor(tu, 'foo')
255        bar = get_cursor(tu, 'bar')
256        self.assertIsNotNone(cls)
257        self.assertIsNotNone(foo)
258        self.assertIsNotNone(bar)
259
260        self.assertTrue(foo.is_pure_virtual_method())
261        self.assertFalse(bar.is_pure_virtual_method())
262
263    def test_is_virtual_method(self):
264        """Ensure Cursor.is_virtual_method works."""
265        source = 'class X { virtual void foo(); void bar(); };'
266        tu = get_tu(source, lang='cpp')
267
268        cls = get_cursor(tu, 'X')
269        foo = get_cursor(tu, 'foo')
270        bar = get_cursor(tu, 'bar')
271        self.assertIsNotNone(cls)
272        self.assertIsNotNone(foo)
273        self.assertIsNotNone(bar)
274
275        self.assertTrue(foo.is_virtual_method())
276        self.assertFalse(bar.is_virtual_method())
277
278    def test_is_abstract_record(self):
279        """Ensure Cursor.is_abstract_record works."""
280        source = 'struct X { virtual void x() = 0; }; struct Y : X { void x(); };'
281        tu = get_tu(source, lang='cpp')
282
283        cls = get_cursor(tu, 'X')
284        self.assertTrue(cls.is_abstract_record())
285
286        cls = get_cursor(tu, 'Y')
287        self.assertFalse(cls.is_abstract_record())
288
289    def test_is_scoped_enum(self):
290        """Ensure Cursor.is_scoped_enum works."""
291        source = 'class X {}; enum RegularEnum {}; enum class ScopedEnum {};'
292        tu = get_tu(source, lang='cpp')
293
294        cls = get_cursor(tu, 'X')
295        regular_enum = get_cursor(tu, 'RegularEnum')
296        scoped_enum = get_cursor(tu, 'ScopedEnum')
297        self.assertIsNotNone(cls)
298        self.assertIsNotNone(regular_enum)
299        self.assertIsNotNone(scoped_enum)
300
301        self.assertFalse(cls.is_scoped_enum())
302        self.assertFalse(regular_enum.is_scoped_enum())
303        self.assertTrue(scoped_enum.is_scoped_enum())
304
305    def test_underlying_type(self):
306        tu = get_tu('typedef int foo;')
307        typedef = get_cursor(tu, 'foo')
308        self.assertIsNotNone(typedef)
309
310        self.assertTrue(typedef.kind.is_declaration())
311        underlying = typedef.underlying_typedef_type
312        self.assertEqual(underlying.kind, TypeKind.INT)
313
314    def test_semantic_parent(self):
315        tu = get_tu(kParentTest, 'cpp')
316        curs = get_cursors(tu, 'f')
317        decl = get_cursor(tu, 'C')
318        self.assertEqual(len(curs), 2)
319        self.assertEqual(curs[0].semantic_parent, curs[1].semantic_parent)
320        self.assertEqual(curs[0].semantic_parent, decl)
321
322    def test_lexical_parent(self):
323        tu = get_tu(kParentTest, 'cpp')
324        curs = get_cursors(tu, 'f')
325        decl = get_cursor(tu, 'C')
326        self.assertEqual(len(curs), 2)
327        self.assertNotEqual(curs[0].lexical_parent, curs[1].lexical_parent)
328        self.assertEqual(curs[0].lexical_parent, decl)
329        self.assertEqual(curs[1].lexical_parent, tu.cursor)
330
331    def test_enum_type(self):
332        tu = get_tu('enum TEST { FOO=1, BAR=2 };')
333        enum = get_cursor(tu, 'TEST')
334        self.assertIsNotNone(enum)
335
336        self.assertEqual(enum.kind, CursorKind.ENUM_DECL)
337        enum_type = enum.enum_type
338        self.assertEqual(enum_type.kind, TypeKind.UINT)
339
340    def test_enum_type_cpp(self):
341        tu = get_tu('enum TEST : long long { FOO=1, BAR=2 };', lang="cpp")
342        enum = get_cursor(tu, 'TEST')
343        self.assertIsNotNone(enum)
344
345        self.assertEqual(enum.kind, CursorKind.ENUM_DECL)
346        self.assertEqual(enum.enum_type.kind, TypeKind.LONGLONG)
347
348    def test_objc_type_encoding(self):
349        tu = get_tu('int i;', lang='objc')
350        i = get_cursor(tu, 'i')
351
352        self.assertIsNotNone(i)
353        self.assertEqual(i.objc_type_encoding, 'i')
354
355    def test_enum_values(self):
356        tu = get_tu('enum TEST { SPAM=1, EGG, HAM = EGG * 20};')
357        enum = get_cursor(tu, 'TEST')
358        self.assertIsNotNone(enum)
359
360        self.assertEqual(enum.kind, CursorKind.ENUM_DECL)
361
362        enum_constants = list(enum.get_children())
363        self.assertEqual(len(enum_constants), 3)
364
365        spam, egg, ham = enum_constants
366
367        self.assertEqual(spam.kind, CursorKind.ENUM_CONSTANT_DECL)
368        self.assertEqual(spam.enum_value, 1)
369        self.assertEqual(egg.kind, CursorKind.ENUM_CONSTANT_DECL)
370        self.assertEqual(egg.enum_value, 2)
371        self.assertEqual(ham.kind, CursorKind.ENUM_CONSTANT_DECL)
372        self.assertEqual(ham.enum_value, 40)
373
374    def test_enum_values_cpp(self):
375        tu = get_tu('enum TEST : long long { SPAM = -1, HAM = 0x10000000000};', lang="cpp")
376        enum = get_cursor(tu, 'TEST')
377        self.assertIsNotNone(enum)
378
379        self.assertEqual(enum.kind, CursorKind.ENUM_DECL)
380
381        enum_constants = list(enum.get_children())
382        self.assertEqual(len(enum_constants), 2)
383
384        spam, ham = enum_constants
385
386        self.assertEqual(spam.kind, CursorKind.ENUM_CONSTANT_DECL)
387        self.assertEqual(spam.enum_value, -1)
388        self.assertEqual(ham.kind, CursorKind.ENUM_CONSTANT_DECL)
389        self.assertEqual(ham.enum_value, 0x10000000000)
390
391    def test_annotation_attribute(self):
392        tu = get_tu('int foo (void) __attribute__ ((annotate("here be annotation attribute")));')
393
394        foo = get_cursor(tu, 'foo')
395        self.assertIsNotNone(foo)
396
397        for c in foo.get_children():
398            if c.kind == CursorKind.ANNOTATE_ATTR:
399                self.assertEqual(c.displayname, "here be annotation attribute")
400                break
401        else:
402            self.fail("Couldn't find annotation")
403
404    def test_annotation_template(self):
405        annotation = '__attribute__ ((annotate("annotation")))'
406        for source, kind in [
407                ('int foo (T value) %s;', CursorKind.FUNCTION_TEMPLATE),
408                ('class %s foo {};', CursorKind.CLASS_TEMPLATE),
409        ]:
410            source = 'template<typename T> ' + (source % annotation)
411            tu = get_tu(source, lang="cpp")
412
413            foo = get_cursor(tu, 'foo')
414            self.assertIsNotNone(foo)
415            self.assertEqual(foo.kind, kind)
416
417            for c in foo.get_children():
418                if c.kind == CursorKind.ANNOTATE_ATTR:
419                    self.assertEqual(c.displayname, "annotation")
420                    break
421            else:
422                self.fail("Couldn't find annotation for {}".format(kind))
423
424    def test_result_type(self):
425        tu = get_tu('int foo();')
426        foo = get_cursor(tu, 'foo')
427
428        self.assertIsNotNone(foo)
429        t = foo.result_type
430        self.assertEqual(t.kind, TypeKind.INT)
431
432    def test_availability(self):
433        tu = get_tu('class A { A(A const&) = delete; };', lang='cpp')
434
435        # AvailabilityKind.AVAILABLE
436        cursor = get_cursor(tu, 'A')
437        self.assertEqual(cursor.kind, CursorKind.CLASS_DECL)
438        self.assertEqual(cursor.availability, AvailabilityKind.AVAILABLE)
439
440        # AvailabilityKind.NOT_AVAILABLE
441        cursors = get_cursors(tu, 'A')
442        for c in cursors:
443            if c.kind == CursorKind.CONSTRUCTOR:
444                self.assertEqual(c.availability, AvailabilityKind.NOT_AVAILABLE)
445                break
446        else:
447            self.fail("Could not find cursor for deleted constructor")
448
449        # AvailabilityKind.DEPRECATED
450        tu = get_tu('void test() __attribute__((deprecated));', lang='cpp')
451        cursor = get_cursor(tu, 'test')
452        self.assertEqual(cursor.availability, AvailabilityKind.DEPRECATED)
453
454        # AvailabilityKind.NOT_ACCESSIBLE is only used in the code completion results
455
456    def test_get_tokens(self):
457        """Ensure we can map cursors back to tokens."""
458        tu = get_tu('int foo(int i);')
459        foo = get_cursor(tu, 'foo')
460
461        tokens = list(foo.get_tokens())
462        self.assertEqual(len(tokens), 6)
463        self.assertEqual(tokens[0].spelling, 'int')
464        self.assertEqual(tokens[1].spelling, 'foo')
465
466    def test_get_token_cursor(self):
467        """Ensure we can map tokens to cursors."""
468        tu = get_tu('class A {}; int foo(A var = A());', lang='cpp')
469        foo = get_cursor(tu, 'foo')
470
471        for cursor in foo.walk_preorder():
472            if cursor.kind.is_expression() and not cursor.kind.is_statement():
473                break
474        else:
475            self.fail("Could not find default value expression")
476
477        tokens = list(cursor.get_tokens())
478        self.assertEqual(len(tokens), 4, [t.spelling for t in tokens])
479        self.assertEqual(tokens[0].spelling, '=')
480        self.assertEqual(tokens[1].spelling, 'A')
481        self.assertEqual(tokens[2].spelling, '(')
482        self.assertEqual(tokens[3].spelling, ')')
483        t_cursor = tokens[1].cursor
484        self.assertEqual(t_cursor.kind, CursorKind.TYPE_REF)
485        r_cursor = t_cursor.referenced # should not raise an exception
486        self.assertEqual(r_cursor.kind, CursorKind.CLASS_DECL)
487
488    def test_get_arguments(self):
489        tu = get_tu('void foo(int i, int j);')
490        foo = get_cursor(tu, 'foo')
491        arguments = list(foo.get_arguments())
492
493        self.assertEqual(len(arguments), 2)
494        self.assertEqual(arguments[0].spelling, "i")
495        self.assertEqual(arguments[1].spelling, "j")
496
497    def test_get_num_template_arguments(self):
498        tu = get_tu(kTemplateArgTest, lang='cpp')
499        foos = get_cursors(tu, 'foo')
500
501        self.assertEqual(foos[1].get_num_template_arguments(), 3)
502
503    def test_get_template_argument_kind(self):
504        tu = get_tu(kTemplateArgTest, lang='cpp')
505        foos = get_cursors(tu, 'foo')
506
507        self.assertEqual(foos[1].get_template_argument_kind(0), TemplateArgumentKind.INTEGRAL)
508        self.assertEqual(foos[1].get_template_argument_kind(1), TemplateArgumentKind.TYPE)
509        self.assertEqual(foos[1].get_template_argument_kind(2), TemplateArgumentKind.INTEGRAL)
510
511    def test_get_template_argument_type(self):
512        tu = get_tu(kTemplateArgTest, lang='cpp')
513        foos = get_cursors(tu, 'foo')
514
515        self.assertEqual(foos[1].get_template_argument_type(1).kind, TypeKind.FLOAT)
516
517    def test_get_template_argument_value(self):
518        tu = get_tu(kTemplateArgTest, lang='cpp')
519        foos = get_cursors(tu, 'foo')
520
521        self.assertEqual(foos[1].get_template_argument_value(0), -7)
522        self.assertEqual(foos[1].get_template_argument_value(2), True)
523
524    def test_get_template_argument_unsigned_value(self):
525        tu = get_tu(kTemplateArgTest, lang='cpp')
526        foos = get_cursors(tu, 'foo')
527
528        self.assertEqual(foos[1].get_template_argument_unsigned_value(0), 2 ** 32 - 7)
529        self.assertEqual(foos[1].get_template_argument_unsigned_value(2), True)
530
531    def test_referenced(self):
532        tu = get_tu('void foo(); void bar() { foo(); }')
533        foo = get_cursor(tu, 'foo')
534        bar = get_cursor(tu, 'bar')
535        for c in bar.get_children():
536            if c.kind == CursorKind.CALL_EXPR:
537                self.assertEqual(c.referenced.spelling, foo.spelling)
538                break
539
540    def test_mangled_name(self):
541        kInputForMangling = """\
542        int foo(int, int);
543        """
544        tu = get_tu(kInputForMangling, lang='cpp')
545        foo = get_cursor(tu, 'foo')
546
547        # Since libclang does not link in targets, we cannot pass a triple to it
548        # and force the target. To enable this test to pass on all platforms, accept
549        # all valid manglings.
550        # [c-index-test handles this by running the source through clang, emitting
551        #  an AST file and running libclang on that AST file]
552        self.assertIn(foo.mangled_name, ('_Z3fooii', '__Z3fooii', '?foo@@YAHHH'))
553