Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/rj45/nanogo

NanoGo a Go (golang) Subset for Homebrew / Hobby CPUs
https://github.com/rj45/nanogo

assembly compiler cpu go golang hobby-compiler hobby-language hobby-project homebrew-computer processor-design

Last synced: 10 days ago
JSON representation

NanoGo a Go (golang) Subset for Homebrew / Hobby CPUs

Awesome Lists containing this project

README

        

# NanoGo

`NanoGo` is a Go (golang) compiler for homebrew / hobby CPUs.

## Status

Currently a major refactor (well more like rewrite) is in progress to switch to a new and improved IR. Along with it the tranforms in xform are also getting a revamp. The new code is in ___2 packages but the old code is still there and should still work.

This is a work in progress! Currently the following should work:

- built in `print()` and `println()`
- word-sized operations (`int` and `uint`)
- string literals and iterating over strings
- memory mapped I/O using `unsafe`
- extern funcs with assembly snippets (useful if you have I/O instructions)

Also, only [rj32](https://github.com/rj45/rj32) and [A32](https://github.com/Artentus/a32emu) are supported, but if you would like assistance adding your CPU, open an issue. The key things needed to support a new CPU are a fully working emulator (that works on mac, linux and windows, arm and x86), and an assembler (customasm is preferred).

## What is it?

This compiler will take a Go package, read in all the packages it depends on in the usual way Go programs work, and compile all the code down into assembly in a style that is compatible with [customasm](https://github.com/hlorenzi/customasm).

As of this writing, customasm does not support linking, so a single large assembly file is produced. A "CPU Def" file can be included which configures the assembly language, as well as the memory layout with `#bank`s.

## Why Go?

C is great, but the language is not the easiest to parse, and while there's many great projects like [LCC](https://github.com/drh/lcc), they are not the easiest to work on and modify for a homebrew CPU.

Go is a very simple language, and is very fast to learn, yet powerful enough to be very productive.

As well, Go has a parser and type checker right in the the standard library. Further, there's an excellent [SSA library](https://golang.org/x/tools/go/ssa) that does most of the work of the frontend of a compiler. So all that's really required is a simple backend, which is what NanoGo is.

## Installing

Install [Go](https://golang.org/) for your system, then there are [releases](https://github.com/rj45/nanogo/releases) with prebuilt binaries and the runtime library source, plus some example programs.

Or you can install with go's installer:

```sh
go install github.com/rj45/nanogo@latest
```

You will also want to install [customasm](https://github.com/hlorenzi/customasm).

You may also want to install [emurj](https://github.com/rj45/rj32/emurj) if you want to be able to run programs in an `rj32` emulator.

You may need to set up a `NANOGOROOT` if you get errors about not being able to find the standard library code. Set it to the folder containing the `src` folder.

## Running

You can run NanoGo to produce assembly like so:

```sh
nanogo -o output.asm asm testdata/seive/seive.go
```

If you have [customasm](https://github.com/hlorenzi/customasm) installed, then you can get a binary or hex file (depending on the architecture) like so:

```sh
nanogo -o output.hex build testdata/seive/seive.go
```

If you have [emurj](https://github.com/rj45/rj32/emurj) installed, you can build and run the program like so:

```sh
nanogo run testdata/seive/seive.go
```

If you'd like to inspect, say, what phases the compiler goes through and all the transformations it does, say, on the `main.main()` function of the above code, you can produce an `ssa.html` using a modified version of the code Go uses for its compiler:

```sh
nanogo -dump main__main asm testdata/seive/seive.go
```

And then you can open `ssa.html` in your browser.

![ssa.html](./docs/img/ssa_html.png)

There is also a way to generate various `.dot` graphs of the flow of values through the program using this:

```sh
nanogo -dump main__main asm testdata/seive/seive.go
```

You can look at the control flow graph at the time of register allocation:

![control flow graph](docs/img/control_flow.png)

Or an exploded view of the flow of values through the program (at register allocation):

![value flow graph](docs/img/value_flow.png)

A `.dot` viewer like `xdot` is recommended because it will highlight which line goes where and allow you to zoom in.

## Limitations

Keep in mind that it took a team of people many years to build the Go compiler and make it as good as it is. There is a lot of work to do to come close to that.

So, while all of Go is parsed, currently many parts of Go are simply not implemented and will result in obscure errors if you try to use them. In the future, a consistent way to track where errors come from and better documentation for them may make this easier.

Defer is ignored, though it could be implemented in the future. There's no allocation yet, nor any freeing of memory. Recovering from panics will not be implemented. Runtime type reflection is not yet implemented. Maps are not yet implemented. Interfaces are similarly not there, nor slices. Global arrays do work however.

`int`s, `uint`s, `byte`s, `rune`s and pointers are 16-bits for rj32. But non-standard sizes can violate some assumptions in the standard library, so anything relying on those assumptions will have bugs.

Almost none of the standard library is supported. You can try it and see if it will work, but some fundamental assumptions are violated, as well as many features relied on are missing. This compiler is meant to help you write your own standard library, kernel, OS and other software for your own homebrew CPU, so you could see the lack of a standard library as a feature.

## Design

NanoGo uses the [golang.org/x/tools/go/ssa](https://golang.org/x/tools/go/ssa) package to build and parse Go into [SSA form](https://en.wikipedia.org/wiki/Static_single_assignment_form).

This IR is then [converted](./parser) into a different [IR](./ir) which is [easier to work with](https://www.hpl.hp.com/techreports/Compaq-DEC/WRL-2000-2.html) and [transform](./xform).

After transformation there is a [SSA based tree register allocator](./regalloc) which uses the properties of SSA to [allocate registers in linear time](https://compilers.cs.uni-saarland.de/projects/ssara/).

After a final transformation pass, then finally [assembly code is generated](./codegen) and output.

## Retargeting / Porting to Your CPU

Some effort was put in to make this easy. See [the retargeting documentation](docs/retargeting.md).

## Contributing

Contributions are welcome! If you spot a bug or an error, and have some time, submit a PR. Otherwise an issue is awesome. Typo fixes or docs are even welcome!

Contributions of your own architecture are welcome! See [the retargeting documentation](docs/retargeting.md).

Please be kind to one another and put effort into determining a kind way to share criticism.

## Forking, commercial use and permission

If you want to fork this project or use it commercially you do not need my permission nor do you need to inform me. But if you feel like letting me know, I would appreciate it, and you can do so via opening an issue.

If you do fork this project, all I ask is that the LICENSE file be kept intact and that your own copyright line be added to it. It's also polite to change the name if the fork should be made public.

## License

Copyright (c) 2021-2023 Ryan "rj45" Sanche ([github.com/rj45](https://github.com/rj45))

Licensed under the MIT License (MIT), see [LICENSE](./LICENSE).

Some parts Copyrighted by the Go Authors, under a BSD like license, see [Go's LICENSE](https://github.com/golang/go/blob/666fc173c02ff3004ac9ef867aa4eec7e243dde3/LICENSE). Files with this license are marked in the header.

Some parts Copyrighted by the TinyGo Authors, under a BSD like license, see [TinyGo's LICENSE](./LICENSE.tinygo). Files with this license are marked in the header.