about summary refs log tree commit diff
path: root/compiler/rustc_builtin_macros/src/define_opaque.rs
blob: cd02e81f5689c56dfe179c0aa3b000e8cf4ff7d2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
use rustc_ast::{DUMMY_NODE_ID, ast};
use rustc_expand::base::{Annotatable, ExtCtxt};
use rustc_span::Span;

pub(crate) fn expand(
    ecx: &mut ExtCtxt<'_>,
    _expand_span: Span,
    meta_item: &ast::MetaItem,
    mut item: Annotatable,
) -> Vec<Annotatable> {
    let define_opaque = match &mut item {
        Annotatable::Item(p) => match &mut p.kind {
            ast::ItemKind::Fn(f) => Some(&mut f.define_opaque),
            ast::ItemKind::Const(ct) => Some(&mut ct.define_opaque),
            ast::ItemKind::Static(si) => Some(&mut si.define_opaque),
            _ => None,
        },
        Annotatable::AssocItem(i, _assoc_ctxt) => match &mut i.kind {
            ast::AssocItemKind::Fn(func) => Some(&mut func.define_opaque),
            ast::AssocItemKind::Const(ct) => Some(&mut ct.define_opaque),
            _ => None,
        },
        Annotatable::Stmt(s) => match &mut s.kind {
            ast::StmtKind::Item(p) => match &mut p.kind {
                ast::ItemKind::Fn(f) => Some(&mut f.define_opaque),
                ast::ItemKind::Const(ct) => Some(&mut ct.define_opaque),
                ast::ItemKind::Static(si) => Some(&mut si.define_opaque),
                _ => None,
            },
            _ => None,
        },
        _ => None,
    };

    let Some(list) = meta_item.meta_item_list() else {
        ecx.dcx().span_err(meta_item.span, "expected list of type aliases");
        return vec![item];
    };

    if let Some(define_opaque) = define_opaque {
        *define_opaque = Some(
            list.iter()
                .filter_map(|entry| match entry {
                    ast::MetaItemInner::MetaItem(meta_item) if meta_item.is_word() => {
                        Some((DUMMY_NODE_ID, meta_item.path.clone()))
                    }
                    _ => {
                        ecx.dcx().span_err(entry.span(), "expected path to type alias");
                        None
                    }
                })
                .collect(),
        );
    } else {
        ecx.dcx().span_err(
            meta_item.span,
            "only functions, statics, and consts can define opaque types",
        );
    }

    vec![item]
}