mirror of
https://github.com/ii64/sonic.git
synced 2026-06-21 00:46:43 +08:00
220 lines
5.5 KiB
Go
220 lines
5.5 KiB
Go
// +build go1.18
|
|
|
|
/*
|
|
* 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 sonic_fuzz
|
|
|
|
import (
|
|
`encoding/json`
|
|
`testing`
|
|
_ `unicode/utf8`
|
|
`os`
|
|
`runtime`
|
|
`runtime/debug`
|
|
`time`
|
|
`io`
|
|
`log`
|
|
`strconv`
|
|
|
|
`github.com/bytedance/sonic`
|
|
`github.com/stretchr/testify/require`
|
|
`github.com/davecgh/go-spew/spew`
|
|
`github.com/bytedance/gopkg/util/gctuner`
|
|
)
|
|
|
|
func FuzzMain(f *testing.F) {
|
|
for _, corp := range(corpus()) {
|
|
f.Add(corp)
|
|
}
|
|
f.Fuzz(fuzzMain)
|
|
}
|
|
|
|
// Used for debug falied fuzz corpus
|
|
func TestCorpus(t *testing.T) {
|
|
fuzzMain(t, []byte("[1\x00"))
|
|
}
|
|
|
|
var target = sonic.ConfigStd
|
|
|
|
func fuzzMain(t *testing.T, data []byte) {
|
|
fuzzValidate(t, data)
|
|
fuzzHtmlEscape(t, data)
|
|
// fuzz ast get api, should not panic here.
|
|
fuzzAst(t, data)
|
|
// Only fuzz the validate json here.
|
|
if !json.Valid(data) {
|
|
return
|
|
}
|
|
for _, typ := range []func() interface{}{
|
|
func() interface{} { return new(interface{}) },
|
|
func() interface{} { return new(map[string]interface{}) },
|
|
func() interface{} { return new([]interface{}) },
|
|
func() interface{} { return new(string) },
|
|
func() interface{} { return new(int64) },
|
|
func() interface{} { return new(uint64) },
|
|
func() interface{} { return new(float64) },
|
|
// func() interface{} { return new(json.Number) },
|
|
// func() interface{} { return new(S) },
|
|
} {
|
|
sv, jv := typ(), typ()
|
|
serr := target.Unmarshal([]byte(data), sv)
|
|
jerr := json.Unmarshal([]byte(data), jv)
|
|
require.Equal(t, serr != nil, jerr != nil,
|
|
dump(data, jv, jerr, sv, serr))
|
|
if jerr != nil {
|
|
continue
|
|
}
|
|
require.Equal(t, sv, jv, dump(data, jv, jerr, sv, serr))
|
|
|
|
v := jv
|
|
sout, serr := target.Marshal(v)
|
|
jout, jerr := json.Marshal(v)
|
|
require.NoError(t, serr, dump(v, jout, jerr, sout, serr))
|
|
require.NoError(t, jerr, dump(v, jout, jerr, sout, serr))
|
|
|
|
{
|
|
sv, jv := typ(), typ()
|
|
serr := target.Unmarshal(sout, sv)
|
|
jerr := json.Unmarshal(jout, jv)
|
|
require.Equalf(t, serr != nil, jerr != nil, dump(data, jv, jerr, sv, serr))
|
|
if jerr != nil {
|
|
continue
|
|
}
|
|
require.Equal(t, sv, jv, dump(data, jv, jerr, sv, serr))
|
|
}
|
|
|
|
if m, ok := sv.(*map[string]interface{}); ok {
|
|
fuzzDynamicStruct(t, jout, *m)
|
|
fuzzASTGetFromObject(t, jout, *m)
|
|
}
|
|
if a, ok := sv.(*[]interface{}); ok {
|
|
fuzzASTGetFromArray(t, jout, *a)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
type S struct {
|
|
A int `json:",omitempty"`
|
|
B string `json:"B1,omitempty"`
|
|
C float64
|
|
D bool
|
|
E uint8
|
|
// F []byte // unmarshal []byte is different with encoding/json
|
|
G interface{}
|
|
H map[string]interface{}
|
|
I map[string]string
|
|
J []interface{}
|
|
K []string
|
|
L S1
|
|
M *S1
|
|
N *int
|
|
O **int
|
|
P int `json:",string"`
|
|
Q float64 `json:",string"`
|
|
R int `json:"-"`
|
|
T struct {}
|
|
U [2]int
|
|
V uintptr
|
|
W json.Number
|
|
// X json.RawMessage
|
|
Y Marshaller
|
|
Z TextMarshaller
|
|
}
|
|
|
|
|
|
type S1 struct {
|
|
A int
|
|
B string
|
|
}
|
|
|
|
type Marshaller struct {
|
|
v string
|
|
}
|
|
|
|
func (m *Marshaller) MarshalJSON() ([]byte, error) {
|
|
return json.Marshal(m.v)
|
|
}
|
|
|
|
func (m *Marshaller) UnmarshalJSON(data []byte) error {
|
|
return json.Unmarshal(data, &m.v)
|
|
}
|
|
|
|
type TextMarshaller struct {
|
|
v int
|
|
}
|
|
|
|
func (k *TextMarshaller) MarshalText() ([]byte, error) {
|
|
return json.Marshal(k.v)
|
|
}
|
|
|
|
func (k *TextMarshaller) UnmarshalText(data []byte) error {
|
|
return json.Unmarshal(data, &k.v)
|
|
}
|
|
|
|
|
|
func dump(args ...interface{}) string {
|
|
return spew.Sdump(args)
|
|
}
|
|
|
|
func fdump(w io.Writer, args ...interface{}) {
|
|
spew.Fdump(w, args)
|
|
}
|
|
|
|
const (
|
|
MemoryLimitEnv = "SONIC_FUZZ_MEM_LIMIT"
|
|
AsynyncGCEnv = "SONIC_NO_ASYNC_GC"
|
|
KB uint64 = 1024
|
|
MB uint64 = 1024 * KB
|
|
GB uint64 = 1024 * MB
|
|
)
|
|
|
|
func setMemLimit(limit uint64) {
|
|
threshold := uint64(float64(limit) * 0.7)
|
|
numWorker := uint64(runtime.GOMAXPROCS(0))
|
|
if os.Getenv(MemoryLimitEnv) != "" {
|
|
if memGB, err := strconv.ParseUint(os.Getenv(MemoryLimitEnv), 10, 64); err == nil {
|
|
limit = memGB * GB
|
|
}
|
|
}
|
|
gctuner.Tuning(threshold / numWorker)
|
|
log.Printf("[%d] Memory Limit: %d GB, Memory Threshold: %d MB\n", os.Getpid(), limit/GB, threshold/MB)
|
|
log.Printf("[%d] Memory Threshold Per Worker: %d MB\n", os.Getpid(), threshold/numWorker/MB)
|
|
}
|
|
|
|
func enableSyncGC() {
|
|
var debugAsyncGC = os.Getenv("AsynyncGCEnv") == ""
|
|
go func () {
|
|
if !debugAsyncGC {
|
|
return
|
|
}
|
|
log.Printf("Begin GC looping...")
|
|
for {
|
|
runtime.GC()
|
|
debug.FreeOSMemory()
|
|
}
|
|
}()
|
|
}
|
|
|
|
func TestMain(m *testing.M) {
|
|
// Avoid OOM
|
|
setMemLimit(12 * GB)
|
|
enableSyncGC()
|
|
time.Sleep(time.Millisecond)
|
|
m.Run()
|
|
}
|