189 lines
4.0 KiB
Go
189 lines
4.0 KiB
Go
/*
|
|
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package runtime
|
|
|
|
import (
|
|
"unsafe"
|
|
|
|
"github.com/goplus/llgo/internal/runtime/c"
|
|
)
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// String is the runtime representation of a string.
|
|
// It cannot be used safely or portably and its representation may
|
|
// change in a later release.
|
|
//
|
|
// Unlike reflect.StringHeader, its Data field is sufficient to guarantee the
|
|
// data it references will not be garbage collected.
|
|
type String struct {
|
|
data unsafe.Pointer
|
|
len int
|
|
}
|
|
|
|
// StringCat concatenates two strings.
|
|
func StringCat(a, b String) String {
|
|
n := a.len + b.len
|
|
dest := AllocU(uintptr(n))
|
|
c.Memcpy(dest, a.data, uintptr(a.len))
|
|
c.Memcpy(c.Advance(dest, a.len), b.data, uintptr(b.len))
|
|
return String{dest, n}
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// CStrCopy copies a Go string to a C string buffer and returns it.
|
|
func CStrCopy(dest unsafe.Pointer, s String) *int8 {
|
|
n := s.len
|
|
c.Memcpy(dest, s.data, uintptr(n))
|
|
arr := (*[1 << 30]int8)(dest)
|
|
arr[n] = 0
|
|
return (*int8)(dest)
|
|
}
|
|
|
|
func CStrDup(s String) *int8 {
|
|
dest := AllocU(uintptr(s.len + 1))
|
|
return CStrCopy(dest, s)
|
|
}
|
|
|
|
func NewStringSlice(base String, i, j int) String {
|
|
if i < 0 || j < i || j > base.len {
|
|
panic("string slice index out of bounds")
|
|
}
|
|
if i < base.len {
|
|
return String{c.Advance(base.data, i), j - i}
|
|
}
|
|
return String{nil, 0}
|
|
}
|
|
|
|
type StringIter struct {
|
|
s string
|
|
pos int
|
|
}
|
|
|
|
func NewStringIter(s string) *StringIter {
|
|
return &StringIter{s, 0}
|
|
}
|
|
|
|
func StringIterNext(it *StringIter) (ok bool, k int, v rune) {
|
|
if it.pos >= len(it.s) {
|
|
return false, 0, 0
|
|
}
|
|
k = it.pos
|
|
if c := it.s[it.pos]; c < runeSelf {
|
|
it.pos++
|
|
v = rune(c)
|
|
} else {
|
|
v, it.pos = decoderune(it.s, it.pos)
|
|
}
|
|
ok = true
|
|
return
|
|
}
|
|
|
|
func StringToBytes(s String) []byte {
|
|
if s.len == 0 {
|
|
return nil
|
|
}
|
|
data := make([]byte, s.len)
|
|
c.Memcpy(unsafe.Pointer(&data[0]), s.data, uintptr(s.len))
|
|
return data
|
|
}
|
|
|
|
func StringToRunes(s string) []rune {
|
|
if len(s) == 0 {
|
|
return nil
|
|
}
|
|
data := make([]rune, len(s))
|
|
var index uint
|
|
for i := 0; i < len(s); {
|
|
if c := s[i]; c < runeSelf {
|
|
data[index] = rune(c)
|
|
i++
|
|
} else {
|
|
data[index], i = decoderune(s, i)
|
|
}
|
|
index++
|
|
}
|
|
return data[:index:index]
|
|
}
|
|
|
|
func StringFromBytes(b Slice) (s String) {
|
|
if b.len == 0 {
|
|
return
|
|
}
|
|
s.len = b.len
|
|
s.data = AllocU(uintptr(s.len))
|
|
c.Memcpy(s.data, b.data, uintptr(b.len))
|
|
return
|
|
}
|
|
|
|
func StringFromRunes(rs []rune) (s String) {
|
|
if len(rs) == 0 {
|
|
return
|
|
}
|
|
data := make([]byte, len(rs)*4)
|
|
var index int
|
|
for _, r := range rs {
|
|
n := encoderune(data[index:], r)
|
|
index += n
|
|
}
|
|
s.len = index
|
|
s.data = unsafe.Pointer(&data[0])
|
|
return
|
|
}
|
|
|
|
func StringFromRune(r rune) (s String) {
|
|
var buf [4]byte
|
|
n := encoderune(buf[:], r)
|
|
s.len = n
|
|
s.data = unsafe.Pointer(&buf[0])
|
|
return
|
|
}
|
|
|
|
func StringEqual(x, y String) bool {
|
|
if x.len != y.len {
|
|
return false
|
|
}
|
|
if x.data != y.data {
|
|
for i := 0; i < x.len; i++ {
|
|
if *(*byte)(c.Advance(x.data, i)) != *(*byte)(c.Advance(y.data, i)) {
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func StringLess(x, y String) bool {
|
|
n := x.len
|
|
if n > y.len {
|
|
n = y.len
|
|
}
|
|
for i := 0; i < n; i++ {
|
|
ix := *(*byte)(c.Advance(x.data, i))
|
|
iy := *(*byte)(c.Advance(y.data, i))
|
|
if ix < iy {
|
|
return true
|
|
} else if ix > iy {
|
|
return false
|
|
}
|
|
}
|
|
return x.len < y.len
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|