blob: c598da317fea9d11afc99e72509a8f20cf17aca3 [file] [log] [blame]
David Why Use Two When One Will Do Presottof6813ec2014-07-11 16:12:54 -07001package netconfig
2
3// We connect to the Netlink Route socket and parse messages to
4// look for network configuration changes. This is very Linux
5// specific, hence the file name.
6
7import (
8 "errors"
9 "fmt"
10 "net"
11 "sync"
12 "syscall"
13 "time"
14 "unsafe"
Jiri Simsa519c5072014-09-17 21:37:57 -070015 "veyron.io/veyron/veyron2/vlog"
David Why Use Two When One Will Do Presottof6813ec2014-07-11 16:12:54 -070016)
17
18/*
19#include <linux/rtnetlink.h>
20*/
21import "C"
22
23// All rtnetlink attributes start with this header.
24type rtAttrHdr C.struct_rtattr
25
26const rtAttrHdrLen = C.sizeof_struct_rtattr
27
28// The address change messages (RTM_NEWADDR, RTM_DELADDR, RTM_GETADDR).
29type ifaddrMsgHdr C.struct_ifaddrmsg
30
31const ifaddrMsgHdrLen = C.sizeof_struct_ifaddrmsg
32
33type rtAttribute fmt.Stringer
34
35type rtAddressMessage struct {
36 name string
37 hdr ifaddrMsgHdr
38 attributes []rtAttribute
39}
40
41// Attribute types (see rtnetlink(7))
42type ifaAddress net.IP
43type ifaLocal net.IP
44type ifaBroadcast net.IP
45type ifaAnycast net.IP
46type ifaMulticast net.IP
47type ifaLabel string
48type ifaCacheInfo C.struct_ifa_cacheinfo
49
50const ifaCacheInfoLen = C.sizeof_struct_ifa_cacheinfo
51
52// String routines to make debugging easier.
53func (a ifaAddress) String() string { return "Address=" + net.IP(a).String() }
54func (a ifaLocal) String() string { return "Local=" + net.IP(a).String() }
55func (a ifaBroadcast) String() string { return "Braodcast=" + net.IP(a).String() }
56func (a ifaAnycast) String() string { return "Anycast=" + net.IP(a).String() }
57func (a ifaMulticast) String() string { return "Anycast=" + net.IP(a).String() }
58func (a ifaLabel) String() string { return "Label=" + string(a) }
59func (a ifaCacheInfo) String() string {
60 return fmt.Sprintf("CacheInfo[preferred %d valid %d cstamp %d tstamp %d]", a.ifa_prefered, a.ifa_valid, a.cstamp, a.tstamp)
61}
62func (m *rtAddressMessage) String() string {
63 return fmt.Sprintf("%s: index %d %v", m.name, m.hdr.ifa_index, m.attributes)
64}
65
66// Address looks for the address attribute in an rtAddressMessage. If it isn't there, just assume the null address.
67func (m *rtAddressMessage) Address() net.IP {
68 for _, a := range m.attributes {
69 switch a.(type) {
70 case ifaAddress:
71 return net.IP(a.(ifaAddress))
72 }
73 }
74 return net.IPv4zero
75}
76
77func parsertAddressAttribute(b []byte) (rtAttribute, []byte, error) {
78 if len(b) == 0 {
79 return nil, nil, nil
80 }
81 if len(b) < rtAttrHdrLen {
82 return nil, nil, errors.New("attribute too short")
83 }
84 ahdr := (*rtAttrHdr)(unsafe.Pointer(&b[0]))
85 rounded := ((ahdr.rta_len + 3) / 4) * 4
86 if len(b) < int(rounded) {
87 return nil, nil, errors.New("attribute too short")
88 }
89 remaining := b[rounded:]
90 b = b[rtAttrHdrLen:ahdr.rta_len]
91 switch ahdr.rta_type {
92 case C.IFA_ADDRESS:
93 return rtAttribute(ifaAddress(net.IP(b))), remaining, nil
94 case C.IFA_LOCAL:
95 return rtAttribute(ifaLocal(b)), remaining, nil
96 case C.IFA_LABEL:
97 return rtAttribute(ifaLabel(b)), remaining, nil
98 case C.IFA_BROADCAST:
99 return rtAttribute(ifaBroadcast(b)), remaining, nil
100 case C.IFA_ANYCAST:
101 return rtAttribute(ifaAnycast(b)), remaining, nil
102 case C.IFA_CACHEINFO:
103 if len(b) < ifaCacheInfoLen {
104 return nil, nil, errors.New("attribute too short")
105 }
106 return rtAttribute(ifaCacheInfo(*(*C.struct_ifa_cacheinfo)(unsafe.Pointer(&b[0])))), remaining, nil
107 case C.IFA_MULTICAST:
108 return rtAttribute(ifaMulticast(b)), remaining, nil
109 }
110 return nil, remaining, errors.New("unknown attribute")
111}
112
113func parsertAddressMessage(nlm syscall.NetlinkMessage) (*rtAddressMessage, error) {
114 var name string
115 switch nlm.Header.Type {
116 case C.RTM_NEWADDR:
117 name = "RTM_NEWADDR"
118 case C.RTM_DELADDR:
119 name = "RTM_DELADDR"
120 case C.RTM_GETADDR:
121 name = "RTM_GETADDR"
122 default:
123 return nil, fmt.Errorf("unknown message type")
124 }
125 if len(nlm.Data) < ifaddrMsgHdrLen {
126 return nil, errors.New("bad length")
127 }
128 m := &rtAddressMessage{name: name, hdr: *(*ifaddrMsgHdr)(unsafe.Pointer(&nlm.Data[0]))}
129 b := nlm.Data[ifaddrMsgHdrLen:]
130 for {
131 var a rtAttribute
132 var err error
133 a, b, err = parsertAddressAttribute(b)
134 if b == nil {
135 break
136 }
137 if err == nil {
138 m.attributes = append(m.attributes, a)
139 }
140 }
141 return m, nil
142}
143
144// The link change messages (RTM_NEWLINK, RTM_DELLINK, RTM_GETLINK).
145type ifInfoMsgHdr C.struct_ifinfomsg
146
147const ifInfoMsgHdrLen = C.sizeof_struct_ifinfomsg
148
149type rtLinkMessage struct {
150 name string
151 hdr ifInfoMsgHdr
152 attributes []rtAttribute
153}
154
155// Attribute types (see rtnetlink(7))
156type iflaAddress []byte
157type iflaBroadcast []byte
158type iflaIFName string
159type iflaMTU uint32
160type iflaLink int
161type iflaQDisc string
162type iflaOperstate int
163type iflaStats C.struct_rtnl_link_stats
164
165const iflaStatsLen = C.sizeof_struct_rtnl_link_stats
166
167// String routines to make debugging easier.
168func (a iflaAddress) String() string { return fmt.Sprintf("HWAddress=%v", []byte(a)) }
169func (a iflaBroadcast) String() string { return fmt.Sprintf("HWBroadcast=%v", []byte(a)) }
170func (a iflaIFName) String() string { return "Name=" + string(a) }
171func (a iflaMTU) String() string { return fmt.Sprintf("MTU=%d", uint32(a)) }
172func (a iflaLink) String() string { return fmt.Sprintf("Type=%d", int(a)) }
173func (a iflaQDisc) String() string { return "Qdisc=" + string(a) }
174func (a iflaStats) String() string {
175 return fmt.Sprintf("Stats[rx %d tx %d ...]", a.rx_packets, a.tx_packets)
176}
177func (a iflaOperstate) String() string { return fmt.Sprintf("Operstate=%d", int(a)) }
178func (m *rtLinkMessage) String() string {
179 return fmt.Sprintf("%s: index %d %v", m.name, m.hdr.ifi_index, m.attributes)
180}
181
182func parseRTLinkAttribute(b []byte) (rtAttribute, []byte, error) {
183 if len(b) == 0 {
184 return nil, nil, nil
185 }
186 if len(b) < rtAttrHdrLen {
187 return nil, nil, errors.New("attribute too short")
188 }
189 ahdr := (*rtAttrHdr)(unsafe.Pointer(&b[0]))
190 rounded := ((ahdr.rta_len + 3) / 4) * 4
191 if len(b) < int(rounded) {
192 return nil, nil, errors.New("attribute too short")
193 }
194 remaining := b[rounded:]
195 b = b[rtAttrHdrLen:ahdr.rta_len]
196 switch ahdr.rta_type {
197 case C.IFLA_ADDRESS:
198 return rtAttribute(iflaAddress(b)), remaining, nil
199 case C.IFLA_BROADCAST:
200 return rtAttribute(iflaBroadcast(b)), remaining, nil
201 case C.IFLA_IFNAME:
202 return rtAttribute(iflaIFName(string(b))), remaining, nil
203 case C.IFLA_MTU:
204 return rtAttribute(iflaMTU(*(*C.uint)(unsafe.Pointer(&b[0])))), remaining, nil
205 case C.IFLA_LINK:
206 return rtAttribute(iflaMTU(*(*C.int)(unsafe.Pointer(&b[0])))), remaining, nil
207 case C.IFLA_QDISC:
208 return rtAttribute(iflaQDisc(string(b))), remaining, nil
209 case C.IFLA_STATS:
210 if len(b) < iflaStatsLen {
211 return nil, remaining, errors.New("attribute too short")
212 }
213 return rtAttribute(iflaStats(*(*C.struct_rtnl_link_stats)(unsafe.Pointer(&b[0])))), remaining, nil
214 case C.IFLA_OPERSTATE:
215 return rtAttribute(iflaOperstate(*(*C.int)(unsafe.Pointer(&b[0])))), remaining, nil
216 }
217 return nil, remaining, errors.New("unknown attribute")
218}
219
220func parsertLinkMessage(nlm syscall.NetlinkMessage) (*rtLinkMessage, error) {
221 var name string
222 switch nlm.Header.Type {
223 case C.RTM_NEWLINK:
224 name = "RTM_NEWLINK"
225 case C.RTM_DELLINK:
226 name = "RTM_DELLINK"
227 case C.RTM_GETLINK:
228 name = "RTM_GETLINK"
229 default:
230 return nil, fmt.Errorf("unknown message type")
231 }
232 if len(nlm.Data) < ifInfoMsgHdrLen {
233 return nil, errors.New("bad length")
234 }
235 m := &rtLinkMessage{name: name, hdr: *(*ifInfoMsgHdr)(unsafe.Pointer(&nlm.Data[0]))}
236 b := nlm.Data[ifInfoMsgHdrLen:]
237 for {
238 var a rtAttribute
239 var err error
240 a, b, err = parseRTLinkAttribute(b)
241 if b == nil {
242 break
243 }
244 if err == nil {
245 m.attributes = append(m.attributes, a)
246 }
247 }
248 return m, nil
249}
250
251type rtnetlinkWatcher struct {
252 sync.Mutex
253 t *time.Timer
254 c chan struct{}
255 s int
256 stopped bool
257}
258
259func (w *rtnetlinkWatcher) Stop() {
260 w.Lock()
261 defer w.Unlock()
262 if w.stopped {
263 return
264 }
265 w.stopped = true
266 syscall.Close(w.s)
267}
268
269func (w *rtnetlinkWatcher) Channel() chan struct{} {
270 return w.c
271}
272
273const (
274 GROUPS = C.RTMGRP_LINK | C.RTMGRP_IPV4_IFADDR | C.RTMGRP_IPV6_IFADDR | C.RTMGRP_NOTIFY
275)
276
277// NewNetConfigWatcher returns a watcher that wakes up anyone
278// calling the Wait routine whenever the configuration changes.
279func NewNetConfigWatcher() (NetConfigWatcher, error) {
280 s, err := syscall.Socket(syscall.AF_NETLINK, syscall.SOCK_RAW, syscall.NETLINK_ROUTE)
281 if err != nil {
282 vlog.Infof("netconfig socket failed: %s", err)
283 return nil, err
284 }
285
286 lsa := &syscall.SockaddrNetlink{Family: syscall.AF_NETLINK, Groups: GROUPS}
287 if err := syscall.Bind(s, lsa); err != nil {
288 vlog.Infof("netconfig bind failed: %s", err)
289 return nil, err
290 }
291
292 w := &rtnetlinkWatcher{c: make(chan struct{}, 1), s: s}
293 go w.watcher()
294 return w, nil
295}
296
297func (w *rtnetlinkWatcher) ding() {
298 w.Lock()
299 defer w.Unlock()
300 w.t = nil
301 if w.stopped {
302 return
303 }
304 // Don't let us hang in the lock. The default is safe because the requirement
305 // is that the client get a message after the last config change. Since this is
306 // a queued chan, we really don't have to stuff anything in it if there's already
307 // something there.
308 select {
309 case w.c <- struct{}{}:
310 default:
311 }
312}
313
314func (w *rtnetlinkWatcher) watcher() {
315 var newAddrs []net.IP
316 for {
317 rb := make([]byte, 4096)
318 nr, _, err := syscall.Recvfrom(w.s, rb, 0)
319 if err != nil {
320 break
321 }
322 rb = rb[:nr]
323 msgs, err := syscall.ParseNetlinkMessage(rb)
324 if err != nil {
325 vlog.Infof("ParseNetlinkMessage failed: %s", err)
326 continue
327 }
328 L:
329 for _, m := range msgs {
330 if am, err := parsertAddressMessage(m); err == nil {
331 // NOTE(p): We get continuous NEWADDR messages about some
332 // IPv6 addresses in Google corp. Just ignore duplicate back
333 // to back NEWADDRs about the same addresses.
334 if am.name == "RTM_NEWADDR" {
335 addr := am.Address()
336 for _, a := range newAddrs {
337 if addr.Equal(a) {
338 break L
339 }
340 }
341 newAddrs = append(newAddrs, addr)
342 } else {
343 newAddrs = nil
344 }
345 } else if _, err := parsertLinkMessage(m); err == nil {
346 newAddrs = nil
347 } else {
348 continue
349 }
350 // Changing networks usually spans many seconds and involves
351 // multiple network config changes. We add histeresis by
352 // setting an alarm when the first change is detected and
353 // not informing the client till the alarm goes off.
354 // NOTE(p): I chose 3 seconds because that covers all the
355 // events involved in moving from one wifi network to another.
356 w.Lock()
357 if w.t == nil {
358 w.t = time.AfterFunc(3*time.Second, w.ding)
359 }
360 w.Unlock()
361 }
362 }
363
364 w.Stop()
365 w.Lock()
366 close(w.c)
367 if w.t != nil {
368 w.t.Stop()
369 }
370 w.Unlock()
371}