Compare commits
32 Commits
5778616965
...
add-bool
| Author | SHA1 | Date | |
|---|---|---|---|
| 0ac26711ac | |||
| 8446ae6186 | |||
| c511ebcb15 | |||
| 00b0f13d3e | |||
| 13d35e54f5 | |||
| 770f6b05b4 | |||
| 2ee954e18f | |||
| cdfccb0f1c | |||
| 5d5d3fd31c | |||
| 190a89fbb3 | |||
| e991b5af67 | |||
| 5a3d0e19ea | |||
| fbc55534f6 | |||
| b6e180f466 | |||
| 8f5f25780e | |||
| f08213cd49 | |||
| 2194198693 | |||
| 5c2b8a0582 | |||
| 4575fa229b | |||
| cbfb513933 | |||
| f10327356e | |||
| f402b46b1c | |||
| c3d0f33700 | |||
| ba2dc6b53f | |||
| 2e03867c66 | |||
| 7a03d8d6b5 | |||
| b2504cda2d | |||
| f6b12d43fb | |||
| c185f5058f | |||
| 813d219580 | |||
| b44d364f0f | |||
| 405b458702 |
@@ -1,48 +1,66 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import "os"
|
import "os"
|
||||||
import "fmt"
|
|
||||||
import "strings"
|
import "strings"
|
||||||
import "path/filepath"
|
import "path/filepath"
|
||||||
|
import "git.tebibyte.media/sashakoshka/go-cli"
|
||||||
import "git.tebibyte.media/sashakoshka/goparse"
|
import "git.tebibyte.media/sashakoshka/goparse"
|
||||||
import "git.tebibyte.media/sashakoshka/hopp/generate"
|
import "git.tebibyte.media/sashakoshka/hopp/generate"
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
name := os.Args[0]
|
flagOutput := cli.NewInputFlag('o', "output", "The output file", "", cli.ValString)
|
||||||
if len(os.Args) != 3 {
|
flagPackageName := cli.NewInputFlag('p', "package-name", "The package name of the file", "", cli.ValString)
|
||||||
fmt.Fprintf(os.Stderr, "Usage: %s SOURCE DESTINATION\n", name)
|
command := cli.New("Compile PDL files to program source code",
|
||||||
|
flagOutput,
|
||||||
|
flagPackageName)
|
||||||
|
command.Syntax = "FILE [OPTIONS]..."
|
||||||
|
command.ParseOrExit(os.Args)
|
||||||
|
|
||||||
|
if len(command.Args) != 1 {
|
||||||
|
command.Usage()
|
||||||
os.Exit(2)
|
os.Exit(2)
|
||||||
}
|
}
|
||||||
source := os.Args[1]
|
source := command.Args[0]
|
||||||
destination := os.Args[2]
|
destination := flagOutput.Value
|
||||||
|
if destination == "" {
|
||||||
|
destination = "protocol.go"
|
||||||
|
}
|
||||||
|
|
||||||
input, err := os.Open(source)
|
input, err := os.Open(source)
|
||||||
handleErr(1, err)
|
handleErr(command, 1, err)
|
||||||
defer input.Close()
|
defer input.Close()
|
||||||
protocol, err := generate.ParseReader(source, input)
|
protocol, err := generate.ParseReader(source, input)
|
||||||
handleErr(1, err)
|
handleErr(command, 1, err)
|
||||||
|
|
||||||
|
packageName := flagPackageName.Value
|
||||||
|
if packageName == "" {
|
||||||
absDestination, err := filepath.Abs(destination)
|
absDestination, err := filepath.Abs(destination)
|
||||||
handleErr(1, err)
|
handleErr(command, 1, err)
|
||||||
packageName := cleanPackageName(strings.ReplaceAll(
|
base := filepath.Base(absDestination)
|
||||||
strings.ToLower(filepath.Base(absDestination)),
|
if scrounged, ok := scroungeForPackageName(base); ok {
|
||||||
" ", "_"))
|
packageName = scrounged
|
||||||
destination = filepath.Join(os.Args[2], "generated.go")
|
} else {
|
||||||
|
packageName = strings.ReplaceAll(
|
||||||
|
strings.ToLower(base),
|
||||||
|
" ", "_")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
packageName = cleanPackageName(packageName)
|
||||||
|
|
||||||
output, err := os.Create(destination)
|
output, err := os.Create(destination)
|
||||||
handleErr(1, err)
|
handleErr(command, 1, err)
|
||||||
generator := generate.Generator {
|
generator := generate.Generator {
|
||||||
Output: output,
|
Output: output,
|
||||||
PackageName: packageName,
|
PackageName: packageName,
|
||||||
}
|
}
|
||||||
_, err = generator.Generate(protocol)
|
_, err = generator.Generate(protocol)
|
||||||
handleErr(1, err)
|
handleErr(command, 1, err)
|
||||||
fmt.Fprintf(os.Stderr, "%s: OK\n", name)
|
command.Println(output, "OK")
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleErr(code int, err error) {
|
func handleErr(command *cli.Cli, code int, err error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Fprintf(os.Stderr, "%s: %v\n", os.Args[0], parse.Format(err))
|
command.Errorln(parse.Format(err))
|
||||||
os.Exit(code)
|
os.Exit(code)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -61,3 +79,32 @@ func cleanPackageName(str string) string {
|
|||||||
}
|
}
|
||||||
return string(buffer[:j])
|
return string(buffer[:j])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func scroungeForPackageName(dir string) (string, bool) {
|
||||||
|
entries, err := os.ReadDir(dir)
|
||||||
|
if err != nil { return "", false}
|
||||||
|
for _, entry := range entries {
|
||||||
|
if !entry.Type().IsRegular() { continue }
|
||||||
|
file, err := os.Open(filepath.Join(dir, entry.Name()))
|
||||||
|
if err != nil { continue }
|
||||||
|
defer file.Close()
|
||||||
|
// FIXME: it is entirely possible that the only file will have
|
||||||
|
// a shitload of doc comments preceeding the package name, and
|
||||||
|
// those comments are usually huge so this is bad
|
||||||
|
buffer := [512]byte { }
|
||||||
|
n, _ := file.Read(buffer[:])
|
||||||
|
text := string(buffer[:n])
|
||||||
|
|
||||||
|
packageIndex := strings.Index(text, "package")
|
||||||
|
if packageIndex < 0 { continue }
|
||||||
|
text = text[packageIndex:]
|
||||||
|
|
||||||
|
newlineIndex := strings.Index(text, "\n")
|
||||||
|
if packageIndex > 0 { text = text[:newlineIndex] }
|
||||||
|
|
||||||
|
fields := strings.Fields(text)
|
||||||
|
if len(fields) < 2 { continue }
|
||||||
|
return fields[1], true
|
||||||
|
}
|
||||||
|
return "", false
|
||||||
|
}
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ PDL allows defining a protocol using HOPP and TAPE.
|
|||||||
| F64 | FP | 7 |
|
| F64 | FP | 7 |
|
||||||
| F128[^2] | FP | 15 |
|
| F128[^2] | FP | 15 |
|
||||||
| F256[^2] | FP | 31 |
|
| F256[^2] | FP | 31 |
|
||||||
|
| Bool | SI | |
|
||||||
| String | SBA/LBA | * | UTF-8 string
|
| String | SBA/LBA | * | UTF-8 string
|
||||||
| Buffer | SBA/LBA | * | Byte array
|
| Buffer | SBA/LBA | * | Byte array
|
||||||
| []\<TYPE\> | OTA | * | Array of any type[^1]
|
| []\<TYPE\> | OTA | * | Array of any type[^1]
|
||||||
@@ -52,6 +53,7 @@ structures. They are separated by whitespace.
|
|||||||
| RBrace | `}` | A right curly brace.
|
| RBrace | `}` | A right curly brace.
|
||||||
| LBracket | `[` | A left square bracket.
|
| LBracket | `[` | A left square bracket.
|
||||||
| RBracket | `]` | A right square bracket.
|
| RBracket | `]` | A right square bracket.
|
||||||
|
| Comment | `\/\/.*$` | A doc comment starting with a double-slash.
|
||||||
|
|
||||||
## Syntax
|
## Syntax
|
||||||
|
|
||||||
@@ -68,18 +70,27 @@ an Ident token respectively. A message consists of the method code (Method), the
|
|||||||
message name (Ident), and the message's root type. This is usually a table, but
|
message name (Ident), and the message's root type. This is usually a table, but
|
||||||
can be anything.
|
can be anything.
|
||||||
|
|
||||||
|
Messages, types, and table fields can all have doc comments preceding them,
|
||||||
|
which are used to generate documentation for the protocol. The syntax is the
|
||||||
|
same as Go's (for now). Comments aren't allowed anywhere else.
|
||||||
|
|
||||||
Here is an example of all that:
|
Here is an example of all that:
|
||||||
|
|
||||||
```
|
```
|
||||||
|
// Connect is sent from the client to the server as the first message of an
|
||||||
|
// authenticated transaction.
|
||||||
M0000 Connect {
|
M0000 Connect {
|
||||||
0000 Name String,
|
0000 Name String,
|
||||||
0001 Password String,
|
0001 Password String,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UserList is sent from the server to the client in response to a Connect
|
||||||
|
// message.
|
||||||
M0001 UserList {
|
M0001 UserList {
|
||||||
0000 Users []User,
|
0000 Users []User,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// User holds profile information about a single user.
|
||||||
User {
|
User {
|
||||||
0000 Name String,
|
0000 Name String,
|
||||||
0001 Bio String,
|
0001 Bio String,
|
||||||
@@ -99,7 +110,7 @@ Below is an EBNF description of the language.
|
|||||||
<field> -> <key> <ident> <type>
|
<field> -> <key> <ident> <type>
|
||||||
<type> -> <ident>
|
<type> -> <ident>
|
||||||
| "[" "]" <type>
|
| "[" "]" <type>
|
||||||
| "{" (<field> ",")* [<field>] "}"
|
| "{" (<comment>* <field> ",")* [<comment>* <field>] "}"
|
||||||
<message> -> <method> <ident> <type>
|
<message> -> <comment>* <method> <ident> <type>
|
||||||
<typedef> -> <ident> <type>
|
<typedef> -> <comment>* <ident> <type>
|
||||||
```
|
```
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ const preamble = `
|
|||||||
|
|
||||||
const static = `
|
const static = `
|
||||||
// Table is a KTV table with an undefined schema.
|
// Table is a KTV table with an undefined schema.
|
||||||
type Table map[uint16] any
|
type Table = map[uint16] any
|
||||||
|
|
||||||
// Message is any message that can be sent along this protocol.
|
// Message is any message that can be sent along this protocol.
|
||||||
type Message interface {
|
type Message interface {
|
||||||
@@ -47,6 +47,15 @@ func canAssign(destination, source tape.Tag) bool {
|
|||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// boolInt converts a bool to an integer.
|
||||||
|
func boolInt(input bool) int {
|
||||||
|
if input {
|
||||||
|
return 1
|
||||||
|
} else {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
}
|
||||||
`
|
`
|
||||||
|
|
||||||
// Generator converts protocols into Go code.
|
// Generator converts protocols into Go code.
|
||||||
@@ -112,13 +121,20 @@ func (this *Generator) Generate(protocol *Protocol) (n int, err error) {
|
|||||||
return n, nil
|
return n, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *Generator) generateTypedef(name string, typ Type) (n int, err error) {
|
func (this *Generator) generateTypedef(name string, typedef Typedef) (n int, err error) {
|
||||||
|
typ := typedef.Type
|
||||||
|
|
||||||
// type definition
|
// type definition
|
||||||
|
if typedef.Doc == "" {
|
||||||
nn, err := this.iprintf(
|
nn, err := this.iprintf(
|
||||||
"\n// %s represents the protocol data type %s.\n",
|
"\n// %s represents the protocol data type %s.\n",
|
||||||
name, name)
|
name, name)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.iprintf("type %s ", name)
|
} else {
|
||||||
|
nn, err := this.iprintf("\n%s\n", this.formatComment(typedef.Doc))
|
||||||
|
n += nn; if err != nil { return n, err }
|
||||||
|
}
|
||||||
|
nn, err := this.iprintf("type %s ", name)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.generateType(typ)
|
nn, err = this.generateType(typ)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
@@ -208,10 +224,16 @@ func (this *Generator) generateTypedef(name string, typ Type) (n int, err error)
|
|||||||
// generateMessage generates the structure, as well as encoding decoding
|
// generateMessage generates the structure, as well as encoding decoding
|
||||||
// functions for the given message.
|
// functions for the given message.
|
||||||
func (this *Generator) generateMessage(method uint16, message Message) (n int, err error) {
|
func (this *Generator) generateMessage(method uint16, message Message) (n int, err error) {
|
||||||
|
if message.Doc == "" {
|
||||||
nn, err := this.iprintf(
|
nn, err := this.iprintf(
|
||||||
"\n// %s represents the protocol message M%04X %s.\n",
|
"\n// %s represents the protocol message M%04X %s.\n",
|
||||||
message.Name, method, message.Name)
|
message.Name, method, message.Name)
|
||||||
nn, err = this.iprintf("type %s ", this.resolveMessageName(message.Name))
|
n += nn; if err != nil { return n, err }
|
||||||
|
} else {
|
||||||
|
nn, err := this.iprintf("\n%s\n", this.formatComment(message.Doc))
|
||||||
|
n += nn; if err != nil { return n, err }
|
||||||
|
}
|
||||||
|
nn, err := this.iprintf("type %s ", this.resolveMessageName(message.Name))
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.generateType(message.Type)
|
nn, err = this.generateType(message.Type)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
@@ -302,6 +324,9 @@ func (this *Generator) generateMessage(method uint16, message Message) (n int, e
|
|||||||
// - nn int
|
// - nn int
|
||||||
func (this *Generator) generateEncodeValue(typ Type, valueSource, tagSource string) (n int, err error) {
|
func (this *Generator) generateEncodeValue(typ Type, valueSource, tagSource string) (n int, err error) {
|
||||||
switch typ := typ.(type) {
|
switch typ := typ.(type) {
|
||||||
|
case TypeBool:
|
||||||
|
// SI: (none)
|
||||||
|
// SI stores the value in the tag, so we write nothing here
|
||||||
case TypeInt:
|
case TypeInt:
|
||||||
// SI: (none)
|
// SI: (none)
|
||||||
// LI/LSI: <value: IntN>
|
// LI/LSI: <value: IntN>
|
||||||
@@ -372,7 +397,7 @@ func (this *Generator) generateEncodeValue(typ Type, valueSource, tagSource stri
|
|||||||
nn, err = this.iprintf("}\n")
|
nn, err = this.iprintf("}\n")
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.iprintf(
|
nn, err = this.iprintf(
|
||||||
"nn, err = encoder.WriteUintN(uint64(len(%s)), %s.CN())\n",
|
"nn, err = encoder.WriteUintN(uint64(len(%s)), %s.CN() + 1)\n",
|
||||||
valueSource, tagSource)
|
valueSource, tagSource)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.generateErrorCheck()
|
nn, err = this.generateErrorCheck()
|
||||||
@@ -407,8 +432,8 @@ func (this *Generator) generateEncodeValue(typ Type, valueSource, tagSource stri
|
|||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.iprintf("nn, err = encoder.WriteTag(itemTag)\n")
|
nn, err = this.iprintf("nn, err = encoder.WriteTag(itemTag)\n")
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
n += nn; if err != nil { return n, err }
|
|
||||||
nn, err = this.generateErrorCheck()
|
nn, err = this.generateErrorCheck()
|
||||||
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.iprintf("for _, item := range %s {\n", valueSource)
|
nn, err = this.iprintf("for _, item := range %s {\n", valueSource)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
this.push()
|
this.push()
|
||||||
@@ -445,7 +470,7 @@ func (this *Generator) generateEncodeValue(typ Type, valueSource, tagSource stri
|
|||||||
nn, err = this.iprintf("}\n")
|
nn, err = this.iprintf("}\n")
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.iprintf(
|
nn, err = this.iprintf(
|
||||||
"nn, err = encoder.WriteUintN(%d, %s.CN())\n",
|
"nn, err = encoder.WriteUintN(%d, %s.CN() + 1)\n",
|
||||||
len(typ.Fields), tagSource)
|
len(typ.Fields), tagSource)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.generateErrorCheck()
|
nn, err = this.generateErrorCheck()
|
||||||
@@ -505,6 +530,11 @@ func (this *Generator) generateEncodeValue(typ Type, valueSource, tagSource stri
|
|||||||
// for [Generator.generateDecodeBranch].
|
// for [Generator.generateDecodeBranch].
|
||||||
func (this *Generator) generateDecodeValue(typ Type, typeName, valueSource, tagSource string) (n int, err error) {
|
func (this *Generator) generateDecodeValue(typ Type, typeName, valueSource, tagSource string) (n int, err error) {
|
||||||
switch typ := typ.(type) {
|
switch typ := typ.(type) {
|
||||||
|
case TypeBool:
|
||||||
|
// SI: (none)
|
||||||
|
// SI stores the value in the tag
|
||||||
|
nn, err := this.iprintf("*%s = %s.CN() > 0\n", valueSource, tagSource)
|
||||||
|
n += nn; if err != nil { return n, err }
|
||||||
case TypeInt:
|
case TypeInt:
|
||||||
// SI: (none)
|
// SI: (none)
|
||||||
// LI/LSI: <value: IntN>
|
// LI/LSI: <value: IntN>
|
||||||
@@ -709,7 +739,7 @@ func (this *Generator) generateDecodeBranch(hash [16]byte, typ Type, typeName st
|
|||||||
this.pop()
|
this.pop()
|
||||||
nn, err = this.iprintf("}\n")
|
nn, err = this.iprintf("}\n")
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.iprintf("%s, nn, err = decoder.ReadUintN(int(tag.CN()))\n", lengthVar)
|
nn, err = this.iprintf("%s, nn, err = decoder.ReadUintN(int(tag.CN()) + 1)\n", lengthVar)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.generateErrorCheck()
|
nn, err = this.generateErrorCheck()
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
@@ -785,7 +815,7 @@ func (this *Generator) generateDecodeBranch(hash [16]byte, typ Type, typeName st
|
|||||||
this.pop()
|
this.pop()
|
||||||
nn, err = this.iprintf("}\n")
|
nn, err = this.iprintf("}\n")
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.iprintf("%s, nn, err = decoder.ReadUintN(int(tag.CN()))\n", lengthVar)
|
nn, err = this.iprintf("%s, nn, err = decoder.ReadUintN(int(tag.CN()) + 1)\n", lengthVar)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.generateErrorCheck()
|
nn, err = this.generateErrorCheck()
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
@@ -931,6 +961,9 @@ func (this *Generator) generateBareErrorCheck() (n int, err error) {
|
|||||||
func (this *Generator) generateTag(typ Type, source string) (tagVar string, n int, err error) {
|
func (this *Generator) generateTag(typ Type, source string) (tagVar string, n int, err error) {
|
||||||
tagVar = this.newTemporaryVar("tag")
|
tagVar = this.newTemporaryVar("tag")
|
||||||
switch typ := typ.(type) {
|
switch typ := typ.(type) {
|
||||||
|
case TypeBool:
|
||||||
|
nn, err := this.iprintf("%s := tape.SI.WithCN(boolInt(bool(%s)))\n", tagVar, source)
|
||||||
|
n += nn; if err != nil { return tagVar, n, err }
|
||||||
case TypeInt:
|
case TypeInt:
|
||||||
if typ.Bits <= 5 {
|
if typ.Bits <= 5 {
|
||||||
nn, err := this.iprintf("%s := tape.SI.WithCN(int(%s))\n", tagVar, source)
|
nn, err := this.iprintf("%s := tape.SI.WithCN(int(%s))\n", tagVar, source)
|
||||||
@@ -952,13 +985,13 @@ func (this *Generator) generateTag(typ Type, source string) (tagVar string, n in
|
|||||||
nn, err := this.iprintf("%s := tape.BufferTag([]byte(%s))\n", tagVar, source)
|
nn, err := this.iprintf("%s := tape.BufferTag([]byte(%s))\n", tagVar, source)
|
||||||
n += nn; if err != nil { return tagVar, n, err }
|
n += nn; if err != nil { return tagVar, n, err }
|
||||||
case TypeArray:
|
case TypeArray:
|
||||||
nn, err := this.iprintf("%s := tape.OTA.WithCN(tape.IntBytes(uint64(len(%s))))\n", tagVar, source)
|
nn, err := this.iprintf("%s := tape.OTA.WithCN(tape.IntBytes(uint64(len(%s))) - 1)\n", tagVar, source)
|
||||||
n += nn; if err != nil { return tagVar, n, err }
|
n += nn; if err != nil { return tagVar, n, err }
|
||||||
case TypeTable:
|
case TypeTable:
|
||||||
nn, err := this.iprintf("%s := tape.KTV.WithCN(tape.IntBytes(uint64(len(%s))))\n", tagVar, source)
|
nn, err := this.iprintf("%s := tape.KTV.WithCN(tape.IntBytes(uint64(len(%s))) - 1)\n", tagVar, source)
|
||||||
n += nn; if err != nil { return tagVar, n, err }
|
n += nn; if err != nil { return tagVar, n, err }
|
||||||
case TypeTableDefined:
|
case TypeTableDefined:
|
||||||
nn, err := this.iprintf("%s := tape.KTV.WithCN(%d)\n", tagVar, tape.IntBytes(uint64(len(typ.Fields))))
|
nn, err := this.iprintf("%s := tape.KTV.WithCN(%d)\n", tagVar, tape.IntBytes(uint64(len(typ.Fields))) - 1)
|
||||||
n += nn; if err != nil { return tagVar, n, err }
|
n += nn; if err != nil { return tagVar, n, err }
|
||||||
case TypeNamed:
|
case TypeNamed:
|
||||||
resolved, err := this.resolveTypeName(typ.Name)
|
resolved, err := this.resolveTypeName(typ.Name)
|
||||||
@@ -984,6 +1017,9 @@ func (this *Generator) generateTag(typ Type, source string) (tagVar string, n in
|
|||||||
// information is chosen.
|
// information is chosen.
|
||||||
func (this *Generator) generateTN(typ Type) (n int, err error) {
|
func (this *Generator) generateTN(typ Type) (n int, err error) {
|
||||||
switch typ := typ.(type) {
|
switch typ := typ.(type) {
|
||||||
|
case TypeBool:
|
||||||
|
nn, err := this.printf("tape.SI")
|
||||||
|
n += nn; if err != nil { return n, err }
|
||||||
case TypeInt:
|
case TypeInt:
|
||||||
if typ.Bits <= 5 {
|
if typ.Bits <= 5 {
|
||||||
nn, err := this.printf("tape.SI")
|
nn, err := this.printf("tape.SI")
|
||||||
@@ -1027,6 +1063,9 @@ func (this *Generator) generateTN(typ Type) (n int, err error) {
|
|||||||
|
|
||||||
func (this *Generator) generateType(typ Type) (n int, err error) {
|
func (this *Generator) generateType(typ Type) (n int, err error) {
|
||||||
switch typ := typ.(type) {
|
switch typ := typ.(type) {
|
||||||
|
case TypeBool:
|
||||||
|
nn, err := this.printf("bool")
|
||||||
|
n += nn; if err != nil { return n, err }
|
||||||
case TypeInt:
|
case TypeInt:
|
||||||
if err := this.validateIntBitSize(typ.Bits); err != nil {
|
if err := this.validateIntBitSize(typ.Bits); err != nil {
|
||||||
return n, err
|
return n, err
|
||||||
@@ -1090,7 +1129,9 @@ func (this *Generator) generateTypeTableDefined(typ TypeTableDefined) (n int, er
|
|||||||
|
|
||||||
for _, key := range slices.Sorted(maps.Keys(typ.Fields)) {
|
for _, key := range slices.Sorted(maps.Keys(typ.Fields)) {
|
||||||
field := typ.Fields[key]
|
field := typ.Fields[key]
|
||||||
nn, err := this.iprintf("%s ", field.Name)
|
nn, err := this.iprintf("%s\n", this.formatComment(field.Doc))
|
||||||
|
n += nn; if err != nil { return n, err }
|
||||||
|
nn, err = this.iprintf("%s ", field.Name)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
nn, err = this.generateType(field.Type)
|
nn, err = this.generateType(field.Type)
|
||||||
n += nn; if err != nil { return n, err }
|
n += nn; if err != nil { return n, err }
|
||||||
@@ -1175,17 +1216,21 @@ func (this *Generator) iprintf(format string, args ...any) (n int, err error) {
|
|||||||
return fmt.Fprintf(this.Output, this.indent() + format, args...)
|
return fmt.Fprintf(this.Output, this.indent() + format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *Generator) formatComment(comment string) string {
|
||||||
|
return "// " + strings.ReplaceAll(comment, "\n", "\n" + this.indent() + "// ")
|
||||||
|
}
|
||||||
|
|
||||||
func (this *Generator) resolveMessageName(message string) string {
|
func (this *Generator) resolveMessageName(message string) string {
|
||||||
return "Message" + message
|
return "Message" + message
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *Generator) resolveTypeName(name string) (Type, error) {
|
func (this *Generator) resolveTypeName(name string) (Type, error) {
|
||||||
if typ, ok := this.protocol.Types[name]; ok {
|
if typedef, ok := this.protocol.Types[name]; ok {
|
||||||
if typ, ok := typ.(TypeNamed); ok {
|
if typ, ok := typedef.Type.(TypeNamed); ok {
|
||||||
return this.resolveTypeName(typ.Name)
|
return this.resolveTypeName(typ.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
return typ, nil
|
return typedef.Type, nil
|
||||||
}
|
}
|
||||||
return nil, fmt.Errorf("no type exists called %s", name)
|
return nil, fmt.Errorf("no type exists called %s", name)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -59,6 +59,7 @@ func init() {
|
|||||||
0x000C: Field { Name: "NI16",Type: TypeInt { Bits: 16, Signed: true } },
|
0x000C: Field { Name: "NI16",Type: TypeInt { Bits: 16, Signed: true } },
|
||||||
0x000D: Field { Name: "NI32",Type: TypeInt { Bits: 32, Signed: true } },
|
0x000D: Field { Name: "NI32",Type: TypeInt { Bits: 32, Signed: true } },
|
||||||
0x000E: Field { Name: "NI64",Type: TypeInt { Bits: 64, Signed: true } },
|
0x000E: Field { Name: "NI64",Type: TypeInt { Bits: 64, Signed: true } },
|
||||||
|
0x000F: Field { Name: "Bool",Type: TypeBool { } },
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@@ -83,12 +84,14 @@ func init() {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
exampleProtocol.Types["User"] = TypeTableDefined {
|
exampleProtocol.Types["User"] = Typedef {
|
||||||
|
Type: TypeTableDefined {
|
||||||
Fields: map[uint16] Field {
|
Fields: map[uint16] Field {
|
||||||
0x0000: Field { Name: "Name", Type: TypeString { } },
|
0x0000: Field { Name: "Name", Type: TypeString { } },
|
||||||
0x0001: Field { Name: "Bio", Type: TypeString { } },
|
0x0001: Field { Name: "Bio", Type: TypeString { } },
|
||||||
0x0002: Field { Name: "Followers", Type: TypeInt { Bits: 32 } },
|
0x0002: Field { Name: "Followers", Type: TypeInt { Bits: 32 } },
|
||||||
},
|
},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -103,7 +106,7 @@ func TestGenerateRunEncodeDecode(test *testing.T) {
|
|||||||
}
|
}
|
||||||
testEncodeDecode(
|
testEncodeDecode(
|
||||||
&messageConnect,
|
&messageConnect,
|
||||||
tu.S(0xE1, 0x02).AddVar(
|
tu.S(0xE0, 0x02).AddVar(
|
||||||
[]byte { 0x00, 0x00, 0x86, 'r', 'a', 'r', 'i', 't', 'y' },
|
[]byte { 0x00, 0x00, 0x86, 'r', 'a', 'r', 'i', 't', 'y' },
|
||||||
[]byte { 0x00, 0x01, 0x84, 'g', 'e', 'm', 's' },
|
[]byte { 0x00, 0x01, 0x84, 'g', 'e', 'm', 's' },
|
||||||
))
|
))
|
||||||
@@ -129,8 +132,8 @@ func TestGenerateRunEncodeDecode(test *testing.T) {
|
|||||||
}
|
}
|
||||||
testEncodeDecode(
|
testEncodeDecode(
|
||||||
&messageUserList,
|
&messageUserList,
|
||||||
tu.S(0xE1, 0x01, 0x00, 0x00,
|
tu.S(0xE0, 0x01, 0x00, 0x00,
|
||||||
0xC1, 0x03, 0xE1,
|
0xC0, 0x03, 0xE0,
|
||||||
).Add(0x03).AddVar(
|
).Add(0x03).AddVar(
|
||||||
[]byte { 0x00, 0x00, 0x86, 'r', 'a', 'r', 'i', 't', 'y' },
|
[]byte { 0x00, 0x00, 0x86, 'r', 'a', 'r', 'i', 't', 'y' },
|
||||||
[]byte { 0x00, 0x01, 0x87, 'a', 's', 'd', 'j', 'a', 'd', 's' },
|
[]byte { 0x00, 0x01, 0x87, 'a', 's', 'd', 'j', 'a', 'd', 's' },
|
||||||
@@ -155,7 +158,7 @@ func TestGenerateRunEncodeDecode(test *testing.T) {
|
|||||||
}
|
}
|
||||||
testEncodeDecode(
|
testEncodeDecode(
|
||||||
&messagePulse,
|
&messagePulse,
|
||||||
tu.S(0xE1, 0x05).AddVar(
|
tu.S(0xE0, 0x05).AddVar(
|
||||||
[]byte { 0x00, 0x00, 0x09 },
|
[]byte { 0x00, 0x00, 0x09 },
|
||||||
[]byte { 0x00, 0x01, 0x41, 0xCA, 0xDF },
|
[]byte { 0x00, 0x01, 0x41, 0xCA, 0xDF },
|
||||||
[]byte { 0x00, 0x02, 0x61, 0x51, 0xAC },
|
[]byte { 0x00, 0x02, 0x61, 0x51, 0xAC },
|
||||||
@@ -176,7 +179,7 @@ func TestGenerateRunEncodeDecode(test *testing.T) {
|
|||||||
}
|
}
|
||||||
testEncodeDecode(
|
testEncodeDecode(
|
||||||
&messageNestedArray,
|
&messageNestedArray,
|
||||||
tu.S(0xC1, 0x02, 0xC1,
|
tu.S(0xC0, 0x02, 0xC0,
|
||||||
0x06, 0x20, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6,
|
0x06, 0x20, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6,
|
||||||
35, 0x20, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6,
|
35, 0x20, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6,
|
||||||
0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC,
|
0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC,
|
||||||
@@ -199,10 +202,11 @@ func TestGenerateRunEncodeDecode(test *testing.T) {
|
|||||||
NI16: -0x34C9,
|
NI16: -0x34C9,
|
||||||
NI32: -0x10E134C9,
|
NI32: -0x10E134C9,
|
||||||
NI64: -0x639109BC10E134C9,
|
NI64: -0x639109BC10E134C9,
|
||||||
|
Bool: true,
|
||||||
}
|
}
|
||||||
testEncodeDecode(
|
testEncodeDecode(
|
||||||
&messageIntegers,
|
&messageIntegers,
|
||||||
tu.S(0xE1, 13).AddVar(
|
tu.S(0xE0, 14).AddVar(
|
||||||
[]byte { 0x00, 0x00, 0x13 },
|
[]byte { 0x00, 0x00, 0x13 },
|
||||||
[]byte { 0x00, 0x01, 0x20, 0xC9 },
|
[]byte { 0x00, 0x01, 0x20, 0xC9 },
|
||||||
[]byte { 0x00, 0x02, 0x21, 0x34, 0xC9 },
|
[]byte { 0x00, 0x02, 0x21, 0x34, 0xC9 },
|
||||||
@@ -216,6 +220,7 @@ func TestGenerateRunEncodeDecode(test *testing.T) {
|
|||||||
[]byte { 0x00, 0x0C, 0x41, 0xCB, 0x37 },
|
[]byte { 0x00, 0x0C, 0x41, 0xCB, 0x37 },
|
||||||
[]byte { 0x00, 0x0D, 0x43, 0xEF, 0x1E, 0xCB, 0x37 },
|
[]byte { 0x00, 0x0D, 0x43, 0xEF, 0x1E, 0xCB, 0x37 },
|
||||||
[]byte { 0x00, 0x0E, 0x47, 0x9C, 0x6E, 0xF6, 0x43, 0xEF, 0x1E, 0xCB, 0x37 },
|
[]byte { 0x00, 0x0E, 0x47, 0x9C, 0x6E, 0xF6, 0x43, 0xEF, 0x1E, 0xCB, 0x37 },
|
||||||
|
[]byte { 0x00, 0x0F, 0x01 },
|
||||||
))
|
))
|
||||||
log.Println("MessageDynamic")
|
log.Println("MessageDynamic")
|
||||||
messageDynamic := MessageDynamic {
|
messageDynamic := MessageDynamic {
|
||||||
@@ -243,7 +248,7 @@ func TestGenerateRunEncodeDecode(test *testing.T) {
|
|||||||
}
|
}
|
||||||
testEncodeDecode(
|
testEncodeDecode(
|
||||||
&messageDynamic,
|
&messageDynamic,
|
||||||
tu.S(0xE1, 14).AddVar(
|
tu.S(0xE0, 14).AddVar(
|
||||||
[]byte { 0x00, 0x00, 0x20, 0x23 },
|
[]byte { 0x00, 0x00, 0x20, 0x23 },
|
||||||
[]byte { 0x00, 0x01, 0x21, 0x32, 0x47 },
|
[]byte { 0x00, 0x01, 0x21, 0x32, 0x47 },
|
||||||
[]byte { 0x00, 0x02, 0x23, 0x87, 0x32, 0x45, 0x23 },
|
[]byte { 0x00, 0x02, 0x23, 0x87, 0x32, 0x45, 0x23 },
|
||||||
@@ -255,11 +260,15 @@ func TestGenerateRunEncodeDecode(test *testing.T) {
|
|||||||
[]byte { 0x00, 0x08, 0x63, 0x45, 0x12, 0x63, 0xCE },
|
[]byte { 0x00, 0x08, 0x63, 0x45, 0x12, 0x63, 0xCE },
|
||||||
[]byte { 0x00, 0x09, 0x67, 0x40, 0x74, 0x4E, 0x3D, 0x6F, 0xCD, 0x17, 0x75 },
|
[]byte { 0x00, 0x09, 0x67, 0x40, 0x74, 0x4E, 0x3D, 0x6F, 0xCD, 0x17, 0x75 },
|
||||||
[]byte { 0x00, 0x0A, 0x87, 'f', 'o', 'x', ' ', 'b', 'e', 'd' },
|
[]byte { 0x00, 0x0A, 0x87, 'f', 'o', 'x', ' ', 'b', 'e', 'd' },
|
||||||
[]byte { 0x00, 0x0B, 0xC4, 0x00, 0x07, 0x00, 0x06, 0x00, 0x05, 0x00, 0x04 },
|
[]byte { 0x00, 0x0B, 0xC0, 0x04, 0x41,
|
||||||
[]byte { 0x00, 0x0C, 0xE1, 0x02,
|
0x00, 0x07,
|
||||||
0x00, 0x01, 0x20, 0x08,
|
0x00, 0x06,
|
||||||
|
0x00, 0x05,
|
||||||
|
0x00, 0x04 },
|
||||||
|
[]byte { 0x00, 0x0C, 0xE0, 0x02,
|
||||||
|
0x00, 0x01, 0x40, 0x08,
|
||||||
0x00, 0x02, 0x67, 0x40, 0x11, 0x99, 0x99, 0x99, 0x99, 0x99, 0x9A },
|
0x00, 0x02, 0x67, 0x40, 0x11, 0x99, 0x99, 0x99, 0x99, 0x99, 0x9A },
|
||||||
[]byte { 0x00, 0x0D, 0xE1, 0x03,
|
[]byte { 0x00, 0x0D, 0xE0, 0x03, // ERR
|
||||||
0x00, 0x01, 0x63, 0x43, 0xF4, 0xC0, 0x00,
|
0x00, 0x01, 0x63, 0x43, 0xF4, 0xC0, 0x00,
|
||||||
0x00, 0x02, 0x82, 'h', 'i',
|
0x00, 0x02, 0x82, 'h', 'i',
|
||||||
0x00, 0x03, 0x21, 0x39, 0x92 },
|
0x00, 0x03, 0x21, 0x39, 0x92 },
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ const (
|
|||||||
TokenRBrace
|
TokenRBrace
|
||||||
TokenLBracket
|
TokenLBracket
|
||||||
TokenRBracket
|
TokenRBracket
|
||||||
|
TokenComment
|
||||||
)
|
)
|
||||||
|
|
||||||
var tokenNames = map[parse.TokenKind] string {
|
var tokenNames = map[parse.TokenKind] string {
|
||||||
@@ -26,6 +27,7 @@ var tokenNames = map[parse.TokenKind] string {
|
|||||||
TokenRBrace: "RBrace",
|
TokenRBrace: "RBrace",
|
||||||
TokenLBracket: "LBracket",
|
TokenLBracket: "LBracket",
|
||||||
TokenRBracket: "RBracket",
|
TokenRBracket: "RBracket",
|
||||||
|
TokenComment: "Comment",
|
||||||
}
|
}
|
||||||
|
|
||||||
func Lex(fileName string, reader io.Reader) (parse.Lexer, error) {
|
func Lex(fileName string, reader io.Reader) (parse.Lexer, error) {
|
||||||
@@ -81,6 +83,18 @@ func (this *lexer) nextInternal() (token parse.Token, err error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
unexpected := func() error {
|
||||||
|
if unicode.IsPrint(this.rune) {
|
||||||
|
return parse.Errorf (
|
||||||
|
this.pos(), "unexpected rune '%c'",
|
||||||
|
this.rune)
|
||||||
|
} else {
|
||||||
|
return parse.Errorf (
|
||||||
|
this.pos(), "unexpected rune %U",
|
||||||
|
this.rune)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
defer func () {
|
defer func () {
|
||||||
newPos := this.pos()
|
newPos := this.pos()
|
||||||
newPos.End -- // TODO figure out why tf we have to do this
|
newPos.End -- // TODO figure out why tf we have to do this
|
||||||
@@ -133,14 +147,21 @@ func (this *lexer) nextInternal() (token parse.Token, err error) {
|
|||||||
token.Kind = TokenRBracket
|
token.Kind = TokenRBracket
|
||||||
appendRune()
|
appendRune()
|
||||||
if this.eof { err = nil; return }
|
if this.eof { err = nil; return }
|
||||||
case unicode.IsPrint(this.rune):
|
// Comment
|
||||||
err = parse.Errorf (
|
case this.rune == '/':
|
||||||
this.pos(), "unexpected rune '%c'",
|
token.Kind = TokenComment
|
||||||
this.rune)
|
appendRune()
|
||||||
|
if this.eof { return }
|
||||||
|
if this.rune != '/' {
|
||||||
|
err = unexpected()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for this.rune != '\n' {
|
||||||
|
appendRune()
|
||||||
|
if this.eof { err = nil; return }
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
err = parse.Errorf (
|
err = unexpected()
|
||||||
this.pos(), "unexpected rune %U",
|
|
||||||
this.rune)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return
|
return
|
||||||
|
|||||||
@@ -6,14 +6,21 @@ import "git.tebibyte.media/sashakoshka/goparse"
|
|||||||
|
|
||||||
func TestLex(test *testing.T) {
|
func TestLex(test *testing.T) {
|
||||||
lexer, err := Lex("test.pdl", strings.NewReader(`
|
lexer, err := Lex("test.pdl", strings.NewReader(`
|
||||||
|
// User holds profile information about a single user.
|
||||||
M0001 User {
|
M0001 User {
|
||||||
0000 Name String,
|
0000 Name String,
|
||||||
|
// dog water comment
|
||||||
|
|
||||||
|
// Users is asdkjsagkj why
|
||||||
|
//
|
||||||
|
// wow
|
||||||
0001 Users []User,
|
0001 Users []User,
|
||||||
0002 Followers U32,
|
0002 Followers U32,
|
||||||
}`))
|
}`))
|
||||||
if err != nil { test.Fatal(parse.Format(err)) }
|
if err != nil { test.Fatal(parse.Format(err)) }
|
||||||
|
|
||||||
correctTokens := []parse.Token {
|
correctTokens := []parse.Token {
|
||||||
|
tok(TokenComment, "// User holds profile information about a single user."),
|
||||||
tok(TokenMethod, "0001"),
|
tok(TokenMethod, "0001"),
|
||||||
tok(TokenIdent, "User"),
|
tok(TokenIdent, "User"),
|
||||||
tok(TokenLBrace, "{"),
|
tok(TokenLBrace, "{"),
|
||||||
@@ -21,6 +28,10 @@ func TestLex(test *testing.T) {
|
|||||||
tok(TokenIdent, "Name"),
|
tok(TokenIdent, "Name"),
|
||||||
tok(TokenIdent, "String"),
|
tok(TokenIdent, "String"),
|
||||||
tok(TokenComma, ","),
|
tok(TokenComma, ","),
|
||||||
|
tok(TokenComment, "// dog water comment"),
|
||||||
|
tok(TokenComment, "// Users is asdkjsagkj why"),
|
||||||
|
tok(TokenComment, "// "),
|
||||||
|
tok(TokenComment, "// wow"),
|
||||||
tok(TokenKey, "0001"),
|
tok(TokenKey, "0001"),
|
||||||
tok(TokenIdent, "Users"),
|
tok(TokenIdent, "Users"),
|
||||||
tok(TokenLBracket, "["),
|
tok(TokenLBracket, "["),
|
||||||
|
|||||||
@@ -100,12 +100,12 @@ func testGenerateRun(test *testing.T, protocol *Protocol, title, imports, testCa
|
|||||||
log.Println("decoding:")
|
log.Println("decoding:")
|
||||||
destination := reflect.New(reflect.ValueOf(message).Elem().Type()).Interface().(Message)
|
destination := reflect.New(reflect.ValueOf(message).Elem().Type()).Interface().(Message)
|
||||||
flat := data.Flatten()
|
flat := data.Flatten()
|
||||||
log.Println("before: ", destination)
|
log.Println("before: ", tu.Describe(destination))
|
||||||
decoder := tape.NewDecoder(bytes.NewBuffer(flat))
|
decoder := tape.NewDecoder(bytes.NewBuffer(flat))
|
||||||
n, err = destination.Decode(decoder)
|
n, err = destination.Decode(decoder)
|
||||||
if err != nil { log.Fatalf("at %d: %v\n", n, err) }
|
if err != nil { log.Fatalf("at %d: %v\n", n, err) }
|
||||||
log.Println("got: ", destination)
|
log.Println("got: ", tu.Describe(destination))
|
||||||
log.Println("correct:", message)
|
log.Println("correct:", tu.Describe(message))
|
||||||
if n != len(flat) {
|
if n != len(flat) {
|
||||||
log.Fatalf("n incorrect: %d != %d\n", n, len(flat))
|
log.Fatalf("n incorrect: %d != %d\n", n, len(flat))
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package generate
|
package generate
|
||||||
|
|
||||||
import "io"
|
import "io"
|
||||||
|
import "strings"
|
||||||
import "strconv"
|
import "strconv"
|
||||||
import "git.tebibyte.media/sashakoshka/goparse"
|
import "git.tebibyte.media/sashakoshka/goparse"
|
||||||
|
|
||||||
@@ -21,7 +22,7 @@ func Parse(lx parse.Lexer) (*Protocol, error) {
|
|||||||
func defaultProtocol() Protocol {
|
func defaultProtocol() Protocol {
|
||||||
return Protocol {
|
return Protocol {
|
||||||
Messages: make(map[uint16] Message),
|
Messages: make(map[uint16] Message),
|
||||||
Types: map[string] Type { },
|
Types: map[string] Typedef { },
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -47,18 +48,28 @@ func (this *parser) parse() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (this *parser) parseTopLevel() error {
|
func (this *parser) parseTopLevel() error {
|
||||||
err := this.ExpectDesc("message or typedef", TokenMethod, TokenIdent)
|
doc := ""
|
||||||
|
for {
|
||||||
|
err := this.ExpectDesc("message or typedef", TokenMethod, TokenIdent, TokenComment)
|
||||||
if err != nil { return err }
|
if err != nil { return err }
|
||||||
if this.EOF() { return nil }
|
if this.EOF() { return nil }
|
||||||
|
if this.Kind() == TokenComment {
|
||||||
|
if doc != "" { doc += "\n" }
|
||||||
|
doc += this.parseComment(this.Value())
|
||||||
|
this.Next()
|
||||||
|
} else {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
switch this.Kind() {
|
switch this.Kind() {
|
||||||
case TokenMethod: return this.parseMessage()
|
case TokenMethod: return this.parseMessage(doc)
|
||||||
case TokenIdent: return this.parseTypedef()
|
case TokenIdent: return this.parseTypedef(doc)
|
||||||
}
|
}
|
||||||
panic("bug")
|
panic("bug")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *parser) parseMessage() error {
|
func (this *parser) parseMessage(doc string) error {
|
||||||
err := this.Expect(TokenMethod)
|
err := this.Expect(TokenMethod)
|
||||||
if err != nil { return err }
|
if err != nil { return err }
|
||||||
method, err := this.parseHexNumber(this.Value(), 0xFFFF)
|
method, err := this.parseHexNumber(this.Value(), 0xFFFF)
|
||||||
@@ -72,12 +83,13 @@ func (this *parser) parseMessage() error {
|
|||||||
if err != nil { return err }
|
if err != nil { return err }
|
||||||
this.protocol.Messages[uint16(method)] = Message {
|
this.protocol.Messages[uint16(method)] = Message {
|
||||||
Name: name,
|
Name: name,
|
||||||
|
Doc: doc,
|
||||||
Type: typ,
|
Type: typ,
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *parser) parseTypedef() error {
|
func (this *parser) parseTypedef(doc string) error {
|
||||||
err := this.Expect(TokenIdent)
|
err := this.Expect(TokenIdent)
|
||||||
if err != nil { return err }
|
if err != nil { return err }
|
||||||
name := this.Value()
|
name := this.Value()
|
||||||
@@ -85,7 +97,10 @@ func (this *parser) parseTypedef() error {
|
|||||||
if err != nil { return err }
|
if err != nil { return err }
|
||||||
typ, err := this.parseType()
|
typ, err := this.parseType()
|
||||||
if err != nil { return err }
|
if err != nil { return err }
|
||||||
this.protocol.Types[name] = typ
|
this.protocol.Types[name] = Typedef {
|
||||||
|
Doc: doc,
|
||||||
|
Type: typ,
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -117,6 +132,7 @@ func (this *parser) parseType() (Type, error) {
|
|||||||
case "Buffer": return TypeBuffer { }, this.Next()
|
case "Buffer": return TypeBuffer { }, this.Next()
|
||||||
case "Table": return TypeTable { }, this.Next()
|
case "Table": return TypeTable { }, this.Next()
|
||||||
case "Any": return TypeAny { }, this.Next()
|
case "Any": return TypeAny { }, this.Next()
|
||||||
|
case "Bool": return TypeBool { }, this.Next()
|
||||||
}
|
}
|
||||||
return this.parseTypeNamed()
|
return this.parseTypeNamed()
|
||||||
case TokenLBracket:
|
case TokenLBracket:
|
||||||
@@ -157,12 +173,22 @@ func (this *parser) parseTypeTable() (TypeTableDefined, error) {
|
|||||||
Fields: make(map[uint16] Field),
|
Fields: make(map[uint16] Field),
|
||||||
}
|
}
|
||||||
for {
|
for {
|
||||||
err := this.ExpectDesc("table field", TokenKey, TokenRBrace)
|
doc := ""
|
||||||
|
for {
|
||||||
|
err := this.ExpectDesc("table field", TokenKey, TokenRBrace, TokenComment)
|
||||||
if err != nil { return TypeTableDefined { }, err }
|
if err != nil { return TypeTableDefined { }, err }
|
||||||
|
if this.Kind() == TokenComment {
|
||||||
|
if doc != "" { doc += "\n" }
|
||||||
|
doc += this.parseComment(this.Value())
|
||||||
|
this.Next()
|
||||||
|
} else {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
if this.Is(TokenRBrace) {
|
if this.Is(TokenRBrace) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
key, field, err := this.parseField()
|
key, field, err := this.parseField(doc)
|
||||||
if err != nil { return TypeTableDefined { }, err }
|
if err != nil { return TypeTableDefined { }, err }
|
||||||
typ.Fields[key] = field
|
typ.Fields[key] = field
|
||||||
err = this.Expect(TokenComma, TokenRBrace)
|
err = this.Expect(TokenComma, TokenRBrace)
|
||||||
@@ -178,7 +204,7 @@ func (this *parser) parseTypeTable() (TypeTableDefined, error) {
|
|||||||
return typ, nil
|
return typ, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *parser) parseField() (uint16, Field, error) {
|
func (this *parser) parseField(doc string) (uint16, Field, error) {
|
||||||
err := this.Expect(TokenKey)
|
err := this.Expect(TokenKey)
|
||||||
if err != nil { return 0, Field { }, err }
|
if err != nil { return 0, Field { }, err }
|
||||||
key, err := this.parseHexNumber(this.Value(), 0xFFFF)
|
key, err := this.parseHexNumber(this.Value(), 0xFFFF)
|
||||||
@@ -192,6 +218,7 @@ func (this *parser) parseField() (uint16, Field, error) {
|
|||||||
if err != nil { return 0, Field { }, err }
|
if err != nil { return 0, Field { }, err }
|
||||||
return uint16(key), Field {
|
return uint16(key), Field {
|
||||||
Name: name,
|
Name: name,
|
||||||
|
Doc: doc,
|
||||||
Type: typ,
|
Type: typ,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
@@ -206,3 +233,7 @@ func (this *parser) parseHexNumber(input string, maxValue int64) (int64, error)
|
|||||||
}
|
}
|
||||||
return number, nil
|
return number, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *parser) parseComment(input string) string {
|
||||||
|
return strings.TrimPrefix(strings.TrimPrefix(input, "//"), " ")
|
||||||
|
}
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ func TestParse(test *testing.T) {
|
|||||||
correct := defaultProtocol()
|
correct := defaultProtocol()
|
||||||
correct.Messages[0x0000] = Message {
|
correct.Messages[0x0000] = Message {
|
||||||
Name: "Connect",
|
Name: "Connect",
|
||||||
|
Doc: "Connect is sent from the client to the server as the first message of an\nauthenticated transaction.",
|
||||||
Type: TypeTableDefined {
|
Type: TypeTableDefined {
|
||||||
Fields: map[uint16] Field {
|
Fields: map[uint16] Field {
|
||||||
0x0000: Field { Name: "Name", Type: TypeString { } },
|
0x0000: Field { Name: "Name", Type: TypeString { } },
|
||||||
@@ -18,36 +19,50 @@ func TestParse(test *testing.T) {
|
|||||||
}
|
}
|
||||||
correct.Messages[0x0001] = Message {
|
correct.Messages[0x0001] = Message {
|
||||||
Name: "UserList",
|
Name: "UserList",
|
||||||
|
Doc: "UserList is sent from the server to the client in response to a Connect\nmessage.",
|
||||||
Type: TypeTableDefined {
|
Type: TypeTableDefined {
|
||||||
Fields: map[uint16] Field {
|
Fields: map[uint16] Field {
|
||||||
0x0000: Field { Name: "Users", Type: TypeArray { Element: TypeNamed { Name: "User" } } },
|
0x0000: Field { Name: "Users", Type: TypeArray { Element: TypeNamed { Name: "User" } } },
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
correct.Types["User"] = TypeTableDefined {
|
correct.Types["User"] = Typedef {
|
||||||
|
Doc: "User holds profile information about a single user.",
|
||||||
|
Type: TypeTableDefined {
|
||||||
Fields: map[uint16] Field {
|
Fields: map[uint16] Field {
|
||||||
0x0000: Field { Name: "Name", Type: TypeString { } },
|
0x0000: Field { Name: "Name", Type: TypeString { } },
|
||||||
0x0001: Field { Name: "Bio", Type: TypeString { } },
|
0x0001: Field { Name: "Bio", Type: TypeString { } },
|
||||||
0x0002: Field { Name: "Followers", Type: TypeInt { Bits: 32 } },
|
0x0002: Field { Name: "Followers", Type: TypeInt { Bits: 32 } },
|
||||||
|
0x0003: Field { Name: "Bouncy", Type: TypeBool { } },
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
correct.Types["Anything"] = TypeAny { }
|
correct.Types["Anything"] = Typedef {
|
||||||
|
Type: TypeAny { },
|
||||||
|
}
|
||||||
test.Log("CORRECT:", &correct)
|
test.Log("CORRECT:", &correct)
|
||||||
|
|
||||||
got, err := ParseReader("test.pdl", strings.NewReader(`
|
got, err := ParseReader("test.pdl", strings.NewReader(`
|
||||||
|
// Connect is sent from the client to the server as the first message of an
|
||||||
|
// authenticated transaction.
|
||||||
M0000 Connect {
|
M0000 Connect {
|
||||||
0000 Name String,
|
0000 Name String,
|
||||||
|
// Password is where you put your secrets, your shameful secrets
|
||||||
0001 Password String,
|
0001 Password String,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UserList is sent from the server to the client in response to a Connect
|
||||||
|
// message.
|
||||||
M0001 UserList {
|
M0001 UserList {
|
||||||
0000 Users []User,
|
0000 Users []User,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// User holds profile information about a single user.
|
||||||
User {
|
User {
|
||||||
0000 Name String,
|
0000 Name String,
|
||||||
0001 Bio String,
|
0001 Bio String,
|
||||||
0002 Followers U32,
|
0002 Followers U32,
|
||||||
|
0003 Bouncy Bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
Anything Any
|
Anything Any
|
||||||
|
|||||||
@@ -7,11 +7,17 @@ import "crypto/md5"
|
|||||||
|
|
||||||
type Protocol struct {
|
type Protocol struct {
|
||||||
Messages map[uint16] Message
|
Messages map[uint16] Message
|
||||||
Types map[string] Type
|
Types map[string] Typedef
|
||||||
}
|
}
|
||||||
|
|
||||||
type Message struct {
|
type Message struct {
|
||||||
Name string
|
Name string
|
||||||
|
Doc string
|
||||||
|
Type Type
|
||||||
|
}
|
||||||
|
|
||||||
|
type Typedef struct {
|
||||||
|
Doc string
|
||||||
Type Type
|
Type Type
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -43,6 +49,12 @@ func (typ TypeFloat) String() string {
|
|||||||
return fmt.Sprintf("F%d", typ.Bits)
|
return fmt.Sprintf("F%d", typ.Bits)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type TypeBool struct { }
|
||||||
|
|
||||||
|
func (TypeBool) String() string {
|
||||||
|
return "Bool"
|
||||||
|
}
|
||||||
|
|
||||||
type TypeString struct { }
|
type TypeString struct { }
|
||||||
|
|
||||||
func (TypeString) String() string {
|
func (TypeString) String() string {
|
||||||
@@ -84,6 +96,7 @@ func (typ TypeTableDefined) String() string {
|
|||||||
|
|
||||||
type Field struct {
|
type Field struct {
|
||||||
Name string
|
Name string
|
||||||
|
Doc string
|
||||||
Type Type
|
Type Type
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
1
go.mod
1
go.mod
@@ -3,6 +3,7 @@ module git.tebibyte.media/sashakoshka/hopp
|
|||||||
go 1.23.0
|
go 1.23.0
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
git.tebibyte.media/sashakoshka/go-cli v0.1.3
|
||||||
git.tebibyte.media/sashakoshka/go-util v0.9.1
|
git.tebibyte.media/sashakoshka/go-util v0.9.1
|
||||||
git.tebibyte.media/sashakoshka/goparse v0.2.0
|
git.tebibyte.media/sashakoshka/goparse v0.2.0
|
||||||
)
|
)
|
||||||
|
|||||||
2
go.sum
2
go.sum
@@ -1,3 +1,5 @@
|
|||||||
|
git.tebibyte.media/sashakoshka/go-cli v0.1.3 h1:tSkWjyx2JrGu6KotbXWSTKSYGGS1D4O3qwCrRoZuwbs=
|
||||||
|
git.tebibyte.media/sashakoshka/go-cli v0.1.3/go.mod h1:JFA3wSdRkXxa4iQJWHfe3DokiG7Dh2XUJBzPmuVlbuY=
|
||||||
git.tebibyte.media/sashakoshka/go-util v0.9.1 h1:eGAbLwYhOlh4aq/0w+YnJcxT83yPhXtxnYMzz6K7xGo=
|
git.tebibyte.media/sashakoshka/go-util v0.9.1 h1:eGAbLwYhOlh4aq/0w+YnJcxT83yPhXtxnYMzz6K7xGo=
|
||||||
git.tebibyte.media/sashakoshka/go-util v0.9.1/go.mod h1:0Q1t+PePdx6tFYkRuJNcpM1Mru7wE6X+it1kwuOH+6Y=
|
git.tebibyte.media/sashakoshka/go-util v0.9.1/go.mod h1:0Q1t+PePdx6tFYkRuJNcpM1Mru7wE6X+it1kwuOH+6Y=
|
||||||
git.tebibyte.media/sashakoshka/goparse v0.2.0 h1:uQmKvOCV2AOlCHEDjg9uclZCXQZzq2PxaXfZ1aIMiQI=
|
git.tebibyte.media/sashakoshka/goparse v0.2.0 h1:uQmKvOCV2AOlCHEDjg9uclZCXQZzq2PxaXfZ1aIMiQI=
|
||||||
|
|||||||
@@ -2,8 +2,9 @@ package testutil
|
|||||||
|
|
||||||
import "fmt"
|
import "fmt"
|
||||||
import "slices"
|
import "slices"
|
||||||
import "strings"
|
|
||||||
import "reflect"
|
import "reflect"
|
||||||
|
import "strings"
|
||||||
|
import "unicode"
|
||||||
|
|
||||||
// Snake lets you compare blocks of data where the ordering of certain parts may
|
// Snake lets you compare blocks of data where the ordering of certain parts may
|
||||||
// be swapped every which way. It is designed for comparing the encoding of
|
// be swapped every which way. It is designed for comparing the encoding of
|
||||||
@@ -124,6 +125,10 @@ func (this *describer) describe(value reflect.Value) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
value = reflect.ValueOf(value.Interface())
|
value = reflect.ValueOf(value.Interface())
|
||||||
|
if !value.IsValid() {
|
||||||
|
this.printf("<invalid>")
|
||||||
|
return
|
||||||
|
}
|
||||||
switch value.Kind() {
|
switch value.Kind() {
|
||||||
case reflect.Array, reflect.Slice:
|
case reflect.Array, reflect.Slice:
|
||||||
this.printf("[\n")
|
this.printf("[\n")
|
||||||
@@ -141,12 +146,20 @@ func (this *describer) describe(value reflect.Value) {
|
|||||||
typ := value.Type()
|
typ := value.Type()
|
||||||
for index := range typ.NumField() {
|
for index := range typ.NumField() {
|
||||||
indexBuffer := [1]int { index }
|
indexBuffer := [1]int { index }
|
||||||
this.iprintf("%s: ", typ.Field(index).Name)
|
field := typ.Field(index)
|
||||||
|
this.iprintf("%s: ", field.Name)
|
||||||
|
for _, char := range field.Name {
|
||||||
|
if unicode.IsUpper(char) {
|
||||||
this.describe(value.FieldByIndex(indexBuffer[:]))
|
this.describe(value.FieldByIndex(indexBuffer[:]))
|
||||||
|
} else {
|
||||||
|
this.printf("<private>")
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
this.iprintf("\n")
|
this.iprintf("\n")
|
||||||
}
|
}
|
||||||
this.indent -= 1
|
this.indent -= 1
|
||||||
this.iprintf("}\n")
|
this.iprintf("}")
|
||||||
case reflect.Map:
|
case reflect.Map:
|
||||||
this.printf("map {\n")
|
this.printf("map {\n")
|
||||||
this.indent += 1
|
this.indent += 1
|
||||||
@@ -159,7 +172,7 @@ func (this *describer) describe(value reflect.Value) {
|
|||||||
this.iprintf("\n")
|
this.iprintf("\n")
|
||||||
}
|
}
|
||||||
this.indent -= 1
|
this.indent -= 1
|
||||||
this.iprintf("}\n")
|
this.iprintf("}")
|
||||||
case reflect.Pointer:
|
case reflect.Pointer:
|
||||||
this.printf("& ")
|
this.printf("& ")
|
||||||
this.describe(value.Elem())
|
this.describe(value.Elem())
|
||||||
|
|||||||
@@ -59,6 +59,7 @@ func EncodeAny(encoder *Encoder, value any, tag Tag) (n int, err error) {
|
|||||||
case reflect.Uint64: return encoder.WriteUint64(uint64(reflectValue.Uint()))
|
case reflect.Uint64: return encoder.WriteUint64(uint64(reflectValue.Uint()))
|
||||||
case reflect.Float32: return encoder.WriteFloat32(float32(reflectValue.Float()))
|
case reflect.Float32: return encoder.WriteFloat32(float32(reflectValue.Float()))
|
||||||
case reflect.Float64: return encoder.WriteFloat64(float64(reflectValue.Float()))
|
case reflect.Float64: return encoder.WriteFloat64(float64(reflectValue.Float()))
|
||||||
|
case reflect.Bool: return // SI has no payload
|
||||||
case reflect.String:
|
case reflect.String:
|
||||||
if reflectValue.Len() > MaxStructureLength {
|
if reflectValue.Len() > MaxStructureLength {
|
||||||
return 0, ErrTooLong
|
return 0, ErrTooLong
|
||||||
@@ -116,9 +117,9 @@ func DecodeAnyInto(decoder *Decoder, destination any, tag Tag) (n int, err error
|
|||||||
func DecodeAny(decoder *Decoder, tag Tag) (value any, n int, err error) {
|
func DecodeAny(decoder *Decoder, tag Tag) (value any, n int, err error) {
|
||||||
destination, err := skeletonPointer(decoder, tag)
|
destination, err := skeletonPointer(decoder, tag)
|
||||||
if err != nil { return nil, n, err }
|
if err != nil { return nil, n, err }
|
||||||
nn, err := DecodeAnyInto(decoder, destination, tag)
|
nn, err := decodeAny(decoder, destination.Elem(), tag)
|
||||||
n += nn; if err != nil { return nil, n, err }
|
n += nn; if err != nil { return nil, n, err }
|
||||||
return destination, n, err
|
return destination.Elem().Interface(), n, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// unknownSlicePlaceholder is inserted by skeletonValue and informs the program
|
// unknownSlicePlaceholder is inserted by skeletonValue and informs the program
|
||||||
@@ -242,12 +243,22 @@ func decodeAnyOrError(decoder *Decoder, destination reflect.Value, tag Tag) (n i
|
|||||||
}
|
}
|
||||||
lengthCast, err := Uint64ToIntSafe(length)
|
lengthCast, err := Uint64ToIntSafe(length)
|
||||||
if err != nil { return n, err }
|
if err != nil { return n, err }
|
||||||
if isTypeAny(destination.Type()) {
|
|
||||||
// need a skeleton value if we are assigning to any.
|
// im fucking so done dude. im so fucking done. this was
|
||||||
|
// supposed to only run when we need it but i guess it runs all
|
||||||
|
// the time, because when we get a map destination (a valid,
|
||||||
|
// allocated one) we break apart on SetMapIndex because of a nil
|
||||||
|
// map. yeah thats right. a fucking nil map panic. on the map we
|
||||||
|
// just allocated. but running this unconditionally (whether or
|
||||||
|
// not we receive an empty any value) actually makes it fucking
|
||||||
|
// work. go figure().
|
||||||
|
//
|
||||||
|
// (the map allocation functionality in skeletonPointer was
|
||||||
|
// removed after this comment was written)
|
||||||
value := reflect.MakeMapWithSize(reflect.TypeOf(dummyMap), lengthCast)
|
value := reflect.MakeMapWithSize(reflect.TypeOf(dummyMap), lengthCast)
|
||||||
destination.Set(value)
|
destination.Set(value)
|
||||||
destination = value
|
destination = value
|
||||||
}
|
|
||||||
destination.Clear()
|
destination.Clear()
|
||||||
for _ = range lengthCast {
|
for _ = range lengthCast {
|
||||||
key, nn, err := decoder.ReadUint16()
|
key, nn, err := decoder.ReadUint16()
|
||||||
@@ -288,6 +299,12 @@ func tagAny(reflectValue reflect.Value) (Tag, error) {
|
|||||||
case reflect.Uint64: return LI.WithCN(7), nil
|
case reflect.Uint64: return LI.WithCN(7), nil
|
||||||
case reflect.Float32: return FP.WithCN(3), nil
|
case reflect.Float32: return FP.WithCN(3), nil
|
||||||
case reflect.Float64: return FP.WithCN(7), nil
|
case reflect.Float64: return FP.WithCN(7), nil
|
||||||
|
case reflect.Bool:
|
||||||
|
if reflectValue.Bool() {
|
||||||
|
return SI.WithCN(1), nil
|
||||||
|
} else {
|
||||||
|
return SI.WithCN(0), nil
|
||||||
|
}
|
||||||
case reflect.String: return bufferLenTag(reflectValue.Len()), nil
|
case reflect.String: return bufferLenTag(reflectValue.Len()), nil
|
||||||
}
|
}
|
||||||
if reflectValue.CanConvert(reflect.TypeOf(dummyBuffer)) {
|
if reflectValue.CanConvert(reflect.TypeOf(dummyBuffer)) {
|
||||||
@@ -364,7 +381,8 @@ func canSet(destination reflect.Type, tag Tag) error {
|
|||||||
switch destination.Kind() {
|
switch destination.Kind() {
|
||||||
case
|
case
|
||||||
reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
|
reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
|
||||||
reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64,
|
||||||
|
reflect.Bool:
|
||||||
default:
|
default:
|
||||||
return errCantAssignf("cannot assign integer to %v", destination)
|
return errCantAssignf("cannot assign integer to %v", destination)
|
||||||
}
|
}
|
||||||
@@ -387,7 +405,11 @@ func canSet(destination reflect.Type, tag Tag) error {
|
|||||||
return errCantAssignf("cannot assign array to %v", destination)
|
return errCantAssignf("cannot assign array to %v", destination)
|
||||||
}
|
}
|
||||||
case KTV:
|
case KTV:
|
||||||
if destination != reflect.TypeOf(dummyMap) {
|
cantAssign :=
|
||||||
|
destination.Kind() != reflect.Map ||
|
||||||
|
destination.Key().Kind() != reflect.Uint16 ||
|
||||||
|
!isTypeAny(destination.Elem())
|
||||||
|
if cantAssign {
|
||||||
return errCantAssignf("cannot assign table to %v", destination)
|
return errCantAssignf("cannot assign table to %v", destination)
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
@@ -418,6 +440,8 @@ func setInt(destination reflect.Value, value int64, bytes int) {
|
|||||||
// setUint expects a settable destination.
|
// setUint expects a settable destination.
|
||||||
func setUint(destination reflect.Value, value uint64, bytes int) {
|
func setUint(destination reflect.Value, value uint64, bytes int) {
|
||||||
switch {
|
switch {
|
||||||
|
case destination.Kind() == reflect.Bool:
|
||||||
|
destination.Set(reflect.ValueOf(value > 0))
|
||||||
case destination.CanInt():
|
case destination.CanInt():
|
||||||
destination.Set(reflect.ValueOf(int64(value)).Convert(destination.Type()))
|
destination.Set(reflect.ValueOf(int64(value)).Convert(destination.Type()))
|
||||||
case destination.CanUint():
|
case destination.CanUint():
|
||||||
@@ -496,7 +520,8 @@ func skeletonValue(decoder *Decoder, tag Tag) (reflect.Value, error) {
|
|||||||
func skeletonPointer(decoder *Decoder, tag Tag) (reflect.Value, error) {
|
func skeletonPointer(decoder *Decoder, tag Tag) (reflect.Value, error) {
|
||||||
typ, err := typeOf(decoder, tag)
|
typ, err := typeOf(decoder, tag)
|
||||||
if err != nil { return reflect.Value { }, err }
|
if err != nil { return reflect.Value { }, err }
|
||||||
return reflect.New(typ), nil
|
value := reflect.New(typ)
|
||||||
|
return value, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// typeOf returns the type of the current tag being decoded. It does not use up
|
// typeOf returns the type of the current tag being decoded. It does not use up
|
||||||
|
|||||||
@@ -15,6 +15,8 @@ var samplePayloads = [][]byte {
|
|||||||
/* uint16 */ []byte { byte(LI.WithCN(1)), 0x45, 0x67 },
|
/* uint16 */ []byte { byte(LI.WithCN(1)), 0x45, 0x67 },
|
||||||
/* uint32 */ []byte { byte(LI.WithCN(3)), 0x45, 0x67, 0x89, 0xAB },
|
/* uint32 */ []byte { byte(LI.WithCN(3)), 0x45, 0x67, 0x89, 0xAB },
|
||||||
/* uint64 */ []byte { byte(LI.WithCN(7)), 0x45, 0x67, 0x89, 0xAB, 0xCD, 0xEF, 0x01, 0x23 },
|
/* uint64 */ []byte { byte(LI.WithCN(7)), 0x45, 0x67, 0x89, 0xAB, 0xCD, 0xEF, 0x01, 0x23 },
|
||||||
|
/* bool */ []byte { byte(SI.WithCN(0)) },
|
||||||
|
/* bool */ []byte { byte(SI.WithCN(1)) },
|
||||||
/* string */ []byte { byte(SBA.WithCN(7)), 'p', 'u', 'p', 'e', 'v', 'e', 'r' },
|
/* string */ []byte { byte(SBA.WithCN(7)), 'p', 'u', 'p', 'e', 'v', 'e', 'r' },
|
||||||
/* []byte */ []byte { byte(SBA.WithCN(5)), 'b', 'l', 'a', 'r', 'g' },
|
/* []byte */ []byte { byte(SBA.WithCN(5)), 'b', 'l', 'a', 'r', 'g' },
|
||||||
/* []string */ []byte {
|
/* []string */ []byte {
|
||||||
@@ -27,6 +29,12 @@ var samplePayloads = [][]byte {
|
|||||||
0x02, 0x23, byte(LSI.WithCN(1)), 0x45, 0x67,
|
0x02, 0x23, byte(LSI.WithCN(1)), 0x45, 0x67,
|
||||||
0x02, 0x24, byte(LI.WithCN(3)), 0x45, 0x67, 0x89, 0xAB,
|
0x02, 0x24, byte(LI.WithCN(3)), 0x45, 0x67, 0x89, 0xAB,
|
||||||
},
|
},
|
||||||
|
/* map[uint16] any */ []byte {
|
||||||
|
byte(KTV.WithCN(0)), 3,
|
||||||
|
0x00, 0x01, 0x63, 0x43, 0xF4, 0xC0, 0x00,
|
||||||
|
0x00, 0x02, 0x82, 'h', 'i',
|
||||||
|
0x00, 0x03, 0x21, 0x39, 0x92,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
var sampleValues = []any {
|
var sampleValues = []any {
|
||||||
@@ -39,6 +47,8 @@ var sampleValues = []any {
|
|||||||
/* uint16 */ uint16(0x4567),
|
/* uint16 */ uint16(0x4567),
|
||||||
/* uint32 */ uint32(0x456789AB),
|
/* uint32 */ uint32(0x456789AB),
|
||||||
/* uint64 */ uint64(0x456789ABCDEF0123),
|
/* uint64 */ uint64(0x456789ABCDEF0123),
|
||||||
|
/* bool */ false,
|
||||||
|
/* bool */ true,
|
||||||
/* string */ "pupever",
|
/* string */ "pupever",
|
||||||
/* []byte */ "blarg",
|
/* []byte */ "blarg",
|
||||||
/* []string */ []string {
|
/* []string */ []string {
|
||||||
@@ -49,6 +59,11 @@ var sampleValues = []any {
|
|||||||
0x0223: int16(0x4567),
|
0x0223: int16(0x4567),
|
||||||
0x0224: uint32(0x456789AB),
|
0x0224: uint32(0x456789AB),
|
||||||
},
|
},
|
||||||
|
/* map[uint16] any */ map[uint16] any {
|
||||||
|
0x0001: float32(489.5),
|
||||||
|
0x0002: "hi",
|
||||||
|
0x0003: uint16(0x3992),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
type userDefinedInteger int16
|
type userDefinedInteger int16
|
||||||
@@ -73,7 +88,9 @@ func TestEncodeAnyTable(test *testing.T) {
|
|||||||
0x3456: userDefinedInteger(0x3921),
|
0x3456: userDefinedInteger(0x3921),
|
||||||
0x1F1F: float32(67.26),
|
0x1F1F: float32(67.26),
|
||||||
0x0F0F: float64(5.3),
|
0x0F0F: float64(5.3),
|
||||||
}, KTV.WithCN(0), tu.S(9).AddVar(
|
0xAAAA: false,
|
||||||
|
0xBBBB: true,
|
||||||
|
}, KTV.WithCN(0), tu.S(11).AddVar(
|
||||||
[]byte {
|
[]byte {
|
||||||
0xF3, 0xB9,
|
0xF3, 0xB9,
|
||||||
byte(LSI.WithCN(3)),
|
byte(LSI.WithCN(3)),
|
||||||
@@ -127,6 +144,14 @@ func TestEncodeAnyTable(test *testing.T) {
|
|||||||
byte(FP.WithCN(7)),
|
byte(FP.WithCN(7)),
|
||||||
0x40, 0x15, 0x33, 0x33, 0x33, 0x33, 0x33, 0x33,
|
0x40, 0x15, 0x33, 0x33, 0x33, 0x33, 0x33, 0x33,
|
||||||
},
|
},
|
||||||
|
[]byte {
|
||||||
|
0xAA, 0xAA,
|
||||||
|
byte(SI.WithCN(0)),
|
||||||
|
},
|
||||||
|
[]byte {
|
||||||
|
0xBB, 0xBB,
|
||||||
|
byte(SI.WithCN(1)),
|
||||||
|
},
|
||||||
))
|
))
|
||||||
if err != nil { test.Fatal(err) }
|
if err != nil { test.Fatal(err) }
|
||||||
}
|
}
|
||||||
@@ -135,7 +160,7 @@ func TestDecodeWrongType(test *testing.T) {
|
|||||||
for index, data := range samplePayloads {
|
for index, data := range samplePayloads {
|
||||||
test.Logf("data %2d %v [%s]", index, Tag(data[0]), tu.HexBytes(data[1:]))
|
test.Logf("data %2d %v [%s]", index, Tag(data[0]), tu.HexBytes(data[1:]))
|
||||||
// integers should only assign to other integers
|
// integers should only assign to other integers
|
||||||
if index > 8 {
|
if index > 10 {
|
||||||
cas := func(destination any) {
|
cas := func(destination any) {
|
||||||
n, err := DecodeAnyInto(NewDecoder(bytes.NewBuffer(data[1:])), destination, Tag(data[0]))
|
n, err := DecodeAnyInto(NewDecoder(bytes.NewBuffer(data[1:])), destination, Tag(data[0]))
|
||||||
if err != nil { test.Fatalf("error: %v | n: %d", err, n) }
|
if err != nil { test.Fatalf("error: %v | n: %d", err, n) }
|
||||||
@@ -144,7 +169,7 @@ func TestDecodeWrongType(test *testing.T) {
|
|||||||
if reflectValue.Int() != 0 {
|
if reflectValue.Int() != 0 {
|
||||||
test.Fatalf("destination not zero: %v", reflectValue.Elem().Interface())
|
test.Fatalf("destination not zero: %v", reflectValue.Elem().Interface())
|
||||||
}
|
}
|
||||||
} else {
|
} else if reflectValue.Kind() != reflect.Bool {
|
||||||
if reflectValue.Uint() != 0 {
|
if reflectValue.Uint() != 0 {
|
||||||
test.Fatalf("destination not zero: %v", reflectValue.Elem().Interface())
|
test.Fatalf("destination not zero: %v", reflectValue.Elem().Interface())
|
||||||
}
|
}
|
||||||
@@ -169,6 +194,8 @@ func TestDecodeWrongType(test *testing.T) {
|
|||||||
{ var dest uint32; cas(&dest) }
|
{ var dest uint32; cas(&dest) }
|
||||||
test.Log("- uint64")
|
test.Log("- uint64")
|
||||||
{ var dest uint64; cas(&dest) }
|
{ var dest uint64; cas(&dest) }
|
||||||
|
test.Log("- bool")
|
||||||
|
{ var dest bool; cas(&dest) }
|
||||||
}
|
}
|
||||||
arrayCase := func(destination any) {
|
arrayCase := func(destination any) {
|
||||||
n, err := DecodeAnyInto(NewDecoder(bytes.NewBuffer(data[1:])), destination, Tag(data[0]))
|
n, err := DecodeAnyInto(NewDecoder(bytes.NewBuffer(data[1:])), destination, Tag(data[0]))
|
||||||
@@ -183,19 +210,19 @@ func TestDecodeWrongType(test *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// SBA/LBA types should only assign to other SBA/LBA types
|
// SBA/LBA types should only assign to other SBA/LBA types
|
||||||
if index != 9 && index != 10 {
|
if index != 11 && index != 12 {
|
||||||
test.Log("- string")
|
test.Log("- string")
|
||||||
{ var dest string; arrayCase(&dest) }
|
{ var dest string; arrayCase(&dest) }
|
||||||
test.Log("- []byte")
|
test.Log("- []byte")
|
||||||
{ var dest []byte; arrayCase(&dest) }
|
{ var dest []byte; arrayCase(&dest) }
|
||||||
}
|
}
|
||||||
// arrays should only assign to other arrays
|
// arrays should only assign to other arrays
|
||||||
if index != 11 {
|
if index != 13 {
|
||||||
test.Log("- []string")
|
test.Log("- []string")
|
||||||
{ var dest []string; arrayCase(&dest) }
|
{ var dest []string; arrayCase(&dest) }
|
||||||
}
|
}
|
||||||
// tables should only assign to other tables
|
// tables should only assign to other tables
|
||||||
if index != 12 {
|
if index != 14 && index != 15 {
|
||||||
test.Log("- map[uint16] any")
|
test.Log("- map[uint16] any")
|
||||||
{ var dest = map[uint16] any { }; arrayCase(&dest) }
|
{ var dest = map[uint16] any { }; arrayCase(&dest) }
|
||||||
}
|
}
|
||||||
@@ -220,6 +247,12 @@ func TestEncodeDecodeAnyTable(test *testing.T) {
|
|||||||
func TestEncodeDecodeAnyDestination(test *testing.T) {
|
func TestEncodeDecodeAnyDestination(test *testing.T) {
|
||||||
var destination any
|
var destination any
|
||||||
for index, data := range samplePayloads {
|
for index, data := range samplePayloads {
|
||||||
|
if _, isBool := sampleValues[index].(bool); isBool {
|
||||||
|
// test is invalid for bools because they are never
|
||||||
|
// created as a skeleton value
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
tag := Tag(data[0])
|
tag := Tag(data[0])
|
||||||
payload := data[1:]
|
payload := data[1:]
|
||||||
test.Logf("data %2d %v [%s]", index, tag, tu.HexBytes(payload))
|
test.Logf("data %2d %v [%s]", index, tag, tu.HexBytes(payload))
|
||||||
@@ -296,3 +329,69 @@ func TestPeekSliceOnce(test *testing.T) {
|
|||||||
test.Fatalf("wrong n: %d != %d", got, correct)
|
test.Fatalf("wrong n: %d != %d", got, correct)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestTagAny(test *testing.T) {
|
||||||
|
cases := [][2]any {
|
||||||
|
[2]any { LSI.WithCN(3), int(9) },
|
||||||
|
[2]any { LSI.WithCN(0), int8(9) },
|
||||||
|
[2]any { LSI.WithCN(1), int16(9) },
|
||||||
|
[2]any { LSI.WithCN(3), int32(9) },
|
||||||
|
[2]any { LSI.WithCN(7), int64(9) },
|
||||||
|
[2]any { LI.WithCN(3), uint(9) },
|
||||||
|
[2]any { LI.WithCN(0), uint8(9) },
|
||||||
|
[2]any { LI.WithCN(1), uint16(9) },
|
||||||
|
[2]any { LI.WithCN(3), uint32(9) },
|
||||||
|
[2]any { LI.WithCN(7), uint64(9) },
|
||||||
|
[2]any { FP.WithCN(3), float32(9) },
|
||||||
|
[2]any { FP.WithCN(7), float64(9) },
|
||||||
|
[2]any { SBA.WithCN(12), "small string" },
|
||||||
|
[2]any { SBA.WithCN(12), []byte("small string") },
|
||||||
|
[2]any { LBA.WithCN(0), "this is a very long string that is long" },
|
||||||
|
[2]any { LBA.WithCN(0), []byte("this is a very long string that is long") },
|
||||||
|
[2]any { LBA.WithCN(1), lipsum },
|
||||||
|
[2]any { OTA.WithCN(0), []int { 1, 2, 3, 4, 5 } },
|
||||||
|
[2]any { OTA.WithCN(0), []string { "1, 2, 3, 4, 5" } },
|
||||||
|
[2]any { KTV.WithCN(0), map[uint16] any {
|
||||||
|
0: 1,
|
||||||
|
1: "wow",
|
||||||
|
2: 10.238,
|
||||||
|
45: -9,
|
||||||
|
9: map[uint16] any { },
|
||||||
|
}},
|
||||||
|
}
|
||||||
|
for _, cas := range cases {
|
||||||
|
test.Log(cas)
|
||||||
|
got, err := TagAny(cas[1])
|
||||||
|
if err != nil { test.Fatal(err) }
|
||||||
|
if correct := cas[0].(Tag); correct != got {
|
||||||
|
test.Fatalf("wrong tag: %v != %v", got, correct)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDecodeAny(test *testing.T) {
|
||||||
|
for index, payload := range samplePayloads {
|
||||||
|
if _, isBool := sampleValues[index].(bool); isBool {
|
||||||
|
// test is invalid for bools because they are never
|
||||||
|
// created as a skeleton value
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
correctValue := sampleValues[index]
|
||||||
|
data := payload[1:]
|
||||||
|
decoder := NewDecoder(bytes.NewBuffer(data))
|
||||||
|
tag := Tag(payload[0])
|
||||||
|
decoded, n, err := DecodeAny(decoder, tag)
|
||||||
|
test.Log("n: ", n)
|
||||||
|
test.Log("tag: ", tag)
|
||||||
|
test.Log("got: ", tu.Describe(decoded))
|
||||||
|
test.Log("correct:", tu.Describe(correctValue))
|
||||||
|
if err != nil { test.Fatal(err) }
|
||||||
|
if !reflect.DeepEqual(decoded, correctValue) {
|
||||||
|
test.Fatal("values not equal")
|
||||||
|
}
|
||||||
|
if n != len(data) {
|
||||||
|
test.Fatalf("n not equal: %d != %d", n, len(data))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
11
tape/strings.go
Normal file
11
tape/strings.go
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
package tape
|
||||||
|
|
||||||
|
const lipsum = `Lorem ipsum dolor sit amet consectetur adipiscing elit. Quisque faucibus ex sapien vitae pellentesque sem placerat. In id cursus mi pretium tellus duis convallis. Tempus leo eu aenean sed diam urna tempor. Pulvinar vivamus fringilla lacus nec metus bibendum egestas. Iaculis massa nisl malesuada lacinia integer nunc posuere. Ut hendrerit semper vel class aptent taciti sociosqu. Ad litora torquent per conubia nostra inceptos himenaeos.
|
||||||
|
|
||||||
|
Lorem ipsum dolor sit amet consectetur adipiscing elit. Quisque faucibus ex sapien vitae pellentesque sem placerat. In id cursus mi pretium tellus duis convallis. Tempus leo eu aenean sed diam urna tempor. Pulvinar vivamus fringilla lacus nec metus bibendum egestas. Iaculis massa nisl malesuada lacinia integer nunc posuere. Ut hendrerit semper vel class aptent taciti sociosqu. Ad litora torquent per conubia nostra inceptos himenaeos.
|
||||||
|
|
||||||
|
Lorem ipsum dolor sit amet consectetur adipiscing elit. Quisque faucibus ex sapien vitae pellentesque sem placerat. In id cursus mi pretium tellus duis convallis. Tempus leo eu aenean sed diam urna tempor. Pulvinar vivamus fringilla lacus nec metus bibendum egestas. Iaculis massa nisl malesuada lacinia integer nunc posuere. Ut hendrerit semper vel class aptent taciti sociosqu. Ad litora torquent per conubia nostra inceptos himenaeos.
|
||||||
|
|
||||||
|
Lorem ipsum dolor sit amet consectetur adipiscing elit. Quisque faucibus ex sapien vitae pellentesque sem placerat. In id cursus mi pretium tellus duis convallis. Tempus leo eu aenean sed diam urna tempor. Pulvinar vivamus fringilla lacus nec metus bibendum egestas. Iaculis massa nisl malesuada lacinia integer nunc posuere. Ut hendrerit semper vel class aptent taciti sociosqu. Ad litora torquent per conubia nostra inceptos himenaeos.
|
||||||
|
|
||||||
|
Lorem ipsum dolor sit amet consectetur adipiscing elit. Quisque faucibus ex sapien vitae pellentesque sem placerat. In id cursus mi pretium tellus duis convallis. Tempus leo eu aenean sed diam urna tempor. Pulvinar vivamus fringilla lacus nec metus bibendum egestas. Iaculis massa nisl malesuada lacinia integer nunc posuere. Ut hendrerit semper vel class aptent taciti sociosqu. Ad litora torquent per conubia nostra inceptos himenaeos.`
|
||||||
13
tape/tag.go
13
tape/tag.go
@@ -18,6 +18,17 @@ type Tag byte; const (
|
|||||||
CNLimit Tag = 32 // All valid CNs are < CNLimit
|
CNLimit Tag = 32 // All valid CNs are < CNLimit
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// what the first nybble of a tag means:
|
||||||
|
//
|
||||||
|
// 0-1 : SI
|
||||||
|
// 2-3 : LI
|
||||||
|
// 4-5 : LSI
|
||||||
|
// 6-7 : FP
|
||||||
|
// 8-9 : SBA
|
||||||
|
// A-B : LBA
|
||||||
|
// C-D : OTA
|
||||||
|
// E-F : KTV
|
||||||
|
|
||||||
func (tag Tag) TN() int {
|
func (tag Tag) TN() int {
|
||||||
return int(tag >> 5)
|
return int(tag >> 5)
|
||||||
}
|
}
|
||||||
@@ -67,6 +78,6 @@ func bufferLenTag(length int) Tag {
|
|||||||
if length < int(CNLimit) {
|
if length < int(CNLimit) {
|
||||||
return SBA.WithCN(length)
|
return SBA.WithCN(length)
|
||||||
} else {
|
} else {
|
||||||
return LBA.WithCN(IntBytes(uint64(length)))
|
return LBA.WithCN(IntBytes(uint64(length)) - 1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user