-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpm_v6_route.lua
160 lines (139 loc) · 4.7 KB
/
pm_v6_route.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
#!/usr/bin/env lua
-- -*-lua-*-
--
-- $Id: pm_v6_route.lua $
--
-- Author: Markus Stenberg <markus [email protected]>
--
-- Copyright (c) 2012 cisco Systems, Inc.
--
-- Created: Thu Nov 8 06:48:34 2012 mstenber
-- Last modified: Mon Sep 30 15:36:07 2013 mstenber
-- Edit time: 27 min
--
-- pm_v6_route is responsible for syncing the real state to
-- ospf_lap/usp by manipulating the configured interfaces' addresses
-- (and implicitly, routes). we could also directly interact with the
-- RIB if we chose to.
require 'pm_handler'
require 'ipv6s'
module(..., package.seeall)
local ipv4_end='/24' -- as it's really v4 looking string
local _parent = pm_handler.pm_handler_with_pa
pm_v6_route = _parent:new_subclass{class='pm_v6_route'}
function pm_v6_route:init()
-- superclass init
_parent.init(self)
-- connect our changed to v6_addr_changed
-- added to 'us' even if destination is the self._pm, as we expire
-- before pm does
self:connect_event(self.changed, self._pm.v6_addr_changed)
end
local function laplist_to_map(l)
local t = mst.map:new{}
for i, v in ipairs(l)
do
local ov = t[v.prefix]
-- if we don't have old value, or old one is
-- depracated, we clearly prefer the new one
if not ov or ov.depracate
then
t[v.prefix] = v
end
end
return t
end
function pm_v6_route:run()
local valid_end='::/64'
local lap = self.lap:get_ipv6()
local rlap = self:get_real_lap()
self:d('lap_changed - rlap/lap', #rlap, #lap)
-- both are lists of map's, with prefix+ifname keys
--
-- convert them to single table
-- (prefixes should be unique, interfaces not necessarily)
local ospf_lap = laplist_to_map(lap)
local real_lap = laplist_to_map(rlap)
local ospf_keys = ospf_lap:keys():to_set()
local real_keys = real_lap:keys():to_set()
local c
c = mst.sync_tables(real_keys, ospf_keys,
-- remove (only in real)
function (prefix)
mst.a(mst.string_endswith(prefix, valid_end),
'invalid prefix', prefix)
self:handle_real_prefix(prefix, real_lap[prefix])
end,
-- add (only in ospf)
function (prefix)
mst.a(mst.string_endswith(prefix, valid_end),
'invalid prefix', prefix)
self:handle_ospf_prefix(prefix, ospf_lap[prefix])
end,
-- are values same?
function (prefix)
mst.a(mst.string_endswith(prefix, valid_end),
'invalid prefix', prefix)
local v1 = ospf_lap[prefix]
local v2 = real_lap[prefix]
return v1.ifname == v2.ifname
end)
return c
end
function pm_v6_route:get_real_lap()
local r = mst.array:new{}
local m = self:get_if_table():read_ip_ipv6()
for _, ifo in ipairs(m:values())
do
for _, addr in ipairs(ifo.ipv6 or {})
do
local prefix = ipv6s.new_prefix_from_ascii(addr)
local bits = prefix:get_binary_bits()
if bits == 64
then
-- non-64 bit prefixes can't be eui64 either
-- consider if we should even care about this prefix
local found = nil
prefix:clear_tailing_bits()
for _, p2 in pairs(self._pm.all_ipv6_binary_prefixes)
do
--self:d('considering', v.prefix, prefix)
if p2:contains(prefix)
then
found = true
break
end
end
if not found
then
self:d('ignoring prefix', prefix)
else
local o = {ifname=ifo.name,
prefix=prefix:get_ascii(),
addr=addr}
self:d('found', o)
r:insert(o)
end
end
end
end
return r
end
function pm_v6_route:handle_ospf_prefix(prefix, po)
self:a(po.ifname, 'no ifname in lap', po)
self:a(po.address)
self:d('handle_ospf_prefix', po)
local ifo = self:get_if_table():get_if(po.ifname)
return ifo:add_ipv6(po.address .. '/64')
end
function pm_v6_route:handle_real_prefix(prefix, po)
-- prefix is only on real interface, but not in OSPF
-- that means that we want to get rid of it.. let's do so
self:d('handle_real_prefix', po)
local addr = po.addr
self:a(addr)
local ifname = po.ifname
self:a(ifname)
local ifo = self:get_if_table():get_if(ifname)
return ifo:del_ipv6(addr)
end