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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
use dirs::home_dir;
use std::{
    collections::BTreeMap,
    path::{Path, PathBuf},
};
use sway_types::{LineCol, SourceEngine};

use serde::{Deserialize, Serialize};

use sway_types::span::Span;

/// Index of an interned path string
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash, Serialize, Deserialize)]
#[serde(transparent)]
pub struct PathIndex(pub usize);

#[derive(Debug, Clone, Default, Serialize, Deserialize)]
pub struct SourceMap {
    /// Paths of dependencies in the `~/.forc` directory, with the prefix stripped.
    /// This makes inverse source mapping work on any machine with deps downloaded.
    pub dependency_paths: Vec<PathBuf>,
    /// Paths to source code files, defined separately to avoid repetition.
    pub paths: Vec<PathBuf>,
    /// Mapping from opcode index to source location
    // count of instructions, multiply the opcode by 4 to get the byte offset
    pub map: BTreeMap<usize, SourceMapSpan>,
}
impl SourceMap {
    pub fn new() -> Self {
        Self::default()
    }

    /// Inserts dependency path. Unsupported locations are ignored for now.
    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());
            }
        }
        // TODO: Only dependencies in ~/.forc are supported for now
    }

    pub fn insert(&mut self, source_engine: &SourceEngine, pc: usize, span: &Span) {
        if let Some(source_id) = span.source_id() {
            let path = source_engine.get_path(source_id);
            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_pos().line_col(),
                        end: span.end_pos().line_col(),
                    },
                },
            );
        }
    }

    /// Inverse source mapping
    pub fn addr_to_span(&self, pc: usize) -> Option<(PathBuf, LocationRange)> {
        self.map
            .get(&pc)
            .map(|sms| sms.to_span(&self.paths, &self.dependency_paths))
    }
}

#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct SourceMapSpan {
    pub path: PathIndex,
    pub range: LocationRange,
}

impl SourceMapSpan {
    pub fn to_span(
        &self,
        paths: &[PathBuf],
        dependency_paths: &[PathBuf],
    ) -> (PathBuf, LocationRange) {
        let p = &paths[self.path.0];
        for dep in 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), self.range);
            }
        }

        (p.to_owned(), self.range)
    }
}

#[derive(Debug, Clone, Copy, Serialize, Deserialize)]
pub struct LocationRange {
    pub start: LineCol,
    pub end: LineCol,
}