Skip to content

Commit 73b46a0

Browse files
committed
Fix source file mapping case sensitive bug, avoid Timeline assert
1 parent 200e64b commit 73b46a0

File tree

7 files changed

+51
-38
lines changed

7 files changed

+51
-38
lines changed

installer/arm64/prepare-installer.cmd

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
1-
set %_VERSION="1.1.3"
1+
set %_VERSION="1.1.4"
22

33
iscc.exe installer.iss /DAPP_VERSION=%_VERSION% /O%cd%

installer/x64/prepare-installer.cmd

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
1-
set %_VERSION="1.1.3"
1+
set %_VERSION="1.1.4"
22

33
iscc.exe installer.iss /DAPP_VERSION=%_VERSION% /O%cd%

src/ProfileExplorerCore/Utilities/SourceFileMapper.cs

+28-10
Original file line numberDiff line numberDiff line change
@@ -7,15 +7,17 @@
77
namespace ProfileExplorer.Core;
88

99
public class SourceFileMapper {
10-
private readonly Dictionary<string, string> map_;
10+
private readonly Dictionary<string, string> sourcePathMap_;
11+
private readonly Dictionary<string, string> sourceFileCache_;
1112
private readonly object lockObject_ = new();
1213

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.
1618
}
1719

18-
public Dictionary<string, string> SourceMap => map_;
20+
public Dictionary<string, string> SourceMap => sourcePathMap_;
1921

2022
public string Map(string sourceFile, Func<string> lookup = null) {
2123
if (string.IsNullOrEmpty(sourceFile)) {
@@ -43,15 +45,23 @@ public string Map(string sourceFile, Func<string> lookup = null) {
4345

4446
public void Reset() {
4547
lock (lockObject_) {
46-
map_.Clear();
48+
sourcePathMap_.Clear();
49+
sourcePathMap_.Clear();
4750
}
4851
}
4952

5053
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.
5161
int index = sourceFile.LastIndexOf(Path.DirectorySeparatorChar);
5262

5363
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)) {
5565
result = Path.Combine(mappedDirectory, sourceFile.Substring(index + 1));
5666
return true;
5767
}
@@ -69,18 +79,26 @@ public void UpdateMap(string originalPath, string mappedPath) {
6979
return;
7080
}
7181

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.
7287
int prevOriginalPath = originalPath.Length;
7388
int prevMappedPath = mappedPath.Length;
7489
int originalPathIndex = originalPath.LastIndexOf(Path.DirectorySeparatorChar);
7590
int mappedPathIndex = mappedPath.LastIndexOf(Path.DirectorySeparatorChar);
7691

7792
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)) {
8098
return;
8199
}
82100

83-
map_[originalPath.Substring(0, originalPathIndex)] = mappedPath.Substring(0, mappedPathIndex);
101+
sourcePathMap_[originalPath.Substring(0, originalPathIndex)] = mappedPath.Substring(0, mappedPathIndex);
84102
prevOriginalPath = originalPathIndex;
85103
prevMappedPath = mappedPathIndex;
86104

src/ProfileExplorerUI/Binary/PDBDebugInfoProvider.cs

+6-18
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,8 @@ namespace ProfileExplorer.UI.Binary;
2727
public sealed class PDBDebugInfoProvider : IDebugInfoProvider {
2828
private const int MaxDemangledFunctionNameLength = 8192;
2929
private const int FunctionCacheMissThreshold = 100;
30+
private const int MaxLogEntryLength = 10240;
31+
3032
private static ConcurrentDictionary<SymbolFileDescriptor, DebugFileSearchResult> resolvedSymbolsCache_ = new();
3133
private static readonly StringWriter authLogWriter_;
3234
private static readonly SymwebHandler authSymwebHandler_;
@@ -608,14 +610,14 @@ private SourceFileDebugInfo FindFunctionSourceFilePathImpl(SourceLineDebugInfo l
608610
Trace.WriteLine($"Query source server for {sourceLine?.SourceFile?.BuildTimeFilePath}");
609611
string filePath = sourceLine.SourceFile.GetSourceFile();
610612

611-
if (ValidateDownloadedSourceFile(filePath)) {
613+
if (SourceFileChecksumMatchesPDB(sourceFile, filePath)) {
612614
Trace.WriteLine($"Downloaded source file {filePath}");
613615
localFilePath = filePath;
614616
hasChecksumMismatch = !SourceFileChecksumMatchesPDB(sourceFile, localFilePath);
615617
}
616618
else {
617619
Trace.WriteLine($"Failed to download source file {localFilePath}");
618-
Trace.WriteLine(symbolReaderLog_.ToString());
620+
Trace.WriteLine(symbolReaderLog_.ToString().TrimToLength(MaxLogEntryLength));
619621
symbolReaderLog_.GetStringBuilder().Clear();
620622
Trace.WriteLine("---------------------------------");
621623
}
@@ -630,25 +632,11 @@ private SourceFileDebugInfo FindFunctionSourceFilePathImpl(SourceLineDebugInfo l
630632
return new SourceFileDebugInfo(localFilePath, originalFilePath, lineInfo.Line, hasChecksumMismatch);
631633
}
632634

633-
private bool ValidateDownloadedSourceFile(string filePath) {
634-
try {
635-
if (!File.Exists(filePath)) {
636-
return false;
637-
}
638-
639-
// If the source server requires authentication, but it's not properly set up,
640-
// usually an HTML error page is returned instead, treat it as a failure.
641-
//? TODO: Better way to detect this, may need change in TraceEvent lib.
642-
string fileText = File.ReadAllText(filePath);
643-
return !fileText.Contains(@"<!DOCTYPE html");
644-
}
645-
catch (Exception ex) {
646-
Trace.WriteLine($"Failed to validate downloaded source file {filePath}: {ex.Message}");
635+
private bool SourceFileChecksumMatchesPDB(IDiaSourceFile sourceFile, string filePath) {
636+
if (string.IsNullOrEmpty(filePath)) {
647637
return false;
648638
}
649-
}
650639

651-
private bool SourceFileChecksumMatchesPDB(IDiaSourceFile sourceFile, string filePath) {
652640
var hashAlgo = GetSourceFileChecksumHashAlgorithm(sourceFile);
653641
byte[] pdbChecksum = GetSourceFileChecksum(sourceFile);
654642
byte[] fileChecksum = ComputeSourceFileChecksum(filePath, hashAlgo);

src/ProfileExplorerUI/Profile/Timeline/ActivityView.xaml.cs

+6-5
Original file line numberDiff line numberDiff line change
@@ -305,7 +305,7 @@ public Task Initialize(ProfileData profile, Rect visibleArea, int threadId = -1)
305305
}
306306

307307
samplingInterval_ = profile_.Report.SamplingInterval;
308-
maxWidth_ = prevMaxWidth_ = visibleArea.Width;
308+
maxWidth_ = prevMaxWidth_ = Math.Max(0, visibleArea.Width);
309309
visual_ = new DrawingVisual();
310310
visual_.Drawing?.Freeze();
311311
AddVisualChild(visual_);
@@ -329,8 +329,8 @@ public void SetMaxWidth(double maxWidth) {
329329
return;
330330
}
331331

332-
prevMaxWidth_ = maxWidth;
333-
maxWidth_ = maxWidth;
332+
prevMaxWidth_ = Math.Max(0, maxWidth);
333+
maxWidth_ = Math.Max(0, maxWidth);
334334
Redraw();
335335
InvalidateMeasure();
336336
}
@@ -565,13 +565,14 @@ private void ActivityView_MouseMove(object sender, MouseEventArgs e) {
565565
}
566566

567567
private List<SliceList> ComputeSampleSlices(ProfileData profile, int threadId = -1) {
568-
if (profile.Samples.Count == 0) {
568+
if (profile.Samples.Count == 0 || maxWidth_ < double.Epsilon) {
569569
return new List<SliceList>();
570570
}
571571

572572
startTime_ = profile.Samples[0].Sample.Time;
573573
endTime_ = profile.Samples[^1].Sample.Time;
574-
double slices = maxWidth_ / sliceWidth_ * (prevMaxWidth_ / maxWidth_);
574+
double slices = (maxWidth_ / sliceWidth_) * (prevMaxWidth_ / maxWidth_);
575+
575576
var timeDiff = endTime_ - startTime_;
576577
double timePerSlice = timeDiff.Ticks / slices;
577578
double timePerSliceReciproc = 1.0 / timePerSlice;

src/ProfileExplorerUI/Profile/Utils/SourceFileFinder.cs

+6
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,10 @@ public void SaveSettings(SourceFileFinderSettings settings) {
5555
return (SourceFileDebugInfo.Unknown, FailureReason.DebugInfoNotFound);
5656
}
5757

58+
// First get the source file path from the debug file (this may download it
59+
// from a source file server, if enabled). If the file path is not found
60+
// on this machine, try to map it to a local path either automatically
61+
// or by asking the user to manually locate the source file.
5862
var sourceInfo = SourceFileDebugInfo.Unknown;
5963
var funcProfile = session_.ProfileData?.GetFunctionProfile(function);
6064

@@ -81,6 +85,8 @@ public void SaveSettings(SourceFileFinderSettings settings) {
8185
// Check if the file can be found. If it's from another machine,
8286
// a mapping is done after the user is asked to pick the new location of the file.
8387
if (File.Exists(sourceInfo.FilePath)) {
88+
// This assumes that a checksum match was done already
89+
// and this is the right source file.
8490
return (sourceInfo, FailureReason.None);
8591
}
8692
else if (!IsDisabledSourceFilePath(sourceInfo.FilePath)) {

src/ProfileExplorerUI/ProfileExplorerUI.csproj

+3-3
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,9 @@
55
<TargetFramework>net8.0-windows</TargetFramework>
66
<UseWPF>true</UseWPF>
77
<ApplicationIcon>main.ico</ApplicationIcon>
8-
<AssemblyVersion>1.1.3</AssemblyVersion>
9-
<FileVersion>1.1.3</FileVersion>
10-
<Version>1.1.3</Version>
8+
<AssemblyVersion>1.1.4</AssemblyVersion>
9+
<FileVersion>1.1.4</FileVersion>
10+
<Version>1.1.4</Version>
1111
<Authors></Authors>
1212
<Company>Microsoft Corporation</Company>
1313
<Product>Profile Explorer</Product>

0 commit comments

Comments
 (0)