cc8a25e2f6
Signed-off-by: Jason A. Donenfeld <Jason@zx2c4.com>
187 lines
6.9 KiB
Go
187 lines
6.9 KiB
Go
/* Copyright (C) 2015-2018 Jason A. Donenfeld <Jason@zx2c4.com>. All Rights Reserved. */
|
|
|
|
package main
|
|
|
|
import (
|
|
"crypto/rand"
|
|
"encoding/base64"
|
|
"encoding/binary"
|
|
"log"
|
|
"net"
|
|
"time"
|
|
|
|
"github.com/titanous/noise"
|
|
"golang.org/x/net/icmp"
|
|
"golang.org/x/net/ipv4"
|
|
"golang.org/x/crypto/blake2s"
|
|
)
|
|
|
|
func ipChecksum(buf []byte) uint16 {
|
|
sum := uint32(0)
|
|
for ; len(buf) >= 2; buf = buf[2:] {
|
|
sum += uint32(buf[0])<<8 | uint32(buf[1])
|
|
}
|
|
if len(buf) > 0 {
|
|
sum += uint32(buf[0]) << 8
|
|
}
|
|
for sum > 0xffff {
|
|
sum = (sum >> 16) + (sum & 0xffff)
|
|
}
|
|
csum := ^uint16(sum)
|
|
if csum == 0 {
|
|
csum = 0xffff
|
|
}
|
|
return csum
|
|
}
|
|
|
|
func main() {
|
|
ourPrivate, _ := base64.StdEncoding.DecodeString("WAmgVYXkbT2bCtdcDwolI88/iVi/aV3/PHcUBTQSYmo=")
|
|
ourPublic, _ := base64.StdEncoding.DecodeString("K5sF9yESrSBsOXPd6TcpKNgqoy1Ik3ZFKl4FolzrRyI=")
|
|
theirPublic, _ := base64.StdEncoding.DecodeString("qRCwZSKInrMAq5sepfCdaCsRJaoLe5jhtzfiw7CjbwM=")
|
|
preshared, _ := base64.StdEncoding.DecodeString("FpCyhws9cxwWoV4xELtfJvjJN+zQVRPISllRWgeopVE=")
|
|
cs := noise.NewCipherSuite(noise.DH25519, noise.CipherChaChaPoly, noise.HashBLAKE2s)
|
|
hs, _ := noise.NewHandshakeState(noise.Config{
|
|
CipherSuite: cs,
|
|
Random: rand.Reader,
|
|
Pattern: noise.HandshakeIK,
|
|
Initiator: true,
|
|
Prologue: []byte("WireGuard v1 zx2c4 Jason@zx2c4.com"),
|
|
PresharedKey: preshared,
|
|
PresharedKeyPlacement: 2,
|
|
StaticKeypair: noise.DHKey{Private: ourPrivate, Public: ourPublic},
|
|
PeerStatic: theirPublic,
|
|
})
|
|
conn, err := net.Dial("udp", "demo.wireguard.com:12913")
|
|
if err != nil {
|
|
log.Fatalf("error dialing udp socket: %s", err)
|
|
}
|
|
defer conn.Close()
|
|
|
|
// write handshake initiation packet
|
|
now := time.Now()
|
|
tai64n := make([]byte, 12)
|
|
binary.BigEndian.PutUint64(tai64n[:], 4611686018427387914+uint64(now.Unix()))
|
|
binary.BigEndian.PutUint32(tai64n[8:], uint32(now.Nanosecond()))
|
|
initiationPacket := make([]byte, 8)
|
|
initiationPacket[0] = 1 // Type: Initiation
|
|
initiationPacket[1] = 0 // Reserved
|
|
initiationPacket[2] = 0 // Reserved
|
|
initiationPacket[3] = 0 // Reserved
|
|
binary.LittleEndian.PutUint32(initiationPacket[4:], 28) // Sender index: 28 (arbitrary)
|
|
initiationPacket, _, _, _ = hs.WriteMessage(initiationPacket, tai64n)
|
|
hasher, _ := blake2s.New256(nil)
|
|
hasher.Write([]byte("mac1----"))
|
|
hasher.Write(theirPublic)
|
|
hasher, _ = blake2s.New128(hasher.Sum(nil))
|
|
hasher.Write(initiationPacket)
|
|
initiationPacket = append(initiationPacket, hasher.Sum(nil)[:16]...)
|
|
initiationPacket = append(initiationPacket, make([]byte, 16)...)
|
|
if _, err := conn.Write(initiationPacket); err != nil {
|
|
log.Fatalf("error writing initiation packet: %s", err)
|
|
}
|
|
|
|
// read handshake response packet
|
|
responsePacket := make([]byte, 92)
|
|
n, err := conn.Read(responsePacket)
|
|
if err != nil {
|
|
log.Fatalf("error reading response packet: %s", err)
|
|
}
|
|
if n != len(responsePacket) {
|
|
log.Fatalf("response packet too short: want %d, got %d", len(responsePacket), n)
|
|
}
|
|
if responsePacket[0] != 2 { // Type: Response
|
|
log.Fatalf("response packet type wrong: want %d, got %d", 2, responsePacket[0])
|
|
}
|
|
if responsePacket[1] != 0 || responsePacket[2] != 0 || responsePacket[3] != 0 {
|
|
log.Fatalf("response packet has non-zero reserved fields")
|
|
}
|
|
theirIndex := binary.LittleEndian.Uint32(responsePacket[4:])
|
|
ourIndex := binary.LittleEndian.Uint32(responsePacket[8:])
|
|
if ourIndex != 28 {
|
|
log.Fatalf("response packet index wrong: want %d, got %d", 28, ourIndex)
|
|
}
|
|
payload, sendCipher, receiveCipher, err := hs.ReadMessage(nil, responsePacket[12:60])
|
|
if err != nil {
|
|
log.Fatalf("error reading handshake message: %s", err)
|
|
}
|
|
if len(payload) > 0 {
|
|
log.Fatalf("unexpected payload: %x", payload)
|
|
}
|
|
|
|
// write ICMP Echo packet
|
|
pingMessage, _ := (&icmp.Message{
|
|
Type: ipv4.ICMPTypeEcho,
|
|
Body: &icmp.Echo{
|
|
ID: 921,
|
|
Seq: 438,
|
|
Data: []byte("WireGuard"),
|
|
},
|
|
}).Marshal(nil)
|
|
pingHeader, err := (&ipv4.Header{
|
|
Version: ipv4.Version,
|
|
Len: ipv4.HeaderLen,
|
|
TotalLen: ipv4.HeaderLen + len(pingMessage),
|
|
Protocol: 1, // ICMP
|
|
TTL: 20,
|
|
Src: net.IPv4(10, 189, 129, 2),
|
|
Dst: net.IPv4(10, 189, 129, 1),
|
|
}).Marshal()
|
|
binary.BigEndian.PutUint16(pingHeader[2:], uint16(ipv4.HeaderLen+len(pingMessage))) // fix the length endianness on BSDs
|
|
pingData := append(append(pingHeader, pingMessage...), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0)
|
|
binary.BigEndian.PutUint16(pingData[10:], ipChecksum(pingData))
|
|
pingPacket := make([]byte, 16)
|
|
pingPacket[0] = 4 // Type: Data
|
|
pingPacket[1] = 0 // Reserved
|
|
pingPacket[2] = 0 // Reserved
|
|
pingPacket[3] = 0 // Reserved
|
|
binary.LittleEndian.PutUint32(pingPacket[4:], theirIndex) // Their index
|
|
binary.LittleEndian.PutUint64(pingPacket[8:], 0) // Nonce
|
|
pingPacket = sendCipher.Encrypt(pingPacket, nil, pingData) // Payload data
|
|
if _, err := conn.Write(pingPacket); err != nil {
|
|
log.Fatalf("error writing ping message: %s", err)
|
|
}
|
|
|
|
// read ICMP Echo Reply packet
|
|
replyPacket := make([]byte, 80)
|
|
n, err = conn.Read(replyPacket)
|
|
if err != nil {
|
|
log.Fatalf("error reading ping reply message: %s", err)
|
|
}
|
|
replyPacket = replyPacket[:n]
|
|
if replyPacket[0] != 4 { // Type: Data
|
|
log.Fatalf("unexpected reply packet type: %d", replyPacket[0])
|
|
}
|
|
if replyPacket[1] != 0 || replyPacket[2] != 0 || replyPacket[3] != 0 {
|
|
log.Fatalf("reply packet has non-zero reserved fields")
|
|
}
|
|
replyPacket, err = receiveCipher.Decrypt(nil, nil, replyPacket[16:])
|
|
if err != nil {
|
|
log.Fatalf("error decrypting reply packet: %s", err)
|
|
}
|
|
replyHeaderLen := int(replyPacket[0]&0x0f) << 2
|
|
replyLen := binary.BigEndian.Uint16(replyPacket[2:])
|
|
replyMessage, err := icmp.ParseMessage(1, replyPacket[replyHeaderLen:replyLen])
|
|
if err != nil {
|
|
log.Fatalf("error parsing echo: %s", err)
|
|
}
|
|
echo, ok := replyMessage.Body.(*icmp.Echo)
|
|
if !ok {
|
|
log.Fatalf("unexpected reply body type %T", replyMessage.Body)
|
|
}
|
|
|
|
if echo.ID != 921 || echo.Seq != 438 || string(echo.Data) != "WireGuard" {
|
|
log.Fatalf("incorrect echo response: %#v", echo)
|
|
}
|
|
|
|
keepalivePacket := make([]byte, 16)
|
|
keepalivePacket[0] = 4 // Type: Data
|
|
keepalivePacket[1] = 0 // Reserved
|
|
keepalivePacket[2] = 0 // Reserved
|
|
keepalivePacket[3] = 0 // Reserved
|
|
binary.LittleEndian.PutUint32(keepalivePacket[4:], theirIndex) // Their index
|
|
binary.LittleEndian.PutUint64(keepalivePacket[8:], 1) // Nonce
|
|
keepalivePacket = sendCipher.Encrypt(keepalivePacket, nil, nil) // Empty data means keepalive
|
|
if _, err := conn.Write(keepalivePacket); err != nil {
|
|
log.Fatalf("error writing keepalive message: %s", err)
|
|
}
|
|
}
|