1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
use dirs::home_dir;
use std::collections::HashMap;
use std::path::{Path, PathBuf};
use serde::{Deserialize, Serialize};
use sway_types::span::Span;
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash, Serialize, Deserialize)]
#[serde(transparent)]
pub struct PathIndex(usize);
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
pub struct SourceMap {
dependency_paths: Vec<PathBuf>,
paths: Vec<PathBuf>,
map: HashMap<usize, SourceMapSpan>,
}
impl SourceMap {
pub fn new() -> Self {
Self::default()
}
pub fn insert_dependency<P: AsRef<Path>>(&mut self, path: P) {
if let Some(home) = home_dir() {
let forc = home.join(".forc/");
if let Ok(unprefixed) = path.as_ref().strip_prefix(forc) {
self.dependency_paths.push(unprefixed.to_owned());
}
}
}
pub fn insert(&mut self, pc: usize, span: &Span) {
if let Some(path) = span.path.as_ref() {
let path_index = self
.paths
.iter()
.position(|p| *p == **path)
.unwrap_or_else(|| {
self.paths.push((**path).to_owned());
self.paths.len() - 1
});
self.map.insert(
pc,
SourceMapSpan {
path: PathIndex(path_index),
range: LocationRange {
start: span.start(),
end: span.end(),
},
},
);
}
}
pub fn addr_to_span(&self, pc: usize) -> Option<(PathBuf, LocationRange)> {
self.map.get(&pc).map(|sms| {
let p = &self.paths[sms.path.0];
for dep in &self.dependency_paths {
if p.starts_with(dep.file_name().unwrap()) {
let mut path = home_dir().expect("Could not get homedir").join(".forc");
if let Some(dp) = dep.parent() {
path = path.join(dp);
}
return (path.join(p), sms.range);
}
}
(p.to_owned(), sms.range)
})
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct SourceMapSpan {
pub path: PathIndex,
pub range: LocationRange,
}
#[derive(Debug, Clone, Copy, Serialize, Deserialize)]
pub struct LocationRange {
pub start: usize,
pub end: usize,
}