2014-09-01 03:27:34 +00:00
|
|
|
package netlink
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"syscall"
|
|
|
|
"unsafe"
|
|
|
|
)
|
|
|
|
|
|
|
|
type RtMsg struct {
|
|
|
|
syscall.RtMsg
|
|
|
|
}
|
|
|
|
|
|
|
|
func newRtMsg() *RtMsg {
|
|
|
|
return &RtMsg{
|
|
|
|
RtMsg: syscall.RtMsg{
|
|
|
|
Table: syscall.RT_TABLE_MAIN,
|
|
|
|
Scope: syscall.RT_SCOPE_UNIVERSE,
|
|
|
|
Protocol: syscall.RTPROT_BOOT,
|
|
|
|
Type: syscall.RTN_UNICAST,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (msg *RtMsg) Len() int {
|
|
|
|
return syscall.SizeofRtMsg
|
|
|
|
}
|
|
|
|
|
|
|
|
func DeserializeRtMsg(b []byte) *RtMsg {
|
|
|
|
return (*RtMsg)(unsafe.Pointer(&b[0:syscall.SizeofRtMsg][0]))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (msg *RtMsg) Serialize() []byte {
|
|
|
|
return (*(*[syscall.SizeofRtMsg]byte)(unsafe.Pointer(msg)))[:]
|
|
|
|
}
|
|
|
|
|
|
|
|
// RtAttr is shared so it is in netlink_linux.go
|
|
|
|
|
|
|
|
// RouteAdd will add a route to the system.
|
|
|
|
// Equivalent to: `ip route add $route`
|
|
|
|
func RouteAdd(route *Route) error {
|
|
|
|
req := newNetlinkRequest(syscall.RTM_NEWROUTE, syscall.NLM_F_CREATE|syscall.NLM_F_EXCL|syscall.NLM_F_ACK)
|
|
|
|
return routeHandle(route, req)
|
|
|
|
}
|
|
|
|
|
|
|
|
// RouteAdd will delete a route from the system.
|
|
|
|
// Equivalent to: `ip route del $route`
|
|
|
|
func RouteDel(route *Route) error {
|
|
|
|
req := newNetlinkRequest(syscall.RTM_DELROUTE, syscall.NLM_F_ACK)
|
|
|
|
return routeHandle(route, req)
|
|
|
|
}
|
|
|
|
|
|
|
|
func routeHandle(route *Route, req *NetlinkRequest) error {
|
|
|
|
if route.Dst.IP == nil && route.Src == nil && route.Gw == nil {
|
|
|
|
return fmt.Errorf("one of Dst.IP, Src, or Gw must not be nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := newRtMsg()
|
2014-09-15 00:44:56 +00:00
|
|
|
msg.Scope = uint8(route.Scope)
|
2014-09-01 03:27:34 +00:00
|
|
|
family := -1
|
|
|
|
var rtAttrs []*RtAttr
|
|
|
|
|
|
|
|
if route.Dst.IP != nil {
|
|
|
|
dstLen, _ := route.Dst.Mask.Size()
|
|
|
|
msg.Dst_len = uint8(dstLen)
|
|
|
|
dstFamily := GetIPFamily(route.Dst.IP)
|
|
|
|
family = dstFamily
|
|
|
|
var dstData []byte
|
2014-09-17 16:33:53 +00:00
|
|
|
if dstFamily == FAMILY_V4 {
|
2014-09-01 03:27:34 +00:00
|
|
|
dstData = route.Dst.IP.To4()
|
|
|
|
} else {
|
|
|
|
dstData = route.Dst.IP.To16()
|
|
|
|
}
|
|
|
|
rtAttrs = append(rtAttrs, newRtAttr(syscall.RTA_DST, dstData))
|
|
|
|
}
|
|
|
|
|
|
|
|
if route.Src != nil {
|
|
|
|
srcFamily := GetIPFamily(route.Src)
|
|
|
|
if family != -1 && family != srcFamily {
|
|
|
|
return fmt.Errorf("source and destination ip are not the same IP family")
|
|
|
|
}
|
|
|
|
family = srcFamily
|
|
|
|
var srcData []byte
|
2014-09-17 16:33:53 +00:00
|
|
|
if srcFamily == FAMILY_V4 {
|
2014-09-01 03:27:34 +00:00
|
|
|
srcData = route.Src.To4()
|
|
|
|
} else {
|
|
|
|
srcData = route.Src.To16()
|
|
|
|
}
|
|
|
|
// The commonly used src ip for routes is actually PREFSRC
|
|
|
|
rtAttrs = append(rtAttrs, newRtAttr(syscall.RTA_PREFSRC, srcData))
|
|
|
|
}
|
|
|
|
|
|
|
|
if route.Gw != nil {
|
|
|
|
gwFamily := GetIPFamily(route.Gw)
|
|
|
|
if family != -1 && family != gwFamily {
|
|
|
|
return fmt.Errorf("gateway, source, and destination ip are not the same IP family")
|
|
|
|
}
|
|
|
|
family = gwFamily
|
|
|
|
var gwData []byte
|
2014-09-17 16:33:53 +00:00
|
|
|
if gwFamily == FAMILY_V4 {
|
2014-09-01 03:27:34 +00:00
|
|
|
gwData = route.Gw.To4()
|
|
|
|
} else {
|
|
|
|
gwData = route.Gw.To16()
|
|
|
|
}
|
|
|
|
rtAttrs = append(rtAttrs, newRtAttr(syscall.RTA_GATEWAY, gwData))
|
|
|
|
}
|
|
|
|
|
|
|
|
msg.Family = uint8(family)
|
|
|
|
|
|
|
|
req.AddData(msg)
|
|
|
|
for _, attr := range rtAttrs {
|
|
|
|
req.AddData(attr)
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
b = make([]byte, 4)
|
|
|
|
native = nativeEndian()
|
|
|
|
)
|
|
|
|
native.PutUint32(b, uint32(route.Link.Index))
|
|
|
|
|
|
|
|
req.AddData(newRtAttr(syscall.RTA_OIF, b))
|
|
|
|
|
|
|
|
_, err := req.Execute(syscall.NETLINK_ROUTE, 0)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// RouteList gets a list of routes in the system.
|
|
|
|
// Equivalent to: `ip route show`.
|
|
|
|
// The list can be filtered by link and ip family.
|
|
|
|
func RouteList(link *Link, family int) ([]Route, error) {
|
|
|
|
req := newNetlinkRequest(syscall.RTM_GETROUTE, syscall.NLM_F_DUMP)
|
|
|
|
msg := newIfInfomsg(family)
|
|
|
|
req.AddData(msg)
|
|
|
|
|
|
|
|
msgs, err := req.Execute(syscall.NETLINK_ROUTE, syscall.RTM_NEWROUTE)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
native := nativeEndian()
|
|
|
|
res := make([]Route, 0)
|
|
|
|
for _, m := range msgs {
|
|
|
|
msg := DeserializeRtMsg(m)
|
|
|
|
|
|
|
|
if msg.Flags&syscall.RTM_F_CLONED != 0 {
|
|
|
|
// Ignore cloned routes
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if msg.Table != syscall.RT_TABLE_MAIN {
|
|
|
|
// Ignore non-main tables
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
attrs, err := parseRouteAttr(m[msg.Len():])
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2014-09-15 00:44:56 +00:00
|
|
|
route := Route{Scope: Scope(msg.Scope)}
|
2014-09-01 03:27:34 +00:00
|
|
|
for _, attr := range attrs {
|
|
|
|
switch attr.Attr.Type {
|
|
|
|
case syscall.RTA_GATEWAY:
|
|
|
|
route.Gw = net.IP(attr.Value)
|
|
|
|
case syscall.RTA_PREFSRC:
|
|
|
|
route.Src = net.IP(attr.Value)
|
|
|
|
case syscall.RTA_DST:
|
2014-09-07 18:19:59 +00:00
|
|
|
route.Dst = &net.IPNet{
|
2014-09-01 03:27:34 +00:00
|
|
|
IP: attr.Value,
|
|
|
|
Mask: net.CIDRMask(int(msg.Dst_len), 8*len(attr.Value)),
|
|
|
|
}
|
|
|
|
case syscall.RTA_OIF:
|
|
|
|
index := int(native.Uint32(attr.Value[0:4]))
|
|
|
|
if link != nil && index != link.Index {
|
|
|
|
// Ignore routes from other interfaces
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
resLink, _ := LinkByIndex(index)
|
2014-09-07 18:19:59 +00:00
|
|
|
route.Link = resLink
|
2014-09-01 03:27:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
res = append(res, route)
|
|
|
|
}
|
|
|
|
|
|
|
|
return res, nil
|
|
|
|
}
|