1
0
forked from extern/smegmesh
smegmesh/pkg/cga/cga.go
2023-09-28 16:55:37 +01:00

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 &params, 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[:]
}