llgo/ssa: Alloca, ArrayAlloca

This commit is contained in:
xushiwei
2024-04-29 17:58:10 +08:00
parent 7a347d4563
commit 2b82af519c
7 changed files with 84 additions and 1 deletions

14
cl/_testcgo/alloca/in.go Normal file
View File

@@ -0,0 +1,14 @@
package main
import (
"unsafe"
"github.com/goplus/llgo/internal/runtime/c"
)
func main() {
s := c.String("Hi\n")
s2 := c.Alloca(4)
c.Memcpy(s2, unsafe.Pointer(s), 4)
c.Printf(c.String("%s"), s)
}

36
cl/_testcgo/alloca/out.ll Normal file
View File

@@ -0,0 +1,36 @@
; ModuleID = 'main'
source_filename = "main"
%"github.com/goplus/llgo/internal/runtime.String" = type { ptr, i64 }
@"main.init$guard" = global ptr null
define void @main.init() {
_llgo_0:
%0 = load i1, ptr @"main.init$guard", align 1
br i1 %0, label %_llgo_2, label %_llgo_1
_llgo_1: ; preds = %_llgo_0
store i1 true, ptr @"main.init$guard", align 1
br label %_llgo_2
_llgo_2: ; preds = %_llgo_1, %_llgo_0
ret void
}
define void @main() {
_llgo_0:
call void @main.init()
%0 = call ptr @"github.com/goplus/llgo/internal/runtime/c.String"([4 x i8] c"Hi\0A\00")
%1 = alloca i8, i64 4, align 1
%2 = call ptr @memcpy(ptr %1, ptr %0, i64 4)
%3 = call ptr @"github.com/goplus/llgo/internal/runtime/c.String"([3 x i8] c"%s\00")
%4 = call i32 (ptr, ...) @printf(ptr %3, ptr %0)
ret void
}
declare ptr @"github.com/goplus/llgo/internal/runtime/c.String"(%"github.com/goplus/llgo/internal/runtime.String")
declare ptr @memcpy(ptr, ptr, i64)
declare i32 @printf(ptr, ...)

View File

@@ -282,6 +282,14 @@ func cstr(b llssa.Builder, args []ssa.Value) (ret llssa.Expr) {
panic("cstr(<string-literal>): invalid arguments") panic("cstr(<string-literal>): invalid arguments")
} }
func (p *context) alloca(b llssa.Builder, args []ssa.Value) (ret llssa.Expr) {
if len(args) == 1 {
n := p.compileValue(b, args[0])
return b.Alloca(n)
}
panic("alloca(size uintptr): invalid arguments")
}
func (p *context) compileInstrOrValue(b llssa.Builder, iv instrOrValue, asValue bool) (ret llssa.Expr) { func (p *context) compileInstrOrValue(b llssa.Builder, iv instrOrValue, asValue bool) (ret llssa.Expr) {
if asValue { if asValue {
if v, ok := p.bvals[iv]; ok { if v, ok := p.bvals[iv]; ok {
@@ -319,6 +327,8 @@ func (p *context) compileInstrOrValue(b llssa.Builder, iv instrOrValue, asValue
ret = b.Call(fn.Expr, args...) ret = b.Call(fn.Expr, args...)
case llgoCstr: case llgoCstr:
ret = cstr(b, call.Args) ret = cstr(b, call.Args)
case llgoAlloca:
ret = p.alloca(b, call.Args)
default: default:
panic("todo") panic("todo")
} }

View File

@@ -16,6 +16,7 @@
package c package c
import "C"
import "unsafe" import "unsafe"
const ( const (
@@ -38,4 +39,4 @@ func Malloc(size uintptr) unsafe.Pointer
func Memcpy(dst, src unsafe.Pointer, n uintptr) unsafe.Pointer func Memcpy(dst, src unsafe.Pointer, n uintptr) unsafe.Pointer
//go:linkname Printf C.printf //go:linkname Printf C.printf
func Printf(format *int8, __llgo_va_list ...any) func Printf(format *int8, __llgo_va_list ...any) C.int

View File

@@ -420,6 +420,28 @@ func (b Builder) Alloc(t *types.Pointer, heap bool) (ret Expr) {
return return
} }
// Alloca allocates space for n bytes.
func (b Builder) Alloca(n Expr) (ret Expr) {
if debugInstr {
log.Printf("Alloca %v\n", n.impl)
}
prog := b.Prog
telem := prog.tyInt8()
ret.impl = llvm.CreateArrayAlloca(b.impl, telem, n.impl)
ret.Type = &aType{prog.tyVoidPtr(), types.Typ[types.UnsafePointer], vkPtr}
return
}
// ArrayAlloca reserves space for an array of n elements of type telem.
func (b Builder) ArrayAlloca(telem Type, n Expr) (ret Expr) {
if debugInstr {
log.Printf("ArrayAlloca %v, %v\n", telem.t, n.impl)
}
ret.impl = llvm.CreateArrayAlloca(b.impl, telem.ll, n.impl)
ret.Type = b.Prog.Pointer(telem)
return
}
// The ChangeType instruction applies to X a value-preserving type // The ChangeType instruction applies to X a value-preserving type
// change to Type(). // change to Type().
// //