7
7
use crate :: ir:: context:: { BindgenContext , IncludeLocation } ;
8
8
use clang_sys:: * ;
9
9
use std:: cmp;
10
-
11
10
use std:: convert:: TryInto ;
12
11
use std:: ffi:: { CStr , CString } ;
13
12
use std:: fmt;
14
13
use std:: hash:: Hash ;
15
14
use std:: hash:: Hasher ;
16
15
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 ;
17
18
use std:: { mem, ptr, slice} ;
18
19
19
20
/// Type representing a clang attribute.
@@ -395,8 +396,9 @@ impl Cursor {
395
396
offset : offset. try_into ( ) . unwrap ( ) ,
396
397
}
397
398
} else {
399
+ let file_name = cxstring_into_string ( clang_getFileName ( file) ) ;
398
400
SourceLocation :: File {
399
- file_name : cxstring_into_string ( clang_getFileName ( file ) ) ,
401
+ file_path : absolutize_path ( file_name ) ,
400
402
line : line. try_into ( ) . unwrap ( ) ,
401
403
column : column. try_into ( ) . unwrap ( ) ,
402
404
offset : offset. try_into ( ) . unwrap ( ) ,
@@ -534,8 +536,12 @@ impl Cursor {
534
536
let mut children = self . collect_children ( ) ;
535
537
for child in & children {
536
538
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
+ ) ;
539
545
}
540
546
}
541
547
}
@@ -1574,7 +1580,7 @@ pub(crate) enum SourceLocation {
1574
1580
/// Location in a source file.
1575
1581
File {
1576
1582
/// Name of the source file.
1577
- file_name : String ,
1583
+ file_path : PathBuf ,
1578
1584
/// Line in the source file.
1579
1585
line : usize ,
1580
1586
/// Column in the source file.
@@ -1584,6 +1590,18 @@ pub(crate) enum SourceLocation {
1584
1590
} ,
1585
1591
}
1586
1592
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
+
1587
1605
impl SourceLocation {
1588
1606
/// Locations of built-in items provided by the compiler (which don't have a source file),
1589
1607
/// are sorted first. Remaining locations are sorted by their position in the source file.
@@ -1608,26 +1626,26 @@ impl SourceLocation {
1608
1626
}
1609
1627
(
1610
1628
SourceLocation :: File {
1611
- file_name , offset, ..
1629
+ file_path , offset, ..
1612
1630
} ,
1613
1631
SourceLocation :: File {
1614
- file_name : other_file_name ,
1632
+ file_path : other_file_path ,
1615
1633
offset : other_offset,
1616
1634
..
1617
1635
} ,
1618
1636
) => {
1619
- if file_name == other_file_name {
1637
+ if file_path == other_file_path {
1620
1638
return offset. cmp ( other_offset) ;
1621
1639
}
1622
1640
1623
1641
// If `file` is transitively included via `ancestor_file`,
1624
1642
// 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 | {
1626
1644
let mut file = file;
1627
1645
while file != ancestor_file {
1628
1646
let include_location = ctx. include_location ( file) ;
1629
1647
file = if let IncludeLocation :: File {
1630
- file_name : file,
1648
+ file_path : file,
1631
1649
offset,
1632
1650
..
1633
1651
} = include_location
@@ -1646,20 +1664,20 @@ impl SourceLocation {
1646
1664
} ;
1647
1665
1648
1666
if let Some ( offset) =
1649
- offset_in_ancestor ( file_name , other_file_name )
1667
+ offset_in_ancestor ( file_path , other_file_path )
1650
1668
{
1651
1669
return offset. cmp ( other_offset) ;
1652
1670
}
1653
1671
1654
1672
if let Some ( other_offset) =
1655
- offset_in_ancestor ( other_file_name , file_name )
1673
+ offset_in_ancestor ( other_file_path , file_path )
1656
1674
{
1657
1675
return offset. cmp ( other_offset) ;
1658
1676
}
1659
1677
1660
1678
// 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 ) ;
1663
1681
parent. cmp_by_source_order ( other_parent, ctx)
1664
1682
}
1665
1683
}
@@ -1671,11 +1689,11 @@ impl fmt::Display for SourceLocation {
1671
1689
match self {
1672
1690
Self :: Builtin { .. } => "built-in" . fmt ( f) ,
1673
1691
Self :: File {
1674
- file_name ,
1692
+ file_path ,
1675
1693
line,
1676
1694
column,
1677
1695
..
1678
- } => write ! ( f, "{}:{}:{}" , file_name , line, column) ,
1696
+ } => write ! ( f, "{}:{}:{}" , file_path . display ( ) , line, column) ,
1679
1697
}
1680
1698
}
1681
1699
}
@@ -1906,6 +1924,15 @@ impl TranslationUnit {
1906
1924
}
1907
1925
}
1908
1926
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
+
1909
1936
/// Is this the null translation unit?
1910
1937
pub ( crate ) fn is_null ( & self ) -> bool {
1911
1938
self . x . is_null ( )
0 commit comments