forked from extern/smegmesh
127 lines
2.4 KiB
Go
127 lines
2.4 KiB
Go
package cga
|
|
|
|
/*
|
|
* Use a WireGuard public key to generate a unique interface ID
|
|
*/
|
|
|
|
import (
|
|
"crypto/rand"
|
|
"crypto/sha1"
|
|
"net"
|
|
|
|
"golang.zx2c4.com/wireguard/wgctrl/wgtypes"
|
|
)
|
|
|
|
const (
|
|
ModifierLength = 16
|
|
ZeroLength = 9
|
|
hash2Length = 57
|
|
hash1Length = 58
|
|
Hash2Prefix = 14
|
|
Hash1Prefix = 8
|
|
InterfaceIdLen = 8
|
|
SubnetPrefixLen = 8
|
|
)
|
|
|
|
/*
|
|
* Cga parameters used to generate an IPV6 interface ID
|
|
*/
|
|
type CgaParameters struct {
|
|
Modifier [ModifierLength]byte
|
|
SubnetPrefix [SubnetPrefixLen]byte
|
|
CollisionCount uint8
|
|
PublicKey wgtypes.Key
|
|
interfaceId [2 * InterfaceIdLen]byte
|
|
flag byte
|
|
}
|
|
|
|
func NewCga(key wgtypes.Key, subnetPrefix [SubnetPrefixLen]byte) (*CgaParameters, error) {
|
|
var params CgaParameters
|
|
|
|
_, err := rand.Read(params.Modifier[:])
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
params.PublicKey = key
|
|
params.SubnetPrefix = subnetPrefix
|
|
return ¶ms, nil
|
|
}
|
|
|
|
func (c *CgaParameters) generateHash2() []byte {
|
|
var byteVal [hash2Length]byte
|
|
|
|
for i := 0; i < ModifierLength; i++ {
|
|
byteVal[i] = c.Modifier[i]
|
|
}
|
|
|
|
for i := 0; i < wgtypes.KeyLen; i++ {
|
|
byteVal[ModifierLength+ZeroLength+i] = c.PublicKey[i]
|
|
}
|
|
|
|
hash := sha1.Sum(byteVal[:])
|
|
|
|
return hash[:Hash2Prefix]
|
|
}
|
|
|
|
func (c *CgaParameters) generateHash1() []byte {
|
|
var byteVal [hash1Length]byte
|
|
|
|
for i := 0; i < ModifierLength; i++ {
|
|
byteVal[i] = c.Modifier[i]
|
|
}
|
|
|
|
for i := 0; i < wgtypes.KeyLen; i++ {
|
|
byteVal[ModifierLength+ZeroLength+i] = c.PublicKey[i]
|
|
}
|
|
|
|
byteVal[hash1Length-1] = c.CollisionCount
|
|
|
|
hash := sha1.Sum(byteVal[:])
|
|
|
|
return hash[:Hash1Prefix]
|
|
}
|
|
|
|
func clearBit(num, pos int) byte {
|
|
mask := ^(1 << pos)
|
|
result := num & mask
|
|
|
|
return byte(result)
|
|
}
|
|
|
|
func (c *CgaParameters) generateInterface() []byte {
|
|
// TODO: On duplicate address detection increment collision.
|
|
// Also incorporate SEC
|
|
|
|
hash1 := c.generateHash1()
|
|
|
|
var interfaceId []byte = make([]byte, InterfaceIdLen)
|
|
|
|
copy(interfaceId[:], hash1)
|
|
|
|
interfaceId[0] = clearBit(int(interfaceId[0]), 6)
|
|
interfaceId[0] = clearBit(int(interfaceId[1]), 7)
|
|
|
|
return interfaceId
|
|
}
|
|
|
|
func (c *CgaParameters) GetIpv6() net.IP {
|
|
if c.flag == 1 {
|
|
return c.interfaceId[:]
|
|
}
|
|
|
|
bytes := c.generateInterface()
|
|
|
|
for i := 0; i < InterfaceIdLen; i++ {
|
|
c.interfaceId[i] = c.SubnetPrefix[i]
|
|
}
|
|
|
|
for i := InterfaceIdLen; i < 2*InterfaceIdLen; i++ {
|
|
c.interfaceId[i] = bytes[i-8]
|
|
}
|
|
|
|
c.flag = 1
|
|
return c.interfaceId[:]
|
|
}
|