packet.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489
  1. // Copyright 2019 Yunion
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package dhcp
  15. import (
  16. "encoding/binary"
  17. "errors"
  18. "fmt"
  19. "net"
  20. "time"
  21. )
  22. // Option is a DHCP option.
  23. type Option struct {
  24. Code OptionCode
  25. Value []byte
  26. }
  27. type OptionCode byte
  28. type OpCode byte
  29. type MessageType byte // Option 53
  30. // A DHCP packet
  31. type Packet []byte
  32. func (p Packet) OpCode() OpCode { return OpCode(p[0]) }
  33. func (p Packet) HType() byte { return p[1] }
  34. func (p Packet) HLen() byte { return p[2] }
  35. func (p Packet) Hops() byte { return p[3] }
  36. func (p Packet) XId() []byte { return p[4:8] }
  37. func (p Packet) TransactionID() string { return string(p.XId()) }
  38. func (p Packet) Secs() []byte { return p[8:10] } // Never Used?
  39. func (p Packet) Flags() []byte { return p[10:12] }
  40. func (p Packet) CIAddr() net.IP { return net.IP(p[12:16]) } // Client's current IP address (it will respond to ARP for this IP)
  41. func (p Packet) YIAddr() net.IP { return net.IP(p[16:20]) } // Client IP address offered/assigned by server
  42. func (p Packet) SIAddr() net.IP { return net.IP(p[20:24]) } // Responding server's IP address
  43. func (p Packet) GIAddr() net.IP { return net.IP(p[24:28]) } // IP address of DHCP relay agent, if an agent forwarded the request
  44. func (p Packet) CHAddr() net.HardwareAddr {
  45. hLen := p.HLen()
  46. if hLen > 16 { // Prevent chaddr exceeding p boundary
  47. hLen = 16
  48. }
  49. return net.HardwareAddr(p[28 : 28+hLen]) // max endPos 44
  50. }
  51. func (p Packet) RelayAddr() net.IP {
  52. return p.GIAddr()
  53. }
  54. func Unmarshal(b []byte) Packet {
  55. p := Packet(b)
  56. return p
  57. }
  58. func (p Packet) Marshal() []byte {
  59. return p
  60. }
  61. func (p Packet) Type() MessageType {
  62. return MessageType(p.ParseOptions()[OptionDHCPMessageType][0])
  63. }
  64. // 192 bytes of zeros BOOTP legacy
  65. // BOOTP legacy
  66. func (p Packet) SName() []byte { return trimNull(p[44:108]) }
  67. // BOOTP legacy
  68. func (p Packet) File() []byte { return trimNull(p[108:236]) }
  69. func trimNull(d []byte) []byte {
  70. for i, v := range d {
  71. if v == 0 {
  72. return d[:i]
  73. }
  74. }
  75. return d
  76. }
  77. func (p Packet) Cookie() []byte { return p[236:240] }
  78. func (p Packet) Options() []byte {
  79. if len(p) > 240 {
  80. return p[240:]
  81. }
  82. return nil
  83. }
  84. func (p Packet) Broadcast() bool { return p.Flags()[0] > 127 }
  85. func (p Packet) SetBroadcast(broadcast bool) {
  86. if p.Broadcast() != broadcast {
  87. p.Flags()[0] ^= 128
  88. }
  89. }
  90. func (p Packet) SetOpCode(c OpCode) { p[0] = byte(c) }
  91. func (p Packet) SetCHAddr(a net.HardwareAddr) {
  92. copy(p[28:44], a)
  93. p[2] = byte(len(a))
  94. }
  95. func (p Packet) SetHType(hType byte) { p[1] = hType }
  96. func (p Packet) SetCookie(cookie []byte) { copy(p.Cookie(), cookie) }
  97. func (p Packet) SetHops(hops byte) { p[3] = hops }
  98. func (p Packet) SetXId(xId []byte) { copy(p.XId(), xId) }
  99. func (p Packet) SetSecs(secs []byte) { copy(p.Secs(), secs) }
  100. func (p Packet) SetFlags(flags []byte) { copy(p.Flags(), flags) }
  101. func (p Packet) SetCIAddr(ip net.IP) { copy(p.CIAddr(), ip.To4()) }
  102. func (p Packet) SetYIAddr(ip net.IP) { copy(p.YIAddr(), ip.To4()) }
  103. func (p Packet) SetSIAddr(ip net.IP) { copy(p.SIAddr(), ip.To4()) }
  104. func (p Packet) SetGIAddr(ip net.IP) { copy(p.GIAddr(), ip.To4()) }
  105. // BOOTP legacy
  106. func (p Packet) SetSName(sName []byte) {
  107. copy(p[44:108], sName)
  108. if len(sName) < 64 {
  109. p[44+len(sName)] = 0
  110. }
  111. }
  112. // BOOTP legacy
  113. func (p Packet) SetFile(file []byte) {
  114. copy(p[108:236], file)
  115. if len(file) < 128 {
  116. p[108+len(file)] = 0
  117. }
  118. }
  119. func (p Packet) GetOptionValue(code OptionCode) []byte {
  120. return p.ParseOptions()[code]
  121. }
  122. // Map of DHCP options
  123. type Options map[OptionCode][]byte
  124. var (
  125. ErrOptionWrongSize = errors.New("option value is the wrong size")
  126. )
  127. func (o Options) Bytes(code OptionCode) ([]byte, error) {
  128. bs := o[code]
  129. if bs == nil {
  130. return nil, fmt.Errorf("option %d not found", code)
  131. }
  132. return bs, nil
  133. }
  134. func (o Options) String(n OptionCode) (string, error) {
  135. bs, err := o.Bytes(n)
  136. if err != nil {
  137. return "", err
  138. }
  139. return string(bs), err
  140. }
  141. func (o Options) Byte(n OptionCode) (byte, error) {
  142. bs, err := o.Bytes(n)
  143. if err != nil {
  144. return 0, err
  145. }
  146. if len(bs) != 1 {
  147. return 0, fmt.Errorf("option value %d is the wrong size", n)
  148. }
  149. return bs[0], nil
  150. }
  151. func (o Options) Uint16(n OptionCode) (uint16, error) {
  152. bs, err := o.Bytes(n)
  153. if err != nil {
  154. return 0, err
  155. }
  156. if len(bs) != 2 {
  157. return 0, ErrOptionWrongSize
  158. }
  159. return binary.BigEndian.Uint16(bs), nil
  160. }
  161. func (o Options) Uint32(n OptionCode) (uint32, error) {
  162. bs, err := o.Bytes(n)
  163. if err != nil {
  164. return 0, err
  165. }
  166. if len(bs) != 4 {
  167. return 0, ErrOptionWrongSize
  168. }
  169. return binary.BigEndian.Uint32(bs), nil
  170. }
  171. func (o Options) IPs(n OptionCode) ([]net.IP, error) {
  172. bs, err := o.Bytes(n)
  173. if err != nil {
  174. return nil, err
  175. }
  176. if len(bs) < 4 || len(bs)%4 != 0 {
  177. return nil, ErrOptionWrongSize
  178. }
  179. ret := make([]net.IP, 0, len(bs)/4)
  180. for i := 0; i < len(bs); i += 4 {
  181. ret = append(ret, net.IP(bs[i:i+4]))
  182. }
  183. return ret, nil
  184. }
  185. func (o Options) IP(n OptionCode) (net.IP, error) {
  186. ips, err := o.IPs(n)
  187. if err != nil {
  188. return nil, err
  189. }
  190. if len(ips) != 1 {
  191. return nil, ErrOptionWrongSize
  192. }
  193. return ips[0], nil
  194. }
  195. func (o Options) IPMask(n OptionCode) (net.IPMask, error) {
  196. bs := o[n]
  197. if bs == nil {
  198. return nil, fmt.Errorf("option %d not found", n)
  199. }
  200. if len(bs) != 4 {
  201. return nil, fmt.Errorf("option %d is the wrong size for an IPMask", n)
  202. }
  203. return net.IPMask(bs), nil
  204. }
  205. // Parses the packet's options into an Options map
  206. func (p Packet) ParseOptions() Options {
  207. opts := p.Options()
  208. options := make(Options, 10)
  209. for len(opts) >= 2 && OptionCode(opts[0]) != End {
  210. if OptionCode(opts[0]) == Pad {
  211. opts = opts[1:]
  212. continue
  213. }
  214. size := int(opts[1])
  215. if len(opts) < 2+size {
  216. break
  217. }
  218. options[OptionCode(opts[0])] = opts[2 : 2+size]
  219. opts = opts[2+size:]
  220. }
  221. return options
  222. }
  223. func NewPacket(opCode OpCode) Packet {
  224. var magic = []byte{99, 130, 83, 99}
  225. p := make(Packet, 241)
  226. p.SetOpCode(opCode)
  227. p.SetHType(1) // Ethernet
  228. p.SetCookie(magic)
  229. p[240] = byte(End)
  230. return p
  231. }
  232. // Appends a DHCP option to the end of a packet
  233. func (p *Packet) AddOption(o OptionCode, value []byte) {
  234. *p = append((*p)[:len(*p)-1], []byte{byte(o), byte(len(value))}...) // Strip off End, Add OptionCode and Length
  235. *p = append(*p, value...) // Add Option Value
  236. *p = append(*p, byte(End)) // Add on new End
  237. }
  238. // Removes all options from packet.
  239. func (p *Packet) StripOptions() {
  240. *p = append((*p)[:240], byte(End))
  241. }
  242. // Creates a request packet that a Client would send to a server.
  243. func RequestPacket(mt MessageType, chAddr net.HardwareAddr, cIAddr net.IP, xId []byte, broadcast bool, options []Option) Packet {
  244. p := NewPacket(BootRequest)
  245. p.SetCHAddr(chAddr)
  246. p.SetXId(xId)
  247. if cIAddr != nil {
  248. p.SetCIAddr(cIAddr)
  249. }
  250. p.SetBroadcast(broadcast)
  251. p.AddOption(OptionDHCPMessageType, []byte{byte(mt)})
  252. for _, o := range options {
  253. p.AddOption(o.Code, o.Value)
  254. }
  255. p.PadToMinSize()
  256. return p
  257. }
  258. // ReplyPacket creates a reply packet that a Server would send to a client.
  259. // It uses the req Packet param to copy across common/necessary fields to
  260. // associate the reply the request.
  261. func ReplyPacket(req Packet, mt MessageType, serverId, yIAddr net.IP, leaseDuration time.Duration, options []Option) Packet {
  262. p := NewPacket(BootReply)
  263. p.SetXId(req.XId())
  264. p.SetFlags(req.Flags())
  265. p.SetYIAddr(yIAddr)
  266. p.SetGIAddr(req.GIAddr())
  267. p.SetCHAddr(req.CHAddr())
  268. p.AddOption(OptionDHCPMessageType, []byte{byte(mt)})
  269. p.AddOption(OptionServerIdentifier, []byte(serverId.To4()))
  270. if leaseDuration > 0 {
  271. p.AddOption(OptionIPAddressLeaseTime, GetOptTime(leaseDuration))
  272. }
  273. for _, o := range options {
  274. p.AddOption(o.Code, o.Value)
  275. }
  276. p.PadToMinSize()
  277. return p
  278. }
  279. // PadToMinSize pads a packet so that when sent over UDP, the entire packet,
  280. // is 300 bytes (BOOTP min), to be compatible with really old devices.
  281. var padder [272]byte
  282. func (p *Packet) PadToMinSize() {
  283. if n := len(*p); n < 272 {
  284. *p = append(*p, padder[:272-n]...)
  285. }
  286. }
  287. // decides how to send Packet on the wire, based on its field values.
  288. //
  289. // This implements the transmission decision process in section 4.1 of
  290. // RFC 2131.
  291. func (p *Packet) txType() txType {
  292. switch {
  293. case p.GIAddr() != nil && p.GIAddr().IsGlobalUnicast():
  294. return txRelayAddr
  295. case p.Type() == NAK:
  296. return txBroadcast
  297. case p.CIAddr() != nil && (p.CIAddr().IsGlobalUnicast() || p.CIAddr().IsLoopback()):
  298. return txClientAddr
  299. case p.Broadcast():
  300. return txBroadcast
  301. default:
  302. return txHardwareAddr
  303. }
  304. }
  305. //go:generate stringer -type=OpCode
  306. // OpCodes
  307. const (
  308. BootRequest OpCode = 1 // From Client
  309. BootReply OpCode = 2 // From Server
  310. )
  311. //go:generate stringer -type=MessageType
  312. // DHCP Message Type 53
  313. const (
  314. Discover MessageType = 1 // Broadcast Packet From Client - Can I have an IP?
  315. Offer MessageType = 2 // Broadcast From Server - Here's an IP
  316. Request MessageType = 3 // Broadcast From Client - I'll take that IP (Also start for renewals)
  317. Decline MessageType = 4 // Broadcast From Client - Sorry I can't use that IP
  318. ACK MessageType = 5 // From Server, Yes you can have that IP
  319. NAK MessageType = 6 // From Server, No you cannot have that IP
  320. Release MessageType = 7 // From Client, I don't need that IP anymore
  321. Inform MessageType = 8 // From Client, I have this IP and there's nothing you can do about it
  322. )
  323. //go:generate stringer -type=OptionCode
  324. // DHCP Options
  325. const (
  326. End OptionCode = 255
  327. Pad OptionCode = 0
  328. OptionSubnetMask OptionCode = 1
  329. OptionTimeOffset OptionCode = 2
  330. OptionRouter OptionCode = 3
  331. OptionTimeServer OptionCode = 4
  332. OptionNameServer OptionCode = 5
  333. OptionDomainNameServer OptionCode = 6
  334. OptionLogServer OptionCode = 7
  335. OptionCookieServer OptionCode = 8
  336. OptionLPRServer OptionCode = 9
  337. OptionImpressServer OptionCode = 10
  338. OptionResourceLocationServer OptionCode = 11
  339. OptionHostName OptionCode = 12
  340. OptionBootFileSize OptionCode = 13
  341. OptionMeritDumpFile OptionCode = 14
  342. OptionDomainName OptionCode = 15
  343. OptionSwapServer OptionCode = 16
  344. OptionRootPath OptionCode = 17
  345. OptionExtensionsPath OptionCode = 18
  346. // IP Layer Parameters per Host
  347. OptionIPForwardingEnableDisable OptionCode = 19
  348. OptionNonLocalSourceRoutingEnableDisable OptionCode = 20
  349. OptionPolicyFilter OptionCode = 21
  350. OptionMaximumDatagramReassemblySize OptionCode = 22
  351. OptionDefaultIPTimeToLive OptionCode = 23
  352. OptionPathMTUAgingTimeout OptionCode = 24
  353. OptionPathMTUPlateauTable OptionCode = 25
  354. // IP Layer Parameters per Interface
  355. OptionInterfaceMTU OptionCode = 26
  356. OptionAllSubnetsAreLocal OptionCode = 27
  357. OptionBroadcastAddress OptionCode = 28
  358. OptionPerformMaskDiscovery OptionCode = 29
  359. OptionMaskSupplier OptionCode = 30
  360. OptionPerformRouterDiscovery OptionCode = 31
  361. OptionRouterSolicitationAddress OptionCode = 32
  362. OptionStaticRoute OptionCode = 33
  363. // Link Layer Parameters per Interface
  364. OptionTrailerEncapsulation OptionCode = 34
  365. OptionARPCacheTimeout OptionCode = 35
  366. OptionEthernetEncapsulation OptionCode = 36
  367. // TCP Parameters
  368. OptionTCPDefaultTTL OptionCode = 37
  369. OptionTCPKeepaliveInterval OptionCode = 38
  370. OptionTCPKeepaliveGarbage OptionCode = 39
  371. // Application and Service Parameters
  372. OptionNetworkInformationServiceDomain OptionCode = 40
  373. OptionNetworkInformationServers OptionCode = 41
  374. OptionNetworkTimeProtocolServers OptionCode = 42
  375. OptionVendorSpecificInformation OptionCode = 43
  376. OptionNetBIOSOverTCPIPNameServer OptionCode = 44
  377. OptionNetBIOSOverTCPIPDatagramDistributionServer OptionCode = 45
  378. OptionNetBIOSOverTCPIPNodeType OptionCode = 46
  379. OptionNetBIOSOverTCPIPScope OptionCode = 47
  380. OptionXWindowSystemFontServer OptionCode = 48
  381. OptionXWindowSystemDisplayManager OptionCode = 49
  382. OptionNetworkInformationServicePlusDomain OptionCode = 64
  383. OptionNetworkInformationServicePlusServers OptionCode = 65
  384. OptionMobileIPHomeAgent OptionCode = 68
  385. OptionSimpleMailTransportProtocol OptionCode = 69
  386. OptionPostOfficeProtocolServer OptionCode = 70
  387. OptionNetworkNewsTransportProtocol OptionCode = 71
  388. OptionDefaultWorldWideWebServer OptionCode = 72
  389. OptionDefaultFingerServer OptionCode = 73
  390. OptionDefaultInternetRelayChatServer OptionCode = 74
  391. OptionStreetTalkServer OptionCode = 75
  392. OptionStreetTalkDirectoryAssistance OptionCode = 76
  393. OptionRelayAgentInformation OptionCode = 82
  394. // DHCP Extensions
  395. OptionRequestedIPAddress OptionCode = 50
  396. OptionIPAddressLeaseTime OptionCode = 51
  397. OptionOverload OptionCode = 52
  398. OptionDHCPMessageType OptionCode = 53
  399. OptionServerIdentifier OptionCode = 54
  400. OptionParameterRequestList OptionCode = 55
  401. OptionMessage OptionCode = 56
  402. OptionMaximumDHCPMessageSize OptionCode = 57
  403. OptionRenewalTimeValue OptionCode = 58
  404. OptionRebindingTimeValue OptionCode = 59
  405. OptionVendorClassIdentifier OptionCode = 60
  406. OptionClientIdentifier OptionCode = 61
  407. OptionTFTPServerName OptionCode = 66
  408. OptionBootFileName OptionCode = 67
  409. OptionUserClass OptionCode = 77
  410. OptionClientArchitecture OptionCode = 93
  411. OptionClientNetworkInterfaceIdentifier OptionCode = 94
  412. OptionClientMachineIdentifier OptionCode = 97
  413. OptionTZPOSIXString OptionCode = 100
  414. OptionTZDatabaseString OptionCode = 101
  415. // https://datatracker.ietf.org/doc/html/rfc8925
  416. OptionIPv6Only OptionCode = 108
  417. OptionDomainSearch OptionCode = 119
  418. OptionClasslessRouteFormat OptionCode = 121
  419. // From RFC3942 - Options Used by PXELINUX
  420. OptionPxelinuxMagic OptionCode = 208
  421. OptionPxelinuxConfigfile OptionCode = 209
  422. OptionPxelinuxPathprefix OptionCode = 210
  423. OptionPxelinuxReboottime OptionCode = 211
  424. )
  425. /* Notes
  426. A DHCP server always returns its own address in the 'server identifier' option.
  427. DHCP defines a new 'client identifier' option that is used to pass an explicit client identifier to a DHCP server.
  428. */