Added separate value type for extern calls
All checks were successful
/ test (push) Successful in 30s
All checks were successful
/ test (push) Successful in 30s
This commit is contained in:
parent
4aa4d613a1
commit
3f481d0bd3
4 changed files with 110 additions and 28 deletions
|
@ -145,12 +145,11 @@ func (f *Function) Evaluate(expr *expression.Expression) (ssa.Value, error) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
call := &ssa.Call{
|
if fn.IsExtern() {
|
||||||
Arguments: args,
|
return f.Append(&ssa.CallExtern{Arguments: args, Source: ssa.Source(expr.Source)}), nil
|
||||||
Source: ssa.Source(expr.Source),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return f.Append(call), nil
|
return f.Append(&ssa.Call{Arguments: args, Source: ssa.Source(expr.Source)}), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
case token.Dot:
|
case token.Dot:
|
||||||
|
|
73
src/ssa/CallExtern.go
Normal file
73
src/ssa/CallExtern.go
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
package ssa
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"git.urbach.dev/cli/q/src/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CallExtern struct {
|
||||||
|
Id
|
||||||
|
Arguments
|
||||||
|
Liveness
|
||||||
|
Source
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *CallExtern) Equals(v Value) bool {
|
||||||
|
b, sameType := v.(*CallExtern)
|
||||||
|
|
||||||
|
if !sameType {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return a.Arguments.Equals(b.Arguments)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *CallExtern) IsConst() bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *CallExtern) Debug(expand bool) string {
|
||||||
|
tmp := strings.Builder{}
|
||||||
|
|
||||||
|
if expand {
|
||||||
|
tmp.WriteString(v.Arguments[0].String())
|
||||||
|
} else {
|
||||||
|
tmp.WriteString("%")
|
||||||
|
tmp.WriteString(strconv.Itoa(v.Arguments[0].ID()))
|
||||||
|
}
|
||||||
|
|
||||||
|
tmp.WriteString("(")
|
||||||
|
args := v.Arguments[1:]
|
||||||
|
|
||||||
|
for i, arg := range args {
|
||||||
|
if expand {
|
||||||
|
tmp.WriteString(arg.String())
|
||||||
|
} else {
|
||||||
|
tmp.WriteString("%")
|
||||||
|
tmp.WriteString(strconv.Itoa(arg.ID()))
|
||||||
|
}
|
||||||
|
|
||||||
|
if i != len(args)-1 {
|
||||||
|
tmp.WriteString(", ")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tmp.WriteString(")")
|
||||||
|
return tmp.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *CallExtern) String() string {
|
||||||
|
return v.Debug(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *CallExtern) Type() types.Type {
|
||||||
|
typ := v.Arguments[0].(*Function).Typ
|
||||||
|
|
||||||
|
if len(typ.Output) == 0 {
|
||||||
|
return types.Void
|
||||||
|
}
|
||||||
|
|
||||||
|
return typ.Output[0]
|
||||||
|
}
|
|
@ -21,7 +21,7 @@ func (f *Compiler) GenerateAssembly(ir ssa.IR, isLeaf bool) {
|
||||||
}
|
}
|
||||||
|
|
||||||
switch instr := instr.(type) {
|
switch instr := instr.(type) {
|
||||||
case *ssa.Call, *ssa.Syscall:
|
case *ssa.Call, *ssa.CallExtern, *ssa.Syscall:
|
||||||
f.ValueToRegister(instr, f.CPU.Return[0])
|
f.ValueToRegister(instr, f.CPU.Return[0])
|
||||||
|
|
||||||
case *ssa.Return:
|
case *ssa.Return:
|
||||||
|
|
|
@ -25,17 +25,6 @@ func (f *Compiler) ValueToRegister(instr ssa.Value, destination cpu.Register) {
|
||||||
case *ssa.Call:
|
case *ssa.Call:
|
||||||
fn := instr.Arguments[0].(*ssa.Function)
|
fn := instr.Arguments[0].(*ssa.Function)
|
||||||
args := instr.Arguments[1:]
|
args := instr.Arguments[1:]
|
||||||
|
|
||||||
if fn.IsExtern {
|
|
||||||
for i := range slices.Backward(args) {
|
|
||||||
f.ValueToRegister(args[i], f.CPU.ExternCall[i])
|
|
||||||
}
|
|
||||||
|
|
||||||
dot := strings.IndexByte(fn.UniqueName, '.')
|
|
||||||
library := fn.UniqueName[:dot]
|
|
||||||
function := fn.UniqueName[dot+1:]
|
|
||||||
f.Assembler.Append(&asm.CallExtern{Library: library, Function: function})
|
|
||||||
} else {
|
|
||||||
offset := 0
|
offset := 0
|
||||||
|
|
||||||
for i := range slices.Backward(args) {
|
for i := range slices.Backward(args) {
|
||||||
|
@ -52,8 +41,29 @@ func (f *Compiler) ValueToRegister(instr ssa.Value, destination cpu.Register) {
|
||||||
}
|
}
|
||||||
|
|
||||||
f.Assembler.Append(&asm.Call{Label: fn.UniqueName})
|
f.Assembler.Append(&asm.Call{Label: fn.UniqueName})
|
||||||
|
|
||||||
|
if destination == f.CPU.Return[0] {
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
f.Assembler.Append(&asm.MoveRegisterRegister{
|
||||||
|
Destination: destination,
|
||||||
|
Source: f.CPU.Return[0],
|
||||||
|
})
|
||||||
|
|
||||||
|
case *ssa.CallExtern:
|
||||||
|
fn := instr.Arguments[0].(*ssa.Function)
|
||||||
|
args := instr.Arguments[1:]
|
||||||
|
|
||||||
|
for i := range slices.Backward(args) {
|
||||||
|
f.ValueToRegister(args[i], f.CPU.ExternCall[i])
|
||||||
|
}
|
||||||
|
|
||||||
|
dot := strings.IndexByte(fn.UniqueName, '.')
|
||||||
|
library := fn.UniqueName[:dot]
|
||||||
|
function := fn.UniqueName[dot+1:]
|
||||||
|
f.Assembler.Append(&asm.CallExtern{Library: library, Function: function})
|
||||||
|
|
||||||
if destination == f.CPU.Return[0] {
|
if destination == f.CPU.Return[0] {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue