Files
five/hbrdd/dbf/mmap_windows.go
CharlesKWON f4ed42556b checkpoint: season-wide bug fix campaign + infra
Cumulative season's silent-bug hunting (~62 fixes) across the FiveSql2
SQL engine, the Five compiler/runtime, and the hbrdd RDD layer. Saved
as a single checkpoint before refactoring the parser to delegate xBase
command translation to the preprocessor.

Highlights:

FiveSql2 engine (_FiveSql2/src/)
- prefix-glob index attach -> explicit convention (<table>_pk.ntx,
  <table>_uq.ntx, <table>.cdx) — fixes silent multi-row INSERT row-drop
- DROP/CREATE TABLE FErase chain extended (.cdx, .fsc, .fsv, .dbt, .fpt)
- COUNT(DISTINCT col) parsed + aggregated via hSeen hash
- UNION column-count mismatch returns SQL_ERR_GRAMMAR (was silent)
- DISTINCT + ORDER BY hidden-col leak fixed (trim before DISTINCT)
- Derived table FROM (SELECT...) + JOIN right-side derived
- Self-FK CASCADE depth 2+ via SqlGetSingleColPK pre-collect
- LAG/LEAD default arg uses SqlEvalRowExpr (handles -N const exprs)
- DATE literal round-trip validation (Feb 29 non-leap rejected)
- CREATE OR REPLACE VIEW; CREATE VIEW errors on already-exists
- AlterTable type dispatcher comma-wrapped (1-char type "A" no longer
  matches CHARACTER)

Compiler / runtime
- gengo: HB_ -> FV_ prefix on emitted Go function names (Five identity)
- gengo split: emit_block.go, emit_stmt.go, folding.go extracted
- parser/stmtreg.go nudges
- hbrt: debug TUI/CLI restructure (debugcmd, debugkey, termios_*),
  windows debug stubs collapsed
- thread/vm/value/class/pcinterp tightening from panic traces

RDD layer (hbrdd/)
- dbf: null bitmap support (null.go + null_test.go), mmap split
  (mmap_posix.go / mmap_windows.go), byte-level numeric parse
- ntx/cdx: windows mmap parity
- workarea + mem RDD: cross-area state-bleed fixes

RTL (hbrtl/)
- errorlog rewrite with platform-specific FD (errorlog_fd_unix /
  errorlog_fd_other)
- sqlscan, sqlhelpers, indexrtl, datetime extensions

Gates green at checkpoint:
- go test ./...        : PASS
- FiveSql2 SQL:1999    : 43/43
- Harbour compat       : 56/56

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-30 09:26:25 +09:00

93 lines
2.3 KiB
Go

// Copyright (c) 2026 Charles KWON OhJun (charleskwonohjun@gmail.com)
// All rights reserved.
//go:build windows
// Windows mmap for DBF record data — CreateFileMappingW + MapViewOfFile
// with PAGE_READONLY. Parallels the POSIX syscall.Mmap path: on mmap
// failure we leave a.mmapData == nil so reads fall back to ReadAt.
//
// Mapping handles are tracked in a package-local registry keyed by
// view address so unmapDBF can recover the HANDLE given only the
// []byte we stored on the Area. Matches the hbrdd/ntx and hbrdd/cdx
// implementations byte-for-byte to stay maintainable.
package dbf
import (
"fmt"
"sync"
"syscall"
"unsafe"
)
const (
pageReadonly = 0x02
fileMapRead = 0x0004
)
var (
kernel32 = syscall.NewLazyDLL("kernel32.dll")
procCreateFileMappingW = kernel32.NewProc("CreateFileMappingW")
procMapViewOfFile = kernel32.NewProc("MapViewOfFile")
procUnmapViewOfFile = kernel32.NewProc("UnmapViewOfFile")
procCloseHandle = kernel32.NewProc("CloseHandle")
mappingMu sync.Mutex
mappings = map[uintptr]syscall.Handle{}
)
func (a *DBFArea) mmapDBF() {
fi, err := a.dataFile.Stat()
if err != nil || fi.Size() < int64(a.header.HeaderLen) {
return
}
size := int(fi.Size())
if size <= 0 {
return
}
hFile := syscall.Handle(a.dataFile.Fd())
sizeHigh := uint32(uint64(size) >> 32)
sizeLow := uint32(uint64(size) & 0xFFFFFFFF)
hMap, _, _ := procCreateFileMappingW.Call(
uintptr(hFile), 0, pageReadonly,
uintptr(sizeHigh), uintptr(sizeLow), 0,
)
if hMap == 0 {
return
}
addr, _, _ := procMapViewOfFile.Call(hMap, fileMapRead, 0, 0, uintptr(size))
if addr == 0 {
procCloseHandle.Call(hMap)
return
}
a.mmapData = unsafe.Slice((*byte)(unsafe.Pointer(addr)), size)
mappingMu.Lock()
mappings[addr] = syscall.Handle(hMap)
mappingMu.Unlock()
}
func (a *DBFArea) unmapDBF() {
if a.mmapData == nil {
return
}
addr := uintptr(unsafe.Pointer(&a.mmapData[0]))
mappingMu.Lock()
hMap, ok := mappings[addr]
delete(mappings, addr)
mappingMu.Unlock()
r, _, _ := procUnmapViewOfFile.Call(addr)
if r == 0 {
// Best-effort — log and continue. Unmap failure usually
// indicates a corrupted handle table, recoverable only via
// process exit.
_ = fmt.Sprint("UnmapViewOfFile failed")
}
if ok {
procCloseHandle.Call(uintptr(hMap))
}
a.mmapData = nil
}