mirror of
https://github.com/status-im/keycard-go.git
synced 2025-01-09 11:32:34 +00:00
132 lines
2.8 KiB
Go
132 lines
2.8 KiB
Go
package apdu
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/binary"
|
|
"errors"
|
|
)
|
|
|
|
// ErrBadRawCommand is an error returned by ParseCommand in case the command data is not long enough.
|
|
var ErrBadRawCommand = errors.New("command must be at least 4 bytes")
|
|
|
|
// Command struct represent the data sent as an APDU command with CLA, Ins, P1, P2, Lc, Data, and Le.
|
|
type Command struct {
|
|
Cla uint8
|
|
Ins uint8
|
|
P1 uint8
|
|
P2 uint8
|
|
Data []byte
|
|
le uint8
|
|
requiresLe bool
|
|
}
|
|
|
|
// NewCommand returns a new apdu Command.
|
|
func NewCommand(cla, ins, p1, p2 uint8, data []byte) *Command {
|
|
return &Command{
|
|
Cla: cla,
|
|
Ins: ins,
|
|
P1: p1,
|
|
P2: p2,
|
|
Data: data,
|
|
requiresLe: false,
|
|
}
|
|
}
|
|
|
|
// SetLe sets the expected Le value and makes sure the Le value is sent in the apdu Command.
|
|
func (c *Command) SetLe(le uint8) {
|
|
c.requiresLe = true
|
|
c.le = le
|
|
}
|
|
|
|
// Le returns if Le is set and its value.
|
|
func (c *Command) Le() (bool, uint8) {
|
|
return c.requiresLe, c.le
|
|
}
|
|
|
|
// Serialize serielizes the command into a raw bytes sequence.
|
|
func (c *Command) Serialize() ([]byte, error) {
|
|
buf := new(bytes.Buffer)
|
|
|
|
if err := binary.Write(buf, binary.BigEndian, c.Cla); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := binary.Write(buf, binary.BigEndian, c.Ins); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := binary.Write(buf, binary.BigEndian, c.P1); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := binary.Write(buf, binary.BigEndian, c.P2); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(c.Data) > 0 {
|
|
if err := binary.Write(buf, binary.BigEndian, uint8(len(c.Data))); err != nil {
|
|
return nil, err
|
|
}
|
|
if err := binary.Write(buf, binary.BigEndian, c.Data); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
if c.requiresLe {
|
|
if err := binary.Write(buf, binary.BigEndian, c.le); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return buf.Bytes(), nil
|
|
}
|
|
|
|
func (c *Command) deserialize(data []byte) error {
|
|
if len(data) < 4 {
|
|
return ErrBadRawCommand
|
|
}
|
|
|
|
buf := bytes.NewReader(data)
|
|
|
|
if err := binary.Read(buf, binary.BigEndian, &c.Cla); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := binary.Read(buf, binary.BigEndian, &c.Ins); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := binary.Read(buf, binary.BigEndian, &c.P1); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := binary.Read(buf, binary.BigEndian, &c.P2); err != nil {
|
|
return err
|
|
}
|
|
|
|
var lc uint8
|
|
if err := binary.Read(buf, binary.BigEndian, &lc); err != nil {
|
|
return nil
|
|
}
|
|
|
|
cmdData := make([]byte, lc)
|
|
if err := binary.Read(buf, binary.BigEndian, &cmdData); err != nil {
|
|
return nil
|
|
}
|
|
c.Data = cmdData
|
|
|
|
var le uint8
|
|
if err := binary.Read(buf, binary.BigEndian, &le); err != nil {
|
|
return nil
|
|
}
|
|
c.SetLe(le)
|
|
|
|
return nil
|
|
}
|
|
|
|
// ParseCommand parses a raw command and returns a Command
|
|
func ParseCommand(raw []byte) (*Command, error) {
|
|
cmd := &Command{}
|
|
return cmd, cmd.deserialize(raw)
|
|
}
|