ir,build: pragma support (#97)
Adds support for arbitrary compiler directives. Fixes #15
This commit is contained in:
committed by
GitHub
parent
0bcbe82731
commit
c8004ba627
@@ -174,6 +174,7 @@ For demonstrations of `avo` features:
|
|||||||
* **[complex](examples/complex):** Working with `complex{64,128}` types.
|
* **[complex](examples/complex):** Working with `complex{64,128}` types.
|
||||||
* **[data](examples/data):** Defining `DATA` sections.
|
* **[data](examples/data):** Defining `DATA` sections.
|
||||||
* **[ext](examples/ext):** Interacting with types from external packages.
|
* **[ext](examples/ext):** Interacting with types from external packages.
|
||||||
|
* **[pragma](examples/pragma):** Apply compiler directives to generated functions.
|
||||||
|
|
||||||
### Real Examples
|
### Real Examples
|
||||||
|
|
||||||
|
|||||||
@@ -92,6 +92,11 @@ func (c *Context) Doc(lines ...string) {
|
|||||||
c.activefunc().Doc = lines
|
c.activefunc().Doc = lines
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Pragma adds a compiler directive to the currently active function.
|
||||||
|
func (c *Context) Pragma(directive string, args ...string) {
|
||||||
|
c.activefunc().AddPragma(directive, args...)
|
||||||
|
}
|
||||||
|
|
||||||
// Attributes sets function attributes for the currently active function.
|
// Attributes sets function attributes for the currently active function.
|
||||||
func (c *Context) Attributes(a attr.Attribute) {
|
func (c *Context) Attributes(a attr.Attribute) {
|
||||||
c.activefunc().Attributes = a
|
c.activefunc().Attributes = a
|
||||||
|
|||||||
@@ -118,6 +118,9 @@ func Dereference(ptr gotypes.Component) gotypes.Component { return ctx.Dereferen
|
|||||||
// Doc sets documentation comment lines for the currently active function.
|
// Doc sets documentation comment lines for the currently active function.
|
||||||
func Doc(lines ...string) { ctx.Doc(lines...) }
|
func Doc(lines ...string) { ctx.Doc(lines...) }
|
||||||
|
|
||||||
|
// Pragma adds a compiler directive to the currently active function.
|
||||||
|
func Pragma(directive string, args ...string) { ctx.Pragma(directive, args...) }
|
||||||
|
|
||||||
// Attributes sets function attributes for the currently active function.
|
// Attributes sets function attributes for the currently active function.
|
||||||
func Attributes(a attr.Attribute) { ctx.Attributes(a) }
|
func Attributes(a attr.Attribute) { ctx.Attributes(a) }
|
||||||
|
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ Features:
|
|||||||
* **[complex](complex):** Working with `complex{64,128}` types.
|
* **[complex](complex):** Working with `complex{64,128}` types.
|
||||||
* **[data](data):** Defining `DATA` sections.
|
* **[data](data):** Defining `DATA` sections.
|
||||||
* **[ext](ext):** Interacting with types from external packages.
|
* **[ext](ext):** Interacting with types from external packages.
|
||||||
|
* **[pragma](pragma):** Apply compiler directives to generated functions.
|
||||||
|
|
||||||
"Real" examples:
|
"Real" examples:
|
||||||
|
|
||||||
|
|||||||
33
examples/pragma/README.md
Normal file
33
examples/pragma/README.md
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
# pragma
|
||||||
|
|
||||||
|
Apply [compiler directives](https://golang.org/pkg/cmd/compile/#hdr-Compiler_Directives) to `avo` functions.
|
||||||
|
|
||||||
|
The [code generator](asm.go) uses the `Pragma` function to apply the `//go:noescape` directive to the `Add` function:
|
||||||
|
|
||||||
|
[embedmd]:# (asm.go go /func main/ /^}/)
|
||||||
|
```go
|
||||||
|
func main() {
|
||||||
|
TEXT("Add", NOSPLIT, "func(z, x, y *uint64)")
|
||||||
|
Pragma("noescape")
|
||||||
|
Doc("Add adds the values at x and y and writes the result to z.")
|
||||||
|
zptr := Mem{Base: Load(Param("z"), GP64())}
|
||||||
|
xptr := Mem{Base: Load(Param("x"), GP64())}
|
||||||
|
yptr := Mem{Base: Load(Param("y"), GP64())}
|
||||||
|
x, y := GP64(), GP64()
|
||||||
|
MOVQ(xptr, x)
|
||||||
|
MOVQ(yptr, y)
|
||||||
|
ADDQ(x, y)
|
||||||
|
MOVQ(y, zptr)
|
||||||
|
RET()
|
||||||
|
Generate()
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Note the directive is applied in the generated stub file:
|
||||||
|
|
||||||
|
[embedmd]:# (stub.go go /\/\/ Add/ /func/)
|
||||||
|
```go
|
||||||
|
// Add adds the values at x and y and writes the result to z.
|
||||||
|
//go:noescape
|
||||||
|
func
|
||||||
|
```
|
||||||
24
examples/pragma/asm.go
Normal file
24
examples/pragma/asm.go
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
// +build ignore
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
. "github.com/mmcloughlin/avo/build"
|
||||||
|
. "github.com/mmcloughlin/avo/operand"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
TEXT("Add", NOSPLIT, "func(z, x, y *uint64)")
|
||||||
|
Pragma("noescape")
|
||||||
|
Doc("Add adds the values at x and y and writes the result to z.")
|
||||||
|
zptr := Mem{Base: Load(Param("z"), GP64())}
|
||||||
|
xptr := Mem{Base: Load(Param("x"), GP64())}
|
||||||
|
yptr := Mem{Base: Load(Param("y"), GP64())}
|
||||||
|
x, y := GP64(), GP64()
|
||||||
|
MOVQ(xptr, x)
|
||||||
|
MOVQ(yptr, y)
|
||||||
|
ADDQ(x, y)
|
||||||
|
MOVQ(y, zptr)
|
||||||
|
RET()
|
||||||
|
Generate()
|
||||||
|
}
|
||||||
2
examples/pragma/doc.go
Normal file
2
examples/pragma/doc.go
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
// Package pragma demonstrates the use of compiler directives.
|
||||||
|
package pragma
|
||||||
14
examples/pragma/pragma.s
Normal file
14
examples/pragma/pragma.s
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
// Code generated by command: go run asm.go -out pragma.s -stubs stub.go. DO NOT EDIT.
|
||||||
|
|
||||||
|
#include "textflag.h"
|
||||||
|
|
||||||
|
// func Add(z *uint64, x *uint64, y *uint64)
|
||||||
|
TEXT ·Add(SB), NOSPLIT, $0-24
|
||||||
|
MOVQ z+0(FP), AX
|
||||||
|
MOVQ x+8(FP), CX
|
||||||
|
MOVQ y+16(FP), DX
|
||||||
|
MOVQ (CX), CX
|
||||||
|
MOVQ (DX), DX
|
||||||
|
ADDQ CX, DX
|
||||||
|
MOVQ DX, (AX)
|
||||||
|
RET
|
||||||
16
examples/pragma/pragma_test.go
Normal file
16
examples/pragma/pragma_test.go
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
package pragma
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"testing/quick"
|
||||||
|
)
|
||||||
|
|
||||||
|
//go:generate go run asm.go -out pragma.s -stubs stub.go
|
||||||
|
|
||||||
|
func TestAdd(t *testing.T) {
|
||||||
|
got := func(x, y uint64) (z uint64) { Add(&z, &x, &y); return }
|
||||||
|
expect := func(x, y uint64) uint64 { return x + y }
|
||||||
|
if err := quick.CheckEqual(got, expect, nil); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
7
examples/pragma/stub.go
Normal file
7
examples/pragma/stub.go
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
// Code generated by command: go run asm.go -out pragma.s -stubs stub.go. DO NOT EDIT.
|
||||||
|
|
||||||
|
package pragma
|
||||||
|
|
||||||
|
// Add adds the values at x and y and writes the result to z.
|
||||||
|
//go:noescape
|
||||||
|
func Add(z *uint64, x *uint64, y *uint64)
|
||||||
15
ir/ir.go
15
ir/ir.go
@@ -144,10 +144,17 @@ func (f *File) Functions() []*Function {
|
|||||||
return fns
|
return fns
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Pragma represents a function compiler directive.
|
||||||
|
type Pragma struct {
|
||||||
|
Directive string
|
||||||
|
Arguments []string
|
||||||
|
}
|
||||||
|
|
||||||
// Function represents an assembly function.
|
// Function represents an assembly function.
|
||||||
type Function struct {
|
type Function struct {
|
||||||
Name string
|
Name string
|
||||||
Attributes attr.Attribute
|
Attributes attr.Attribute
|
||||||
|
Pragmas []Pragma
|
||||||
Doc []string
|
Doc []string
|
||||||
Signature *gotypes.Signature
|
Signature *gotypes.Signature
|
||||||
LocalSize int
|
LocalSize int
|
||||||
@@ -171,6 +178,14 @@ func NewFunction(name string) *Function {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// AddPragma adds a pragma to this function.
|
||||||
|
func (f *Function) AddPragma(directive string, args ...string) {
|
||||||
|
f.Pragmas = append(f.Pragmas, Pragma{
|
||||||
|
Directive: directive,
|
||||||
|
Arguments: args,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
// SetSignature sets the function signature.
|
// SetSignature sets the function signature.
|
||||||
func (f *Function) SetSignature(s *gotypes.Signature) {
|
func (f *Function) SetSignature(s *gotypes.Signature) {
|
||||||
f.Signature = s
|
f.Signature = s
|
||||||
|
|||||||
@@ -28,7 +28,18 @@ func (s *stubs) Print(f *ir.File) ([]byte, error) {
|
|||||||
for _, fn := range f.Functions() {
|
for _, fn := range f.Functions() {
|
||||||
s.NL()
|
s.NL()
|
||||||
s.Comment(fn.Doc...)
|
s.Comment(fn.Doc...)
|
||||||
|
for _, pragma := range fn.Pragmas {
|
||||||
|
s.pragma(pragma)
|
||||||
|
}
|
||||||
s.Printf("%s\n", fn.Stub())
|
s.Printf("%s\n", fn.Stub())
|
||||||
}
|
}
|
||||||
return s.Result()
|
return s.Result()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *stubs) pragma(p ir.Pragma) {
|
||||||
|
s.Printf("//go:%s", p.Directive)
|
||||||
|
for _, arg := range p.Arguments {
|
||||||
|
s.Printf(" %s", arg)
|
||||||
|
}
|
||||||
|
s.NL()
|
||||||
|
}
|
||||||
|
|||||||
28
printer/stubs_test.go
Normal file
28
printer/stubs_test.go
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
package printer_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/mmcloughlin/avo/build"
|
||||||
|
"github.com/mmcloughlin/avo/printer"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestStubsPragmas(t *testing.T) {
|
||||||
|
ctx := build.NewContext()
|
||||||
|
ctx.Function("f")
|
||||||
|
ctx.Pragma("noescape")
|
||||||
|
ctx.Pragma("linkname f remote.f")
|
||||||
|
ctx.SignatureExpr("func(x *uint64)")
|
||||||
|
ctx.RET()
|
||||||
|
|
||||||
|
AssertPrintsLines(t, ctx, printer.NewStubs, []string{
|
||||||
|
"// Code generated by avo. DO NOT EDIT.",
|
||||||
|
"",
|
||||||
|
"package printer",
|
||||||
|
"",
|
||||||
|
"//go:noescape",
|
||||||
|
"//go:linkname f remote.f",
|
||||||
|
"func f(x *uint64)",
|
||||||
|
"",
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -15,6 +15,7 @@ func AssertPrintsLines(t *testing.T, ctx *build.Context, pb printer.Builder, exp
|
|||||||
lines := strings.Split(output, "\n")
|
lines := strings.Split(output, "\n")
|
||||||
|
|
||||||
if len(expect) != len(lines) {
|
if len(expect) != len(lines) {
|
||||||
|
t.Logf("output:\n%s", output)
|
||||||
t.Fatalf("have %d lines of output; expected %d", len(lines), len(expect))
|
t.Fatalf("have %d lines of output; expected %d", len(lines), len(expect))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user