summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
-rw-r--r--dhcpv6/dhcpv6.go10
-rw-r--r--dhcpv6/dhcpv6_test.go8
-rw-r--r--dhcpv6/dhcpv6message.go24
-rw-r--r--dhcpv6/dhcpv6relay_test.go8
-rw-r--r--dhcpv6/modifiers.go3
-rw-r--r--dhcpv6/modifiers_test.go12
-rw-r--r--dhcpv6/option_archtype.go41
-rw-r--r--dhcpv6/option_archtype_test.go14
-rw-r--r--dhcpv6/option_relaymsg_test.go6
-rw-r--r--dhcpv6/options.go2
-rw-r--r--iana/archtype.go10
11 files changed, 71 insertions, 67 deletions
diff --git a/dhcpv6/dhcpv6.go b/dhcpv6/dhcpv6.go
index 4f7fbbb..41b4726 100644
--- a/dhcpv6/dhcpv6.go
+++ b/dhcpv6/dhcpv6.go
@@ -204,13 +204,9 @@ func IsUsingUEFI(msg *Message) bool {
// 7 EFI BC
// 8 EFI Xscale
// 9 EFI x86-64
- if opt := msg.GetOneOption(OptionClientArchType); opt != nil {
- optat := opt.(*OptClientArchType)
- for _, at := range optat.ArchTypes {
- // TODO investigate if other types are appropriate
- if at == iana.EFI_BC || at == iana.EFI_X86_64 {
- return true
- }
+ if archTypes := msg.Options.ArchTypes(); archTypes != nil {
+ if archTypes.Contains(iana.EFI_BC) || archTypes.Contains(iana.EFI_X86_64) {
+ return true
}
}
if opt := msg.GetOneOption(OptionUserClass); opt != nil {
diff --git a/dhcpv6/dhcpv6_test.go b/dhcpv6/dhcpv6_test.go
index d097095..b23a858 100644
--- a/dhcpv6/dhcpv6_test.go
+++ b/dhcpv6/dhcpv6_test.go
@@ -114,7 +114,7 @@ func TestAddOption(t *testing.T) {
require.Empty(t, d.Options)
opt := OptionGeneric{OptionCode: 0, OptionData: []byte{}}
d.AddOption(&opt)
- require.Equal(t, Options{&opt}, d.Options)
+ require.Equal(t, Options{&opt}, d.Options.Options)
}
func TestToBytes(t *testing.T) {
@@ -269,15 +269,13 @@ func TestNewMessageTypeSolicit(t *testing.T) {
func TestIsUsingUEFIArchTypeTrue(t *testing.T) {
msg := Message{}
- opt := OptClientArchType{ArchTypes: []iana.Arch{iana.EFI_BC}}
- msg.AddOption(&opt)
+ msg.AddOption(OptClientArchType(iana.EFI_BC))
require.True(t, IsUsingUEFI(&msg))
}
func TestIsUsingUEFIArchTypeFalse(t *testing.T) {
msg := Message{}
- opt := OptClientArchType{ArchTypes: []iana.Arch{iana.INTEL_X86PC}}
- msg.AddOption(&opt)
+ msg.AddOption(OptClientArchType(iana.INTEL_X86PC))
require.False(t, IsUsingUEFI(&msg))
}
diff --git a/dhcpv6/dhcpv6message.go b/dhcpv6/dhcpv6message.go
index f4f5eb1..d67882a 100644
--- a/dhcpv6/dhcpv6message.go
+++ b/dhcpv6/dhcpv6message.go
@@ -13,11 +13,27 @@ import (
const MessageHeaderSize = 4
+// MessageOptions are the options that may appear in a normal DHCPv6 message.
+//
+// RFC 3315 Appendix B lists the valid options that can be used.
+type MessageOptions struct {
+ Options
+}
+
+// ArchTypes returns the architecture type option.
+func (mo MessageOptions) ArchTypes() iana.Archs {
+ opt := mo.GetOne(OptionClientArchType)
+ if opt == nil {
+ return nil
+ }
+ return opt.(*optClientArchType).Archs
+}
+
// Message represents a DHCPv6 Message as defined by RFC 3315 Section 6.
type Message struct {
MessageType MessageType
TransactionID TransactionID
- Options Options
+ Options MessageOptions
}
var randomRead = rand.Read
@@ -250,7 +266,7 @@ func (m *Message) IsOptionRequested(requested OptionCode) bool {
// String returns a short human-readable string for this message.
func (m *Message) String() string {
return fmt.Sprintf("Message(messageType=%s transactionID=%s, %d options)",
- m.MessageType, m.TransactionID, len(m.Options))
+ m.MessageType, m.TransactionID, len(m.Options.Options))
}
// Summary prints all options associated with this message.
@@ -263,10 +279,10 @@ func (m *Message) Summary() string {
m.TransactionID,
)
ret += " options=["
- if len(m.Options) > 0 {
+ if len(m.Options.Options) > 0 {
ret += "\n"
}
- for _, opt := range m.Options {
+ for _, opt := range m.Options.Options {
ret += fmt.Sprintf(" %v\n", opt.String())
}
ret += " ]\n"
diff --git a/dhcpv6/dhcpv6relay_test.go b/dhcpv6/dhcpv6relay_test.go
index 3e7853f..4070f54 100644
--- a/dhcpv6/dhcpv6relay_test.go
+++ b/dhcpv6/dhcpv6relay_test.go
@@ -63,9 +63,11 @@ func TestRelayMessageToBytes(t *testing.T) {
relayMessage: &Message{
MessageType: MessageTypeSolicit,
TransactionID: TransactionID{0xaa, 0xbb, 0xcc},
- Options: []Option{
- &OptElapsedTime{
- ElapsedTime: 0,
+ Options: MessageOptions{
+ Options: []Option{
+ &OptElapsedTime{
+ ElapsedTime: 0,
+ },
},
},
},
diff --git a/dhcpv6/modifiers.go b/dhcpv6/modifiers.go
index 06eb9d9..d5e24c0 100644
--- a/dhcpv6/modifiers.go
+++ b/dhcpv6/modifiers.go
@@ -63,8 +63,7 @@ func WithUserClass(uc []byte) Modifier {
// WithArchType adds an arch type option to the packet
func WithArchType(at iana.Arch) Modifier {
return func(d DHCPv6) {
- ao := OptClientArchType{ArchTypes: []iana.Arch{at}}
- d.AddOption(&ao)
+ d.AddOption(OptClientArchType(at))
}
}
diff --git a/dhcpv6/modifiers_test.go b/dhcpv6/modifiers_test.go
index dfef08c..ec8ff0b 100644
--- a/dhcpv6/modifiers_test.go
+++ b/dhcpv6/modifiers_test.go
@@ -60,8 +60,8 @@ func TestWithIANA(t *testing.T) {
PreferredLifetime: 3600,
ValidLifetime: 5200,
})(&d)
- require.Equal(t, 1, len(d.Options))
- require.Equal(t, OptionIANA, d.Options[0].Code())
+ require.Equal(t, 1, len(d.Options.Options))
+ require.Equal(t, OptionIANA, d.Options.Options[0].Code())
}
func TestWithDNS(t *testing.T) {
@@ -70,8 +70,8 @@ func TestWithDNS(t *testing.T) {
net.ParseIP("fe80::1"),
net.ParseIP("fe80::2"),
}...)(&d)
- require.Equal(t, 1, len(d.Options))
- dns := d.Options[0].(*OptDNSRecursiveNameServer)
+ require.Equal(t, 1, len(d.Options.Options))
+ dns := d.Options.Options[0].(*OptDNSRecursiveNameServer)
log.Printf("DNS %+v", dns)
require.Equal(t, OptionDNSRecursiveNameServer, dns.Code())
require.Equal(t, 2, len(dns.NameServers))
@@ -86,8 +86,8 @@ func TestWithDomainSearchList(t *testing.T) {
"slackware.it",
"dhcp.slackware.it",
}...)(&d)
- require.Equal(t, 1, len(d.Options))
- osl := d.Options[0].(*OptDomainSearchList)
+ require.Equal(t, 1, len(d.Options.Options))
+ osl := d.Options.Options[0].(*OptDomainSearchList)
require.Equal(t, OptionDomainSearchList, osl.Code())
require.NotNil(t, osl.DomainSearchList)
labels := osl.DomainSearchList.Labels
diff --git a/dhcpv6/option_archtype.go b/dhcpv6/option_archtype.go
index 83ebf0e..1461056 100644
--- a/dhcpv6/option_archtype.go
+++ b/dhcpv6/option_archtype.go
@@ -2,49 +2,34 @@ package dhcpv6
import (
"fmt"
- "strings"
"github.com/insomniacslk/dhcp/iana"
- "github.com/u-root/u-root/pkg/uio"
)
-// OptClientArchType represents an option CLIENT_ARCH_TYPE
+// OptClientArchType represents an option CLIENT_ARCH_TYPE.
//
// This module defines the OptClientArchType structure.
// https://www.ietf.org/rfc/rfc5970.txt
-type OptClientArchType struct {
- ArchTypes []iana.Arch
+func OptClientArchType(a ...iana.Arch) Option {
+ return &optClientArchType{Archs: a}
}
-func (op *OptClientArchType) Code() OptionCode {
- return OptionClientArchType
+type optClientArchType struct {
+ iana.Archs
}
-// ToBytes marshals the client arch type as defined by RFC 5970.
-func (op *OptClientArchType) ToBytes() []byte {
- buf := uio.NewBigEndianBuffer(nil)
- for _, at := range op.ArchTypes {
- buf.Write16(uint16(at))
- }
- return buf.Data()
+func (op *optClientArchType) Code() OptionCode {
+ return OptionClientArchType
}
-func (op *OptClientArchType) String() string {
- atStrings := make([]string, 0)
- for _, at := range op.ArchTypes {
- atStrings = append(atStrings, at.String())
- }
- return fmt.Sprintf("OptClientArchType{archtype=%v}", strings.Join(atStrings, ", "))
+func (op optClientArchType) String() string {
+ return fmt.Sprintf("ClientArchType: %s", op.Archs.String())
}
-// ParseOptClientArchType builds an OptClientArchType structure from
+// parseOptClientArchType builds an OptClientArchType structure from
// a sequence of bytes The input data does not include option code and
// length bytes.
-func ParseOptClientArchType(data []byte) (*OptClientArchType, error) {
- var opt OptClientArchType
- buf := uio.NewBigEndianBuffer(data)
- for buf.Has(2) {
- opt.ArchTypes = append(opt.ArchTypes, iana.Arch(buf.Read16()))
- }
- return &opt, buf.FinError()
+func parseOptClientArchType(data []byte) (*optClientArchType, error) {
+ var opt optClientArchType
+ return &opt, opt.FromBytes(data)
}
diff --git a/dhcpv6/option_archtype_test.go b/dhcpv6/option_archtype_test.go
index b01db08..0481c1a 100644
--- a/dhcpv6/option_archtype_test.go
+++ b/dhcpv6/option_archtype_test.go
@@ -11,14 +11,14 @@ func TestParseOptClientArchType(t *testing.T) {
data := []byte{
0, 6, // EFI_IA32
}
- opt, err := ParseOptClientArchType(data)
+ opt, err := parseOptClientArchType(data)
require.NoError(t, err)
- require.Equal(t, iana.EFI_IA32, opt.ArchTypes[0])
+ require.Equal(t, iana.EFI_IA32, opt.Archs[0])
}
func TestParseOptClientArchTypeInvalid(t *testing.T) {
data := []byte{42}
- _, err := ParseOptClientArchType(data)
+ _, err := parseOptClientArchType(data)
require.Error(t, err)
}
@@ -26,15 +26,13 @@ func TestOptClientArchTypeParseAndToBytes(t *testing.T) {
data := []byte{
0, 8, // EFI_XSCALE
}
- opt, err := ParseOptClientArchType(data)
+ opt, err := parseOptClientArchType(data)
require.NoError(t, err)
require.Equal(t, data, opt.ToBytes())
}
func TestOptClientArchType(t *testing.T) {
- opt := OptClientArchType{
- ArchTypes: []iana.Arch{iana.EFI_ITANIUM},
- }
+ opt := OptClientArchType(iana.EFI_ITANIUM)
require.Equal(t, OptionClientArchType, opt.Code())
- require.Contains(t, opt.String(), "archtype=EFI Itanium", "String() should contain the correct ArchType output")
+ require.Contains(t, opt.String(), "EFI Itanium", "String() should contain the correct ArchType output")
}
diff --git a/dhcpv6/option_relaymsg_test.go b/dhcpv6/option_relaymsg_test.go
index dd82d28..887bf95 100644
--- a/dhcpv6/option_relaymsg_test.go
+++ b/dhcpv6/option_relaymsg_test.go
@@ -106,10 +106,10 @@ func TestRelayMsgParseOptRelayMsgSingleEncapsulation(t *testing.T) {
if tID := innerDHCP.TransactionID; tID != xid {
t.Fatalf("Invalid inner DHCP transaction ID. Expected 0xaabbcc, got %v", tID)
}
- if len(innerDHCP.Options) != 1 {
- t.Fatalf("Invalid inner DHCP options length. Expected 1, got %v", len(innerDHCP.Options))
+ if len(innerDHCP.Options.Options) != 1 {
+ t.Fatalf("Invalid inner DHCP options length. Expected 1, got %v", len(innerDHCP.Options.Options))
}
- innerOpt := innerDHCP.Options[0]
+ innerOpt := innerDHCP.Options.Options[0]
eto, ok := innerOpt.(*OptElapsedTime)
if !ok {
t.Fatalf("Invalid inner option type. Expected OptElapsedTime, got %v",
diff --git a/dhcpv6/options.go b/dhcpv6/options.go
index e653c01..bf37600 100644
--- a/dhcpv6/options.go
+++ b/dhcpv6/options.go
@@ -80,7 +80,7 @@ func ParseOption(code OptionCode, optData []byte) (Option, error) {
case OptionBootfileParam:
opt, err = ParseOptBootFileParam(optData)
case OptionClientArchType:
- opt, err = ParseOptClientArchType(optData)
+ opt, err = parseOptClientArchType(optData)
case OptionNII:
opt, err = ParseOptNetworkInterfaceId(optData)
case Option4RD:
diff --git a/iana/archtype.go b/iana/archtype.go
index 255687c..865659b 100644
--- a/iana/archtype.go
+++ b/iana/archtype.go
@@ -49,6 +49,16 @@ func (a Arch) String() string {
// Archs represents multiple Arch values.
type Archs []Arch
+// Contains returns whether b is one of the Archs in a.
+func (a Archs) Contains(b Arch) bool {
+ for _, t := range a {
+ if t == b {
+ return true
+ }
+ }
+ return false
+}
+
// ToBytes returns the serialized option defined by RFC 4578 (DHCPv4) and RFC
// 5970 (DHCPv6) as the Client System Architecture Option.
func (a Archs) ToBytes() []byte {