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
//! The core part of Koopa IR.
//!
//! This module provides in-memory form IR related implementations,
//! including:
//!
//! * Structs of Koopa IR programs ([`Program`]), functions ([`Function`],
//! [`FunctionData`]), basic blocks ([`BasicBlock`],
//! [`BasicBlockData`](entities::BasicBlockData)) and values ([`Value`],
//! [`ValueData`](entities::ValueData)).
//! * Types of IR values ([`Type`]).
//! * IR builders and IR builder traits ([`builder`]).
//!
//! # Example
//!
//! Here is a Fibonacci function represented in Koopa IR:
//!
//! ```koopa
//! fun @fib(@n: i32): i32 {
//! %entry:
//! %cond = le @n, 2
//! br %cond, %then, %else
//!
//! %then:
//! ret 1
//!
//! %else:
//! %0 = sub @n, 1
//! %x = call @fib(%0)
//! %1 = sub @n, 2
//! %y = call @fib(%1)
//! %ans = add %x, %y
//! ret %ans
//! }
//! ```
//!
//! You can build the corresponding Koopa IR program:
//!
//! ```
//! use koopa::ir::*;
//! use koopa::ir::builder_traits::*;
//!
//! // create program and function
//! let mut program = Program::new();
//! let fib = program.new_func_def_with_param_names(
//! "@fib".into(),
//! vec![(Some("@n".into()), Type::get_i32())],
//! Type::get_i32(),
//! );
//! let fib_data = program.func_mut(fib);
//! let n = fib_data.params()[0];
//!
//! // entry/then/else basic block
//! let entry = fib_data.dfg_mut().new_bb().basic_block(Some("%entry".into()));
//! let then = fib_data.dfg_mut().new_bb().basic_block(Some("%then".into()));
//! let else_bb = fib_data.dfg_mut().new_bb().basic_block(Some("%else".into()));
//! fib_data.layout_mut().bbs_mut().extend([entry, then, else_bb]);
//!
//! // instructions in entry basic block
//! let two = fib_data.dfg_mut().new_value().integer(2);
//! let cond = fib_data.dfg_mut().new_value().binary(BinaryOp::Le, n, two);
//! let br = fib_data.dfg_mut().new_value().branch(cond, then, else_bb);
//! fib_data.layout_mut().bb_mut(entry).insts_mut().extend([cond, br]);
//!
//! // instructions in `then` basic block
//! let one = fib_data.dfg_mut().new_value().integer(1);
//! let ret = fib_data.dfg_mut().new_value().ret(Some(one));
//! fib_data.layout_mut().bb_mut(then).insts_mut().push_key_back(ret);
//!
//! // instructions in `else` basic block
//! let sub1 = fib_data.dfg_mut().new_value().binary(BinaryOp::Sub, n, one);
//! let call1 = fib_data.dfg_mut().new_value().call(fib, vec![sub1]);
//! let sub2 = fib_data.dfg_mut().new_value().binary(BinaryOp::Sub, n, two);
//! let call2 = fib_data.dfg_mut().new_value().call(fib, vec![sub2]);
//! let ans = fib_data.dfg_mut().new_value().binary(BinaryOp::Add, call1, call2);
//! let ret = fib_data.dfg_mut().new_value().ret(Some(ans));
//! fib_data.layout_mut().bb_mut(else_bb).insts_mut().extend([sub1, call1, sub2, call2, ans, ret]);
//! ```
pub use ;
pub use ;
pub use BinaryOp;