From adf82692888d43e520baa49ae4fc1298a14f99d9 Mon Sep 17 00:00:00 2001 From: Sean DuBois Date: Wed, 6 Mar 2024 11:36:55 -0500 Subject: [PATCH] Optimize unmarshal type handling Dispatch type as a byte, instead of allocating a string Before ''' 14433 ns/op 2576 B/op 108 allocs/op ''' After ''' 13616 ns/op 2512 B/op 83 allocs/op ''' --- base_lexer.go | 36 +++-------- unmarshal.go | 162 +++++++++++++++++++++++++------------------------- util.go | 4 +- 3 files changed, 92 insertions(+), 110 deletions(-) diff --git a/base_lexer.go b/base_lexer.go index 904b059..99542ab 100644 --- a/base_lexer.go +++ b/base_lexer.go @@ -178,45 +178,27 @@ func (l *baseLexer) readLine() (string, error) { } } -func (l *baseLexer) readString(until byte) (string, error) { - start := l.pos +func (l *baseLexer) readType() (byte, error) { for { - ch, err := l.readByte() + firstByte, err := l.readByte() if err != nil { - return "", err - } - if ch == until { - return string(l.value[start:l.pos]), nil + return 0, err } - } -} -func (l *baseLexer) readType() (string, error) { - for { - b, err := l.readByte() - if err != nil { - return "", err - } - - if isNewline(b) { + if isNewline(firstByte) { continue } - err = l.unreadByte() - if err != nil { - return "", err - } - - key, err := l.readString('=') + secondByte, err := l.readByte() if err != nil { - return key, err + return 0, err } - if len(key) == 2 { - return key, nil + if secondByte != '=' { + return firstByte, l.syntaxError() } - return key, l.syntaxError() + return firstByte, nil } } diff --git a/unmarshal.go b/unmarshal.go index 6dc493f..0963459 100644 --- a/unmarshal.go +++ b/unmarshal.go @@ -114,8 +114,8 @@ func (s *SessionDescription) Unmarshal(value []byte) error { } func s1(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { - if key == "v=" { + return l.handleType(func(key byte) stateFn { + if key == 'v' { return unmarshalProtocolVersion } return nil @@ -123,8 +123,8 @@ func s1(l *lexer) (stateFn, error) { } func s2(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { - if key == "o=" { + return l.handleType(func(key byte) stateFn { + if key == 'o' { return unmarshalOrigin } return nil @@ -132,8 +132,8 @@ func s2(l *lexer) (stateFn, error) { } func s3(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { - if key == "s=" { + return l.handleType(func(key byte) stateFn { + if key == 's' { return unmarshalSessionName } return nil @@ -141,21 +141,21 @@ func s3(l *lexer) (stateFn, error) { } func s4(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "i=": + case 'i': return unmarshalSessionInformation - case "u=": + case 'u': return unmarshalURI - case "e=": + case 'e': return unmarshalEmail - case "p=": + case 'p': return unmarshalPhone - case "c=": + case 'c': return unmarshalSessionConnectionInformation - case "b=": + case 'b': return unmarshalSessionBandwidth - case "t=": + case 't': return unmarshalTiming } return nil @@ -163,11 +163,11 @@ func s4(l *lexer) (stateFn, error) { } func s5(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "b=": + case 'b': return unmarshalSessionBandwidth - case "t=": + case 't': return unmarshalTiming } return nil @@ -175,15 +175,15 @@ func s5(l *lexer) (stateFn, error) { } func s6(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "p=": + case 'p': return unmarshalPhone - case "c=": + case 'c': return unmarshalSessionConnectionInformation - case "b=": + case 'b': return unmarshalSessionBandwidth - case "t=": + case 't': return unmarshalTiming } return nil @@ -191,19 +191,19 @@ func s6(l *lexer) (stateFn, error) { } func s7(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "u=": + case 'u': return unmarshalURI - case "e=": + case 'e': return unmarshalEmail - case "p=": + case 'p': return unmarshalPhone - case "c=": + case 'c': return unmarshalSessionConnectionInformation - case "b=": + case 'b': return unmarshalSessionBandwidth - case "t=": + case 't': return unmarshalTiming } return nil @@ -211,13 +211,13 @@ func s7(l *lexer) (stateFn, error) { } func s8(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "c=": + case 'c': return unmarshalSessionConnectionInformation - case "b=": + case 'b': return unmarshalSessionBandwidth - case "t=": + case 't': return unmarshalTiming } return nil @@ -225,19 +225,19 @@ func s8(l *lexer) (stateFn, error) { } func s9(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "z=": + case 'z': return unmarshalTimeZones - case "k=": + case 'k': return unmarshalSessionEncryptionKey - case "a=": + case 'a': return unmarshalSessionAttribute - case "r=": + case 'r': return unmarshalRepeatTimes - case "t=": + case 't': return unmarshalTiming - case "m=": + case 'm': return unmarshalMediaDescription } return nil @@ -245,17 +245,17 @@ func s9(l *lexer) (stateFn, error) { } func s10(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "e=": + case 'e': return unmarshalEmail - case "p=": + case 'p': return unmarshalPhone - case "c=": + case 'c': return unmarshalSessionConnectionInformation - case "b=": + case 'b': return unmarshalSessionBandwidth - case "t=": + case 't': return unmarshalTiming } return nil @@ -263,11 +263,11 @@ func s10(l *lexer) (stateFn, error) { } func s11(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "a=": + case 'a': return unmarshalSessionAttribute - case "m=": + case 'm': return unmarshalMediaDescription } return nil @@ -275,19 +275,19 @@ func s11(l *lexer) (stateFn, error) { } func s12(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "a=": + case 'a': return unmarshalMediaAttribute - case "k=": + case 'k': return unmarshalMediaEncryptionKey - case "b=": + case 'b': return unmarshalMediaBandwidth - case "c=": + case 'c': return unmarshalMediaConnectionInformation - case "i=": + case 'i': return unmarshalMediaTitle - case "m=": + case 'm': return unmarshalMediaDescription } return nil @@ -295,13 +295,13 @@ func s12(l *lexer) (stateFn, error) { } func s13(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "a=": + case 'a': return unmarshalSessionAttribute - case "k=": + case 'k': return unmarshalSessionEncryptionKey - case "m=": + case 'm': return unmarshalMediaDescription } return nil @@ -309,23 +309,23 @@ func s13(l *lexer) (stateFn, error) { } func s14(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "a=": + case 'a': return unmarshalMediaAttribute - case "k=": + case 'k': // Non-spec ordering return unmarshalMediaEncryptionKey - case "b=": + case 'b': // Non-spec ordering return unmarshalMediaBandwidth - case "c=": + case 'c': // Non-spec ordering return unmarshalMediaConnectionInformation - case "i=": + case 'i': // Non-spec ordering return unmarshalMediaTitle - case "m=": + case 'm': return unmarshalMediaDescription } return nil @@ -333,20 +333,20 @@ func s14(l *lexer) (stateFn, error) { } func s15(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "a=": + case 'a': return unmarshalMediaAttribute - case "k=": + case 'k': return unmarshalMediaEncryptionKey - case "b=": + case 'b': return unmarshalMediaBandwidth - case "c=": + case 'c': return unmarshalMediaConnectionInformation - case "i=": + case 'i': // Non-spec ordering return unmarshalMediaTitle - case "m=": + case 'm': return unmarshalMediaDescription } return nil @@ -354,20 +354,20 @@ func s15(l *lexer) (stateFn, error) { } func s16(l *lexer) (stateFn, error) { - return l.handleType(func(key string) stateFn { + return l.handleType(func(key byte) stateFn { switch key { - case "a=": + case 'a': return unmarshalMediaAttribute - case "k=": + case 'k': return unmarshalMediaEncryptionKey - case "c=": + case 'c': return unmarshalMediaConnectionInformation - case "b=": + case 'b': return unmarshalMediaBandwidth - case "i=": + case 'i': // Non-spec ordering return unmarshalMediaTitle - case "m=": + case 'm': return unmarshalMediaDescription } return nil diff --git a/util.go b/util.go index 6c2ab53..5449dca 100644 --- a/util.go +++ b/util.go @@ -315,11 +315,11 @@ type lexer struct { baseLexer } -type keyToState func(key string) stateFn +type keyToState func(key byte) stateFn func (l *lexer) handleType(fn keyToState) (stateFn, error) { key, err := l.readType() - if errors.Is(err, io.EOF) && key == "" { + if errors.Is(err, io.EOF) && key == 0 { return nil, nil //nolint:nilnil } else if err != nil { return nil, err