| 1 | //! Intermediate representation for modules (AKA C++ namespaces). |
| 2 | |
| 3 | use super::context::BindgenContext; |
| 4 | use super::dot::DotAttributes; |
| 5 | use super::item::ItemSet; |
| 6 | use crate::clang; |
| 7 | use crate::parse::{ClangSubItemParser, ParseError, ParseResult}; |
| 8 | use crate::parse_one; |
| 9 | |
| 10 | use std::io; |
| 11 | |
| 12 | /// Whether this module is inline or not. |
| 13 | #[derive (Debug, Copy, Clone, PartialEq, Eq)] |
| 14 | pub(crate) enum ModuleKind { |
| 15 | /// This module is not inline. |
| 16 | Normal, |
| 17 | /// This module is inline, as in `inline namespace foo {}`. |
| 18 | Inline, |
| 19 | } |
| 20 | |
| 21 | /// A module, as in, a C++ namespace. |
| 22 | #[derive (Clone, Debug)] |
| 23 | pub(crate) struct Module { |
| 24 | /// The name of the module, or none if it's anonymous. |
| 25 | name: Option<String>, |
| 26 | /// The kind of module this is. |
| 27 | kind: ModuleKind, |
| 28 | /// The children of this module, just here for convenience. |
| 29 | children: ItemSet, |
| 30 | } |
| 31 | |
| 32 | impl Module { |
| 33 | /// Construct a new `Module`. |
| 34 | pub(crate) fn new(name: Option<String>, kind: ModuleKind) -> Self { |
| 35 | Module { |
| 36 | name, |
| 37 | kind, |
| 38 | children: ItemSet::new(), |
| 39 | } |
| 40 | } |
| 41 | |
| 42 | /// Get this module's name. |
| 43 | pub(crate) fn name(&self) -> Option<&str> { |
| 44 | self.name.as_deref() |
| 45 | } |
| 46 | |
| 47 | /// Get a mutable reference to this module's children. |
| 48 | pub(crate) fn children_mut(&mut self) -> &mut ItemSet { |
| 49 | &mut self.children |
| 50 | } |
| 51 | |
| 52 | /// Get this module's children. |
| 53 | pub(crate) fn children(&self) -> &ItemSet { |
| 54 | &self.children |
| 55 | } |
| 56 | |
| 57 | /// Whether this namespace is inline. |
| 58 | pub(crate) fn is_inline(&self) -> bool { |
| 59 | self.kind == ModuleKind::Inline |
| 60 | } |
| 61 | } |
| 62 | |
| 63 | impl DotAttributes for Module { |
| 64 | fn dot_attributes<W>( |
| 65 | &self, |
| 66 | _ctx: &BindgenContext, |
| 67 | out: &mut W, |
| 68 | ) -> io::Result<()> |
| 69 | where |
| 70 | W: io::Write, |
| 71 | { |
| 72 | writeln!(out, "<tr><td>ModuleKind</td><td> {:?}</td></tr>" , self.kind) |
| 73 | } |
| 74 | } |
| 75 | |
| 76 | impl ClangSubItemParser for Module { |
| 77 | fn parse( |
| 78 | cursor: clang::Cursor, |
| 79 | ctx: &mut BindgenContext, |
| 80 | ) -> Result<ParseResult<Self>, ParseError> { |
| 81 | use clang_sys::*; |
| 82 | match cursor.kind() { |
| 83 | CXCursor_Namespace => { |
| 84 | let module_id: ModuleId = ctx.module(cursor); |
| 85 | ctx.with_module(module_id, |ctx: &mut BindgenContext| { |
| 86 | cursor.visit_sorted(ctx, |ctx: &mut BindgenContext, child: Cursor| { |
| 87 | parse_one(ctx, cursor:child, parent:Some(module_id.into())) |
| 88 | }) |
| 89 | }); |
| 90 | |
| 91 | Ok(ParseResult::AlreadyResolved(module_id.into())) |
| 92 | } |
| 93 | _ => Err(ParseError::Continue), |
| 94 | } |
| 95 | } |
| 96 | } |
| 97 | |