Skip to content

Commit

Permalink
Simplify test setup.
Browse files Browse the repository at this point in the history
  • Loading branch information
reitermarkus committed Sep 17, 2022
1 parent 5449aa3 commit 9e59566
Show file tree
Hide file tree
Showing 2 changed files with 58 additions and 138 deletions.
6 changes: 5 additions & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -16,4 +16,8 @@ travis-ci = { repository = "jethrogb/rust-cexpr" }
nom = { version = "7", default-features = false, features = ["std"] }

[dev-dependencies]
clang-sys = ">= 0.13.0, < 0.29.0"
clang = { version = "2", features = ["runtime"] }

[patch.crates-io]
clang-sys = { git = "https://github.com/reitermarkus/clang-sys", branch = "load-api" }
clang = { git = "https://github.com/reitermarkus/clang-rs", branch = "load-api" }
190 changes: 53 additions & 137 deletions tests/clang.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,19 +6,18 @@
// option. This file may not be copied, modified, or distributed
// except according to those terms.
extern crate cexpr;
extern crate clang_sys;

use std::collections::HashMap;
use std::io::Write;
use std::str::{self, FromStr};
use std::{char, ffi, mem, ptr, slice};
use std::char;
use std::num::Wrapping;

use cexpr::assert_full_parse;
use cexpr::expr::{fn_macro_declaration, EvalResult, IdentifierParser};
use cexpr::literal::CChar;
use cexpr::token::Token;
use clang_sys::*;
use clang::{source::SourceRange, token::TokenKind, EntityKind};

// main testing routine
fn test_definition(
Expand Down Expand Up @@ -179,162 +178,79 @@ fn test_definition(
}
}

// support code for the clang lexer
unsafe fn clang_str_to_vec(s: CXString) -> Vec<u8> {
let vec = ffi::CStr::from_ptr(clang_getCString(s))
.to_bytes()
.to_owned();
clang_disposeString(s);
vec
}

#[allow(non_upper_case_globals)]
unsafe fn token_clang_to_cexpr(tu: CXTranslationUnit, orig: &CXToken) -> Token {
fn token_clang_to_cexpr(token: &clang::token::Token) -> Token {
Token {
kind: match clang_getTokenKind(*orig) {
CXToken_Comment => cexpr::token::Kind::Comment,
CXToken_Identifier => cexpr::token::Kind::Identifier,
CXToken_Keyword => cexpr::token::Kind::Keyword,
CXToken_Literal => cexpr::token::Kind::Literal,
CXToken_Punctuation => cexpr::token::Kind::Punctuation,
_ => panic!("invalid token kind: {:?}", *orig),
kind: match token.get_kind() {
TokenKind::Comment => cexpr::token::Kind::Comment,
TokenKind::Identifier => cexpr::token::Kind::Identifier,
TokenKind::Keyword => cexpr::token::Kind::Keyword,
TokenKind::Literal => cexpr::token::Kind::Literal,
TokenKind::Punctuation => cexpr::token::Kind::Punctuation,
},
raw: clang_str_to_vec(clang_getTokenSpelling(tu, *orig)).into_boxed_slice(),
raw: token.get_spelling().into_bytes().into_boxed_slice(),
}
}

extern "C" fn visit_children_thunk<F>(
cur: CXCursor,
parent: CXCursor,
closure: CXClientData,
) -> CXChildVisitResult
where
F: FnMut(CXCursor, CXCursor) -> CXChildVisitResult,
{
unsafe { (&mut *(closure as *mut F))(cur, parent) }
}

unsafe fn visit_children<F>(cursor: CXCursor, mut f: F)
where
F: FnMut(CXCursor, CXCursor) -> CXChildVisitResult,
{
clang_visitChildren(
cursor,
visit_children_thunk::<F> as _,
&mut f as *mut F as CXClientData,
);
fn location_in_scope(r: &SourceRange) -> bool {
let start = r.get_start();
let location = start.get_spelling_location();
start.is_in_main_file() && !start.is_in_system_header() && location.file.is_some()
}

unsafe fn location_in_scope(r: CXSourceRange) -> bool {
let start = clang_getRangeStart(r);
let mut file = ptr::null_mut();
clang_getSpellingLocation(
start,
&mut file,
ptr::null_mut(),
ptr::null_mut(),
ptr::null_mut(),
);
clang_Location_isFromMainFile(start) != 0
&& clang_Location_isInSystemHeader(start) == 0
&& file != ptr::null_mut()
}

/// tokenize_range_adjust can be used to work around LLVM bug 9069
/// https://bugs.llvm.org//show_bug.cgi?id=9069
fn file_visit_macros<F: FnMut(Vec<u8>, Vec<Token>)>(
file: &str,
tokenize_range_adjust: bool,
mut visitor: F,
) {
unsafe {
let tu = {
let index = clang_createIndex(true as _, false as _);
let cfile = ffi::CString::new(file).unwrap();
let mut tu = mem::MaybeUninit::uninit();
assert!(
clang_parseTranslationUnit2(
index,
cfile.as_ptr(),
[b"-std=c11\0".as_ptr() as *const ::std::os::raw::c_char].as_ptr(),
1,
ptr::null_mut(),
0,
CXTranslationUnit_DetailedPreprocessingRecord,
&mut *tu.as_mut_ptr()
) == CXError_Success,
"Failure reading test case {}",
file
);
tu.assume_init()
};
visit_children(clang_getTranslationUnitCursor(tu), |cur, _parent| {
if cur.kind == CXCursor_MacroDefinition {
let mut range = clang_getCursorExtent(cur);
if !location_in_scope(range) {
return CXChildVisit_Continue;
}
range.end_int_data -= if tokenize_range_adjust { 1 } else { 0 };
let mut token_ptr = ptr::null_mut();
let mut num = 0;
clang_tokenize(tu, range, &mut token_ptr, &mut num);
if token_ptr != ptr::null_mut() {
let tokens = slice::from_raw_parts(token_ptr, num as usize);
let tokens: Vec<_> = tokens
.iter()
.filter_map(|t| {
if clang_getTokenKind(*t) != CXToken_Comment {
Some(token_clang_to_cexpr(tu, t))
} else {
None
}
})
.collect();
clang_disposeTokens(tu, token_ptr, num);
visitor(clang_str_to_vec(clang_getCursorSpelling(cur)), tokens)
}
let clang = clang::Clang::new().unwrap();

let index = clang::Index::new(&clang, false, true);

let tu = index
.parser(file)
.arguments(&["-std=c11"])
.detailed_preprocessing_record(true)
.skip_function_bodies(true)
.parse()
.unwrap();

let entity = tu.get_entity();

entity.visit_children(|cur, _parent| {
if cur.get_kind() == EntityKind::MacroDefinition {
let range = cur.get_range().unwrap();
if !location_in_scope(&range) {
return clang::EntityVisitResult::Continue;
}
CXChildVisit_Continue
});
clang_disposeTranslationUnit(tu);
};

let tokens: Vec<_> = range
.tokenize()
.into_iter()
.filter_map(|token| {
if token.get_kind() == TokenKind::Comment {
return None;
}

Some(token_clang_to_cexpr(&token))
})
.collect();

let display_name = cur.get_display_name().unwrap();
visitor(display_name.into_bytes(), tokens)
}

clang::EntityVisitResult::Continue
});
}

fn test_file(file: &str) -> bool {
let mut idents = HashMap::new();
let mut all_succeeded = true;
file_visit_macros(file, fix_bug_9069(), |ident, tokens| {
file_visit_macros(file, |ident, tokens| {
all_succeeded &= test_definition(ident, &tokens, &mut idents)
});
all_succeeded
}

fn fix_bug_9069() -> bool {
fn check_bug_9069() -> bool {
let mut token_sets = vec![];
file_visit_macros(
"tests/input/test_llvm_bug_9069.h",
false,
|ident, tokens| {
assert_eq!(&ident, b"A");
token_sets.push(tokens);
},
);
assert_eq!(token_sets.len(), 2);
token_sets[0] != token_sets[1]
}

use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::Once;

static CHECK_FIX: Once = Once::new();
static FIX: AtomicBool = AtomicBool::new(false);

CHECK_FIX.call_once(|| FIX.store(check_bug_9069(), Ordering::SeqCst));

FIX.load(Ordering::SeqCst)
}

macro_rules! test_file {
($f:ident) => {
#[test]
Expand Down

0 comments on commit 9e59566

Please sign in to comment.