Skip to content
This repository has been archived by the owner on Dec 10, 2022. It is now read-only.

Relooper.2018 #291

Open
wants to merge 61 commits into
base: master
Choose a base branch
from
Open
Changes from 1 commit
Commits
Show all changes
61 commits
Select commit Hold shift + click to select a range
1678d02
Added @JTranscRelooper annotation to manual control relooping
soywiz Jan 14, 2018
565962a
Some unfinished work
soywiz Jan 14, 2018
aef4ecf
Merge branch 'master' of github.com:jtransc/jtransc into relooper.2018
soywiz Jan 15, 2018
8092d1c
Not working: Some work on new relooper
soywiz Jan 15, 2018
6e7aba8
Small fix in relooper
soywiz Jan 15, 2018
30c3c28
More work on relooper
soywiz Jan 15, 2018
ad07e44
Some more work on relooper
soywiz Jan 15, 2018
39d89b7
Some more work on relooper
soywiz Jan 15, 2018
ba7e03d
Some Relooper adjustments
soywiz Jan 15, 2018
404e005
Some more work on relooper
soywiz Jan 15, 2018
8ca8e22
Some more work on relooper
soywiz Jan 15, 2018
3ec897c
Some more work on relooper
soywiz Jan 15, 2018
5ff3685
Some more work on relooper
soywiz Jan 15, 2018
ecd691b
Some more work on relooper
soywiz Jan 15, 2018
3f1c637
Some more work on relooper
soywiz Jan 15, 2018
19f8ee1
Some more work on relooper
soywiz Jan 15, 2018
c7ed617
Some more work on relooper
soywiz Jan 16, 2018
1f5e449
Some more work on relooper
soywiz Jan 16, 2018
829393d
Remove old annotations
soywiz Jan 16, 2018
887797b
Output jtransc version with JS for convenience
soywiz Jan 16, 2018
ca3697f
Old comment
soywiz Jan 16, 2018
0f6ece9
Add a flag for generators to specify if they support labels (most of …
soywiz Jan 16, 2018
61d9b81
Optimize DO_WHILE + if continue + break externally
soywiz Jan 16, 2018
f384cd0
Some more work on relooper
soywiz Jan 16, 2018
91b2168
Supported white and for loops
soywiz Jan 16, 2018
f1faab9
Enable relooper in benchmark
soywiz Jan 16, 2018
ce878e2
This PR will start 0.7 branch
soywiz Jan 16, 2018
84a323c
Inline errors
soywiz Jan 16, 2018
009e53d
Added failing test case using relooper + test code generator when deb…
soywiz Jan 16, 2018
dd4dd2e
Remove empty nodes, that were making the strong component contract (t…
soywiz Jan 16, 2018
1679db2
Do not create additional exit node if not required
soywiz Jan 16, 2018
ae6db58
Revert previos: we need end node for this to work properly
soywiz Jan 16, 2018
1e4c067
Fixed some relooper cases
soywiz Jan 16, 2018
90b2fc1
Some loop optimizations
soywiz Jan 16, 2018
fa640a7
Updated version
soywiz Jan 16, 2018
bc1c646
Detects if && chains
soywiz Jan 17, 2018
26b5b0d
Detects if || chains
soywiz Jan 17, 2018
2932f7e
Small fixes
soywiz Jan 17, 2018
2f13dd0
ConcurrentModification fix
soywiz Jan 17, 2018
fea7cf9
Convert recursive functions into normal queues
soywiz Jan 17, 2018
0a1611f
Optimize IF_ELSE with return on both sides
soywiz Jan 17, 2018
d1bf5d4
Optimize some IF_ELSE to act as guard clauses
soywiz Jan 17, 2018
7dd53ef
Kotlin 1.2.20
soywiz Jan 17, 2018
f39d87c
Fixed a problematic bug related to indices not matching in Relooper
soywiz Jan 17, 2018
3bd4e7a
org.gradle.caching=true
soywiz Jan 17, 2018
e7627df
Throw statements can be also guard clauses
soywiz Jan 17, 2018
8e80373
Remove exit node cycle
soywiz Jan 17, 2018
29f92a8
Detect some ifs leading to the exit point (return and throws)
soywiz Jan 17, 2018
7707735
Prepare to support switches in relooper
soywiz Jan 17, 2018
a083765
Some more work on supporting switch on relooper
soywiz Jan 17, 2018
4b04237
First switch implementation with ifs
soywiz Jan 17, 2018
9cbd1bb
Compact else-if chains to reduce indentation
soywiz Jan 17, 2018
b98197f
Disable switch in relooper temporarily
soywiz Jan 17, 2018
2e84278
Some work on fixing switches with a new/better conversor to basic Stm…
soywiz Jan 17, 2018
d1b3f37
switch fixes
soywiz Jan 17, 2018
34521b8
Some switch fixes
soywiz Jan 17, 2018
10a5ed4
Proper switch working (common node finding still requires some work t…
soywiz Jan 17, 2018
4159fcf
Prefer break/continue over return/throw as guard clauses
soywiz Jan 17, 2018
28699bc
Proper generate switch
soywiz Jan 17, 2018
6ef715c
Fix floating point comparisons
soywiz Jan 18, 2018
a8ebb76
Update badges to represent master instead of last build
soywiz Jan 18, 2018
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
Prev Previous commit
Next Next commit
Detects if || chains
  • Loading branch information
soywiz committed Jan 17, 2018
commit 26b5b0d160a89cc86c081689a00f2146bde8b561
110 changes: 73 additions & 37 deletions jtransc-core/src/com/jtransc/graph/Relooper.kt
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,7 @@ class Relooper(val types: AstTypes, val name: String = "unknown", val debug: Boo
}

override fun toString(): String = "L$index: " + dump(types, body.stm()).toString().replace('\n', ' ').trim() + " EDGES: $dstEdges. SRC_EDGES: ${srcEdges.size}"
fun isEmpty(): Boolean = body.isEmpty()
}

class Edge(val types: AstTypes, val src: Node, val dst: Node, val cond: AstExpr? = null) {
Expand Down Expand Up @@ -102,29 +103,39 @@ class Relooper(val types: AstTypes, val name: String = "unknown", val debug: Boo

data class Prepare(val nodes: List<Node>, val entry: Node, val exit: Node)

/**
* - Remove empty nodes
* - Combine nodes with && and ||
* - Create a single artificial exit node
*/
private fun prepare(entry: Node): Prepare {
val explored = LinkedHashSet<Node>()
val result = LinkedHashSet<Node>()
val exitNodes = arrayListOf<Node>()
fun explore(node: Node): Unit {
if (node in explored) return
explored += node
private fun Node.removeEmptyNodes(): Node {
var nentry = this
val processed = LinkedHashSet<Node>()
fun explore(node: Node) {
if (node in processed) return
processed += node

// Remove empty node
if (node.body.isEmpty() && node.dstEdges.size == 1 && node.dstEdgesButNext.isEmpty()) {
val dstNode = node.dstEdges.first().dst
for (e in node.srcEdges.toList()) {
e.remove()
e.src.edgeTo(dstNode, e.cond)
}
if (nentry == node) {
nentry = dstNode
}
explore(dstNode)
return
} else {
for (edge in node.dstEdges.toList()) {
explore(edge.dst)
}
}
}
explore(nentry)
return nentry
}

private fun Node.combineBooleanOpsEdges(): Node {
val entry = this
val processed = LinkedHashSet<Node>()
fun explore(node: Node) {
if (node in processed) return
processed += node

// This node may be a || or a &&
if (node.body.isEmpty() && node.srcEdges.size == 1 && node.srcEdges[0].src.next == node && node.dstEdges.size == 2) {
Expand All @@ -140,7 +151,11 @@ class Relooper(val types: AstTypes, val name: String = "unknown", val debug: Boo
val prevCondNode = prevCond.dst
val currCondNode = currCond.dst

// OR (after negating, acts as an && in code)
if (debug) {
println("!!")
}

// && in the original code
//* L0: EDGES: [goto L1;, IF ((p0 >= p1)) goto L2;]. SRC_EDGES: 0
//* L1: NOP(empty stm) EDGES: [goto L3;, IF ((p0 < 0)) goto L2;]. SRC_EDGES: 1
if (prevCondNode == currCondNode) {
Expand All @@ -151,44 +166,65 @@ class Relooper(val types: AstTypes, val name: String = "unknown", val debug: Boo
explore(currCond.dst)
explore(currNext)
return
} else {
//TODO()
}
// || in the original code
//* L0: EDGES: [goto L1;, IF ((p0 < p1)) goto L2;]. SRC_EDGES: 0
//* L1: NOP(empty stm) EDGES: [IF ((p0 < 0)) goto L4;, goto L2;]. SRC_EDGES: 1
else if (prevCondNode == currNext) {
prevCond.remove()
prev.edgeTo(currNext, prevCond.condOrTrue bor currCond.condOrTrue.not())
prev.next = currCond.dst
//println("-------")
explore(currCond.dst)
explore(currNext)
return
}
// None
else {

}
}
}

result += node
if (node.next == null) exitNodes += node
if (node.next != null) explore(node.next!!)
for (edge in node.dstEdges) explore(edge.dst)
}
explore(entry)
return entry
}

// Ensure just one single exit node
//val actualExit = if (exitNodes.size != 1) {
// val exit = node(listOf())
// for (node in exitNodes) edge(node, exit)
// result += exit
// exit
//} else {
// exitNodes.first()
//}

val actualExit = run {
val exit = node(listOf())
for (node in exitNodes) edge(node, exit)
result += exit
exit
/**
* - Create a single artificial exit node
* - Creates a list of nodes for the graph
*/
private fun prepare(entry: Node): Prepare {
val exit = node(listOf())
val explored = LinkedHashSet<Node>()
val result = LinkedHashSet<Node>()
fun explore(node: Node): Unit {
if (node in explored) return
explored += node
result += node
if (node.next == null) {
edge(node, exit)
}
if (node.next != null) explore(node.next!!)
for (edge in node.dstEdges) explore(edge.dst)
}
explore(entry)
result += exit

return Prepare(result.toList(), entry, actualExit)
return Prepare(result.toList(), entry, exit)
}

inline private fun trace(msg: () -> String) {
if (debug) println(msg())
}

fun render(entry: Node): AstStm {
fun render(rentry: Node): AstStm {
val entry = rentry
.removeEmptyNodes()
.combineBooleanOpsEdges()

val gresult = prepare(entry)
val g = graphList(gresult.nodes.map { it to it.possibleNextNodes })
if (debug) {
Expand Down