diff options
author | Hitoshi Irino <irino@sfc.wide.ad.jp> | 2019-03-02 13:54:38 +0900 |
---|---|---|
committer | FUJITA Tomonori <fujita.tomonori@gmail.com> | 2019-03-14 21:20:31 +0900 |
commit | 3a79ad3fdcbb7af0b7911a9c610c6e7af970ad6d (patch) | |
tree | 40911a6e5be205a15752aa11ea3b2e4f7335ebfa /pkg | |
parent | dc4d9c6d1b253bc3a381f82941796b6bb8c71384 (diff) |
Supporting BGP/MPLS L3VPNs with Frrouting Zebra API 6
- This commit aims to solve reported problem on issues #1611, #1648 and #1912
- Partial changes of this commit duplicate with changes on PR #1587 (not merged) and PR #1766 (not merged and already closed)
- This commit is tested with only FRRouting version 6.0.2 (which uses Zebra API 6)
- This commit fixes lack of LABEL_MANAGER_CONNECT_ASYNC for ZAPI6.
(This bug is introduced on commit 2bdb76f2dcf24b891f2b6327a57c31b26463b2dd "Supporting Zebra API version 6 which is used in FRRouting version 6")
Diffstat (limited to 'pkg')
-rw-r--r-- | pkg/server/server.go | 32 | ||||
-rw-r--r-- | pkg/server/zclient.go | 69 |
2 files changed, 84 insertions, 17 deletions
diff --git a/pkg/server/server.go b/pkg/server/server.go index ee0498e5..17937fff 100644 --- a/pkg/server/server.go +++ b/pkg/server/server.go @@ -958,6 +958,7 @@ func (s *BgpServer) propagateUpdate(peer *peer, pathList []*table.Path) { peer.fsm.lock.RLock() peerVrf := peer.fsm.pConf.Config.Vrf peer.fsm.lock.RUnlock() + path.SetReceiveVrfId(rib.Vrfs[peerVrf].Id) path = path.ToGlobal(rib.Vrfs[peerVrf]) } @@ -1590,7 +1591,10 @@ func (s *BgpServer) EnableZebra(ctx context.Context, r *api.EnableZebraRequest) protos = append(protos, string(p)) } var err error - s.zclient, err = newZebraClient(s, r.Url, protos, uint8(r.Version), r.NexthopTriggerEnable, uint8(r.NexthopTriggerDelay)) + s.zclient, err = newZebraClient(s, r.Url, protos, uint8(r.Version), r.NexthopTriggerEnable, uint8(r.NexthopTriggerDelay), r.MplsLabelRangeSize) + if err == nil && s.globalRib != nil && s.zclient.client.Version >= 4 && r.MplsLabelRangeSize > 0 { + err = s.globalRib.EnableMplsLabelAllocation() + } return err }, false) } @@ -2029,6 +2033,26 @@ func (s *BgpServer) ListVrf(ctx context.Context, _ *api.ListVrfRequest, fn func( return nil } +func assignMplsLabel(s *BgpServer, vrf *table.Vrf) error { + var mplsLabel uint32 + var err error + for mplsLabel, err = s.globalRib.AssignMplsLabel(); err != nil; mplsLabel, err = s.globalRib.AssignMplsLabel() { + if err = s.zclient.client.SendGetLabelChunk(&zebra.GetLabelChunkBody{ChunkSize: s.zclient.mplsLabelRangeSize}); err != nil { + return err + } + labels := <-s.zclient.mplsLabelRangeCh + if err = s.globalRib.AllocateMplsLabelRange(labels[0], labels[1]); err != nil { + return err + } + } + vrf.MplsLabel = mplsLabel + err = s.zclient.client.SendVrfLabel(vrf.MplsLabel, vrf.Id) + if err != nil { + return err + } + return nil +} + func (s *BgpServer) AddVrf(ctx context.Context, r *api.AddVrfRequest) error { return s.mgmtOperation(func() error { if r == nil || r.Vrf == nil { @@ -2055,11 +2079,17 @@ func (s *BgpServer) AddVrf(ctx context.Context, r *api.AddVrfRequest) error { AS: s.bgpConfig.Global.Config.As, LocalID: net.ParseIP(s.bgpConfig.Global.Config.RouterId).To4(), } + if pathList, err := s.globalRib.AddVrf(name, id, rd, im, ex, pi); err != nil { return err } else if len(pathList) > 0 { s.propagateUpdate(nil, pathList) } + if vrf, ok := s.globalRib.Vrfs[name]; ok { + if s.zclient.mplsLabelRangeSize > 0 { + go assignMplsLabel(s, vrf) + } + } return nil }, true) } diff --git a/pkg/server/zclient.go b/pkg/server/zclient.go index 3e4fb3e6..f4276e5b 100644 --- a/pkg/server/zclient.go +++ b/pkg/server/zclient.go @@ -116,18 +116,19 @@ func filterOutExternalPath(paths []*table.Path) []*table.Path { return filteredPaths } -func newIPRouteBody(dst []*table.Path) (body *zebra.IPRouteBody, isWithdraw bool) { +func newIPRouteBody(dst []*table.Path, vrfId uint32, version uint8) (body *zebra.IPRouteBody, isWithdraw bool) { paths := filterOutExternalPath(dst) if len(paths) == 0 { return nil, false } path := paths[0] + receiveVrfId := path.ReceiveVrfId() l := strings.SplitN(path.GetNlri().String(), "/", 2) var prefix net.IP - //nexthops := make([]net.IP, 0, len(paths)) var nexthop zebra.Nexthop nexthops := make([]zebra.Nexthop, 0, len(paths)) + msgFlags := zebra.MESSAGE_NEXTHOP switch path.GetRouteFamily() { case bgp.RF_IPv4_UC, bgp.RF_IPv4_VPN: if path.GetRouteFamily() == bgp.RF_IPv4_UC { @@ -137,6 +138,14 @@ func newIPRouteBody(dst []*table.Path) (body *zebra.IPRouteBody, isWithdraw bool } for _, p := range paths { nexthop.Gate = p.GetNexthop().To4() + nexthop.VrfId = receiveVrfId + if version > 4 && path.GetRouteFamily() == bgp.RF_IPv4_VPN && receiveVrfId != vrfId { + msgFlags |= zebra.FRR_ZAPI5_MESSAGE_LABEL + for _, label := range path.GetNlri().(*bgp.LabeledVPNIPAddrPrefix).Labels.Labels { + nexthop.LabelNum++ + nexthop.MplsLabels = append(nexthop.MplsLabels, label) + } + } nexthops = append(nexthops, nexthop) } case bgp.RF_IPv6_UC, bgp.RF_IPv6_VPN: @@ -147,16 +156,27 @@ func newIPRouteBody(dst []*table.Path) (body *zebra.IPRouteBody, isWithdraw bool } for _, p := range paths { nexthop.Gate = p.GetNexthop().To16() + nexthop.VrfId = receiveVrfId + if version > 4 && path.GetRouteFamily() == bgp.RF_IPv6_VPN && receiveVrfId != vrfId { + msgFlags |= zebra.FRR_ZAPI5_MESSAGE_LABEL + for _, label := range path.GetNlri().(*bgp.LabeledVPNIPAddrPrefix).Labels.Labels { + nexthop.LabelNum++ + nexthop.MplsLabels = append(nexthop.MplsLabels, label) + } + } nexthops = append(nexthops, nexthop) } default: return nil, false } - msgFlags := zebra.MESSAGE_NEXTHOP plen, _ := strconv.ParseUint(l[1], 10, 8) med, err := path.GetMed() if err == nil { - msgFlags |= zebra.MESSAGE_METRIC + if version < 5 { + msgFlags |= zebra.MESSAGE_METRIC + } else { + msgFlags |= zebra.FRR_ZAPI5_MESSAGE_METRIC + } } var flags zebra.FLAG if path.IsIBGP() { @@ -282,10 +302,12 @@ func newPathFromIPRouteMessage(m *zebra.Message, version uint8) *table.Path { } type zebraClient struct { - client *zebra.Client - server *BgpServer - nexthopCache nexthopStateCache - dead chan struct{} + client *zebra.Client + server *BgpServer + nexthopCache nexthopStateCache + mplsLabelRangeCh chan [2]uint32 // {start, end} + mplsLabelRangeSize uint32 + dead chan struct{} } func (z *zebraClient) getPathListWithNexthopUpdate(body *zebra.NexthopUpdateBody) []*table.Path { @@ -336,6 +358,10 @@ func (z *zebraClient) loop() { }...) defer w.Stop() + if z.mplsLabelRangeSize > 0 { + defer close(z.mplsLabelRangeCh) + } + for { select { case <-z.dead: @@ -367,6 +393,8 @@ func (z *zebraClient) loop() { delete(z.nexthopCache, body.Prefix.Prefix.String()) } z.updatePathByNexthopCache(paths) + case *zebra.GetLabelChunkBody: + z.mplsLabelRangeCh <- [2]uint32{body.Start, body.End} } case ev := <-w.Event(): switch msg := ev.(type) { @@ -374,7 +402,7 @@ func (z *zebraClient) loop() { if table.UseMultiplePaths.Enabled { for _, paths := range msg.MultiPathList { z.updatePathByNexthopCache(paths) - if body, isWithdraw := newIPRouteBody(paths); body != nil { + if body, isWithdraw := newIPRouteBody(paths, 0, z.client.Version); body != nil { z.client.SendIPRoute(0, body, isWithdraw) } if body := newNexthopRegisterBody(paths, z.nexthopCache); body != nil { @@ -391,8 +419,15 @@ func (z *zebraClient) loop() { } } for _, i := range vrfs { - if body, isWithdraw := newIPRouteBody([]*table.Path{path}); body != nil { - z.client.SendIPRoute(i, body, isWithdraw) + routeFamily := path.GetRouteFamily() + if i == zebra.VRF_DEFAULT && (routeFamily == bgp.RF_IPv4_VPN || routeFamily == bgp.RF_IPv6_VPN) { + continue + } + if body, isWithdraw := newIPRouteBody([]*table.Path{path}, i, z.client.Version); body != nil { + err := z.client.SendIPRoute(i, body, isWithdraw) + if err != nil { + continue + } } if body := newNexthopRegisterBody([]*table.Path{path}, z.nexthopCache); body != nil { z.client.SendNexthopRegister(i, body, false) @@ -415,7 +450,7 @@ func (z *zebraClient) loop() { } } -func newZebraClient(s *BgpServer, url string, protos []string, version uint8, nhtEnable bool, nhtDelay uint8) (*zebraClient, error) { +func newZebraClient(s *BgpServer, url string, protos []string, version uint8, nhtEnable bool, nhtDelay uint8, mplsLabelRangeSize uint32) (*zebraClient, error) { l := strings.SplitN(url, ":", 2) if len(l) != 2 { return nil, fmt.Errorf("unsupported url: %s", url) @@ -468,10 +503,12 @@ func newZebraClient(s *BgpServer, url string, protos []string, version uint8, nh cli.SendRedistribute(t, zebra.VRF_DEFAULT) } w := &zebraClient{ - client: cli, - server: s, - nexthopCache: make(nexthopStateCache), - dead: make(chan struct{}), + client: cli, + server: s, + nexthopCache: make(nexthopStateCache), + mplsLabelRangeSize: mplsLabelRangeSize, + dead: make(chan struct{}), + mplsLabelRangeCh: make(chan [2]uint32), } go w.loop() return w, nil |