Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Merged by Bors] - Implement instruction flowgraph generator #2422

Closed
wants to merge 25 commits into from
Closed
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
5bda107
Implement generic graph with graphviz format
HalidOdat Nov 8, 2022
0a87747
Generate mermaid graphs
HalidOdat Nov 9, 2022
c8426b5
generate colored nodes and edges
HalidOdat Nov 10, 2022
6acf808
Simplify adding nodes and edges
HalidOdat Nov 10, 2022
6b19f04
Add subgraphs
HalidOdat Nov 10, 2022
6315aef
Add recursive subgraphs
HalidOdat Nov 10, 2022
2b1b457
Move to_graph to graph module
HalidOdat Nov 10, 2022
130d3eb
Add CLI option for generating flowgraphs
HalidOdat Nov 10, 2022
6e945a1
Put under feature-gate
HalidOdat Nov 10, 2022
0676727
Programmatically generate colors
HalidOdat Nov 10, 2022
b4fc149
Make Return instruction point to End
HalidOdat Nov 10, 2022
12e11db
Handle Switch and Try/Catch instructions
HalidOdat Nov 10, 2022
4d9bf79
Make returns point to finally if it exists
HalidOdat Nov 10, 2022
40f828f
Make Throw point to catch block
HalidOdat Nov 10, 2022
ad149b8
Add CLI option for flowgraph direction
HalidOdat Nov 11, 2022
fe440a3
Include opcode locations
HalidOdat Nov 11, 2022
12757d9
Documentation
HalidOdat Nov 13, 2022
1dd9022
Document mermaid block fix
HalidOdat Nov 13, 2022
050fc64
Revent node name collisions
HalidOdat Nov 13, 2022
b0ca0fd
Split into sub-modules
HalidOdat Nov 13, 2022
5f48fb2
Refactor `Color`
HalidOdat Nov 13, 2022
60d3333
Add documentation to flowgraph feature
HalidOdat Nov 13, 2022
c27e382
Run rrettier on `debugging.md`
HalidOdat Nov 13, 2022
a56b0ee
Merge branch 'main' into feature/instruction-flowgraph
HalidOdat Nov 22, 2022
c6e956f
Fix clippy error
HalidOdat Nov 22, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion boa_cli/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ repository.workspace = true
rust-version.workspace = true

[dependencies]
boa_engine = { workspace = true, features = ["deser", "console"] }
boa_engine = { workspace = true, features = ["deser", "console", "flowgraph"] }
boa_ast = { workspace = true, features = ["serde"]}
boa_interner.workspace = true
boa_parser.workspace = true
Expand Down
91 changes: 88 additions & 3 deletions boa_cli/src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,10 @@
)]

use boa_ast::StatementList;
use boa_engine::Context;
use boa_engine::{
vm::graph::{Direction, Graph},
Context, JsResult,
};
use clap::{Parser, ValueEnum, ValueHint};
use colored::{Color, Colorize};
use rustyline::{config::Config, error::ReadlineError, EditMode, Editor};
Expand Down Expand Up @@ -96,17 +99,38 @@ struct Opt {
short = 'a',
value_name = "FORMAT",
ignore_case = true,
value_enum
value_enum,
conflicts_with = "graph"
)]
dump_ast: Option<Option<DumpFormat>>,

/// Dump the AST to stdout with the given format.
#[arg(long, short)]
#[arg(long, short, conflicts_with = "graph")]
trace: bool,

/// Use vi mode in the REPL
#[arg(long = "vi")]
vi_mode: bool,

/// Generate instruction flowgraph. Default is Graphviz.
#[arg(
long,
value_name = "FORMAT",
ignore_case = true,
value_enum,
group = "graph"
)]
flowgraph: Option<Option<FlowgraphFormat>>,

/// Specifies the direction of the flowgraph. Default is TopToBottom.
#[arg(
long,
value_name = "FORMAT",
ignore_case = true,
value_enum,
requires = "graph"
)]
flowgraph_direction: Option<FlowgraphDirection>,
}

impl Opt {
Expand Down Expand Up @@ -136,6 +160,22 @@ enum DumpFormat {
JsonPretty,
}

#[derive(Debug, Clone, Copy, ValueEnum)]
HalidOdat marked this conversation as resolved.
Show resolved Hide resolved
enum FlowgraphFormat {
/// Generates in graphviz format.
Graphviz,
/// Generates in mermaid format.
Mermaid,
}

#[derive(Debug, Clone, Copy, ValueEnum)]
enum FlowgraphDirection {
TopToBottom,
BottomToTop,
LeftToRight,
RightToLeft,
}
HalidOdat marked this conversation as resolved.
Show resolved Hide resolved

/// Parses the the token stream into an AST and returns it.
///
/// Returns a error of type String with a message,
Expand Down Expand Up @@ -184,6 +224,31 @@ where
Ok(())
}

fn generate_flowgraph(
context: &mut Context,
src: &[u8],
flowgraph: FlowgraphFormat,
HalidOdat marked this conversation as resolved.
Show resolved Hide resolved
flowgraph_direction: Option<FlowgraphDirection>,
) -> JsResult<String> {
let ast = context.parse(src)?;
let code = context.compile(&ast)?;

let flowgraph_direction = match flowgraph_direction {
Some(FlowgraphDirection::TopToBottom) | None => Direction::TopToBottom,
Some(FlowgraphDirection::BottomToTop) => Direction::BottomToTop,
Some(FlowgraphDirection::LeftToRight) => Direction::LeftToRight,
Some(FlowgraphDirection::RightToLeft) => Direction::RightToLeft,
};

let mut graph = Graph::new(flowgraph_direction);
code.to_graph(context.interner(), graph.subgraph(String::default()));
let result = match flowgraph {
FlowgraphFormat::Graphviz => graph.to_graphviz_format(),
FlowgraphFormat::Mermaid => graph.to_mermaid_format(),
};
Ok(result)
}

pub fn main() -> Result<(), io::Error> {
let args = Opt::parse();

Expand All @@ -199,6 +264,16 @@ pub fn main() -> Result<(), io::Error> {
if let Err(e) = dump(&buffer, &args, &mut context) {
eprintln!("{e}");
}
} else if let Some(flowgraph) = args.flowgraph {
match generate_flowgraph(
&mut context,
&buffer,
flowgraph.unwrap_or(FlowgraphFormat::Graphviz),
args.flowgraph_direction,
) {
Ok(v) => println!("{}", v),
Err(v) => eprintln!("Uncaught {v}"),
}
} else {
match context.eval(&buffer) {
Ok(v) => println!("{}", v.display()),
Expand Down Expand Up @@ -245,6 +320,16 @@ pub fn main() -> Result<(), io::Error> {
if let Err(e) = dump(&line, &args, &mut context) {
eprintln!("{e}");
}
} else if let Some(flowgraph) = args.flowgraph {
match generate_flowgraph(
&mut context,
line.trim_end().as_bytes(),
flowgraph.unwrap_or(FlowgraphFormat::Graphviz),
args.flowgraph_direction,
) {
Ok(v) => println!("{}", v),
Err(v) => eprintln!("Uncaught {v}"),
}
} else {
match context.eval(line.trim_end()) {
Ok(v) => println!("{}", v.display()),
Expand Down
2 changes: 2 additions & 0 deletions boa_engine/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,8 @@ intl = [

fuzz = ["boa_ast/fuzz", "boa_interner/fuzz"]

flowgraph = []

# Enable Boa's WHATWG console object implementation.
console = []

Expand Down
Loading