7
7
namespace ProfileExplorer . Core ;
8
8
9
9
public class SourceFileMapper {
10
- private readonly Dictionary < string , string > map_ ;
10
+ private readonly Dictionary < string , string > sourcePathMap_ ;
11
+ private readonly Dictionary < string , string > sourceFileCache_ ;
11
12
private readonly object lockObject_ = new ( ) ;
12
13
13
- public SourceFileMapper ( Dictionary < string , string > map = null ) {
14
- map_ = map ;
15
- map_ ??= new Dictionary < string , string > ( ) ;
14
+ public SourceFileMapper ( Dictionary < string , string > sourcePathMap = null ) {
15
+ sourcePathMap_ = sourcePathMap ;
16
+ sourcePathMap_ ??= new Dictionary < string , string > ( ) ; // Saved across sessions.
17
+ sourceFileCache_ = new Dictionary < string , string > ( ) ; // Active per session.
16
18
}
17
19
18
- public Dictionary < string , string > SourceMap => map_ ;
20
+ public Dictionary < string , string > SourceMap => sourcePathMap_ ;
19
21
20
22
public string Map ( string sourceFile , Func < string > lookup = null ) {
21
23
if ( string . IsNullOrEmpty ( sourceFile ) ) {
@@ -43,15 +45,23 @@ public string Map(string sourceFile, Func<string> lookup = null) {
43
45
44
46
public void Reset ( ) {
45
47
lock ( lockObject_ ) {
46
- map_ . Clear ( ) ;
48
+ sourcePathMap_ . Clear ( ) ;
49
+ sourcePathMap_ . Clear ( ) ;
47
50
}
48
51
}
49
52
50
53
private bool TryLookupInMap ( string sourceFile , out string result ) {
54
+ // Check the direct mapping cache first.
55
+ if ( sourceFileCache_ . TryGetValue ( sourceFile , out result ) ) {
56
+ return true ;
57
+ }
58
+
59
+ // Use the past directory mappings to build the equivalent
60
+ // local path for the source file.
51
61
int index = sourceFile . LastIndexOf ( Path . DirectorySeparatorChar ) ;
52
62
53
63
while ( index > 0 ) {
54
- if ( map_ . TryGetValue ( sourceFile . Substring ( 0 , index ) , out string mappedDirectory ) ) {
64
+ if ( sourcePathMap_ . TryGetValue ( sourceFile . Substring ( 0 , index ) , out string mappedDirectory ) ) {
55
65
result = Path . Combine ( mappedDirectory , sourceFile . Substring ( index + 1 ) ) ;
56
66
return true ;
57
67
}
@@ -69,18 +79,26 @@ public void UpdateMap(string originalPath, string mappedPath) {
69
79
return ;
70
80
}
71
81
82
+ sourceFileCache_ [ originalPath ] = mappedPath ;
83
+
84
+ // Try to create a mapping between the directory paths,
85
+ // to be used later with another source file part of the same
86
+ // directory structure.
72
87
int prevOriginalPath = originalPath . Length ;
73
88
int prevMappedPath = mappedPath . Length ;
74
89
int originalPathIndex = originalPath . LastIndexOf ( Path . DirectorySeparatorChar ) ;
75
90
int mappedPathIndex = mappedPath . LastIndexOf ( Path . DirectorySeparatorChar ) ;
76
91
77
92
while ( originalPathIndex > 0 && mappedPathIndex > 0 ) {
78
- if ( originalPath . Substring ( originalPathIndex , prevOriginalPath - originalPathIndex ) !=
79
- mappedPath . Substring ( mappedPathIndex , prevMappedPath - mappedPathIndex ) ) {
93
+ // Stop once there is a mismatch in directory names.
94
+ // Use a case-insensitive compare for Windows paths.
95
+ if ( ! originalPath . Substring ( originalPathIndex , prevOriginalPath - originalPathIndex ) . Equals (
96
+ mappedPath . Substring ( mappedPathIndex , prevMappedPath - mappedPathIndex ) ,
97
+ StringComparison . OrdinalIgnoreCase ) ) {
80
98
return ;
81
99
}
82
100
83
- map_ [ originalPath . Substring ( 0 , originalPathIndex ) ] = mappedPath . Substring ( 0 , mappedPathIndex ) ;
101
+ sourcePathMap_ [ originalPath . Substring ( 0 , originalPathIndex ) ] = mappedPath . Substring ( 0 , mappedPathIndex ) ;
84
102
prevOriginalPath = originalPathIndex ;
85
103
prevMappedPath = mappedPathIndex ;
86
104
0 commit comments