Skip to content

Commit 909d926

Browse files
committed
Don't count include locations of main header file.
1 parent 6553527 commit 909d926

File tree

9 files changed

+197
-70
lines changed

9 files changed

+197
-70
lines changed

bindgen-tests/tests/expectations/tests/source-order-recursive.rs

+67
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
// bindgen-flags: -- -Itests/headers -Itests/headers/source-order-recursive
2+
3+
#ifndef A_H
4+
#define A_H
5+
6+
struct foo {};
7+
8+
#include "source-order-recursive-2.h"
9+
10+
struct baz {
11+
struct bar field;
12+
};
13+
14+
#endif
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
#include "source-order-recursive.h"
2+
3+
struct bar {
4+
struct foo field;
5+
};

bindgen/clang.rs

+43-16
Original file line numberDiff line numberDiff line change
@@ -7,13 +7,14 @@
77
use crate::ir::context::{BindgenContext, IncludeLocation};
88
use clang_sys::*;
99
use std::cmp;
10-
1110
use std::convert::TryInto;
1211
use std::ffi::{CStr, CString};
1312
use std::fmt;
1413
use std::hash::Hash;
1514
use std::hash::Hasher;
1615
use std::os::raw::{c_char, c_int, c_longlong, c_uint, c_ulong, c_ulonglong};
16+
use std::path::Path;
17+
use std::path::PathBuf;
1718
use std::{mem, ptr, slice};
1819

1920
/// Type representing a clang attribute.
@@ -395,8 +396,9 @@ impl Cursor {
395396
offset: offset.try_into().unwrap(),
396397
}
397398
} else {
399+
let file_name = cxstring_into_string(clang_getFileName(file));
398400
SourceLocation::File {
399-
file_name: cxstring_into_string(clang_getFileName(file)),
401+
file_path: absolutize_path(file_name),
400402
line: line.try_into().unwrap(),
401403
column: column.try_into().unwrap(),
402404
offset: offset.try_into().unwrap(),
@@ -534,8 +536,12 @@ impl Cursor {
534536
let mut children = self.collect_children();
535537
for child in &children {
536538
if child.kind() == CXCursor_InclusionDirective {
537-
if let Some(included_file) = child.get_included_file_name() {
538-
ctx.add_include(included_file, child.location());
539+
if let Some(included_file_name) = child.get_included_file_name()
540+
{
541+
ctx.add_include(
542+
absolutize_path(included_file_name),
543+
child.location(),
544+
);
539545
}
540546
}
541547
}
@@ -1574,7 +1580,7 @@ pub(crate) enum SourceLocation {
15741580
/// Location in a source file.
15751581
File {
15761582
/// Name of the source file.
1577-
file_name: String,
1583+
file_path: PathBuf,
15781584
/// Line in the source file.
15791585
line: usize,
15801586
/// Column in the source file.
@@ -1584,6 +1590,18 @@ pub(crate) enum SourceLocation {
15841590
},
15851591
}
15861592

1593+
fn absolutize_path<P: AsRef<Path>>(path: P) -> PathBuf {
1594+
let path = path.as_ref();
1595+
1596+
if path.is_relative() {
1597+
std::env::current_dir()
1598+
.expect("Cannot retrieve current directory")
1599+
.join(path)
1600+
} else {
1601+
path.to_owned()
1602+
}
1603+
}
1604+
15871605
impl SourceLocation {
15881606
/// Locations of built-in items provided by the compiler (which don't have a source file),
15891607
/// are sorted first. Remaining locations are sorted by their position in the source file.
@@ -1608,26 +1626,26 @@ impl SourceLocation {
16081626
}
16091627
(
16101628
SourceLocation::File {
1611-
file_name, offset, ..
1629+
file_path, offset, ..
16121630
},
16131631
SourceLocation::File {
1614-
file_name: other_file_name,
1632+
file_path: other_file_path,
16151633
offset: other_offset,
16161634
..
16171635
},
16181636
) => {
1619-
if file_name == other_file_name {
1637+
if file_path == other_file_path {
16201638
return offset.cmp(other_offset);
16211639
}
16221640

16231641
// If `file` is transitively included via `ancestor_file`,
16241642
// find the offset of the include directive in `ancestor_file`.
1625-
let offset_in_ancestor = |file: &str, ancestor_file: &str| {
1643+
let offset_in_ancestor = |file: &Path, ancestor_file: &Path| {
16261644
let mut file = file;
16271645
while file != ancestor_file {
16281646
let include_location = ctx.include_location(file);
16291647
file = if let IncludeLocation::File {
1630-
file_name: file,
1648+
file_path: file,
16311649
offset,
16321650
..
16331651
} = include_location
@@ -1646,20 +1664,20 @@ impl SourceLocation {
16461664
};
16471665

16481666
if let Some(offset) =
1649-
offset_in_ancestor(file_name, other_file_name)
1667+
offset_in_ancestor(file_path, other_file_path)
16501668
{
16511669
return offset.cmp(other_offset);
16521670
}
16531671

16541672
if let Some(other_offset) =
1655-
offset_in_ancestor(other_file_name, file_name)
1673+
offset_in_ancestor(other_file_path, file_path)
16561674
{
16571675
return offset.cmp(other_offset);
16581676
}
16591677

16601678
// If the source files are siblings, compare their include locations.
1661-
let parent = ctx.include_location(file_name);
1662-
let other_parent = ctx.include_location(other_file_name);
1679+
let parent = ctx.include_location(file_path);
1680+
let other_parent = ctx.include_location(other_file_path);
16631681
parent.cmp_by_source_order(other_parent, ctx)
16641682
}
16651683
}
@@ -1671,11 +1689,11 @@ impl fmt::Display for SourceLocation {
16711689
match self {
16721690
Self::Builtin { .. } => "built-in".fmt(f),
16731691
Self::File {
1674-
file_name,
1692+
file_path,
16751693
line,
16761694
column,
16771695
..
1678-
} => write!(f, "{}:{}:{}", file_name, line, column),
1696+
} => write!(f, "{}:{}:{}", file_path.display(), line, column),
16791697
}
16801698
}
16811699
}
@@ -1906,6 +1924,15 @@ impl TranslationUnit {
19061924
}
19071925
}
19081926

1927+
/// Get the source file path of this translation unit.
1928+
pub(crate) fn path(&self) -> PathBuf {
1929+
let file_name = unsafe {
1930+
cxstring_into_string(clang_getTranslationUnitSpelling(self.x))
1931+
};
1932+
1933+
absolutize_path(file_name)
1934+
}
1935+
19091936
/// Is this the null translation unit?
19101937
pub(crate) fn is_null(&self) -> bool {
19111938
self.x.is_null()

bindgen/codegen/mod.rs

+6-6
Original file line numberDiff line numberDiff line change
@@ -4354,17 +4354,17 @@ fn unsupported_abi_diagnostic(
43544354
);
43554355

43564356
if let Some(crate::clang::SourceLocation::File {
4357-
file_name,
4357+
file_path,
43584358
line,
43594359
column,
43604360
..
43614361
}) = location.cloned()
43624362
{
4363-
if let Ok(Some(source)) = get_line(&file_name, line) {
4363+
if let Ok(Some(source)) = get_line(&file_path, line) {
43644364
let mut slice = Slice::default();
43654365
slice
43664366
.with_source(source)
4367-
.with_location(file_name, line, column);
4367+
.with_location(file_path, line, column);
43684368
diag.add_slice(slice);
43694369
}
43704370
}
@@ -4395,17 +4395,17 @@ fn variadic_fn_diagnostic(
43954395
.add_annotation("No code will be generated for this function.", Level::Note);
43964396

43974397
if let Some(crate::clang::SourceLocation::File {
4398-
file_name,
4398+
file_path,
43994399
line,
44004400
column,
44014401
..
44024402
}) = location.cloned()
44034403
{
4404-
if let Ok(Some(source)) = get_line(&file_name, line) {
4404+
if let Ok(Some(source)) = get_line(&file_path, line) {
44054405
let mut slice = Slice::default();
44064406
slice
44074407
.with_source(source)
4408-
.with_location(file_name, line, column);
4408+
.with_location(file_path, line, column);
44094409
diag.add_slice(slice);
44104410
}
44114411
}

bindgen/diagnostics.rs

+8-9
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,8 @@
22
//!
33
//! The entry point of this module is the [`Diagnostic`] type.
44
5-
use std::fmt::Write;
65
use std::io::{self, BufRead, BufReader};
6+
use std::path::Path;
77
use std::{borrow::Cow, fs::File};
88

99
use annotate_snippets::{
@@ -162,25 +162,24 @@ impl<'a> Slice<'a> {
162162
}
163163

164164
/// Set the file, line and column.
165-
pub(crate) fn with_location(
165+
pub(crate) fn with_location<P: AsRef<Path>>(
166166
&mut self,
167-
mut name: String,
167+
path: P,
168168
line: usize,
169169
col: usize,
170170
) -> &mut Self {
171-
write!(name, ":{}:{}", line, col)
172-
.expect("Writing to a string cannot fail");
173-
self.filename = Some(name);
171+
self.filename =
172+
Some(format!("{}:{}:{}", path.as_ref().display(), line, col));
174173
self.line = Some(line);
175174
self
176175
}
177176
}
178177

179-
pub(crate) fn get_line(
180-
filename: &str,
178+
pub(crate) fn get_line<P: AsRef<Path>>(
179+
file_path: P,
181180
line: usize,
182181
) -> io::Result<Option<String>> {
183-
let file = BufReader::new(File::open(filename)?);
182+
let file = BufReader::new(File::open(file_path.as_ref())?);
184183
if let Some(line) = file.lines().nth(line.wrapping_sub(1)) {
185184
return line.map(Some);
186185
}

0 commit comments

Comments
 (0)