feat:llgo.Asm
test:asm test with `nop`
This commit is contained in:
18
_demo/asmcall/asmcall.go
Normal file
18
_demo/asmcall/asmcall.go
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
//llgo:link asm llgo.asm
|
||||||
|
func asm(instruction string) {}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
start := time.Now()
|
||||||
|
for i := 0; i < 100000; i++ {
|
||||||
|
asm("nop")
|
||||||
|
}
|
||||||
|
duration := time.Since(start)
|
||||||
|
fmt.Println("Duration:", duration)
|
||||||
|
}
|
||||||
10
cl/_testrt/asm/in.go
Normal file
10
cl/_testrt/asm/in.go
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import _ "unsafe"
|
||||||
|
|
||||||
|
//go:linkname asm llgo.asm
|
||||||
|
func asm(instruction string)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
asm("nop")
|
||||||
|
}
|
||||||
23
cl/_testrt/asm/out.ll
Normal file
23
cl/_testrt/asm/out.ll
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
; ModuleID = 'github.com/goplus/llgo/cl/_testrt/asm'
|
||||||
|
source_filename = "github.com/goplus/llgo/cl/_testrt/asm"
|
||||||
|
|
||||||
|
@"github.com/goplus/llgo/cl/_testrt/asm.init$guard" = global i1 false, align 1
|
||||||
|
|
||||||
|
define void @"github.com/goplus/llgo/cl/_testrt/asm.init"() {
|
||||||
|
_llgo_0:
|
||||||
|
%0 = load i1, ptr @"github.com/goplus/llgo/cl/_testrt/asm.init$guard", align 1
|
||||||
|
br i1 %0, label %_llgo_2, label %_llgo_1
|
||||||
|
|
||||||
|
_llgo_1: ; preds = %_llgo_0
|
||||||
|
store i1 true, ptr @"github.com/goplus/llgo/cl/_testrt/asm.init$guard", align 1
|
||||||
|
br label %_llgo_2
|
||||||
|
|
||||||
|
_llgo_2: ; preds = %_llgo_1, %_llgo_0
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
define void @"github.com/goplus/llgo/cl/_testrt/asm.main"() {
|
||||||
|
_llgo_0:
|
||||||
|
call void asm sideeffect "nop", ""()
|
||||||
|
ret void
|
||||||
|
}
|
||||||
@@ -506,6 +506,9 @@ const (
|
|||||||
llgoCgoCheckPointer = llgoCgoBase + 0x6
|
llgoCgoCheckPointer = llgoCgoBase + 0x6
|
||||||
llgoCgoCgocall = llgoCgoBase + 0x7
|
llgoCgoCgocall = llgoCgoBase + 0x7
|
||||||
|
|
||||||
|
llgoAsmBase = llgoInstrBase + 0x40
|
||||||
|
llgoAsm = llgoAsmBase + 0x0
|
||||||
|
|
||||||
llgoAtomicOpLast = llgoAtomicOpBase + int(llssa.OpUMin)
|
llgoAtomicOpLast = llgoAtomicOpBase + int(llssa.OpUMin)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
15
cl/instr.go
15
cl/instr.go
@@ -66,6 +66,17 @@ func cstr(b llssa.Builder, args []ssa.Value) (ret llssa.Expr) {
|
|||||||
panic("cstr(<string-literal>): invalid arguments")
|
panic("cstr(<string-literal>): invalid arguments")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// func asm(string)
|
||||||
|
func asm(b llssa.Builder, args []ssa.Value) (ret llssa.Expr) {
|
||||||
|
if len(args) == 1 {
|
||||||
|
if sv, ok := constStr(args[0]); ok {
|
||||||
|
b.InlineAsm(sv)
|
||||||
|
return llssa.Expr{Type: b.Prog.Void()}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panic("asm(<string-literal>): invalid arguments")
|
||||||
|
}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
// func _Cfunc_CString(s string) *int8
|
// func _Cfunc_CString(s string) *int8
|
||||||
@@ -329,6 +340,8 @@ var llgoInstrs = map[string]int{
|
|||||||
"_Cfunc__CMalloc": llgoCgoCMalloc,
|
"_Cfunc__CMalloc": llgoCgoCMalloc,
|
||||||
"_cgoCheckPointer": llgoCgoCheckPointer,
|
"_cgoCheckPointer": llgoCgoCheckPointer,
|
||||||
"_cgo_runtime_cgocall": llgoCgoCgocall,
|
"_cgo_runtime_cgocall": llgoCgoCgocall,
|
||||||
|
|
||||||
|
"asm": llgoAsm,
|
||||||
}
|
}
|
||||||
|
|
||||||
// funcOf returns a function by name and set ftype = goFunc, cFunc, etc.
|
// funcOf returns a function by name and set ftype = goFunc, cFunc, etc.
|
||||||
@@ -456,6 +469,8 @@ func (p *context) call(b llssa.Builder, act llssa.DoAction, call *ssa.CallCommon
|
|||||||
ret = pystr(b, args)
|
ret = pystr(b, args)
|
||||||
case llgoCstr:
|
case llgoCstr:
|
||||||
ret = cstr(b, args)
|
ret = cstr(b, args)
|
||||||
|
case llgoAsm:
|
||||||
|
ret = asm(b, args)
|
||||||
case llgoCgoCString:
|
case llgoCgoCString:
|
||||||
ret = p.cgoCString(b, args)
|
ret = p.cgoCString(b, args)
|
||||||
case llgoCgoCBytes:
|
case llgoCgoCBytes:
|
||||||
|
|||||||
11
ssa/expr.go
11
ssa/expr.go
@@ -275,6 +275,17 @@ func (b Builder) CBytes(v Expr) Expr {
|
|||||||
return b.Call(fn, v)
|
return b.Call(fn, v)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// InlineAsm generates inline assembly instruction
|
||||||
|
func (b Builder) InlineAsm(instruction string) {
|
||||||
|
if debugInstr {
|
||||||
|
log.Printf("InlineAsm %s\n", instruction)
|
||||||
|
}
|
||||||
|
|
||||||
|
typ := llvm.FunctionType(b.Prog.tyVoid(), nil, false)
|
||||||
|
asm := llvm.InlineAsm(typ, instruction, "", true, false, llvm.InlineAsmDialectATT, false)
|
||||||
|
b.impl.CreateCall(typ, asm, nil, "")
|
||||||
|
}
|
||||||
|
|
||||||
// GoString returns a Go string
|
// GoString returns a Go string
|
||||||
func (b Builder) GoString(v Expr) Expr {
|
func (b Builder) GoString(v Expr) Expr {
|
||||||
fn := b.Pkg.rtFunc("GoString")
|
fn := b.Pkg.rtFunc("GoString")
|
||||||
|
|||||||
Reference in New Issue
Block a user