Files
llgo/ssa/ssa_test.go

253 lines
6.6 KiB
Go
Raw Normal View History

2024-04-15 05:48:48 +08:00
/*
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
2024-04-15 17:12:30 +08:00
package ssa
2024-04-15 05:48:48 +08:00
import (
2024-04-18 17:54:19 +08:00
"go/token"
2024-04-15 05:48:48 +08:00
"go/types"
"testing"
)
2024-04-19 00:05:57 +08:00
func init() {
Initialize(InitAll)
}
2024-04-16 00:43:29 +08:00
/*
func asmPkg(t *testing.T, p *Package) {
2024-04-15 17:12:30 +08:00
b, err := p.CodeGen(AssemblyFile)
2024-04-15 05:48:48 +08:00
if err != nil {
t.Fatal("ctx.ParseIR:", err)
}
2024-04-15 17:12:30 +08:00
if v := string(b); v != "" {
t.Log(v)
}
2024-04-15 05:48:48 +08:00
}
2024-04-16 00:43:29 +08:00
*/
2024-04-18 01:18:41 +08:00
func assertPkg(t *testing.T, p Package, expected string) {
2024-04-16 00:43:29 +08:00
if v := p.String(); v != expected {
t.Fatalf("\n==> got:\n%s\n==> expected:\n%s\n", v, expected)
}
}
2024-04-15 05:48:48 +08:00
func TestVar(t *testing.T) {
2024-04-15 17:12:30 +08:00
prog := NewProgram(nil)
2024-04-16 00:43:29 +08:00
pkg := prog.NewPackage("bar", "foo/bar")
2024-04-15 05:48:48 +08:00
pkg.NewVar("a", types.Typ[types.Int])
2024-04-16 00:43:29 +08:00
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
@a = external global i64
`)
2024-04-15 05:48:48 +08:00
}
2024-04-16 03:05:20 +08:00
2024-04-16 03:08:39 +08:00
func TestStruct(t *testing.T) {
empty := types.NewStruct(nil, nil)
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
pkg.NewVar("a", empty)
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
@a = external global {}
`)
}
func TestNamedStruct(t *testing.T) {
2024-04-16 03:05:20 +08:00
src := types.NewPackage("bar", "foo/bar")
empty := types.NewNamed(types.NewTypeName(0, src, "Empty", nil), types.NewStruct(nil, nil), nil)
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
pkg.NewVar("a", empty)
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
%Empty = type {}
@a = external global %Empty
`)
}
2024-04-17 00:01:42 +08:00
2024-04-18 15:03:10 +08:00
func TestDeclFunc(t *testing.T) {
2024-04-17 00:01:42 +08:00
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
params := types.NewTuple(types.NewVar(0, nil, "a", types.Typ[types.Int]))
sig := types.NewSignatureType(nil, nil, nil, params, nil, false)
pkg.NewFunc("fn", sig)
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
declare void @fn(i64)
`)
}
2024-04-18 15:03:10 +08:00
func TestBasicFunc(t *testing.T) {
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
2024-04-18 17:15:06 +08:00
params := types.NewTuple(
types.NewVar(0, nil, "a", types.Typ[types.Int]),
types.NewVar(0, nil, "b", types.Typ[types.Float64]))
rets := types.NewTuple(types.NewVar(0, nil, "", types.Typ[types.Int]))
sig := types.NewSignatureType(nil, nil, nil, params, rets, false)
pkg.NewFunc("fn", sig).MakeBody("").
Return(prog.Val(1))
2024-04-18 15:03:10 +08:00
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
2024-04-18 17:15:06 +08:00
define i64 @fn(i64 %0, double %1) {
2024-04-18 15:03:10 +08:00
ret i64 1
}
`)
}
2024-04-18 17:15:06 +08:00
func TestFuncParam(t *testing.T) {
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
params := types.NewTuple(
types.NewVar(0, nil, "a", types.Typ[types.Int]),
types.NewVar(0, nil, "b", types.Typ[types.Float64]))
rets := types.NewTuple(types.NewVar(0, nil, "", types.Typ[types.Int]))
sig := types.NewSignatureType(nil, nil, nil, params, rets, false)
fn := pkg.NewFunc("fn", sig)
2024-04-18 17:54:19 +08:00
fn.MakeBody("").Return(fn.Param(0))
2024-04-18 17:15:06 +08:00
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
define i64 @fn(i64 %0, double %1) {
2024-04-18 17:54:19 +08:00
ret i64 %0
}
`)
}
2024-04-19 00:05:57 +08:00
func TestFuncCall(t *testing.T) {
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
params := types.NewTuple(
types.NewVar(0, nil, "a", types.Typ[types.Int]),
types.NewVar(0, nil, "b", types.Typ[types.Float64]))
rets := types.NewTuple(types.NewVar(0, nil, "", types.Typ[types.Int]))
sig := types.NewSignatureType(nil, nil, nil, params, rets, false)
fn := pkg.NewFunc("fn", sig)
fn.MakeBody("").
Return(prog.Val(1))
sigMain := types.NewSignatureType(nil, nil, nil, nil, nil, false)
b := pkg.NewFunc("main", sigMain).MakeBody("")
b.Call(fn.Expr, prog.Val(1), prog.Val(1.2))
b.Return()
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
define i64 @fn(i64 %0, double %1) {
ret i64 1
}
define void @main() {
%1 = call i64 @fn(i64 1, double 1.200000e+00)
ret void
}
`)
}
2024-04-19 00:39:09 +08:00
func TestFuncMultiRet(t *testing.T) {
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
params := types.NewTuple(
types.NewVar(0, nil, "b", types.Typ[types.Float64]))
rets := types.NewTuple(
types.NewVar(0, nil, "c", types.Typ[types.Int]),
types.NewVar(0, nil, "d", types.Typ[types.Float64]))
sig := types.NewSignatureType(nil, nil, nil, params, rets, false)
a := pkg.NewVar("a", types.Typ[types.Int])
fn := pkg.NewFunc("fn", sig)
b := fn.MakeBody("")
b.Return(a.Expr, fn.Param(0))
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
@a = external global i64
define { i64, double } @fn(double %0) {
%mrv = insertvalue { i64, double } { ptr @a, double poison }, double %0, 1
ret { i64, double } %mrv
}
`)
}
2024-04-20 13:50:48 +08:00
func TestPrintf(t *testing.T) {
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
pchar := types.NewPointer(types.Typ[types.Int8])
params := types.NewTuple(types.NewVar(0, nil, "format", pchar), VArg())
rets := types.NewTuple(types.NewVar(0, nil, "", types.Typ[types.Int32]))
sig := types.NewSignatureType(nil, nil, nil, params, rets, false)
pkg.NewFunc("printf", sig)
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
declare i32 @printf(ptr, ...)
`)
}
2024-04-18 17:54:19 +08:00
func TestBinOp(t *testing.T) {
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
params := types.NewTuple(
types.NewVar(0, nil, "a", types.Typ[types.Int]),
types.NewVar(0, nil, "b", types.Typ[types.Float64]))
rets := types.NewTuple(types.NewVar(0, nil, "", types.Typ[types.Int]))
sig := types.NewSignatureType(nil, nil, nil, params, rets, false)
fn := pkg.NewFunc("fn", sig)
b := fn.MakeBody("")
ret := b.BinOp(token.ADD, fn.Param(0), prog.Val(1))
b.Return(ret)
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
define i64 @fn(i64 %0, double %1) {
%3 = add i64 %0, 1
ret i64 %3
2024-04-18 17:15:06 +08:00
}
`)
}
2024-04-20 17:31:49 +08:00
func TestUnOp(t *testing.T) {
prog := NewProgram(nil)
pkg := prog.NewPackage("bar", "foo/bar")
params := types.NewTuple(
types.NewVar(0, nil, "p", types.NewPointer(types.Typ[types.Int])),
)
rets := types.NewTuple(types.NewVar(0, nil, "", types.Typ[types.Int]))
sig := types.NewSignatureType(nil, nil, nil, params, rets, false)
fn := pkg.NewFunc("fn", sig)
b := fn.MakeBody("")
ret := b.UnOp(token.MUL, fn.Param(0))
b.Return(ret)
assertPkg(t, pkg, `; ModuleID = 'foo/bar'
source_filename = "foo/bar"
define i64 @fn(ptr %0) {
%2 = load i64, ptr %0, align 4
ret i64 %2
}
`)
}