Porting code to TinyGo

How to port existing code to TinyGo.

TinyGo tries to stay compatible with the gc compiler toolchain. It should over time become more and more compatible as new features are implemented. However, if you’re porting existing code over then you may encounter a number of issues.

Serialization and deserialization

TinyGo does have limited support for the reflect package, but it is not complete. This is especially noticeable with serialization/deserialization packages such as encoding/json.

How to fix

Your best bet is is to use a different package than the standard library package. Packages optimized for speed will often work because they may avoid using reflection.

reflect.SliceHeader and reflect.StringHeader

These two structs are sometimes used for unsafely casting between strings and slices. In Go, the structs look like this:

type SliceHeader struct {
	Data uintptr
	Len  int
	Cap  int
}

In TinyGo, they look like this:

type SliceHeader struct {
	Data uintptr
	Len  uintptr
	Cap  uintptr
}

The difference is that Len and Cap are of type uintptr instead of int. This is because TinyGo supports AVR, which is a mixed 8/16-bit architecture with 16-bit pointers. The Go language requires int to be either 32-bit or 64-bit. To solve this mismatch, TinyGo has chosen to use uintptr for the Len and Cap fields.

Also, note the comment, which explicitly states that using this struct is not portable (emphasis mine):

SliceHeader is the runtime representation of a slice. It cannot be used safely or portably and its representation may change in a later release. Moreover, the Data field is not sufficient to guarantee the data it references will not be garbage collected, so programs must keep a separate, correctly typed pointer to the underlying data.

How to fix

Cast the types as appropriate. For example, code like this:

func split(s []byte) (*byte, int, int) {
	header := (*reflect.SliceHeader)(unsafe.Pointer(&s))
	return (*byte)(unsafe.Pointer(header.Data)), header.Len, header.Cap
}

Can easily be supported in both Go versions, with some type casts:

func split(s []byte) (*byte, int, int) {
	header := (*reflect.SliceHeader)(unsafe.Pointer(&s))
	return (*byte)(unsafe.Pointer(header.Data)), int(header.Len), int(header.Cap)
}

The difference is that header.Len and header.Cap are now cast to an int before returning.

Last modified September 8, 2022: guides: add page with some porting hints (b7c8b2a)