Implemented length storage of allocated memory
This commit is contained in:
parent
1ff56e0856
commit
2b2e707520
@ -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(" ")
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -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)
|
|
||||||
}
|
}
|
@ -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)
|
|
||||||
}
|
}
|
@ -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)
|
||||||
|
@ -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")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -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
|
||||||
|
@ -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
21
lib/io/io.q
Normal 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))
|
||||||
|
}
|
@ -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
|
||||||
}
|
}
|
@ -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
|
||||||
}
|
}
|
@ -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
|
||||||
}
|
}
|
@ -1,3 +1,9 @@
|
|||||||
|
read(fd Int, address Pointer, length Int) -> Int {
|
||||||
|
kernel32.ReadFile(fd, address, length)
|
||||||
|
return length
|
||||||
|
}
|
||||||
|
|
||||||
write(fd Int, address Pointer, length Int) -> Int {
|
write(fd Int, address Pointer, length Int) -> Int {
|
||||||
return kernel32.WriteFile(fd, address, length)
|
kernel32.WriteFile(fd, address, length)
|
||||||
|
return length
|
||||||
}
|
}
|
@ -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)
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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))
|
||||||
}
|
}
|
||||||
|
@ -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)
|
||||||
|
|
||||||
|
@ -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()),
|
||||||
}
|
}
|
||||||
|
@ -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
26
src/core/CompileLen.go
Normal 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
|
||||||
|
}
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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
|
||||||
|
@ -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)
|
||||||
}
|
}
|
||||||
|
@ -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)
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
}{
|
}{
|
||||||
|
@ -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)
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
|
@ -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
|
||||||
|
Loading…
x
Reference in New Issue
Block a user