Files
five/hbrt/ops_collection.go
Charles KWON OhJun 486e466592 feat: FiveSql2 43/43, @byref, mutable closure, RTL 479, DateTime fix
Major changes since last commit:
- FiveSql2 SQL:1999 engine (10,458 LOC) — 43/43 ALL PASS
- 21 compiler/runtime bugs fixed (short-circuit AND/OR, FOR LOOP, etc.)
- @byref pass-by-reference via RefCell pattern
- Mutable closure capture (EnsureLocalRef + RefCell sharing)
- RTL: 400 → 479 functions (+79: file, string, datetime, hash, UTF-8)
- DateTime/Timestamp fully working (hb_DateTime, hb_Hour/Min/Sec, display)
- Reserved word guard (39 keywords blocked from function calls)
- AEval arg order fix (element before index)
- Closure capture redecl fix (unique _cap_ names per block)
- Hash/string indexing in ArrayPush/ArrayPop
- Harbour compat test suite: 51/51
- 4 docs: Porting Report, Implementation Plan, Optimization Plan, Commercialization

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-11 11:35:37 +09:00

147 lines
3.2 KiB
Go

// Copyright (c) 2026 Charles KWON OhJun (charleskwonohjun@gmail.com)
// All rights reserved.
// Collection operations for the Five runtime.
// Array generation, indexing, hash generation, and code block evaluation.
// Harbour: HB_P_ARRAYGEN, HB_P_ARRAYPUSH, HB_P_ARRAYPOP, etc.
package hbrt
import "unsafe"
// ArrayGen pops n items from stack and creates an array.
// Harbour: HB_P_ARRAYGEN / hb_vmArrayGen
func (t *Thread) ArrayGen(n int) {
items := make([]Value, n)
for i := n - 1; i >= 0; i-- {
items[i] = t.pop()
}
t.push(MakeArrayFrom(items))
}
// HashGen pops n key-value pairs and creates a hash.
// Stack: [key1] [val1] [key2] [val2] ... → Hash
func (t *Thread) HashGen(n int) {
hh := &HbHash{
Keys: make([]Value, n),
Values: make([]Value, n),
}
for i := n - 1; i >= 0; i-- {
hh.Values[i] = t.pop()
hh.Keys[i] = t.pop()
}
t.push(Value{
info: makeInfo(tHash, 0, 0),
ptr: unsafe.Pointer(hh),
})
}
// ArrayPush pops index and array/hash, pushes array[index] or hash[key].
// Harbour: HB_P_ARRAYPUSH
func (t *Thread) ArrayPush() {
idx := t.pop()
arr := t.pop()
// Hash: h[key] → value
if arr.IsHash() {
hh := arr.AsHash()
for i, k := range hh.Keys {
if valueEqual(k, idx) {
t.push(hh.Values[i])
return
}
}
t.push(MakeNil())
return
}
if !arr.IsArray() {
// String indexing: cString[n] → single char
if arr.IsString() {
s := arr.AsString()
n := int(idx.AsNumInt())
if n >= 1 && n <= len(s) {
t.push(MakeString(string(s[n-1])))
return
}
t.push(MakeString(""))
return
}
panic(t.argError("[]", arr, idx))
}
ha := arr.AsArray()
n := int(idx.AsNumInt())
// Harbour: 1-based indexing
if n < 1 || n > len(ha.Items) {
panic(t.runtimeError("array index out of bounds"))
}
t.push(ha.Items[n-1])
}
// ArrayPop pops value, index, array/hash and sets array[index]=value or hash[key]=value.
// Harbour: HB_P_ARRAYPOP
func (t *Thread) ArrayPop() {
val := t.pop()
idx := t.pop()
arr := t.pop()
// Hash: h[key] := value
if arr.IsHash() {
hh := arr.AsHash()
for i, k := range hh.Keys {
if valueEqual(k, idx) {
hh.Values[i] = val
return
}
}
hh.Keys = append(hh.Keys, idx)
hh.Values = append(hh.Values, val)
return
}
if !arr.IsArray() {
panic(t.argError("[]=", arr, idx))
}
ha := arr.AsArray()
n := int(idx.AsNumInt())
if n < 1 || n > len(ha.Items) {
panic(t.runtimeError("array index out of bounds"))
}
ha.Items[n-1] = val
}
// EvalBlock evaluates a code block on the stack with nArgs arguments.
// Stack: [block] [arg1] ... [argN] → [result]
func (t *Thread) EvalBlock(nArgs int) {
args := make([]Value, nArgs)
for i := nArgs - 1; i >= 0; i-- {
args[i] = t.pop()
}
blockVal := t.pop()
if !blockVal.IsBlock() {
panic(t.argError("Eval", blockVal))
}
blk := blockVal.AsBlock()
// Push args for Frame
for _, arg := range args {
t.push(arg)
}
t.pendingParams = nArgs
blk.Fn(t)
t.push(t.retVal)
}
// PushBlock creates a code block and pushes it onto the stack.
func (t *Thread) PushBlock(fn func(*Thread), detachedLocals int) {
t.push(MakeBlock(fn, detachedLocals))
}
// PushSelf pushes the current Self object (for :: access in methods).
func (t *Thread) PushSelf() {
t.push(t.self)
}