Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/dramforever/finlog

Compiling finite generators to digital logic. WIP
https://github.com/dramforever/finlog

compiler digital-logic haskell verilog

Last synced: 2 months ago
JSON representation

Compiling finite generators to digital logic. WIP

Awesome Lists containing this project

README

        

# finlog

Just a toy compiler. *WIP*, *BUGS BEWARE*

## Current progress

Imperative code with loops broken by `yield` (name stolen from generators) like this:

```plain
proc main {
var counter: u32 = 0u32;
output counter;

var a: u32 = 1u32;
var out: u32 = 0u32;
loop {
var b: u32 = a + a;
var c: u32 = b + b;
a = a + b + c;
counter = counter + 1u32;

out = counter;
yield;
out = a;
yield;
}
}
```

… have a finite amount of computation between each yield, which means that they can be thought as automatons running in individual steps.

Synchronous hardware also runs step by step, which means that if the state space the program is finite, the code can be compiled into hardware. Hence the name of the project: *finlog*, a portmanteau of "finite" and "logic", and it's also a way of paying homage to Verilog.

(Curiously, Verilog *also* described hardware in imperative code, but it was originally designed to do so indirectly: *simulation* of hardware in *software*.)

In this case, the code is compiled into a synthesizable Verilog module (output abridged for readability):

```verilog
module main (clk, rst, out$counter);
input clk;
input rst;
output [31:0] out$counter;
reg [31:0] reg$counter$4;
reg [31:0] reg$a$9;
reg [31:0] reg$out$13;
reg [31:0] reg$b$18;
reg [31:0] reg$c$22;
assign out$counter = reg$counter$4;
wire [31:0] _$5 = reg$counter$4;
wire [31:0] _$6 = 32'd0;
// ...
reg [0:0] state$;
always @(posedge clk)
if (rst) begin
case (1'b1)
_$41: state$ <= 1'd0;
endcase
reg$counter$4 <= _$49;
reg$a$9 <= _$46;
reg$out$13 <= _$49;
reg$b$18 <= _$42;
reg$c$22 <= _$43;
end else case (state$)
1'd0: begin
case (1'b1)
_$41: state$ <= 1'd1;
endcase
reg$out$13 <= _$10;
end
1'd1: begin
// ...
end
endcase
endmodule
```

As you can see, the execution of an otherwise pretty imperative-looking program is converted into a state machine.

The feature set is limited but should allow basic imperative programs, But there's no documentation now, so for everything I said above you probably need to just take my word for it.

A bare minimum driver program is available and can be accessed by `stack run` or `cabal run`. An example implementing a prime sieve on a block RAM is available in the `examples/sieve` directory.

Hey, at least I wrote a sorta working compiler.

## The pipeline

- Input text is parsed into an AST using [megaparsec]
- The AST is converted into a control flow graph (CFG), the design of which is inspired greatly by [hoopl]. The 'algebraic' approach really took a large amount of mental burden off the construction of CFGs. Unfortunately, I was unable to find sufficient documentation on hoopl to get me started, and I am worried that the design of hoopl might not be flexible enough for use in finlog.
- The CFG is effectively broken up at `yield` statements and the program is executed symbolically in topological order. We track which state we were in and keep track of the boolean variable indicating whether this branch is taken, and at control flow merging points we combine the values and the indicator.
- The values are saved in a hash-consed DAG
- We gather up all the data in a way that's like a huge control flow merge.
- The program is a state machine now, and we generate Verilog code using a homemade AST and [prettyprinter].

A large amount of plumbing is powered by [mtl] and [microlens].

[megaparsec]: https://github.com/mrkkrp/megaparsec
[hoopl]: http://hackage.haskell.org/package/hoopl
[prettyprinter]: https://github.com/quchen/prettyprinter
[mtl]: http://hackage.haskell.org/package/mtl
[microlens]: https://github.com/monadfix/microlens

A curious note is that although a DAG is used in finlog, since we are compiling to hardware, a scheduler is not necessary.