2023-10-27 18:20:02 +09:00
|
|
|
package hcl
|
|
|
|
|
|
|
|
/*
|
|
|
|
#include <hcl.h>
|
|
|
|
#include <hcl-utl.h>
|
2024-02-22 01:21:11 +09:00
|
|
|
#include <string.h> // for memcpy
|
2023-10-27 18:20:02 +09:00
|
|
|
*/
|
|
|
|
import "C"
|
|
|
|
|
|
|
|
import (
|
2023-10-31 21:17:08 +09:00
|
|
|
"bufio"
|
|
|
|
"io"
|
2023-10-27 18:20:02 +09:00
|
|
|
"os"
|
2023-11-05 23:33:22 +09:00
|
|
|
"path"
|
2024-02-24 12:57:47 +09:00
|
|
|
"path/filepath"
|
2024-02-24 22:53:34 +09:00
|
|
|
"reflect"
|
2023-10-31 21:17:08 +09:00
|
|
|
"sync"
|
|
|
|
"unsafe"
|
2023-10-27 18:20:02 +09:00
|
|
|
)
|
|
|
|
|
2024-02-24 19:41:26 +09:00
|
|
|
//import "fmt"
|
|
|
|
|
2023-10-27 18:20:02 +09:00
|
|
|
type IOHandle struct {
|
|
|
|
file *os.File
|
|
|
|
ioif interface{}
|
|
|
|
}
|
2023-10-30 02:30:19 +09:00
|
|
|
|
2023-10-27 18:20:02 +09:00
|
|
|
type IOHandleTable struct {
|
|
|
|
mtx sync.Mutex
|
|
|
|
handles []IOHandle
|
|
|
|
free_slots []int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (io *IOHandleTable) add_io_handle(f *os.File, ioif interface{}) int {
|
|
|
|
io.mtx.Lock()
|
|
|
|
defer io.mtx.Unlock()
|
|
|
|
|
|
|
|
var n int = len(io.free_slots)
|
|
|
|
|
|
|
|
if n <= 0 { // no free slots
|
|
|
|
io.handles = append(io.handles, IOHandle{file: f, ioif: ioif})
|
|
|
|
return len(io.handles) - 1
|
|
|
|
} else {
|
|
|
|
var slot int
|
|
|
|
n--
|
|
|
|
slot = io.free_slots[n]
|
|
|
|
io.free_slots = io.free_slots[:n]
|
|
|
|
io.handles[slot].file = f
|
|
|
|
io.handles[slot].ioif = ioif
|
|
|
|
return slot
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (io *IOHandleTable) del_io_handle(slot int) IOHandle {
|
|
|
|
var (
|
|
|
|
h IOHandle
|
|
|
|
n int
|
|
|
|
)
|
|
|
|
|
|
|
|
io.mtx.Lock()
|
|
|
|
defer io.mtx.Unlock()
|
|
|
|
|
|
|
|
h = io.handles[slot]
|
|
|
|
io.handles[slot].file = nil
|
|
|
|
io.handles[slot].ioif = nil
|
|
|
|
|
|
|
|
n = len(io.handles)
|
|
|
|
if slot == n-1 {
|
|
|
|
io.handles = io.handles[:n-1]
|
|
|
|
} else {
|
|
|
|
io.free_slots = append(io.free_slots, slot)
|
|
|
|
}
|
|
|
|
|
|
|
|
return h
|
|
|
|
}
|
|
|
|
|
|
|
|
func (io *IOHandleTable) slot_to_io_handle(slot int) IOHandle {
|
|
|
|
io.mtx.Lock()
|
|
|
|
defer io.mtx.Unlock()
|
|
|
|
return io.handles[slot]
|
|
|
|
}
|
|
|
|
|
|
|
|
var io_tab IOHandleTable = IOHandleTable{}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
//export hcl_go_cci_handler
|
|
|
|
func hcl_go_cci_handler(c *C.hcl_t, cmd C.hcl_io_cmd_t, arg unsafe.Pointer) C.int {
|
2023-10-27 18:20:02 +09:00
|
|
|
var (
|
|
|
|
g *HCL
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
g = c_to_go(c)
|
|
|
|
|
|
|
|
switch cmd {
|
|
|
|
case C.HCL_IO_OPEN:
|
|
|
|
var (
|
2023-11-05 22:31:33 +09:00
|
|
|
ioarg *C.hcl_io_cciarg_t
|
2023-10-27 18:20:02 +09:00
|
|
|
name string
|
|
|
|
fd int
|
2024-02-24 12:57:47 +09:00
|
|
|
tptr unsafe.Pointer
|
|
|
|
tlen C.size_t
|
2023-10-27 18:20:02 +09:00
|
|
|
)
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
ioarg = (*C.hcl_io_cciarg_t)(arg)
|
2024-02-24 17:32:42 +09:00
|
|
|
|
|
|
|
if ioarg.includer == nil /* || ioarg.includer.name == nil */ {
|
|
|
|
// main stream
|
|
|
|
name = g.io.cci_main
|
2023-10-27 18:20:02 +09:00
|
|
|
} else {
|
2024-02-24 17:32:42 +09:00
|
|
|
// actual included stream
|
|
|
|
var includer_name string
|
|
|
|
|
2024-02-24 19:41:26 +09:00
|
|
|
name = string(ucstr_to_rune_slice(ioarg.name))
|
2023-10-27 18:20:02 +09:00
|
|
|
|
2024-02-24 12:57:47 +09:00
|
|
|
tptr = ioarg.includer.handle
|
|
|
|
tlen = *(*C.size_t)(unsafe.Pointer(uintptr(tptr) + unsafe.Sizeof(fd)))
|
|
|
|
|
|
|
|
includer_name = C.GoStringN((*C.char)(unsafe.Pointer(uintptr(tptr)+unsafe.Sizeof(fd)+unsafe.Sizeof(tlen))), C.int(tlen))
|
|
|
|
name = filepath.Join(path.Dir(includer_name), name)
|
2023-10-27 18:20:02 +09:00
|
|
|
}
|
|
|
|
|
2024-02-24 17:32:42 +09:00
|
|
|
tlen = C.size_t(len(name)) // number of bytes in the string
|
2024-02-24 19:41:26 +09:00
|
|
|
tptr = C.hcl_allocmem(c, C.size_t(unsafe.Sizeof(fd)) + C.size_t(unsafe.Sizeof(tlen)) + tlen)
|
2024-02-24 12:57:47 +09:00
|
|
|
if tptr == nil {
|
2024-02-24 19:41:26 +09:00
|
|
|
g.set_errmsg(C.HCL_ESYSMEM, "cci name allocation failure")
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
2024-02-24 17:32:42 +09:00
|
|
|
if ioarg.includer == nil {
|
|
|
|
fd = -1
|
|
|
|
} else {
|
|
|
|
fd, err = g.io.cci.Open(g, name)
|
2024-02-24 12:57:47 +09:00
|
|
|
if err != nil {
|
|
|
|
g.set_errmsg(C.HCL_EIOERR, err.Error())
|
2024-02-24 19:41:26 +09:00
|
|
|
C.hcl_freemem(c, tptr)
|
2024-02-24 12:57:47 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-24 19:41:26 +09:00
|
|
|
// | fd | length | name bytes of the length |
|
2024-02-24 22:53:34 +09:00
|
|
|
*(*int)(tptr) = fd;
|
|
|
|
*(*C.size_t)(unsafe.Pointer(uintptr(tptr)+unsafe.Sizeof(fd))) = tlen;
|
|
|
|
|
|
|
|
// C.CString() allocates a memory block. Use a SliceHeader to avoid extra memory allocation
|
|
|
|
// for the string conversion. Create a fake slice header that can be used with copy() instead.
|
|
|
|
var dsthdr reflect.SliceHeader
|
|
|
|
dsthdr.Data = uintptr(tptr)+unsafe.Sizeof(fd)+unsafe.Sizeof(tlen)
|
|
|
|
dsthdr.Len = int(tlen)
|
|
|
|
dsthdr.Cap = int(tlen)
|
|
|
|
copy(*(*[]byte)(unsafe.Pointer(&dsthdr)), name);
|
2024-02-24 12:57:47 +09:00
|
|
|
|
|
|
|
ioarg.handle = tptr
|
2023-10-27 18:20:02 +09:00
|
|
|
return 0
|
|
|
|
|
|
|
|
case C.HCL_IO_CLOSE:
|
2024-02-24 12:57:47 +09:00
|
|
|
var (
|
|
|
|
fd int
|
|
|
|
ioarg *C.hcl_io_cciarg_t
|
|
|
|
)
|
|
|
|
|
|
|
|
ioarg = (*C.hcl_io_cciarg_t)(arg)
|
2024-02-24 19:41:26 +09:00
|
|
|
fd = *(*int)(ioarg.handle) // the descriptor at the beginning
|
2024-02-24 12:57:47 +09:00
|
|
|
if fd >= 0 {
|
|
|
|
g.io.cci.Close(fd)
|
|
|
|
}
|
2024-02-24 19:41:26 +09:00
|
|
|
C.hcl_freemem(c, ioarg.handle)
|
|
|
|
ioarg.handle = nil
|
2023-10-27 18:20:02 +09:00
|
|
|
return 0
|
|
|
|
|
|
|
|
case C.HCL_IO_READ:
|
|
|
|
var (
|
2023-11-05 22:31:33 +09:00
|
|
|
ioarg *C.hcl_io_cciarg_t
|
2023-10-27 18:20:02 +09:00
|
|
|
n int
|
|
|
|
i int
|
|
|
|
buf []rune
|
2024-02-22 01:21:11 +09:00
|
|
|
dummy C.hcl_uch_t
|
2024-02-24 12:57:47 +09:00
|
|
|
fd int
|
2023-10-27 18:20:02 +09:00
|
|
|
)
|
2023-11-05 22:31:33 +09:00
|
|
|
ioarg = (*C.hcl_io_cciarg_t)(arg)
|
2023-10-27 18:20:02 +09:00
|
|
|
|
2024-02-24 19:41:26 +09:00
|
|
|
fd = *(*int)(ioarg.handle) // the descriptor at the beginning
|
2024-02-24 12:57:47 +09:00
|
|
|
|
2023-10-27 18:20:02 +09:00
|
|
|
buf = make([]rune, 1024) // TODO: different size...
|
2024-02-24 12:57:47 +09:00
|
|
|
n, err = g.io.cci.Read(fd, buf)
|
2023-10-27 18:20:02 +09:00
|
|
|
if err != nil {
|
2023-11-05 23:33:22 +09:00
|
|
|
g.set_errmsg(C.HCL_EIOERR, err.Error())
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
2024-05-19 17:09:31 +09:00
|
|
|
ioarg.byte_oriented = 0
|
2024-02-22 01:21:11 +09:00
|
|
|
if unsafe.Sizeof(buf[0]) == unsafe.Sizeof(dummy) {
|
2024-02-23 00:54:36 +09:00
|
|
|
C.memcpy(
|
2024-02-22 01:21:11 +09:00
|
|
|
unsafe.Pointer(&ioarg.buf[0]),
|
|
|
|
unsafe.Pointer(&buf[0]),
|
2024-02-23 00:54:36 +09:00
|
|
|
C.size_t(unsafe.Sizeof(buf[0])*uintptr(n)))
|
2024-02-22 01:21:11 +09:00
|
|
|
} else {
|
|
|
|
var dst uintptr
|
|
|
|
// work around cgo's union issue. not able to access individual union
|
|
|
|
// member fields. cgo treats union as byte aggregates.
|
|
|
|
dst = uintptr(unsafe.Pointer(&ioarg.buf[0]))
|
|
|
|
for i = 0; i < n; i++ {
|
|
|
|
//ioarg.buf.c[i] = C.hcl_uch_t(buf[i])
|
2024-02-23 00:54:36 +09:00
|
|
|
*(*C.hcl_uch_t)(unsafe.Pointer(dst)) = C.hcl_uch_t(buf[i])
|
2024-02-22 01:21:11 +09:00
|
|
|
dst += unsafe.Sizeof(dummy)
|
|
|
|
}
|
2023-10-27 18:20:02 +09:00
|
|
|
}
|
2024-02-22 01:21:11 +09:00
|
|
|
|
2023-10-27 18:20:02 +09:00
|
|
|
ioarg.xlen = C.hcl_oow_t(n)
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2023-11-01 16:06:28 +09:00
|
|
|
C.hcl_seterrnum(c, C.HCL_EIOERR)
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
//export hcl_go_udi_handler
|
|
|
|
func hcl_go_udi_handler(c *C.hcl_t, cmd C.hcl_io_cmd_t, arg unsafe.Pointer) C.int {
|
2023-10-27 18:20:02 +09:00
|
|
|
var (
|
|
|
|
g *HCL
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
g = c_to_go(c)
|
|
|
|
|
|
|
|
switch cmd {
|
|
|
|
case C.HCL_IO_OPEN:
|
2023-11-05 23:33:22 +09:00
|
|
|
err = g.io.udi.Open(g)
|
2023-10-27 18:20:02 +09:00
|
|
|
if err != nil {
|
2023-11-05 23:33:22 +09:00
|
|
|
g.set_errmsg(C.HCL_EIOERR, err.Error())
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
// we don't need to set ioarg.handle because the c object to go object
|
|
|
|
// mapping has been established and it doesn't handle multiple streams
|
|
|
|
return 0
|
|
|
|
|
|
|
|
case C.HCL_IO_CLOSE:
|
2023-11-05 23:33:22 +09:00
|
|
|
g.io.udi.Close()
|
2023-10-27 18:20:02 +09:00
|
|
|
return 0
|
|
|
|
|
|
|
|
case C.HCL_IO_READ:
|
|
|
|
var (
|
2023-11-05 16:58:45 +09:00
|
|
|
ioarg *C.hcl_io_udiarg_t
|
2023-10-27 18:20:02 +09:00
|
|
|
n int
|
|
|
|
err error
|
|
|
|
buf []rune
|
|
|
|
)
|
2023-11-05 16:58:45 +09:00
|
|
|
ioarg = (*C.hcl_io_udiarg_t)(arg)
|
2023-10-27 18:20:02 +09:00
|
|
|
|
|
|
|
buf = make([]rune, 1024) // TODO: different size...
|
2023-11-05 23:33:22 +09:00
|
|
|
n, err = g.io.udi.Read(buf)
|
2023-10-27 18:20:02 +09:00
|
|
|
if err != nil {
|
2023-11-05 23:33:22 +09:00
|
|
|
g.set_errmsg(C.HCL_EIOERR, err.Error())
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
ioarg.xlen = C.ulong(n)
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2023-11-01 16:06:28 +09:00
|
|
|
C.hcl_seterrnum(c, C.HCL_EIOERR)
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
//export hcl_go_udo_handler
|
|
|
|
func hcl_go_udo_handler(c *C.hcl_t, cmd C.hcl_io_cmd_t, arg unsafe.Pointer) C.int {
|
2023-10-27 18:20:02 +09:00
|
|
|
var (
|
|
|
|
g *HCL
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
g = c_to_go(c)
|
|
|
|
|
|
|
|
switch cmd {
|
|
|
|
case C.HCL_IO_OPEN:
|
2023-11-05 23:33:22 +09:00
|
|
|
err = g.io.udo.Open(g)
|
2023-10-27 18:20:02 +09:00
|
|
|
if err != nil {
|
2023-11-05 23:33:22 +09:00
|
|
|
g.set_errmsg(C.HCL_EIOERR, err.Error())
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
// we don't need to set ioarg.handle because the c object to go object
|
|
|
|
// mapping has been established and it doesn't handle multiple streams
|
|
|
|
return 0
|
|
|
|
|
|
|
|
case C.HCL_IO_CLOSE:
|
2023-11-05 23:33:22 +09:00
|
|
|
g.io.udo.Close()
|
2023-10-27 18:20:02 +09:00
|
|
|
return 0
|
|
|
|
|
|
|
|
case C.HCL_IO_WRITE:
|
|
|
|
var (
|
2023-11-05 16:58:45 +09:00
|
|
|
ioarg *C.hcl_io_udoarg_t
|
2023-10-27 18:20:02 +09:00
|
|
|
data []rune
|
|
|
|
err error
|
|
|
|
)
|
2023-11-05 16:58:45 +09:00
|
|
|
ioarg = (*C.hcl_io_udoarg_t)(arg)
|
2023-10-27 18:20:02 +09:00
|
|
|
data = uchars_to_rune_slice((*C.hcl_uch_t)(ioarg.ptr), uintptr(ioarg.len))
|
2023-11-05 23:33:22 +09:00
|
|
|
err = g.io.udo.Write(data)
|
2023-10-27 18:20:02 +09:00
|
|
|
if err != nil {
|
2023-11-05 23:33:22 +09:00
|
|
|
g.set_errmsg(C.HCL_EIOERR, err.Error())
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
ioarg.xlen = C.hcl_oow_t(len(data))
|
|
|
|
return 0
|
|
|
|
|
|
|
|
case C.HCL_IO_WRITE_BYTES:
|
|
|
|
var (
|
2023-11-05 16:58:45 +09:00
|
|
|
ioarg *C.hcl_io_udoarg_t
|
2023-10-27 18:20:02 +09:00
|
|
|
data []byte
|
|
|
|
err error
|
|
|
|
)
|
2023-11-05 16:58:45 +09:00
|
|
|
ioarg = (*C.hcl_io_udoarg_t)(arg)
|
2023-10-27 18:20:02 +09:00
|
|
|
data = unsafe.Slice((*byte)(ioarg.ptr), ioarg.len)
|
2023-11-05 23:33:22 +09:00
|
|
|
err = g.io.udo.WriteBytes(data)
|
2023-10-27 18:20:02 +09:00
|
|
|
if err != nil {
|
2023-11-05 23:33:22 +09:00
|
|
|
g.set_errmsg(C.HCL_EIOERR, err.Error())
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
ioarg.xlen = C.hcl_oow_t(len(data))
|
|
|
|
return 0
|
|
|
|
|
|
|
|
case C.HCL_IO_FLUSH:
|
2023-11-05 23:33:22 +09:00
|
|
|
var err error = g.io.udo.Flush()
|
2023-10-27 18:20:02 +09:00
|
|
|
if err != nil {
|
2023-11-05 23:33:22 +09:00
|
|
|
g.set_errmsg(C.HCL_EIOERR, err.Error())
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2023-11-01 16:06:28 +09:00
|
|
|
C.hcl_seterrnum(c, C.HCL_EIOERR)
|
2023-10-27 18:20:02 +09:00
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------------------------
|
2023-11-05 22:31:33 +09:00
|
|
|
type CciFileHandler struct {
|
2023-10-27 18:20:02 +09:00
|
|
|
g *HCL
|
|
|
|
}
|
|
|
|
|
2024-02-24 17:32:42 +09:00
|
|
|
func (p *CciFileHandler) Open(g *HCL, name string) (int, error) {
|
2023-10-27 18:20:02 +09:00
|
|
|
var (
|
|
|
|
f *os.File
|
|
|
|
r *bufio.Reader
|
|
|
|
fd int
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
if name == "" {
|
|
|
|
f = os.Stdin
|
|
|
|
} else {
|
2024-02-24 17:32:42 +09:00
|
|
|
f, err = os.Open(name)
|
2023-10-27 18:20:02 +09:00
|
|
|
if err != nil {
|
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
r = bufio.NewReader(f)
|
|
|
|
fd = io_tab.add_io_handle(f, r)
|
|
|
|
|
|
|
|
p.g = g
|
|
|
|
return fd, nil
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *CciFileHandler) Close(fd int) {
|
2023-10-27 18:20:02 +09:00
|
|
|
var hnd IOHandle = io_tab.slot_to_io_handle(fd)
|
|
|
|
if hnd.file != nil {
|
|
|
|
if hnd.file != os.Stdout && hnd.file != os.Stderr {
|
|
|
|
hnd.file.Close()
|
|
|
|
}
|
|
|
|
hnd.file = nil
|
|
|
|
hnd.ioif = nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *CciFileHandler) Read(fd int, buf []rune) (int, error) {
|
2023-10-27 18:20:02 +09:00
|
|
|
var (
|
|
|
|
hnd IOHandle
|
|
|
|
i int
|
|
|
|
c rune
|
|
|
|
r *bufio.Reader
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
hnd = io_tab.slot_to_io_handle(fd)
|
|
|
|
r, _ = hnd.ioif.(*bufio.Reader)
|
|
|
|
for i = 0; i < len(buf); i++ {
|
|
|
|
c, _, err = r.ReadRune()
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
} else if err != nil {
|
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
|
|
|
|
buf[i] = c
|
|
|
|
}
|
|
|
|
|
|
|
|
return i, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------------------------
|
2023-11-05 22:31:33 +09:00
|
|
|
type UdiFileHandler struct {
|
2023-10-27 18:20:02 +09:00
|
|
|
g *HCL
|
|
|
|
f *os.File
|
|
|
|
r *bufio.Reader
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *UdiFileHandler) Open(g *HCL) error {
|
2023-10-27 18:20:02 +09:00
|
|
|
var (
|
|
|
|
f *os.File
|
|
|
|
// err error
|
|
|
|
)
|
|
|
|
|
|
|
|
f = os.Stdin
|
2023-10-31 21:17:08 +09:00
|
|
|
//f, err = os.Open("/dev/stdin")
|
|
|
|
//if err != nil {
|
|
|
|
// return err
|
|
|
|
//}
|
2023-10-27 18:20:02 +09:00
|
|
|
|
|
|
|
p.r = bufio.NewReader(f)
|
|
|
|
p.f = f
|
|
|
|
p.g = g
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *UdiFileHandler) Close() {
|
2023-10-27 18:20:02 +09:00
|
|
|
if p.f != nil {
|
|
|
|
if p.f != os.Stdout && p.f != os.Stderr {
|
|
|
|
p.f.Close()
|
|
|
|
}
|
|
|
|
p.f = nil
|
|
|
|
p.r = nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *UdiFileHandler) Read(buf []rune) (int, error) {
|
2023-10-27 18:20:02 +09:00
|
|
|
var (
|
|
|
|
i int
|
|
|
|
c rune
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
// flush all pending print out before reading
|
2023-11-05 23:33:22 +09:00
|
|
|
p.g.io.udo.Flush()
|
2023-10-27 18:20:02 +09:00
|
|
|
|
|
|
|
for i = 0; i < len(buf); i++ {
|
|
|
|
c, _, err = p.r.ReadRune()
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
} else if err != nil {
|
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
|
|
|
|
buf[i] = c
|
|
|
|
}
|
|
|
|
|
|
|
|
return i, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------------------------------
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
type UdoFileHandler struct {
|
2023-10-27 18:20:02 +09:00
|
|
|
f *os.File
|
|
|
|
w *bufio.Writer
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *UdoFileHandler) Open(g *HCL) error {
|
2023-10-27 18:20:02 +09:00
|
|
|
var (
|
|
|
|
f *os.File
|
|
|
|
// err error
|
|
|
|
)
|
2023-10-30 02:30:19 +09:00
|
|
|
|
2023-10-31 21:17:08 +09:00
|
|
|
// f, err = os.OpenFile("/dev/stdout", os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0644)
|
|
|
|
// if err != nil {
|
|
|
|
// return err
|
|
|
|
// }
|
2023-10-27 18:20:02 +09:00
|
|
|
|
|
|
|
f = os.Stdout
|
|
|
|
|
|
|
|
p.w = bufio.NewWriter(f)
|
|
|
|
p.f = f
|
|
|
|
|
|
|
|
//fmt.Fprintf(os.Stderr, "XXXXXXXXXX open porint\n")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *UdoFileHandler) Close() {
|
2023-10-27 18:20:02 +09:00
|
|
|
|
|
|
|
//fmt.Fprintf(os.Stderr, "XXXXXXXXXX close porint\n")
|
|
|
|
if p.f != nil {
|
|
|
|
if p.f != os.Stdout && p.f != os.Stderr {
|
|
|
|
p.f.Close()
|
|
|
|
}
|
|
|
|
p.f = nil
|
|
|
|
p.w = nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *UdoFileHandler) Write(data []rune) error {
|
2023-10-27 18:20:02 +09:00
|
|
|
var err error
|
|
|
|
|
|
|
|
//fmt.Fprintf(os.Stderr, "XXXXXXXXXX write porint\n")
|
|
|
|
_, err = p.w.WriteString(string(data))
|
|
|
|
p.w.Flush() // TODO: is this needed?
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *UdoFileHandler) WriteBytes(data []byte) error {
|
2023-10-27 18:20:02 +09:00
|
|
|
var err error
|
|
|
|
|
|
|
|
//fmt.Fprintf(os.Stderr, "XXXXXXXXXX write porint\n")
|
|
|
|
_, err = p.w.Write(data)
|
|
|
|
p.w.Flush() // TODO: is this needed?
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-11-05 22:31:33 +09:00
|
|
|
func (p *UdoFileHandler) Flush() error {
|
2023-10-27 18:20:02 +09:00
|
|
|
//fmt.Fprintf(os.Stderr, "XXXXXXXXXX flush porint\n")
|
|
|
|
return p.w.Flush()
|
|
|
|
}
|