mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-10-29 19:02:26 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			168 lines
		
	
	
		
			No EOL
		
	
	
		
			4.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			168 lines
		
	
	
		
			No EOL
		
	
	
		
			4.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
|  * Copyright 2021 ByteDance Inc.
 | |
|  *
 | |
|  * 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 encoder
 | |
| 
 | |
| import (
 | |
|     `encoding`
 | |
|     `encoding/json`
 | |
|     `unsafe`
 | |
| 
 | |
|     `github.com/bytedance/sonic/internal/jit`
 | |
|     `github.com/bytedance/sonic/internal/native`
 | |
|     `github.com/bytedance/sonic/internal/rt`
 | |
| )
 | |
| 
 | |
| /** Encoder Primitives **/
 | |
| 
 | |
| func encodeNil(rb *[]byte) error {
 | |
|     *rb = append(*rb, 'n', 'u', 'l', 'l')
 | |
|     return nil
 | |
| }
 | |
| 
 | |
| func encodeString(buf *[]byte, val string) error {
 | |
|     var sidx int
 | |
|     var pbuf *rt.GoSlice
 | |
|     var pstr *rt.GoString
 | |
| 
 | |
|     /* opening quote */
 | |
|     *buf = append(*buf, '"')
 | |
|     pbuf = (*rt.GoSlice)(unsafe.Pointer(buf))
 | |
|     pstr = (*rt.GoString)(unsafe.Pointer(&val))
 | |
| 
 | |
|     /* encode with native library */
 | |
|     for sidx < pstr.Len {
 | |
|         sn := pstr.Len - sidx
 | |
|         dn := pbuf.Cap - pbuf.Len
 | |
|         sp := padd(pstr.Ptr, sidx)
 | |
|         dp := padd(pbuf.Ptr, pbuf.Len)
 | |
|         nb := native.Quote(sp, sn, dp, &dn, 0)
 | |
| 
 | |
|         /* check for errors */
 | |
|         if pbuf.Len += dn; nb >= 0 {
 | |
|             break
 | |
|         }
 | |
| 
 | |
|         /* not enough space, grow the slice and try again */
 | |
|         sidx += ^nb
 | |
|         *pbuf = growslice(rt.UnpackType(byteType), *pbuf, pbuf.Cap * 2)
 | |
|     }
 | |
| 
 | |
|     /* closing quote */
 | |
|     *buf = append(*buf, '"')
 | |
|     return nil
 | |
| }
 | |
| 
 | |
| func encodeTypedPointer(buf *[]byte, vt *rt.GoType, vp *unsafe.Pointer, sb *_Stack, fv uint64) error {
 | |
|     if vt == nil {
 | |
|         return encodeNil(buf)
 | |
|     } else if fn, err := findOrCompile(vt, (fv&(1<<bitPointerValue)) != 0); err != nil {
 | |
|         return err
 | |
|     } else if vt.Indirect() {
 | |
|         rt.MoreStack(_FP_size + native.MaxFrameSize)
 | |
|         rt.StopProf()
 | |
|         err := fn(buf, *vp, sb, fv)
 | |
|         rt.StartProf()
 | |
|         return err
 | |
|     } else {
 | |
|         rt.MoreStack(_FP_size + native.MaxFrameSize)
 | |
|         rt.StopProf()
 | |
|         err := fn(buf, unsafe.Pointer(vp), sb, fv)
 | |
|         rt.StartProf()
 | |
|         return err
 | |
|     }
 | |
| }
 | |
| 
 | |
| func encodeJsonMarshaler(buf *[]byte, val json.Marshaler, opt Options) error {
 | |
|     if ret, err := val.MarshalJSON(); err != nil {
 | |
|         return err
 | |
|     } else {
 | |
|         if opt & CompactMarshaler != 0 {
 | |
|             return compact(buf, ret)
 | |
|         }
 | |
|         if ok, s := Valid(ret); !ok {
 | |
|             return error_marshaler(ret, s)
 | |
|         }
 | |
|         *buf = append(*buf, ret...)
 | |
|         return nil
 | |
|     }
 | |
| }
 | |
| 
 | |
| func encodeTextMarshaler(buf *[]byte, val encoding.TextMarshaler, opt Options) error {
 | |
|     if ret, err := val.MarshalText(); err != nil {
 | |
|         return err
 | |
|     } else {
 | |
|         if opt & NoQuoteTextMarshaler != 0 {
 | |
|             *buf = append(*buf, ret...)
 | |
|             return nil
 | |
|         }
 | |
|         return encodeString(buf, rt.Mem2Str(ret) )
 | |
|     }
 | |
| }
 | |
| 
 | |
| func htmlEscape(dst []byte, src []byte) []byte {
 | |
|     var sidx int
 | |
| 
 | |
|     dst  = append(dst, src[:0]...) // avoid check nil dst
 | |
|     sbuf := (*rt.GoSlice)(unsafe.Pointer(&src))
 | |
|     dbuf := (*rt.GoSlice)(unsafe.Pointer(&dst))
 | |
| 
 | |
|     /* grow dst if it is shorter */
 | |
|     if cap(dst) - len(dst) < len(src) + native.BufPaddingSize {
 | |
|         cap :=  len(src) * 3 / 2 + native.BufPaddingSize
 | |
|         *dbuf = growslice(typeByte, *dbuf, cap)
 | |
|     }
 | |
| 
 | |
|     for sidx < sbuf.Len {
 | |
|         sp := padd(sbuf.Ptr, sidx)
 | |
|         dp := padd(dbuf.Ptr, dbuf.Len)
 | |
| 
 | |
|         sn := sbuf.Len - sidx
 | |
|         dn := dbuf.Cap - dbuf.Len
 | |
|         nb := native.HTMLEscape(sp, sn, dp, &dn)
 | |
| 
 | |
|         /* check for errors */
 | |
|         if dbuf.Len += dn; nb >= 0 {
 | |
|             break
 | |
|         }
 | |
| 
 | |
|         /* not enough space, grow the slice and try again */
 | |
|         sidx += ^nb
 | |
|         *dbuf = growslice(typeByte, *dbuf, dbuf.Cap * 2)
 | |
|     }
 | |
|     return dst
 | |
| }
 | |
| 
 | |
| var (
 | |
|     argPtrs   = []bool { true, true, true, false }
 | |
|     localPtrs = []bool{}
 | |
| )
 | |
| 
 | |
| var (
 | |
|     _F_assertI2I = jit.Func(assertI2I)
 | |
| )
 | |
| 
 | |
| func asText(v unsafe.Pointer) (string, error) {
 | |
|     text := assertI2I(_T_encoding_TextMarshaler, *(*rt.GoIface)(v))
 | |
|     r, e := (*(*encoding.TextMarshaler)(unsafe.Pointer(&text))).MarshalText()
 | |
|     return rt.Mem2Str(r), e
 | |
| }
 | |
| 
 | |
| func asJson(v unsafe.Pointer) (string, error) {
 | |
|     text := assertI2I(_T_json_Marshaler, *(*rt.GoIface)(v))
 | |
|     r, e := (*(*json.Marshaler)(unsafe.Pointer(&text))).MarshalJSON()
 | |
|     return rt.Mem2Str(r), e
 | |
| } |