2018-05-03 13:04:00 +00:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0
|
|
|
|
*
|
|
|
|
* Copyright (C) 2017-2018 Jason A. Donenfeld <Jason@zx2c4.com>. All Rights Reserved.
|
|
|
|
*/
|
|
|
|
|
2017-06-24 13:34:17 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/rand"
|
|
|
|
"sync"
|
2018-05-13 16:23:40 +00:00
|
|
|
"unsafe"
|
2017-06-24 13:34:17 +00:00
|
|
|
)
|
|
|
|
|
2017-06-26 20:07:29 +00:00
|
|
|
type IndexTableEntry struct {
|
|
|
|
peer *Peer
|
|
|
|
handshake *Handshake
|
2018-05-13 16:23:40 +00:00
|
|
|
keypair *Keypair
|
2017-06-26 20:07:29 +00:00
|
|
|
}
|
|
|
|
|
2017-06-24 13:34:17 +00:00
|
|
|
type IndexTable struct {
|
2017-06-26 20:07:29 +00:00
|
|
|
mutex sync.RWMutex
|
|
|
|
table map[uint32]IndexTableEntry
|
2017-06-24 13:34:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func randUint32() (uint32, error) {
|
2018-05-13 16:23:40 +00:00
|
|
|
var integer [4]byte
|
|
|
|
_, err := rand.Read(integer[:])
|
|
|
|
return *(*uint32)(unsafe.Pointer(&integer[0])), err
|
2017-06-24 13:34:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (table *IndexTable) Init() {
|
|
|
|
table.mutex.Lock()
|
2018-05-13 16:23:40 +00:00
|
|
|
defer table.mutex.Unlock()
|
2017-06-26 20:07:29 +00:00
|
|
|
table.table = make(map[uint32]IndexTableEntry)
|
2017-06-24 13:34:17 +00:00
|
|
|
}
|
|
|
|
|
2017-06-30 12:41:08 +00:00
|
|
|
func (table *IndexTable) Delete(index uint32) {
|
2017-06-26 20:07:29 +00:00
|
|
|
table.mutex.Lock()
|
2018-05-13 16:23:40 +00:00
|
|
|
defer table.mutex.Unlock()
|
2017-06-26 20:07:29 +00:00
|
|
|
delete(table.table, index)
|
|
|
|
}
|
|
|
|
|
2018-05-13 16:23:40 +00:00
|
|
|
func (table *IndexTable) SwapIndexForKeypair(index uint32, keypair *Keypair) {
|
2017-06-24 13:34:17 +00:00
|
|
|
table.mutex.Lock()
|
2018-05-13 16:23:40 +00:00
|
|
|
defer table.mutex.Unlock()
|
|
|
|
entry, ok := table.table[index]
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
table.table[index] = IndexTableEntry{
|
|
|
|
peer: entry.peer,
|
|
|
|
keypair: keypair,
|
|
|
|
handshake: nil,
|
|
|
|
}
|
2017-06-26 20:07:29 +00:00
|
|
|
}
|
|
|
|
|
2018-05-13 16:23:40 +00:00
|
|
|
func (table *IndexTable) NewIndexForHandshake(peer *Peer, handshake *Handshake) (uint32, error) {
|
2017-06-24 13:34:17 +00:00
|
|
|
for {
|
|
|
|
// generate random index
|
|
|
|
|
2017-06-26 20:07:29 +00:00
|
|
|
index, err := randUint32()
|
2017-06-24 13:34:17 +00:00
|
|
|
if err != nil {
|
2017-06-26 20:07:29 +00:00
|
|
|
return index, err
|
2017-06-24 13:34:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// check if index used
|
|
|
|
|
2017-06-26 20:07:29 +00:00
|
|
|
table.mutex.RLock()
|
|
|
|
_, ok := table.table[index]
|
2017-07-17 14:16:18 +00:00
|
|
|
table.mutex.RUnlock()
|
2017-06-24 13:34:17 +00:00
|
|
|
if ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-05-13 16:23:40 +00:00
|
|
|
// check again while locked
|
2017-06-24 13:34:17 +00:00
|
|
|
|
2017-06-26 20:07:29 +00:00
|
|
|
table.mutex.Lock()
|
|
|
|
_, found := table.table[index]
|
|
|
|
if found {
|
|
|
|
table.mutex.Unlock()
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
table.table[index] = IndexTableEntry{
|
|
|
|
peer: peer,
|
2018-05-13 16:23:40 +00:00
|
|
|
handshake: handshake,
|
|
|
|
keypair: nil,
|
2017-06-26 20:07:29 +00:00
|
|
|
}
|
|
|
|
table.mutex.Unlock()
|
|
|
|
return index, nil
|
2017-06-24 13:34:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-26 20:07:29 +00:00
|
|
|
func (table *IndexTable) Lookup(id uint32) IndexTableEntry {
|
2017-06-24 13:34:17 +00:00
|
|
|
table.mutex.RLock()
|
|
|
|
defer table.mutex.RUnlock()
|
2017-06-26 20:07:29 +00:00
|
|
|
return table.table[id]
|
2017-06-24 13:34:17 +00:00
|
|
|
}
|