package scalar // TODO: d.Pos check err // TODO: fn for actual? // TODO: dsl import .? own scalar package? // TODO: better IOError op names import ( "bytes" "fmt" "strconv" "strings" "github.com/wader/fq/internal/bitioextra" "github.com/wader/fq/pkg/bitio" ) //go:generate sh -c "cat scalar_gen.go.tmpl | go run ../../dev/tmpl.go ../decode/types.json | gofmt > scalar_gen.go" type DisplayFormat int const ( NumberDecimal DisplayFormat = iota NumberBinary NumberOctal NumberHex ) func (df DisplayFormat) FormatBase() int { switch df { case NumberDecimal: return 10 case NumberBinary: return 2 case NumberOctal: return 8 case NumberHex: return 16 default: return 0 } } type S struct { Actual interface{} // nil, int, int64, uint64, float64, string, bool, []byte, *bit.Int, bitio.BitReaderAtSeeker, ActualDisplay DisplayFormat Sym interface{} SymDisplay DisplayFormat Description string Unknown bool } func (s S) Value() interface{} { if s.Sym != nil { return s.Sym } return s.Actual } type Mapper interface { MapScalar(S) (S, error) } type Fn func(S) (S, error) func (fn Fn) MapScalar(s S) (S, error) { return fn(s) } var Bin = Fn(func(s S) (S, error) { s.ActualDisplay = NumberBinary; return s, nil }) var Oct = Fn(func(s S) (S, error) { s.ActualDisplay = NumberOctal; return s, nil }) var Dec = Fn(func(s S) (S, error) { s.ActualDisplay = NumberDecimal; return s, nil }) var Hex = Fn(func(s S) (S, error) { s.ActualDisplay = NumberHex; return s, nil }) var SymBin = Fn(func(s S) (S, error) { s.SymDisplay = NumberBinary; return s, nil }) var SymOct = Fn(func(s S) (S, error) { s.SymDisplay = NumberOctal; return s, nil }) var SymDec = Fn(func(s S) (S, error) { s.SymDisplay = NumberDecimal; return s, nil }) var SymHex = Fn(func(s S) (S, error) { s.SymDisplay = NumberHex; return s, nil }) func Actual(v interface{}) Mapper { return Fn(func(s S) (S, error) { s.Actual = v; return s, nil }) } func Sym(v interface{}) Mapper { return Fn(func(s S) (S, error) { s.Sym = v; return s, nil }) } func Description(v string) Mapper { return Fn(func(s S) (S, error) { s.Description = v; return s, nil }) } func UAdd(n int) Mapper { return Fn(func(s S) (S, error) { // TODO: use math.Add/Sub? s.Actual = uint64(int64(s.ActualU()) + int64(n)) return s, nil }) } func SAdd(n int) Mapper { return Fn(func(s S) (S, error) { s.Actual = s.ActualS() + int64(n) return s, nil }) } // TODO: nicer api? func Trim(cutset string) Mapper { return Fn(func(s S) (S, error) { s.Actual = strings.Trim(s.ActualStr(), cutset) return s, nil }) } var TrimSpace = Fn(func(s S) (S, error) { s.Actual = strings.TrimSpace(s.ActualStr()) return s, nil }) func strMapToSym(fn func(s string) (interface{}, error)) Mapper { return Fn(func(s S) (S, error) { ts := strings.TrimSpace(s.ActualStr()) if ts != "" { n, err := fn(ts) if err != nil { return s, err } s.Sym = n } return s, nil }) } func StrUintToSym(base int) Mapper { return strMapToSym(func(s string) (interface{}, error) { return strconv.ParseUint(s, base, 64) }) } func StrIntToSym(base int) Mapper { return strMapToSym(func(s string) (interface{}, error) { return strconv.ParseInt(s, base, 64) }) } func StrFToSym(base int) Mapper { return strMapToSym(func(s string) (interface{}, error) { return strconv.ParseFloat(s, base) }) } type URangeEntry struct { Range [2]uint64 S S } // URangeToScalar maps uint64 ranges to a scalar, first in range is chosen type URangeToScalar []URangeEntry func (rs URangeToScalar) MapScalar(s S) (S, error) { n, ok := s.Actual.(uint64) if !ok { return s, nil } for _, re := range rs { if n >= re.Range[0] && n <= re.Range[1] { ns := re.S ns.Actual = s.Actual s = ns break } } return s, nil } // SRangeToScalar maps ranges to a scalar, first in range is chosen type SRangeEntry struct { Range [2]int64 S S } // SRangeToScalar maps sint64 ranges to a scalar, first in range is chosen type SRangeToScalar []SRangeEntry func (rs SRangeToScalar) MapScalar(s S) (S, error) { n, ok := s.Actual.(int64) if !ok { return s, nil } for _, re := range rs { if n >= re.Range[0] && n <= re.Range[1] { ns := re.S ns.Actual = s.Actual s = ns break } } return s, nil } func rawSym(s S, nBytes int, fn func(b []byte) string) (S, error) { br, ok := s.Actual.(bitio.ReadAtSeeker) if !ok { return s, nil } brLen, err := bitioextra.Len(br) if err != nil { return S{}, err } if nBytes < 0 { nBytes = int(brLen) / 8 if brLen%8 != 0 { nBytes++ } } if brLen < int64(nBytes)*8 { return s, nil } // TODO: shared somehow? b := make([]byte, nBytes) if _, err := br.ReadBitsAt(b, int64(nBytes)*8, 0); err != nil { return s, err } s.Sym = fn(b[0:nBytes]) return s, nil } var RawUUID = Fn(func(s S) (S, error) { return rawSym(s, -1, func(b []byte) string { return fmt.Sprintf("%x-%x-%x-%x-%x", b[0:4], b[4:6], b[6:8], b[8:10], b[10:]) }) }) var RawHex = Fn(func(s S) (S, error) { return rawSym(s, -1, func(b []byte) string { return fmt.Sprintf("%x", b) }) }) type BytesToScalar []struct { Bytes []byte Scalar S } func (m BytesToScalar) MapScalar(s S) (S, error) { bb := &bytes.Buffer{} if _, err := bitioextra.CopyBits(bb, s.ActualBitBuf()); err != nil { return s, err } for _, bs := range m { if bytes.Equal(bb.Bytes(), bs.Bytes) { ns := bs.Scalar ns.Actual = s.Actual s = ns break } } return s, nil }