pub mod apidoc;
pub mod apidoc_data;
pub mod apidoc_patches;
pub mod ast;
pub mod c_fn_decl;
pub mod error;
pub mod enum_dict;
pub mod fields_dict;
pub mod global_const_dict;
pub mod infer_api;
pub mod inline_fn;
pub mod intern;
pub mod lexer;
pub mod macro_def;
pub mod macro_infer;
pub mod parser;
pub mod perl_config;
pub mod perlvar_dict;
pub mod perlvar_emitter;
pub mod pipeline;
pub mod pp_expr;
pub mod preprocessor;
pub mod rust_codegen;
pub mod rust_decl;
pub mod static_array_emitter;
pub mod struct_emitter;
pub mod semantic;
pub mod sexp;
pub mod syn_codegen;
pub mod source;
pub mod token;
pub mod token_source;
pub mod type_env;
pub mod type_repr;
pub mod unified_type;
pub use apidoc::{
find_apidoc_dir_from, resolve_apidoc_path,
ApidocArg, ApidocCollector, ApidocDict, ApidocEntry, ApidocFlags, ApidocResolveError, ApidocStats, Nullability,
};
pub use infer_api::{
run_inference_with_preprocessor,
DebugOptions, InferConfig, InferError, InferResult, InferStats, TypedefDict,
};
pub use ast::*;
pub use error::{CompileError, DisplayLocation, LexError, PPError, ParseError, Result};
pub use fields_dict::FieldsDict;
pub use inline_fn::InlineFnDict;
pub use intern::{InternedStr, StringInterner};
pub use rust_decl::RustDeclDict;
pub use lexer::{IdentResolver, Interning, Lexer, LookupOnly, MutableLexer, ReadOnlyLexer};
pub use macro_def::{MacroDef, MacroKind, MacroTable};
pub use macro_infer::{
convert_assert_calls_in_compound_stmt, detect_assert_kind, InferStatus, MacroInferContext,
MacroInferInfo, MacroInferStats, NoExpandSymbols, ParseResult,
};
pub use parser::{parse_expression_from_tokens, parse_expression_from_tokens_ref, parse_type_from_string, Parser};
pub use perl_config::{
build_pp_config_for_perl, get_default_target_dir, get_perl_config, get_perl_version,
PerlConfig, PerlConfigError,
};
pub use perlvar_dict::{
ArrayLength, PerlvarCollector, PerlvarDict, PerlvarEntry, PerlvarKind,
};
pub use preprocessor::{
CallbackPair, CommentCallback, MacroCalledCallback, MacroCallWatcher, MacroDefCallback,
PPConfig, Preprocessor,
};
pub use semantic::{SemanticAnalyzer, Symbol, SymbolKind, Type};
pub use sexp::{SexpPrinter, TypedSexpPrinter};
pub use source::{FileId, FileRegistry, SourceLocation};
pub use token::{Comment, CommentKind, Token, TokenKind};
pub use token_source::{TokenSlice, TokenSliceRef, TokenSource};
pub use type_env::{ParamLink, TypeConstraint, TypeEnv};
pub use type_repr::{
CDerivedType, CPrimitiveKind, CTypeSource, CTypeSpecs, InferredType,
IntSize as TypeReprIntSize, RustPrimitiveKind, RustTypeRepr, RustTypeSource, TypeRepr,
};
pub use unified_type::{IntSize, SourcedType, TypeSource, UnifiedType};
pub use rust_codegen::{CodegenConfig, CodegenDriver, CodegenStats, GeneratedCode, GenerateStatus, RustCodegen};
pub use pipeline::{
Pipeline, PipelineBuilder, PipelineError,
PreprocessConfig, InferConfig as PipelineInferConfig, CodegenConfig as PipelineCodegenConfig,
PreprocessedPipeline, InferredPipeline, GeneratedPipeline,
};
#[cfg(test)]
mod tests {
use super::*;
use std::path::PathBuf;
#[test]
fn test_basic_lexer_integration() {
let source = b"int main(void) { return 0; }";
let mut files = FileRegistry::new();
let file_id = files.register(PathBuf::from("test.c"));
let mut interner = StringInterner::new();
let mut lexer = Lexer::new(source, file_id, &mut interner);
let mut tokens = Vec::new();
loop {
let token = lexer.next_token().unwrap();
if matches!(token.kind, TokenKind::Eof) {
break;
}
tokens.push(token);
}
assert_eq!(tokens.len(), 10);
assert!(matches!(tokens[0].kind, TokenKind::KwInt));
assert!(matches!(tokens[1].kind, TokenKind::Ident(_))); assert!(matches!(tokens[2].kind, TokenKind::LParen));
assert!(matches!(tokens[3].kind, TokenKind::KwVoid));
assert!(matches!(tokens[4].kind, TokenKind::RParen));
assert!(matches!(tokens[5].kind, TokenKind::LBrace));
assert!(matches!(tokens[6].kind, TokenKind::KwReturn));
assert!(matches!(tokens[7].kind, TokenKind::IntLit(0)));
assert!(matches!(tokens[8].kind, TokenKind::Semi));
assert!(matches!(tokens[9].kind, TokenKind::RBrace));
if let TokenKind::Ident(id) = tokens[1].kind {
assert_eq!(interner.get(id), "main");
} else {
panic!("Expected identifier for 'main'");
}
}
#[test]
fn test_comment_preservation() {
let source = b"// doc comment\nint x;";
let mut files = FileRegistry::new();
let file_id = files.register(PathBuf::from("test.c"));
let mut interner = StringInterner::new();
let mut lexer = Lexer::new(source, file_id, &mut interner);
let newline = lexer.next_token().unwrap();
assert!(matches!(newline.kind, TokenKind::Newline));
assert_eq!(newline.leading_comments.len(), 1);
assert!(newline.leading_comments[0].text.contains("doc comment"));
let token = lexer.next_token().unwrap();
assert!(matches!(token.kind, TokenKind::KwInt));
}
}