Implemented length storage of allocated memory

This commit is contained in:
Eduard Urbach 2025-02-09 14:14:41 +01:00
parent 1ff56e0856
commit 2b2e707520
Signed by: akyoto
GPG Key ID: C874F672B1AF20C0
27 changed files with 134 additions and 49 deletions

View File

@ -1,5 +1,5 @@
import io
import log import log
import sys
main() { main() {
collatz(12) collatz(12)
@ -19,6 +19,6 @@ collatz(x Int) {
return return
} }
sys.write(1, " ", 1) io.out(" ")
} }
} }

View File

@ -1,5 +1,5 @@
import io
import log import log
import sys
main() { main() {
fizzbuzz(15) fizzbuzz(15)
@ -10,9 +10,9 @@ fizzbuzz(n Int) {
loop { loop {
switch { switch {
x % 15 == 0 { print("FizzBuzz", 8) } x % 15 == 0 { io.out("FizzBuzz") }
x % 5 == 0 { print("Buzz", 4) } x % 5 == 0 { io.out("Buzz") }
x % 3 == 0 { print("Fizz", 4) } x % 3 == 0 { io.out("Fizz") }
_ { log.number(x) } _ { log.number(x) }
} }
@ -22,10 +22,6 @@ fizzbuzz(n Int) {
return return
} }
print(" ", 1) io.out(" ")
} }
} }
print(address Pointer, length Int) {
sys.write(1, address, length)
}

View File

@ -1,9 +1,5 @@
import sys import io
main() { main() {
print("Hello\n", 6) io.out("Hello\n")
}
print(address Pointer, length Int) {
sys.write(1, address, length)
} }

View File

@ -1,5 +1,5 @@
import io
import log import log
import sys
main() { main() {
n := 100 n := 100
@ -12,7 +12,7 @@ main() {
if isPrime(i) == 1 { if isPrime(i) == 1 {
if i != 2 { if i != 2 {
sys.write(1, " ", 1) io.out(" ")
} }
log.number(i) log.number(i)

View File

@ -1,6 +1,7 @@
// Open server and client in 2 terminals: // Open server and client in 2 terminals:
// [1] q run examples/server // [1] q run examples/server
// [2] curl http://127.0.0.1:8080 // [2] curl http://127.0.0.1:8080
import io
import net import net
import sys import sys
@ -8,30 +9,30 @@ main() {
socket := sys.socket(2, 1, 0) socket := sys.socket(2, 1, 0)
if socket < 0 { if socket < 0 {
sys.write(1, "socket error\n", 13) io.error("socket error\n")
sys.exit(1) sys.exit(1)
} }
if net.bind(socket, 8080) != 0 { if net.bind(socket, 8080) != 0 {
sys.write(1, "bind error\n", 11) io.error("bind error\n")
sys.exit(1) sys.exit(1)
} }
if sys.listen(socket, 128) != 0 { if sys.listen(socket, 128) != 0 {
sys.write(1, "listen error\n", 13) io.error("listen error\n")
sys.exit(1) sys.exit(1)
} }
sys.write(1, "listening...\n", 13) io.out("listening...\n")
loop { loop {
conn := sys.accept(socket, 0, 0) conn := sys.accept(socket, 0, 0)
if conn >= 0 { if conn >= 0 {
sys.write(conn, "HTTP/1.0 200 OK\r\nContent-Length: 6\r\n\r\nHello\n", 44) io.write(conn, "HTTP/1.0 200 OK\r\nContent-Length: 6\r\n\r\nHello\n")
sys.close(conn) sys.close(conn)
} else { } else {
sys.write(1, "error\n", 6) io.error("accept error\n")
} }
} }
} }

View File

@ -1,3 +1,4 @@
import io
import mem import mem
import sys import sys
@ -9,8 +10,8 @@ main() {
info := mem.alloc(24) info := mem.alloc(24)
loop { loop {
sys.write(1, "$ ", 2) io.out("$ ")
n := sys.read(0, command, length) n := io.in(command)
if n <= 0 { if n <= 0 {
return return

View File

@ -1,3 +1,4 @@
import io
import sys import sys
import thread import thread
import time import time
@ -10,8 +11,8 @@ main() {
} }
work() { work() {
sys.write(1, "[ ] start\n", 10) io.out("[ ] start\n")
time.sleep(10 * 1000 * 1000) time.sleep(10 * 1000 * 1000)
sys.write(1, "[x] end\n", 8) io.out("[x] end\n")
sys.exit(0) sys.exit(0)
} }

21
lib/io/io.q Normal file
View File

@ -0,0 +1,21 @@
import sys
in(buffer Pointer) -> Int {
return sys.read(0, buffer, len(buffer))
}
out(message Pointer) -> Int {
return sys.write(1, message, len(message))
}
error(message Pointer) -> Int {
return sys.write(2, message, len(message))
}
read(fd Int, buffer Pointer) -> Int {
return sys.read(fd, buffer, len(buffer))
}
write(fd Int, message Pointer) -> Int {
return sys.write(fd, message, len(message))
}

View File

@ -1,5 +1,12 @@
import sys import sys
alloc(length Int) -> Pointer { alloc(length Int) -> Pointer {
return sys.mmap(0, length, 0x1|0x2, 0x02|0x20) x := sys.mmap(0, length+8, 0x1|0x2, 0x02|0x20)
if x <= 0 {
return x
}
store(x, 8, length)
return x + 8
} }

View File

@ -1,5 +1,12 @@
import sys import sys
alloc(length Int) -> Pointer { alloc(length Int) -> Pointer {
return sys.mmap(0, length, 0x1|0x2, 0x02|0x1000) x := sys.mmap(0, length, 0x1|0x2, 0x02|0x1000)
if x <= 0 {
return x
}
store(x, 8, length)
return x + 8
} }

View File

@ -1,5 +1,12 @@
import sys import sys
alloc(length Int) -> Pointer { alloc(length Int) -> Pointer {
return sys.mmap(0, length, 0x0004, 0x3000) x := sys.mmap(0, length, 0x0004, 0x3000)
if x <= 0 {
return x
}
store(x, 8, length)
return x + 8
} }

View File

@ -1,3 +1,9 @@
write(fd Int, address Pointer, length Int) -> Int { read(fd Int, address Pointer, length Int) -> Int {
return kernel32.WriteFile(fd, address, length) kernel32.ReadFile(fd, address, length)
return length
}
write(fd Int, address Pointer, length Int) -> Int {
kernel32.WriteFile(fd, address, length)
return length
} }

View File

@ -24,5 +24,5 @@ func (a *Assembler) SetData(label string, bytes []byte) {
a.Data = data.Data{} a.Data = data.Data{}
} }
a.Data[label] = bytes a.Data.Insert(label, bytes)
} }

View File

@ -4,7 +4,7 @@ import "git.akyoto.dev/cli/q/src/cpu"
type Memory struct { type Memory struct {
Base cpu.Register Base cpu.Register
Offset byte Offset int8
OffsetRegister cpu.Register OffsetRegister cpu.Register
Length byte Length byte
} }

View File

@ -80,7 +80,7 @@ restart:
dataStart, _ := fs.Align(c.codeStart+Address(len(c.code)), config.Align) dataStart, _ := fs.Align(c.codeStart+Address(len(c.code)), config.Align)
for _, pointer := range c.dataPointers { for _, pointer := range c.dataPointers {
address := config.BaseAddress + dataStart + pointer.Resolve() address := config.BaseAddress + dataStart + pointer.Resolve() + 8
slice := c.code[pointer.Position : pointer.Position+4] slice := c.code[pointer.Position : pointer.Position+4]
binary.LittleEndian.PutUint32(slice, uint32(address)) binary.LittleEndian.PutUint32(slice, uint32(address))
} }

View File

@ -40,7 +40,7 @@ func (f *Function) CompileAssignArray(node *ast.Assign) error {
return err return err
} }
memory.Offset = byte(offset) memory.Offset = int8(offset)
} else { } else {
_, indexRegister, isTemporary, err := f.Evaluate(index) _, indexRegister, isTemporary, err := f.Evaluate(index)

View File

@ -32,7 +32,7 @@ func (f *Function) CompileAssignField(node *ast.Assign) error {
memory := asm.Memory{ memory := asm.Memory{
Base: variable.Register, Base: variable.Register,
Offset: byte(field.Offset), Offset: int8(field.Offset),
OffsetRegister: math.MaxUint8, OffsetRegister: math.MaxUint8,
Length: byte(field.Type.Size()), Length: byte(field.Type.Size()),
} }

View File

@ -27,6 +27,14 @@ func (f *Function) CompileCall(root *expression.Expression) (*Function, error) {
name = nameNode.Token.Text(f.File.Bytes) name = nameNode.Token.Text(f.File.Bytes)
switch name { switch name {
case "len":
return &Function{
Output: []*Output{
{
Type: types.Int,
},
},
}, f.CompileLen(root)
case "syscall": case "syscall":
return nil, f.CompileSyscall(root) return nil, f.CompileSyscall(root)
case "new": case "new":

26
src/core/CompileLen.go Normal file
View File

@ -0,0 +1,26 @@
package core
import (
"math"
"git.akyoto.dev/cli/q/src/asm"
"git.akyoto.dev/cli/q/src/expression"
)
// CompileLen returns the length of a slice.
func (f *Function) CompileLen(root *expression.Expression) error {
_, register, isTemporary, err := f.Evaluate(root.Children[1])
if err != nil {
return err
}
f.SaveRegister(f.CPU.Output[0])
f.MemoryRegister(asm.LOAD, asm.Memory{Base: register, Offset: -8, OffsetRegister: math.MaxUint8, Length: 8}, f.CPU.Output[0])
if isTemporary {
f.FreeRegister(register)
}
return nil
}

View File

@ -40,7 +40,7 @@ func (f *Function) ExpressionToRegister(node *expression.Expression, register cp
if node.Token.Kind == token.Array { if node.Token.Kind == token.Array {
array := f.VariableByName(node.Children[0].Token.Text(f.File.Bytes)) array := f.VariableByName(node.Children[0].Token.Text(f.File.Bytes))
offset, err := f.Number(node.Children[1].Token) offset, err := f.Number(node.Children[1].Token)
f.MemoryRegister(asm.LOAD, asm.Memory{Base: array.Register, Offset: byte(offset), Length: 1}, register) f.MemoryRegister(asm.LOAD, asm.Memory{Base: array.Register, Offset: int8(offset), Length: 1}, register)
return types.Int, err return types.Int, err
} }
@ -51,7 +51,7 @@ func (f *Function) ExpressionToRegister(node *expression.Expression, register cp
right := node.Children[1] right := node.Children[1]
name = right.Token.Text(f.File.Bytes) name = right.Token.Text(f.File.Bytes)
field := variable.Type.(*types.Pointer).To.(*types.Struct).FieldByName(name) field := variable.Type.(*types.Pointer).To.(*types.Struct).FieldByName(name)
f.MemoryRegister(asm.LOAD, asm.Memory{Base: variable.Register, Offset: byte(field.Offset), Length: byte(field.Type.Size())}, register) f.MemoryRegister(asm.LOAD, asm.Memory{Base: variable.Register, Offset: int8(field.Offset), Length: byte(field.Type.Size())}, register)
return field.Type, nil return field.Type, nil
} }

View File

@ -1,6 +1,8 @@
package core package core
import ( import (
"encoding/binary"
"git.akyoto.dev/cli/q/src/asm" "git.akyoto.dev/cli/q/src/asm"
"git.akyoto.dev/cli/q/src/cpu" "git.akyoto.dev/cli/q/src/cpu"
"git.akyoto.dev/cli/q/src/errors" "git.akyoto.dev/cli/q/src/errors"
@ -45,7 +47,12 @@ func (f *Function) TokenToRegister(t token.Token, register cpu.Register) (types.
case token.String: case token.String:
data := t.Bytes(f.File.Bytes) data := t.Bytes(f.File.Bytes)
data = String(data) data = String(data)
label := f.AddBytes(data)
slice := make([]byte, len(data)+8)
binary.LittleEndian.PutUint64(slice, uint64(len(data)))
copy(slice[8:], data)
label := f.AddBytes(slice)
f.SaveRegister(register) f.SaveRegister(register)
f.RegisterLabel(asm.MOVE, register, label) f.RegisterLabel(asm.MOVE, register, label)
return types.PointerAny, nil return types.PointerAny, nil

View File

@ -26,4 +26,5 @@ func (f *Machine) SaveRegister(register cpu.Register) {
newRegister := f.NewRegister() newRegister := f.NewRegister()
f.RegisterRegister(asm.MOVE, newRegister, register) f.RegisterRegister(asm.MOVE, newRegister, register)
variable.Register = newRegister variable.Register = newRegister
f.FreeRegister(register)
} }

View File

@ -3,6 +3,6 @@ package x86
import "git.akyoto.dev/cli/q/src/cpu" import "git.akyoto.dev/cli/q/src/cpu"
// LoadRegister loads from memory into a register. // LoadRegister loads from memory into a register.
func LoadRegister(code []byte, destination cpu.Register, offset byte, length byte, source cpu.Register) []byte { func LoadRegister(code []byte, destination cpu.Register, offset int8, length byte, source cpu.Register) []byte {
return memoryAccess(code, 0x8A, 0x8B, source, offset, length, destination) return memoryAccess(code, 0x8A, 0x8B, source, offset, length, destination)
} }

View File

@ -12,7 +12,7 @@ func TestLoadRegister(t *testing.T) {
usagePatterns := []struct { usagePatterns := []struct {
Destination cpu.Register Destination cpu.Register
Source cpu.Register Source cpu.Register
Offset byte Offset int8
Length byte Length byte
Code []byte Code []byte
}{ }{

View File

@ -7,7 +7,7 @@ import (
) )
// StoreNumber stores a number into the memory address included in the given register. // StoreNumber stores a number into the memory address included in the given register.
func StoreNumber(code []byte, register cpu.Register, offset byte, length byte, number int) []byte { func StoreNumber(code []byte, register cpu.Register, offset int8, length byte, number int) []byte {
code = memoryAccess(code, 0xC6, 0xC7, register, offset, length, 0b000) code = memoryAccess(code, 0xC6, 0xC7, register, offset, length, 0b000)
switch length { switch length {
@ -22,6 +22,6 @@ func StoreNumber(code []byte, register cpu.Register, offset byte, length byte, n
} }
// StoreRegister stores the contents of the `source` register into the memory address included in the given register. // StoreRegister stores the contents of the `source` register into the memory address included in the given register.
func StoreRegister(code []byte, register cpu.Register, offset byte, length byte, source cpu.Register) []byte { func StoreRegister(code []byte, register cpu.Register, offset int8, length byte, source cpu.Register) []byte {
return memoryAccess(code, 0x88, 0x89, register, offset, length, source) return memoryAccess(code, 0x88, 0x89, register, offset, length, source)
} }

View File

@ -11,7 +11,7 @@ import (
func TestStoreNumber(t *testing.T) { func TestStoreNumber(t *testing.T) {
usagePatterns := []struct { usagePatterns := []struct {
Register cpu.Register Register cpu.Register
Offset byte Offset int8
Length byte Length byte
Number int Number int
Code []byte Code []byte
@ -159,7 +159,7 @@ func TestStoreNumber(t *testing.T) {
func TestStoreRegister(t *testing.T) { func TestStoreRegister(t *testing.T) {
usagePatterns := []struct { usagePatterns := []struct {
RegisterTo cpu.Register RegisterTo cpu.Register
Offset byte Offset int8
Length byte Length byte
RegisterFrom cpu.Register RegisterFrom cpu.Register
Code []byte Code []byte

View File

@ -3,7 +3,7 @@ package x86
import "git.akyoto.dev/cli/q/src/cpu" import "git.akyoto.dev/cli/q/src/cpu"
// memoryAccess encodes a memory access. // memoryAccess encodes a memory access.
func memoryAccess(code []byte, opCode8 byte, opCode32 byte, register cpu.Register, offset byte, numBytes byte, source cpu.Register) []byte { func memoryAccess(code []byte, opCode8 byte, opCode32 byte, register cpu.Register, offset int8, numBytes byte, source cpu.Register) []byte {
opCode := opCode32 opCode := opCode32
if numBytes == 1 { if numBytes == 1 {
@ -27,7 +27,7 @@ func memoryAccess(code []byte, opCode8 byte, opCode32 byte, register cpu.Registe
} }
if mod == AddressMemoryOffset8 { if mod == AddressMemoryOffset8 {
code = append(code, offset) code = append(code, byte(offset))
} }
return code return code