blob: 083255808be15a98af57bfa544cfca62eed3200b [file] [log] [blame]
// Copyright 2016 Open Networking Laboratory
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package main
import (
"bytes"
"math"
"net"
"strconv"
"strings"
)
type Storage interface {
Init(networkIn string, low string, high string) error
Get(mac string) (string, error)
GetAll() map[string]string
Put(mac, ip string) error
Remove(mac string) (string, error)
Dequeue() (string, error)
Enqueue(ip string) error
}
type MemoryStorage struct {
allocated map[string]IPv4
available []IPv4
readIdx, writeIdx, size uint
}
func inIPRange(from net.IP, to net.IP, test net.IP) bool {
if from == nil || to == nil || test == nil {
return false
}
from16 := from.To16()
to16 := to.To16()
test16 := test.To16()
if from16 == nil || to16 == nil || test16 == nil {
return false
}
if bytes.Compare(test16, from16) >= 0 && bytes.Compare(test16, to16) <= 0 {
return true
}
return false
}
func (s *MemoryStorage) Init(networkIn string, low string, high string) error {
_, network, err := net.ParseCIDR(networkIn)
if err != nil {
return err
}
start, _, err := net.ParseCIDR(network.String())
if err != nil {
return err
}
parts := strings.Split(network.String(), "/")
ip, err := ParseIP(start.String())
if err != nil {
return err
}
bits, err := strconv.Atoi(parts[1])
if err != nil {
return err
}
hostCount := int(math.Pow(2, float64(32-bits)))
s.readIdx = 0
s.writeIdx = 0
s.size = uint(hostCount)
s.allocated = make(map[string]IPv4)
s.available = make([]IPv4, 0, hostCount)
ipLow := net.ParseIP(low)
ipHigh := net.ParseIP(high)
for i := 0; i < hostCount; i += 1 {
if inIPRange(ipLow, ipHigh, net.ParseIP(ip.String())) {
s.available = append(s.available, ip)
}
ip, err = ip.Next()
if err != nil {
return err
}
}
log.Debugf("AVAILABLE: %+v\n", s.available)
return nil
}
func (s *MemoryStorage) Get(mac string) (string, error) {
ip, ok := s.allocated[mac]
if !ok {
return "", nil
}
return ip.String(), nil
}
func (s *MemoryStorage) GetAll() map[string]string {
all := make(map[string]string)
for k, v := range s.allocated {
all[k] = v.String()
}
return all
}
func (s *MemoryStorage) Put(mac, ip string) error {
data, err := ParseIP(ip)
if err != nil {
return err
}
s.allocated[mac] = data
return nil
}
func (s *MemoryStorage) Remove(mac string) (string, error) {
ip, ok := s.allocated[mac]
if !ok {
return "", nil
}
delete(s.allocated, mac)
return ip.String(), nil
}
func (s *MemoryStorage) Dequeue() (string, error) {
ip := s.available[s.readIdx]
s.readIdx = (s.readIdx + 1) % s.size
return ip.String(), nil
}
func (s *MemoryStorage) Enqueue(ip string) error {
data, err := ParseIP(ip)
if err != nil {
return err
}
s.available[s.writeIdx] = data
s.writeIdx = (s.writeIdx + 1) % s.size
return nil
}