blob: 958ab19feea714499d27e860af27aa0322651c8b [file] [log] [blame]
manikkaraj kbf256be2019-03-25 00:13:48 +05301/*
Joey Armstrong11f5a572024-01-12 19:11:32 -05002 * Copyright 2018-2024 Open Networking Foundation (ONF) and the ONF Contributors
manikkaraj kbf256be2019-03-25 00:13:48 +05303
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7
8 * http://www.apache.org/licenses/LICENSE-2.0
9
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Joey Armstrong3f0e2422023-07-05 18:25:41 -040017// Package core provides the utility for olt devices, flows and statistics
Scott Bakerdbd960e2020-02-28 08:57:51 -080018package core
manikkaraj kbf256be2019-03-25 00:13:48 +053019
20import (
21 "context"
Matteo Scandolo6056e822019-11-13 14:05:29 -080022 "encoding/hex"
Girish Gowdracefae192020-03-19 18:14:10 -070023 "errors"
manikkaraj kbf256be2019-03-25 00:13:48 +053024 "fmt"
Gamze Abaka7650be62021-02-26 10:50:36 +000025 "strconv"
serkant.uluderya4aff1862020-09-17 23:35:26 +030026 "strings"
27 "sync"
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -070028 "time"
serkant.uluderya4aff1862020-09-17 23:35:26 +030029
khenaidoo106c61a2021-08-11 18:05:46 -040030 "github.com/opencord/voltha-lib-go/v7/pkg/meters"
Mahir Gunyel199570a2021-07-04 15:39:36 -070031
khenaidoo106c61a2021-08-11 18:05:46 -040032 "github.com/opencord/voltha-lib-go/v7/pkg/flows"
33 "github.com/opencord/voltha-lib-go/v7/pkg/log"
Mahir Gunyel85f61c12021-10-06 11:53:45 -070034 plt "github.com/opencord/voltha-lib-go/v7/pkg/platform"
khenaidoo106c61a2021-08-11 18:05:46 -040035 tp "github.com/opencord/voltha-lib-go/v7/pkg/techprofile"
Scott Bakerdbd960e2020-02-28 08:57:51 -080036 rsrcMgr "github.com/opencord/voltha-openolt-adapter/internal/pkg/resourcemanager"
khenaidoo106c61a2021-08-11 18:05:46 -040037 "github.com/opencord/voltha-protos/v5/go/common"
khenaidoodc2116e2021-10-19 17:33:19 -040038 ia "github.com/opencord/voltha-protos/v5/go/inter_adapter"
khenaidoo106c61a2021-08-11 18:05:46 -040039 ofp "github.com/opencord/voltha-protos/v5/go/openflow_13"
40 openoltpb2 "github.com/opencord/voltha-protos/v5/go/openolt"
41 tp_pb "github.com/opencord/voltha-protos/v5/go/tech_profile"
42 "github.com/opencord/voltha-protos/v5/go/voltha"
Chaitrashree G S579fe732019-08-20 20:50:47 -040043
Thomas Lee S94109f12020-03-03 16:39:29 +053044 "github.com/opencord/voltha-openolt-adapter/internal/pkg/olterrors"
Daniele Rossi22db98e2019-07-11 11:50:00 +000045 "google.golang.org/grpc/codes"
46 "google.golang.org/grpc/status"
manikkaraj kbf256be2019-03-25 00:13:48 +053047)
48
49const (
Akash Kankanala041a2122024-10-16 15:49:22 +053050 // IPProtoDhcp flow category
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070051 IPProtoDhcp = 17
manikkaraj kbf256be2019-03-25 00:13:48 +053052
Akash Kankanala041a2122024-10-16 15:49:22 +053053 // IgmpProto proto value
Girish Gowdraa09aeab2020-09-14 16:30:52 -070054 IgmpProto = 2
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070055
Akash Kankanala041a2122024-10-16 15:49:22 +053056 // EapEthType eapethtype value
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070057 EapEthType = 0x888e
Akash Kankanala041a2122024-10-16 15:49:22 +053058 // LldpEthType lldp ethtype value
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070059 LldpEthType = 0x88cc
Akash Kankanala041a2122024-10-16 15:49:22 +053060 // IPv4EthType IPv4 ethernet type value
Esin Karamanae41e2b2019-12-17 18:13:13 +000061 IPv4EthType = 0x800
Akash Kankanala041a2122024-10-16 15:49:22 +053062 // PPPoEDEthType PPPoE discovery ethernet type value
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -030063 PPPoEDEthType = 0x8863
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070064
Akash Kankanala041a2122024-10-16 15:49:22 +053065 // ReservedVlan Transparent Vlan (Masked Vlan, VLAN_ANY in ONOS Flows)
Andrea Campanella7acc0b92020-02-14 09:20:49 +010066 ReservedVlan = 4096
Harsh Awasthiea45af72019-08-26 02:39:00 -040067
Akash Kankanala041a2122024-10-16 15:49:22 +053068 // DefaultMgmtVlan default vlan value
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070069 DefaultMgmtVlan = 4091
manikkaraj kbf256be2019-03-25 00:13:48 +053070
manikkaraj kbf256be2019-03-25 00:13:48 +053071 // Openolt Flow
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070072
Akash Kankanala041a2122024-10-16 15:49:22 +053073 // Upstream constant
David K. Bainbridge82efc492019-09-04 09:57:11 -070074 Upstream = "upstream"
Akash Kankanala041a2122024-10-16 15:49:22 +053075 // Downstream constant
David K. Bainbridge82efc492019-09-04 09:57:11 -070076 Downstream = "downstream"
Akash Kankanala041a2122024-10-16 15:49:22 +053077 // Multicast constant
Esin Karamanccb714b2019-11-29 15:02:06 +000078 Multicast = "multicast"
Akash Kankanala041a2122024-10-16 15:49:22 +053079 // PacketTagType constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070080 PacketTagType = "pkt_tag_type"
Akash Kankanala041a2122024-10-16 15:49:22 +053081 // Untagged constant
David K. Bainbridge82efc492019-09-04 09:57:11 -070082 Untagged = "untagged"
Akash Kankanala041a2122024-10-16 15:49:22 +053083 // SingleTag constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070084 SingleTag = "single_tag"
Akash Kankanala041a2122024-10-16 15:49:22 +053085 // DoubleTag constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070086 DoubleTag = "double_tag"
manikkaraj kbf256be2019-03-25 00:13:48 +053087
88 // classifierInfo
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070089
Akash Kankanala041a2122024-10-16 15:49:22 +053090 // EthType constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070091 EthType = "eth_type"
Akash Kankanala041a2122024-10-16 15:49:22 +053092 // EthDst constant
Esin Karamanccb714b2019-11-29 15:02:06 +000093 EthDst = "eth_dst"
Akash Kankanala041a2122024-10-16 15:49:22 +053094 // EthSrc constant
Girish Gowdraffa52e52022-02-16 15:48:10 -080095 EthSrc = "eth_src"
Akash Kankanala041a2122024-10-16 15:49:22 +053096 // TPID constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070097 TPID = "tpid"
Akash Kankanala041a2122024-10-16 15:49:22 +053098 // IPProto constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -070099 IPProto = "ip_proto"
Akash Kankanala041a2122024-10-16 15:49:22 +0530100 // InPort constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700101 InPort = "in_port"
Akash Kankanala041a2122024-10-16 15:49:22 +0530102 // VlanVid constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700103 VlanVid = "vlan_vid"
Akash Kankanala041a2122024-10-16 15:49:22 +0530104 // VlanPcp constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700105 VlanPcp = "vlan_pcp"
106
Akash Kankanala041a2122024-10-16 15:49:22 +0530107 // UDPDst constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700108 UDPDst = "udp_dst"
Akash Kankanala041a2122024-10-16 15:49:22 +0530109 // UDPSrc constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700110 UDPSrc = "udp_src"
Akash Kankanala041a2122024-10-16 15:49:22 +0530111 // Ipv4Dst constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700112 Ipv4Dst = "ipv4_dst"
Akash Kankanala041a2122024-10-16 15:49:22 +0530113 // Ipv4Src constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700114 Ipv4Src = "ipv4_src"
Akash Kankanala041a2122024-10-16 15:49:22 +0530115 // Metadata constant
David K. Bainbridge82efc492019-09-04 09:57:11 -0700116 Metadata = "metadata"
Akash Kankanala041a2122024-10-16 15:49:22 +0530117 // TunnelID constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700118 TunnelID = "tunnel_id"
Akash Kankanala041a2122024-10-16 15:49:22 +0530119 // Output constant
David K. Bainbridge82efc492019-09-04 09:57:11 -0700120 Output = "output"
Akash Kankanala041a2122024-10-16 15:49:22 +0530121 // GroupID constant
Esin Karamanccb714b2019-11-29 15:02:06 +0000122 GroupID = "group_id"
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700123 // Actions
124
Akash Kankanala041a2122024-10-16 15:49:22 +0530125 // PopVlan constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700126 PopVlan = "pop_vlan"
Akash Kankanala041a2122024-10-16 15:49:22 +0530127 // PushVlan constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700128 PushVlan = "push_vlan"
Akash Kankanala041a2122024-10-16 15:49:22 +0530129 // TrapToHost constant
Girish Gowdru6a80bbd2019-07-02 07:36:09 -0700130 TrapToHost = "trap_to_host"
Akash Kankanala041a2122024-10-16 15:49:22 +0530131 // MaxMeterBand constant
Manikkaraj kb1d51442019-07-23 10:41:02 -0400132 MaxMeterBand = 2
Akash Kankanala041a2122024-10-16 15:49:22 +0530133 // VlanPCPMask contant
Manikkaraj kb1d51442019-07-23 10:41:02 -0400134 VlanPCPMask = 0xFF
Akash Kankanala041a2122024-10-16 15:49:22 +0530135 // VlanvIDMask constant
Manikkaraj kb1d51442019-07-23 10:41:02 -0400136 VlanvIDMask = 0xFFF
Akash Kankanala041a2122024-10-16 15:49:22 +0530137 // IntfID constant
Gamze Abakafee36392019-10-03 11:17:24 +0000138 IntfID = "intfId"
Akash Kankanala041a2122024-10-16 15:49:22 +0530139 // OnuID constant
Gamze Abakafee36392019-10-03 11:17:24 +0000140 OnuID = "onuId"
Akash Kankanala041a2122024-10-16 15:49:22 +0530141 // UniID constant
Gamze Abakafee36392019-10-03 11:17:24 +0000142 UniID = "uniId"
Akash Kankanala041a2122024-10-16 15:49:22 +0530143 // PortNo constant
Gamze Abakafee36392019-10-03 11:17:24 +0000144 PortNo = "portNo"
Akash Kankanala041a2122024-10-16 15:49:22 +0530145 // AllocID constant
Gamze Abakafee36392019-10-03 11:17:24 +0000146 AllocID = "allocId"
Akash Kankanala041a2122024-10-16 15:49:22 +0530147 // GemID constant
Gamze Abakafcbd6e72020-12-17 13:25:16 +0000148 GemID = "gemId"
Esin Karamanccb714b2019-11-29 15:02:06 +0000149
Akash Kankanala041a2122024-10-16 15:49:22 +0530150 // NoneOnuID constant
Esin Karamanccb714b2019-11-29 15:02:06 +0000151 NoneOnuID = -1
Akash Kankanala041a2122024-10-16 15:49:22 +0530152 // NoneUniID constant
Esin Karamanccb714b2019-11-29 15:02:06 +0000153 NoneUniID = -1
Matteo Scandolo738c52a2020-08-03 11:14:22 -0700154
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700155 // Max number of flows that can be queued per ONU
156 maxConcurrentFlowsPerOnu = 20
manikkaraj kbf256be2019-03-25 00:13:48 +0530157
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700158 bitMapPrefix = "0b"
159 pbit1 = '1'
160)
manikkaraj k9eb6cac2019-05-09 12:32:03 -0400161
Gamze Abakafee36392019-10-03 11:17:24 +0000162type schedQueue struct {
Akash Kankanala041a2122024-10-16 15:49:22 +0530163 tpInst interface{}
164 flowMetadata *ofp.FlowMetadata
Gamze Abakafee36392019-10-03 11:17:24 +0000165 direction tp_pb.Direction
166 intfID uint32
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530167 nniIntfID uint32
Gamze Abakafee36392019-10-03 11:17:24 +0000168 onuID uint32
169 uniID uint32
170 tpID uint32
171 uniPort uint32
Gamze Abakafee36392019-10-03 11:17:24 +0000172 meterID uint32
Gamze Abakafee36392019-10-03 11:17:24 +0000173}
174
Gamze Abaka7650be62021-02-26 10:50:36 +0000175type flowContext struct {
Akash Kankanala041a2122024-10-16 15:49:22 +0530176 classifier map[string]interface{}
177 action map[string]interface{}
178 logicalFlow *ofp.OfpFlowStats
179 pbitToGem map[uint32]uint32
180 gemToAes map[uint32]bool
Gamze Abaka7650be62021-02-26 10:50:36 +0000181 intfID uint32
182 onuID uint32
183 uniID uint32
184 portNo uint32
Gamze Abaka7650be62021-02-26 10:50:36 +0000185 allocID uint32
186 gemPortID uint32
187 tpID uint32
Gamze Abaka7650be62021-02-26 10:50:36 +0000188}
189
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700190// This control block is created per flow add/remove and pushed on the incomingFlows channel slice
191// The flowControlBlock is then picked by the perOnuFlowHandlerRoutine for further processing.
192// There is on perOnuFlowHandlerRoutine routine per ONU that constantly monitors for any incoming
193// flow and processes it serially
194type flowControlBlock struct {
khenaidoodc2116e2021-10-19 17:33:19 -0400195 ctx context.Context // Flow handler context
khenaidoodc2116e2021-10-19 17:33:19 -0400196 flow *ofp.OfpFlowStats // Flow message
197 flowMetadata *ofp.FlowMetadata // FlowMetadata that contains flow meter information. This can be nil for Flow remove
198 errChan *chan error // channel to report the Flow handling error
Akash Kankanala041a2122024-10-16 15:49:22 +0530199 addFlow bool // if true flow to be added, else removed
Esin Karamanccb714b2019-11-29 15:02:06 +0000200}
201
Joey Armstrong3f0e2422023-07-05 18:25:41 -0400202// OpenOltFlowMgr creates the Structure of OpenOltFlowMgr obj
manikkaraj kbf256be2019-03-25 00:13:48 +0530203type OpenOltFlowMgr struct {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700204 techprofile tp.TechProfileIf
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700205 deviceHandler *DeviceHandler
206 grpMgr *OpenOltGroupMgr
207 resourceMgr *rsrcMgr.OpenOltResourceMgr
208
Akash Kankanala041a2122024-10-16 15:49:22 +0530209 packetInGemPort map[rsrcMgr.PacketInInfoKey]uint32 // packet in gem port local cache
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700210
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700211 // Slice of channels. Each channel in slice, index by ONU ID, queues flows per ONU.
212 // A go routine per ONU, waits on the unique channel (indexed by ONU ID) for incoming flows (add/remove)
Girish Gowdra4736e5c2021-08-25 15:19:10 -0700213 incomingFlows []chan flowControlBlock
214 stopFlowHandlerRoutine []chan bool
215 flowHandlerRoutineActive []bool
Akash Kankanala041a2122024-10-16 15:49:22 +0530216 packetInGemPortLock sync.RWMutex
217
218 ponPortIdx uint32 // Pon Port this FlowManager is responsible for
manikkaraj kbf256be2019-03-25 00:13:48 +0530219}
220
Holger Hildebrandt143b5be2023-02-10 08:28:15 +0000221// CloseKVClient closes open KV clients
222func (f *OpenOltFlowMgr) CloseKVClient(ctx context.Context) {
223 if f.techprofile != nil {
224 f.techprofile.CloseKVClient(ctx)
225 }
226}
227
Joey Armstrong3f0e2422023-07-05 18:25:41 -0400228// NewFlowManager creates OpenOltFlowMgr object and initializes the parameters
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700229func NewFlowManager(ctx context.Context, dh *DeviceHandler, rMgr *rsrcMgr.OpenOltResourceMgr, grpMgr *OpenOltGroupMgr, ponPortIdx uint32) *OpenOltFlowMgr {
Neha Sharma96b7bf22020-06-15 10:37:32 +0000230 logger.Infow(ctx, "initializing-flow-manager", log.Fields{"device-id": dh.device.Id})
manikkaraj kbf256be2019-03-25 00:13:48 +0530231 var flowMgr OpenOltFlowMgr
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +0530232 var err error
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +0530233
manikkaraj kbf256be2019-03-25 00:13:48 +0530234 flowMgr.deviceHandler = dh
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700235 flowMgr.ponPortIdx = ponPortIdx
Girish Gowdra9602eb42020-09-09 15:50:39 -0700236 flowMgr.grpMgr = grpMgr
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +0530237 flowMgr.resourceMgr = rMgr
yasin saplid0566272021-12-21 09:10:30 +0000238 // dh.totalPonPorts is reserved for NNI trap flows. It doesn't need a tech profile
239 if ponPortIdx != dh.totalPonPorts {
240 if err = flowMgr.populateTechProfileForCurrentPonPort(ctx); err != nil {
241 logger.Errorw(ctx, "error-while-populating-tech-profile-mgr", log.Fields{"err": err})
242 return nil
243 }
manikkaraj kbf256be2019-03-25 00:13:48 +0530244 }
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +0530245 flowMgr.packetInGemPort = make(map[rsrcMgr.PacketInInfoKey]uint32)
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700246
247 // Create a slice of buffered channels for handling concurrent flows per ONU.
248 // The additional entry (+1) is to handle the NNI trap flows on a separate channel from individual ONUs channel
Mahir Gunyel85f61c12021-10-06 11:53:45 -0700249 flowMgr.incomingFlows = make([]chan flowControlBlock, plt.MaxOnusPerPon+1)
250 flowMgr.stopFlowHandlerRoutine = make([]chan bool, plt.MaxOnusPerPon+1)
251 flowMgr.flowHandlerRoutineActive = make([]bool, plt.MaxOnusPerPon+1)
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700252 for i := range flowMgr.incomingFlows {
253 flowMgr.incomingFlows[i] = make(chan flowControlBlock, maxConcurrentFlowsPerOnu)
Girish Gowdraae56c722021-11-22 14:31:11 -0800254 flowMgr.stopFlowHandlerRoutine[i] = make(chan bool)
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700255 // Spin up a go routine to handling incoming flows (add/remove).
256 // There will be on go routine per ONU.
257 // This routine will be blocked on the flowMgr.incomingFlows[onu-id] channel for incoming flows.
Girish Gowdra4736e5c2021-08-25 15:19:10 -0700258 flowMgr.flowHandlerRoutineActive[i] = true
259 go flowMgr.perOnuFlowHandlerRoutine(i, flowMgr.incomingFlows[i], flowMgr.stopFlowHandlerRoutine[i])
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700260 }
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700261
Akash Kankanala041a2122024-10-16 15:49:22 +0530262 // load interface to multicast queue map from kv store
Girish Gowdra9602eb42020-09-09 15:50:39 -0700263 flowMgr.grpMgr.LoadInterfaceToMulticastQueueMap(ctx)
balaji.nagarajan31db4ea2026-01-28 09:13:55 +0530264 logger.Debugw(ctx, "initialization-of-flow-manager-success", log.Fields{"device-id": dh.device.Id})
manikkaraj kbf256be2019-03-25 00:13:48 +0530265 return &flowMgr
266}
267
Kent Hagermane6ff1012020-07-14 15:07:53 -0400268func (f *OpenOltFlowMgr) registerFlow(ctx context.Context, flowFromCore *ofp.OfpFlowStats, deviceFlow *openoltpb2.Flow) error {
yasin saplid0566272021-12-21 09:10:30 +0000269 // In case of nni trap flow
270 if deviceFlow.AccessIntfId == -1 {
yasin saplibddc2d72022-02-08 13:10:17 +0000271 return f.resourceMgr.RegisterFlowIDForGem(ctx, uint32(deviceFlow.AccessIntfId), flowFromCore)
yasin saplid0566272021-12-21 09:10:30 +0000272 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700273 if !deviceFlow.ReplicateFlow && deviceFlow.GemportId > 0 {
274 // Flow is not replicated in this case, we need to register the flow for a single gem-port
yasin saplibddc2d72022-02-08 13:10:17 +0000275 return f.resourceMgr.RegisterFlowIDForGem(ctx, uint32(deviceFlow.GemportId), flowFromCore)
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700276 } else if deviceFlow.ReplicateFlow && len(deviceFlow.PbitToGemport) > 0 {
277 // Flow is replicated in this case. We need to register the flow for all the gem-ports it is replicated to.
278 for _, gemPort := range deviceFlow.PbitToGemport {
yasin saplibddc2d72022-02-08 13:10:17 +0000279 if err := f.resourceMgr.RegisterFlowIDForGem(ctx, gemPort, flowFromCore); err != nil {
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700280 return err
281 }
Matteo Scandolo738c52a2020-08-03 11:14:22 -0700282 }
Gamze Abakafee36392019-10-03 11:17:24 +0000283 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700284 return nil
285}
286
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530287func (f *OpenOltFlowMgr) processAddFlow(ctx context.Context, intfID uint32, nni_port uint32, onuID uint32, uniID uint32, portNo uint32,
salmansiddiqui7ac62132019-08-22 03:58:50 +0000288 classifierInfo map[string]interface{}, actionInfo map[string]interface{}, flow *ofp.OfpFlowStats, TpID uint32,
khenaidoodc2116e2021-10-19 17:33:19 -0400289 UsMeterID uint32, DsMeterID uint32, flowMetadata *ofp.FlowMetadata) error {
Gamze Abakafee36392019-10-03 11:17:24 +0000290 var allocID uint32
manikkaraj kbf256be2019-03-25 00:13:48 +0530291 var gemPorts []uint32
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700292 var TpInst interface{}
manikkaraj kbf256be2019-03-25 00:13:48 +0530293
Neha Sharma96b7bf22020-06-15 10:37:32 +0000294 logger.Infow(ctx, "dividing-flow", log.Fields{
Shrey Baid26912972020-04-16 21:02:31 +0530295 "device-id": f.deviceHandler.device.Id,
296 "intf-id": intfID,
297 "onu-id": onuID,
298 "uni-id": uniID,
299 "port-no": portNo,
Matteo Scandolod625b4c2020-04-02 16:16:01 -0700300 "classifier": classifierInfo,
Shrey Baid26912972020-04-16 21:02:31 +0530301 "action": actionInfo,
302 "usmeter-iD": UsMeterID,
303 "dsmeter-iD": DsMeterID,
304 "tp-id": TpID})
Matt Jeanneret77199612019-07-26 18:08:35 -0400305 // only create tcont/gemports if there is actually an onu id. otherwise BAL throws an error. Usually this
306 // is because the flow is an NNI flow and there would be no onu resources associated with it
307 // TODO: properly deal with NNI flows
Kent Hagermane6ff1012020-07-14 15:07:53 -0400308 if onuID == 0 {
Andrea Campanellabfe08432020-09-11 17:07:03 +0200309 cause := "no-onu-id-for-flow"
310 fields := log.Fields{
311 "onu": onuID,
312 "port-no": portNo,
313 "classifer": classifierInfo,
314 "action": actionInfo,
315 "device-id": f.deviceHandler.device.Id}
316 logger.Errorw(ctx, cause, fields)
317 return olterrors.NewErrNotFound(cause, fields, nil)
manikkaraj kbf256be2019-03-25 00:13:48 +0530318 }
319
Matteo Scandolod625b4c2020-04-02 16:16:01 -0700320 uni := getUniPortPath(f.deviceHandler.device.Id, intfID, int32(onuID), int32(uniID))
Neha Sharma96b7bf22020-06-15 10:37:32 +0000321 logger.Debugw(ctx, "uni-port-path", log.Fields{
Shrey Baid26912972020-04-16 21:02:31 +0530322 "uni": uni,
323 "device-id": f.deviceHandler.device.Id})
Girish Gowdra3d633032019-12-10 16:37:05 +0530324
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700325 logger.Debugw(ctx, "dividing-flow-create-tcont-gem-ports", log.Fields{
326 "device-id": f.deviceHandler.device.Id,
327 "intf-id": intfID,
328 "onu-id": onuID,
329 "uni-id": uniID,
330 "port-no": portNo,
331 "classifier": classifierInfo,
332 "action": actionInfo,
333 "usmeter-id": UsMeterID,
334 "dsmeter-id": DsMeterID,
335 "tp-id": TpID})
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530336 allocID, gemPorts, TpInst = f.createTcontGemports(ctx, intfID, nni_port, onuID, uniID, uni, portNo, TpID, UsMeterID, DsMeterID, flowMetadata)
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700337 if allocID == 0 || gemPorts == nil || TpInst == nil {
338 logger.Error(ctx, "alloc-id-gem-ports-tp-unavailable")
339 return olterrors.NewErrNotFound(
340 "alloc-id-gem-ports-tp-unavailable",
341 nil, nil)
Manikkaraj kb1d51442019-07-23 10:41:02 -0400342 }
Girish Gowdrafb3d6102020-10-16 16:32:36 -0700343 args := make(map[string]uint32)
344 args[IntfID] = intfID
345 args[OnuID] = onuID
346 args[UniID] = uniID
347 args[PortNo] = portNo
348 args[AllocID] = allocID
349
350 /* Flows can be added specific to gemport if p-bits are received.
351 * If no pbit mentioned then adding flows for all gemports
352 */
Gamze Abaka6d0a64f2021-11-18 08:08:33 +0000353 return f.checkAndAddFlow(ctx, args, classifierInfo, actionInfo, flow, TpInst, gemPorts, TpID, uni)
manikkaraj kbf256be2019-03-25 00:13:48 +0530354}
355
salmansiddiqui7ac62132019-08-22 03:58:50 +0000356// CreateSchedulerQueues creates traffic schedulers on the device with the given scheduler configuration and traffic shaping info
Girish Gowdraf3728b12022-02-02 21:46:51 -0800357// nolint: gocyclo
npujarec5762e2020-01-01 14:08:48 +0530358func (f *OpenOltFlowMgr) CreateSchedulerQueues(ctx context.Context, sq schedQueue) error {
nikesh.krishnan81ec7442023-10-31 17:19:34 +0530359 logger.Debugw(ctx, "CreateSchedulerQueues",
Shrey Baid26912972020-04-16 21:02:31 +0530360 log.Fields{"dir": sq.direction,
361 "intf-id": sq.intfID,
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530362 "nniIntfID": sq.nniIntfID,
Shrey Baid26912972020-04-16 21:02:31 +0530363 "onu-id": sq.onuID,
364 "uni-id": sq.uniID,
365 "tp-id": sq.tpID,
366 "meter-id": sq.meterID,
367 "tp-inst": sq.tpInst,
368 "flowmetadata": sq.flowMetadata,
369 "device-id": f.deviceHandler.device.Id})
Manikkaraj kb1d51442019-07-23 10:41:02 -0400370
Girish Gowdra6071f382021-12-14 12:52:04 +0530371 direction, err := verifyMeterIDAndGetDirection(sq.meterID, sq.direction)
salmansiddiqui7ac62132019-08-22 03:58:50 +0000372 if err != nil {
373 return err
Manikkaraj kb1d51442019-07-23 10:41:02 -0400374 }
375
Girish Gowdraf3728b12022-02-02 21:46:51 -0800376 var TrafficShaping *tp_pb.TrafficShapingInfo
377 if sq.flowMetadata == nil || len(sq.flowMetadata.Meters) != 1 {
378 return olterrors.NewErrInvalidValue(log.Fields{
379 "reason": "invalid-meter-config",
380 "meter-id": sq.meterID,
381 "device-id": f.deviceHandler.device.Id}, nil)
382 }
383
384 if TrafficShaping, err = meters.GetTrafficShapingInfo(ctx, sq.flowMetadata.Meters[0]); err != nil {
385 return olterrors.NewErrInvalidValue(log.Fields{
386 "reason": "invalid-meter-config",
387 "meter-id": sq.meterID,
388 "device-id": f.deviceHandler.device.Id}, nil)
389 }
390
391 var SchedCfg *tp_pb.SchedulerConfig
mgouda86543582025-10-29 20:58:16 +0530392 switch sq.direction {
393 case tp_pb.Direction_UPSTREAM:
Girish Gowdraf3728b12022-02-02 21:46:51 -0800394 SchedCfg = f.techprofile.GetUsScheduler(sq.tpInst.(*tp_pb.TechProfileInstance))
mgouda86543582025-10-29 20:58:16 +0530395 case tp_pb.Direction_DOWNSTREAM:
Girish Gowdraf3728b12022-02-02 21:46:51 -0800396 SchedCfg = f.techprofile.GetDsScheduler(sq.tpInst.(*tp_pb.TechProfileInstance))
397 }
bseeniva0b9cbcb2026-02-12 19:11:11 +0530398 trafficSched := f.techprofile.GetTrafficScheduler(sq.tpInst.(*tp_pb.TechProfileInstance), SchedCfg, TrafficShaping)
399 trafficSched.TechProfileId = sq.tpID
400 TrafficSched := []*tp_pb.TrafficScheduler{trafficSched}
Girish Gowdraf3728b12022-02-02 21:46:51 -0800401
Manikkaraj kb1d51442019-07-23 10:41:02 -0400402 /* Lets make a simple assumption that if the meter-id is present on the KV store,
403 * then the scheduler and queues configuration is applied on the OLT device
404 * in the given direction.
405 */
yasin saplibddc2d72022-02-08 13:10:17 +0000406 meterInfo, err := f.resourceMgr.GetMeterInfoForOnu(ctx, direction, sq.onuID, sq.uniID, sq.tpID)
Manikkaraj kb1d51442019-07-23 10:41:02 -0400407 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +0530408 return olterrors.NewErrNotFound("meter",
409 log.Fields{"intf-id": sq.intfID,
410 "onu-id": sq.onuID,
411 "uni-id": sq.uniID,
412 "device-id": f.deviceHandler.device.Id}, err)
Manikkaraj kb1d51442019-07-23 10:41:02 -0400413 }
Girish Kumarf26e4882020-03-05 06:49:10 +0000414
Girish Gowdraf3728b12022-02-02 21:46:51 -0800415 // update reference count and return if the meter was already installed before
Girish Gowdra6071f382021-12-14 12:52:04 +0530416 if meterInfo != nil && meterInfo.MeterID == sq.meterID {
Girish Gowdraf3728b12022-02-02 21:46:51 -0800417 logger.Infow(ctx, "scheduler-already-created-for-direction",
Girish Gowdra6071f382021-12-14 12:52:04 +0530418 log.Fields{"device-id": f.deviceHandler.device.Id, "direction": direction, "meter-id": sq.meterID})
balaji.nagarajan29be8bc2026-01-26 16:31:41 +0530419 if err = f.resourceMgr.HandleMeterInfoRefCntUpdate(ctx, meterInfo, direction, sq.onuID, sq.uniID, sq.tpID, true); err != nil {
Girish Gowdraf3728b12022-02-02 21:46:51 -0800420 return err
421 }
422
423 if allocExists := f.isAllocUsedByAnotherUNI(ctx, sq); allocExists {
424 // Alloc object was already created as part of flow setup on another uni of the onu for the same service.
425 // Just create gem ports and traffic queues on the current uni for the given service
426 logger.Infow(ctx, "alloc in use on another uni, schedulers already created, creating queues only",
427 log.Fields{"intf-id": sq.intfID,
428 "onu-id": sq.onuID,
429 "uni-id": sq.uniID,
430 "tp-id": sq.tpID,
431 "device-id": f.deviceHandler.device.Id})
432 // The upstream scheduler is already created. We only need to create the queues
433 // If there are multiple upstream flows on a given uni, then it is possible that
434 // we call pushTrafficQueues multiple times, but that is OK as BAL returns OK.
435 // TODO: Find better mechanism to not duplicate request.
436 if err = f.pushTrafficQueues(ctx, sq, TrafficSched); err != nil {
437 return olterrors.NewErrAdapter("failure-pushing-traffic-queues-to-device",
438 log.Fields{"intf-id": sq.intfID,
439 "direction": sq.direction,
440 "device-id": f.deviceHandler.device.Id}, err)
441 }
442 } else {
443 logger.Infow(ctx, "alloc not in use on another uni, only meter ref cnt updated",
444 log.Fields{"intf-id": sq.intfID,
445 "onu-id": sq.onuID,
446 "uni-id": sq.uniID,
447 "tp-id": sq.tpID,
448 "device-id": f.deviceHandler.device.Id})
449 }
450 return err
Manikkaraj kb1d51442019-07-23 10:41:02 -0400451 }
Girish Kumar8f73fe02019-12-09 13:19:37 +0000452
Neha Sharma96b7bf22020-06-15 10:37:32 +0000453 logger.Debugw(ctx, "meter-does-not-exist-creating-new",
Shrey Baid26912972020-04-16 21:02:31 +0530454 log.Fields{
455 "meter-id": sq.meterID,
Girish Gowdra6071f382021-12-14 12:52:04 +0530456 "direction": direction,
Shrey Baid26912972020-04-16 21:02:31 +0530457 "device-id": f.deviceHandler.device.Id})
Girish Kumar8f73fe02019-12-09 13:19:37 +0000458
Girish Gowdraa482f272021-03-24 23:04:19 -0700459 found := false
460 meterInfo = &rsrcMgr.MeterInfo{}
Gamze Abakafee36392019-10-03 11:17:24 +0000461 if sq.flowMetadata != nil {
462 for _, meter := range sq.flowMetadata.Meters {
463 if sq.meterID == meter.MeterId {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700464 meterInfo.MeterID = meter.MeterId
Girish Gowdraa482f272021-03-24 23:04:19 -0700465 meterInfo.RefCnt = 1 // initialize it to 1, since this is the first flow that referenced the meter id.
Neha Sharma96b7bf22020-06-15 10:37:32 +0000466 logger.Debugw(ctx, "found-meter-config-from-flowmetadata",
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700467 log.Fields{"meter": meter,
Shrey Baid26912972020-04-16 21:02:31 +0530468 "device-id": f.deviceHandler.device.Id})
Girish Gowdraa482f272021-03-24 23:04:19 -0700469 found = true
Manikkaraj kb1d51442019-07-23 10:41:02 -0400470 break
471 }
472 }
473 } else {
Neha Sharma96b7bf22020-06-15 10:37:32 +0000474 logger.Errorw(ctx, "flow-metadata-not-present-in-flow", log.Fields{"device-id": f.deviceHandler.device.Id})
Manikkaraj kb1d51442019-07-23 10:41:02 -0400475 }
Girish Gowdraa482f272021-03-24 23:04:19 -0700476 if !found {
Thomas Lee S94109f12020-03-03 16:39:29 +0530477 return olterrors.NewErrNotFound("meterbands", log.Fields{
David K. Bainbridge794735f2020-02-11 21:01:37 -0800478 "reason": "Could-not-get-meterbands-from-flowMetadata",
479 "flow-metadata": sq.flowMetadata,
Shrey Baid26912972020-04-16 21:02:31 +0530480 "meter-id": sq.meterID,
481 "device-id": f.deviceHandler.device.Id}, nil)
Manikkaraj kb1d51442019-07-23 10:41:02 -0400482 }
Gamze Abaka01174422021-03-10 06:55:27 +0000483
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700484 if err := f.pushSchedulerQueuesToDevice(ctx, sq, TrafficSched); err != nil {
Shrey Baid26912972020-04-16 21:02:31 +0530485 return olterrors.NewErrAdapter("failure-pushing-traffic-scheduler-and-queues-to-device",
486 log.Fields{"intf-id": sq.intfID,
487 "direction": sq.direction,
488 "device-id": f.deviceHandler.device.Id}, err)
Manikkaraj kb1d51442019-07-23 10:41:02 -0400489 }
490
salmansiddiqui7ac62132019-08-22 03:58:50 +0000491 /* After we successfully applied the scheduler configuration on the OLT device,
Manikkaraj kb1d51442019-07-23 10:41:02 -0400492 * store the meter id on the KV store, for further reference.
493 */
yasin saplibddc2d72022-02-08 13:10:17 +0000494 if err := f.resourceMgr.StoreMeterInfoForOnu(ctx, direction, sq.onuID, sq.uniID, sq.tpID, meterInfo); err != nil {
Shrey Baid26912972020-04-16 21:02:31 +0530495 return olterrors.NewErrAdapter("failed-updating-meter-id",
496 log.Fields{"onu-id": sq.onuID,
497 "meter-id": sq.meterID,
498 "device-id": f.deviceHandler.device.Id}, err)
Manikkaraj kb1d51442019-07-23 10:41:02 -0400499 }
Neha Sharma96b7bf22020-06-15 10:37:32 +0000500 logger.Infow(ctx, "updated-meter-info-into-kv-store-successfully",
Girish Gowdra6071f382021-12-14 12:52:04 +0530501 log.Fields{"direction": direction,
Girish Gowdraa482f272021-03-24 23:04:19 -0700502 "meter-info": meterInfo,
503 "device-id": f.deviceHandler.device.Id})
Manikkaraj kb1d51442019-07-23 10:41:02 -0400504 return nil
505}
506
Girish Gowdraf3728b12022-02-02 21:46:51 -0800507func (f *OpenOltFlowMgr) pushTrafficQueues(ctx context.Context, sq schedQueue, TrafficSched []*tp_pb.TrafficScheduler) error {
508 trafficQueues, err := f.techprofile.GetTrafficQueues(ctx, sq.tpInst.(*tp_pb.TechProfileInstance), sq.direction)
509 if err != nil {
510 return olterrors.NewErrAdapter("unable-to-construct-traffic-queue-configuration",
511 log.Fields{"intf-id": sq.intfID,
512 "direction": sq.direction,
513 "device-id": f.deviceHandler.device.Id}, err)
514 }
515 logger.Debugw(ctx, "sending-traffic-queues-create-to-device",
516 log.Fields{"direction": sq.direction,
517 "traffic-queues": trafficQueues,
518 "device-id": f.deviceHandler.device.Id})
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530519 queues := &tp_pb.TrafficQueues{IntfId: sq.intfID, NetworkIntfId: sq.nniIntfID, OnuId: sq.onuID,
Girish Gowdraf3728b12022-02-02 21:46:51 -0800520 UniId: sq.uniID, PortNo: sq.uniPort,
521 TrafficQueues: trafficQueues,
522 TechProfileId: TrafficSched[0].TechProfileId}
bseenivaa1622112025-12-11 18:24:02 +0530523 subCtx1, cancel1 := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
524 if _, err = f.deviceHandler.Client.CreateTrafficQueues(subCtx1, queues); err != nil {
525 cancel1()
Girish Gowdraf3728b12022-02-02 21:46:51 -0800526 if len(queues.TrafficQueues) > 1 {
527 logger.Debug(ctx, "removing-queues-for-1tcont-multi-gem", log.Fields{"intfID": sq.intfID, "onuID": sq.onuID, "dir": sq.direction})
bseenivaa1622112025-12-11 18:24:02 +0530528 subCtx2, cancel2 := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
529 _, _ = f.deviceHandler.Client.RemoveTrafficQueues(subCtx2, queues)
530 cancel2()
Girish Gowdraf3728b12022-02-02 21:46:51 -0800531 }
532 return olterrors.NewErrAdapter("failed-to-create-traffic-queues-in-device", log.Fields{"traffic-queues": trafficQueues}, err)
533 }
bseenivaa1622112025-12-11 18:24:02 +0530534 cancel1()
Girish Gowdraf3728b12022-02-02 21:46:51 -0800535 return err
536}
537
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700538func (f *OpenOltFlowMgr) pushSchedulerQueuesToDevice(ctx context.Context, sq schedQueue, TrafficSched []*tp_pb.TrafficScheduler) error {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700539 trafficQueues, err := f.techprofile.GetTrafficQueues(ctx, sq.tpInst.(*tp_pb.TechProfileInstance), sq.direction)
Girish Kumar8f73fe02019-12-09 13:19:37 +0000540
541 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +0530542 return olterrors.NewErrAdapter("unable-to-construct-traffic-queue-configuration",
543 log.Fields{"intf-id": sq.intfID,
544 "direction": sq.direction,
545 "device-id": f.deviceHandler.device.Id}, err)
Girish Kumar8f73fe02019-12-09 13:19:37 +0000546 }
547
Gamze Abakacb0e6772021-06-10 08:32:12 +0000548 if allocExists := f.isAllocUsedByAnotherUNI(ctx, sq); !allocExists {
549 logger.Debugw(ctx, "sending-traffic-scheduler-create-to-device",
550 log.Fields{
551 "direction": sq.direction,
552 "TrafficScheds": TrafficSched,
553 "device-id": f.deviceHandler.device.Id,
554 "intfID": sq.intfID,
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530555 "nniIntfID": sq.nniIntfID,
Gamze Abakacb0e6772021-06-10 08:32:12 +0000556 "onuID": sq.onuID,
557 "uniID": sq.uniID})
bseenivaa1622112025-12-11 18:24:02 +0530558 subCtx, cancel := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
559 if _, err := f.deviceHandler.Client.CreateTrafficSchedulers(subCtx, &tp_pb.TrafficSchedulers{
Gamze Abakacb0e6772021-06-10 08:32:12 +0000560 IntfId: sq.intfID, OnuId: sq.onuID,
561 UniId: sq.uniID, PortNo: sq.uniPort,
562 TrafficScheds: TrafficSched}); err != nil {
bseenivaa1622112025-12-11 18:24:02 +0530563 cancel()
Gamze Abakacb0e6772021-06-10 08:32:12 +0000564 return olterrors.NewErrAdapter("failed-to-create-traffic-schedulers-in-device", log.Fields{"TrafficScheds": TrafficSched}, err)
565 }
bseenivaa1622112025-12-11 18:24:02 +0530566 cancel()
balaji.nagarajan31db4ea2026-01-28 09:13:55 +0530567 logger.Debugw(ctx, "successfully-created-traffic-schedulers", log.Fields{
Gamze Abakacb0e6772021-06-10 08:32:12 +0000568 "direction": sq.direction,
569 "traffic-queues": trafficQueues,
570 "device-id": f.deviceHandler.device.Id})
Girish Kumar8f73fe02019-12-09 13:19:37 +0000571 }
572
573 // On receiving the CreateTrafficQueues request, the driver should create corresponding
574 // downstream queues.
Neha Sharma96b7bf22020-06-15 10:37:32 +0000575 logger.Debugw(ctx, "sending-traffic-queues-create-to-device",
Shrey Baid26912972020-04-16 21:02:31 +0530576 log.Fields{"direction": sq.direction,
577 "traffic-queues": trafficQueues,
578 "device-id": f.deviceHandler.device.Id})
Gamze Abaka411ef2f2021-11-22 08:38:08 +0000579 queues := &tp_pb.TrafficQueues{IntfId: sq.intfID, OnuId: sq.onuID,
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530580 NetworkIntfId: sq.nniIntfID,
581 UniId: sq.uniID, PortNo: sq.uniPort,
Gamze Abaka411ef2f2021-11-22 08:38:08 +0000582 TrafficQueues: trafficQueues,
583 TechProfileId: TrafficSched[0].TechProfileId}
bseenivaa1622112025-12-11 18:24:02 +0530584 subCtx1, cancel1 := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
585 if _, err := f.deviceHandler.Client.CreateTrafficQueues(subCtx1, queues); err != nil {
586 cancel1()
Gamze Abaka411ef2f2021-11-22 08:38:08 +0000587 if len(queues.TrafficQueues) > 1 {
588 logger.Debug(ctx, "removing-queues-for-1tcont-multi-gem", log.Fields{"intfID": sq.intfID, "onuID": sq.onuID, "dir": sq.direction})
bseenivaa1622112025-12-11 18:24:02 +0530589 subCtx2, cancel2 := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
590 _, _ = f.deviceHandler.Client.RemoveTrafficQueues(subCtx2, queues)
591 cancel2()
Gamze Abaka411ef2f2021-11-22 08:38:08 +0000592 }
593 f.revertScheduler(ctx, sq, TrafficSched)
Shrey Baid26912972020-04-16 21:02:31 +0530594 return olterrors.NewErrAdapter("failed-to-create-traffic-queues-in-device", log.Fields{"traffic-queues": trafficQueues}, err)
Girish Kumar8f73fe02019-12-09 13:19:37 +0000595 }
bseenivaa1622112025-12-11 18:24:02 +0530596 cancel1()
Neha Sharma96b7bf22020-06-15 10:37:32 +0000597 logger.Infow(ctx, "successfully-created-traffic-schedulers", log.Fields{
Shrey Baid26912972020-04-16 21:02:31 +0530598 "direction": sq.direction,
599 "traffic-queues": trafficQueues,
600 "device-id": f.deviceHandler.device.Id})
Girish Kumar8f73fe02019-12-09 13:19:37 +0000601
Esin Karamanccb714b2019-11-29 15:02:06 +0000602 if sq.direction == tp_pb.Direction_DOWNSTREAM {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700603 multicastTrafficQueues := f.techprofile.GetMulticastTrafficQueues(ctx, sq.tpInst.(*tp_pb.TechProfileInstance))
Esin Karamanccb714b2019-11-29 15:02:06 +0000604 if len(multicastTrafficQueues) > 0 {
Akash Kankanala041a2122024-10-16 15:49:22 +0530605 if _, present := f.grpMgr.GetInterfaceToMcastQueueMap(sq.intfID); !present { // assumed that there is only one queue per PON for the multicast service
606 // the default queue with multicastQueuePerPonPort.Priority per a pon interface is used for multicast service
607 // just put it in interfaceToMcastQueueMap to use for building group members
Neha Sharma96b7bf22020-06-15 10:37:32 +0000608 logger.Debugw(ctx, "multicast-traffic-queues", log.Fields{"device-id": f.deviceHandler.device.Id})
Esin Karamanccb714b2019-11-29 15:02:06 +0000609 multicastQueuePerPonPort := multicastTrafficQueues[0]
Girish Gowdra9602eb42020-09-09 15:50:39 -0700610 val := &QueueInfoBrief{
Esin Karamanccb714b2019-11-29 15:02:06 +0000611 gemPortID: multicastQueuePerPonPort.GemportId,
612 servicePriority: multicastQueuePerPonPort.Priority,
613 }
Girish Gowdra9602eb42020-09-09 15:50:39 -0700614 f.grpMgr.UpdateInterfaceToMcastQueueMap(sq.intfID, val)
Akash Kankanala041a2122024-10-16 15:49:22 +0530615 // also store the queue info in kv store
yasin saplibddc2d72022-02-08 13:10:17 +0000616 if err := f.resourceMgr.AddMcastQueueForIntf(ctx, multicastQueuePerPonPort.GemportId, multicastQueuePerPonPort.Priority); err != nil {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700617 logger.Errorw(ctx, "failed-to-add-mcast-queue", log.Fields{"err": err})
Kent Hagermane6ff1012020-07-14 15:07:53 -0400618 return err
619 }
Shrey Baid26912972020-04-16 21:02:31 +0530620
Neha Sharma96b7bf22020-06-15 10:37:32 +0000621 logger.Infow(ctx, "multicast-queues-successfully-updated", log.Fields{"device-id": f.deviceHandler.device.Id})
Esin Karamanccb714b2019-11-29 15:02:06 +0000622 }
623 }
624 }
Girish Kumar8f73fe02019-12-09 13:19:37 +0000625 return nil
626}
627
Girish Gowdraf3728b12022-02-02 21:46:51 -0800628// RemoveQueues removes the traffic queues from the device based on the given schedQueue info
629func (f *OpenOltFlowMgr) RemoveQueues(ctx context.Context, sq schedQueue) error {
630 var err error
631 logger.Infow(ctx, "removing-queue-in-olt",
632 log.Fields{
633 "direction": sq.direction,
634 "intf-id": sq.intfID,
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530635 "nniIntfID": sq.nniIntfID,
Girish Gowdraf3728b12022-02-02 21:46:51 -0800636 "onu-id": sq.onuID,
637 "uni-id": sq.uniID,
638 "uni-port": sq.uniPort,
639 "device-id": f.deviceHandler.device.Id})
Manikkaraj kb1d51442019-07-23 10:41:02 -0400640
Girish Gowdraf3728b12022-02-02 21:46:51 -0800641 TrafficQueues, err := f.techprofile.GetTrafficQueues(ctx, sq.tpInst.(*tp_pb.TechProfileInstance), sq.direction)
642 if err != nil {
643 return olterrors.NewErrAdapter("unable-to-construct-traffic-queue-configuration",
644 log.Fields{
645 "intf-id": sq.intfID,
646 "direction": sq.direction,
647 "device-id": f.deviceHandler.device.Id}, err)
648 }
649
bseenivaa1622112025-12-11 18:24:02 +0530650 subCtx, cancel := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
651 defer cancel()
652 if _, err = f.deviceHandler.Client.RemoveTrafficQueues(subCtx,
Girish Gowdraf3728b12022-02-02 21:46:51 -0800653 &tp_pb.TrafficQueues{IntfId: sq.intfID, OnuId: sq.onuID,
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530654 NetworkIntfId: sq.nniIntfID,
655 UniId: sq.uniID, PortNo: sq.uniPort,
Girish Gowdraf3728b12022-02-02 21:46:51 -0800656 TrafficQueues: TrafficQueues,
657 TechProfileId: sq.tpID}); err != nil {
658 return olterrors.NewErrAdapter("unable-to-remove-traffic-queues-from-device",
659 log.Fields{
660 "intf-id": sq.intfID,
661 "traffic-queues": TrafficQueues,
662 "device-id": f.deviceHandler.device.Id}, err)
663 }
balaji.nagarajan31db4ea2026-01-28 09:13:55 +0530664 logger.Debugw(ctx, "removed-traffic-queues-successfully", log.Fields{"device-id": f.deviceHandler.device.Id, "trafficQueues": TrafficQueues})
Girish Gowdraf3728b12022-02-02 21:46:51 -0800665
666 return err
667}
668
669// RemoveScheduler removes the traffic scheduler from the device based on the given schedQueue info
670func (f *OpenOltFlowMgr) RemoveScheduler(ctx context.Context, sq schedQueue) error {
Manikkaraj kb1d51442019-07-23 10:41:02 -0400671 var Direction string
672 var SchedCfg *tp_pb.SchedulerConfig
673 var err error
Girish Gowdraf3728b12022-02-02 21:46:51 -0800674 logger.Infow(ctx, "removing-scheduler-in-olt",
Shrey Baid26912972020-04-16 21:02:31 +0530675 log.Fields{
676 "direction": sq.direction,
677 "intf-id": sq.intfID,
678 "onu-id": sq.onuID,
679 "uni-id": sq.uniID,
680 "uni-port": sq.uniPort,
681 "device-id": f.deviceHandler.device.Id})
mgouda86543582025-10-29 20:58:16 +0530682 switch sq.direction {
683 case tp_pb.Direction_UPSTREAM:
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700684 SchedCfg = f.techprofile.GetUsScheduler(sq.tpInst.(*tp_pb.TechProfileInstance))
Manikkaraj kb1d51442019-07-23 10:41:02 -0400685 Direction = "upstream"
mgouda86543582025-10-29 20:58:16 +0530686 case tp_pb.Direction_DOWNSTREAM:
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700687 SchedCfg = f.techprofile.GetDsScheduler(sq.tpInst.(*tp_pb.TechProfileInstance))
Manikkaraj kb1d51442019-07-23 10:41:02 -0400688 Direction = "downstream"
689 }
690
Girish Gowdraa482f272021-03-24 23:04:19 -0700691 TrafficShaping := &tp_pb.TrafficShapingInfo{} // this info is not really useful for the agent during flow removal. Just use default values.
Manikkaraj kb1d51442019-07-23 10:41:02 -0400692
bseeniva0b9cbcb2026-02-12 19:11:11 +0530693 trafficSched := f.techprofile.GetTrafficScheduler(sq.tpInst.(*tp_pb.TechProfileInstance), SchedCfg, TrafficShaping)
694 trafficSched.TechProfileId = sq.tpID
695 TrafficSched := []*tp_pb.TrafficScheduler{trafficSched}
Girish Kumar8f73fe02019-12-09 13:19:37 +0000696
bseenivaa1622112025-12-11 18:24:02 +0530697 subCtx, cancel := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
698 if _, err = f.deviceHandler.Client.RemoveTrafficSchedulers(subCtx, &tp_pb.TrafficSchedulers{
Girish Gowdraf3728b12022-02-02 21:46:51 -0800699 IntfId: sq.intfID, OnuId: sq.onuID,
700 UniId: sq.uniID, PortNo: sq.uniPort,
701 TrafficScheds: TrafficSched}); err != nil {
bseenivaa1622112025-12-11 18:24:02 +0530702 cancel()
Girish Gowdraf3728b12022-02-02 21:46:51 -0800703 return olterrors.NewErrAdapter("unable-to-remove-traffic-schedulers-from-device",
Shrey Baid26912972020-04-16 21:02:31 +0530704 log.Fields{
Girish Gowdraf3728b12022-02-02 21:46:51 -0800705 "intf-id": sq.intfID,
706 "traffic-schedulers": TrafficSched,
707 "onu-id": sq.onuID,
708 "uni-id": sq.uniID,
709 "uni-port": sq.uniPort}, err)
Girish Kumar8f73fe02019-12-09 13:19:37 +0000710 }
bseenivaa1622112025-12-11 18:24:02 +0530711 cancel()
Girish Gowdraf3728b12022-02-02 21:46:51 -0800712 logger.Infow(ctx, "removed-traffic-schedulers-successfully",
713 log.Fields{"device-id": f.deviceHandler.device.Id,
714 "intf-id": sq.intfID,
715 "onu-id": sq.onuID,
716 "uni-id": sq.uniID,
717 "uni-port": sq.uniPort})
Manikkaraj kb1d51442019-07-23 10:41:02 -0400718
Girish Gowdraf3728b12022-02-02 21:46:51 -0800719 if sq.direction == tp_pb.Direction_UPSTREAM {
720 allocID := sq.tpInst.(*tp_pb.TechProfileInstance).UsScheduler.AllocId
721 // Delete the TCONT on the ONU.
722 uni := getUniPortPath(f.deviceHandler.device.Id, sq.intfID, int32(sq.onuID), int32(sq.uniID))
yasin saplibddc2d72022-02-08 13:10:17 +0000723 tpPath := f.getTPpath(ctx, uni, sq.tpID)
Akash Kankanala041a2122024-10-16 15:49:22 +0530724 if err = f.sendDeleteTcontToChild(ctx, sq.intfID, sq.onuID, sq.uniID, allocID, tpPath); err != nil {
Girish Gowdraf3728b12022-02-02 21:46:51 -0800725 logger.Errorw(ctx, "error-processing-delete-tcont-towards-onu",
Gamze Abakacb0e6772021-06-10 08:32:12 +0000726 log.Fields{
Girish Gowdraf3728b12022-02-02 21:46:51 -0800727 "intf": sq.intfID,
728 "onu-id": sq.onuID,
729 "uni-id": sq.uniID,
730 "device-id": f.deviceHandler.device.Id,
731 "alloc-id": allocID})
Gamze Abakacb0e6772021-06-10 08:32:12 +0000732 }
733 }
salmansiddiqui7ac62132019-08-22 03:58:50 +0000734
735 /* After we successfully remove the scheduler configuration on the OLT device,
Manikkaraj kb1d51442019-07-23 10:41:02 -0400736 * delete the meter id on the KV store.
737 */
Girish Gowdraf3728b12022-02-02 21:46:51 -0800738 err = f.removeMeterReference(ctx, Direction, sq)
Manikkaraj kb1d51442019-07-23 10:41:02 -0400739 return err
740}
741
Girish Gowdra197acc12021-08-16 10:59:45 -0700742// We are trying to force remove the schedulers and queues here if one exists for the given key.
743// We ignore any errors encountered in the process. The errors most likely are encountered when
744// the schedulers and queues are already cleared for the given key.
745func (f *OpenOltFlowMgr) forceRemoveSchedulerQueues(ctx context.Context, sq schedQueue) {
Girish Gowdra197acc12021-08-16 10:59:45 -0700746 var schedCfg *tp_pb.SchedulerConfig
Girish Gowdra197acc12021-08-16 10:59:45 -0700747 logger.Infow(ctx, "removing-schedulers-and-queues-in-olt",
748 log.Fields{
749 "direction": sq.direction,
750 "intf-id": sq.intfID,
751 "onu-id": sq.onuID,
752 "uni-id": sq.uniID,
753 "uni-port": sq.uniPort,
754 "tp-id": sq.tpID,
755 "device-id": f.deviceHandler.device.Id})
mgouda86543582025-10-29 20:58:16 +0530756 switch sq.direction {
757 case tp_pb.Direction_UPSTREAM:
Girish Gowdra197acc12021-08-16 10:59:45 -0700758 schedCfg = f.techprofile.GetUsScheduler(sq.tpInst.(*tp_pb.TechProfileInstance))
mgouda86543582025-10-29 20:58:16 +0530759 case tp_pb.Direction_DOWNSTREAM:
Girish Gowdra197acc12021-08-16 10:59:45 -0700760 schedCfg = f.techprofile.GetDsScheduler(sq.tpInst.(*tp_pb.TechProfileInstance))
761 }
762
763 TrafficShaping := &tp_pb.TrafficShapingInfo{} // this info is not really useful for the agent during flow removal. Just use default values.
bseeniva0b9cbcb2026-02-12 19:11:11 +0530764 trafficSched := f.techprofile.GetTrafficScheduler(sq.tpInst.(*tp_pb.TechProfileInstance), schedCfg, TrafficShaping)
765 trafficSched.TechProfileId = sq.tpID
766 TrafficSched := []*tp_pb.TrafficScheduler{trafficSched}
Girish Gowdra197acc12021-08-16 10:59:45 -0700767
768 // Remove traffic queues. Ignore any errors, just log them.
769 if TrafficQueues, err := f.techprofile.GetTrafficQueues(ctx, sq.tpInst.(*tp_pb.TechProfileInstance), sq.direction); err != nil {
770 logger.Errorw(ctx, "error retrieving traffic queue", log.Fields{
771 "direction": sq.direction,
772 "intf-id": sq.intfID,
773 "onu-id": sq.onuID,
774 "uni-id": sq.uniID,
775 "uni-port": sq.uniPort,
776 "tp-id": sq.tpID,
777 "device-id": f.deviceHandler.device.Id,
778 "err": err})
779 } else {
bseenivaa1622112025-12-11 18:24:02 +0530780 subCtx, cancel := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
781 if _, err := f.deviceHandler.Client.RemoveTrafficQueues(subCtx,
Girish Gowdra197acc12021-08-16 10:59:45 -0700782 &tp_pb.TrafficQueues{IntfId: sq.intfID, OnuId: sq.onuID,
783 UniId: sq.uniID, PortNo: sq.uniPort,
784 TrafficQueues: TrafficQueues,
785 TechProfileId: TrafficSched[0].TechProfileId}); err != nil {
786 logger.Warnw(ctx, "error removing traffic queue", log.Fields{
787 "direction": sq.direction,
788 "intf-id": sq.intfID,
789 "onu-id": sq.onuID,
790 "uni-id": sq.uniID,
791 "uni-port": sq.uniPort,
792 "tp-id": sq.tpID,
793 "device-id": f.deviceHandler.device.Id,
794 "err": err})
Girish Gowdra197acc12021-08-16 10:59:45 -0700795 } else {
balaji.nagarajan31db4ea2026-01-28 09:13:55 +0530796 logger.Debugw(ctx, "removed-traffic-queues-successfully", log.Fields{"device-id": f.deviceHandler.device.Id,
Girish Gowdra197acc12021-08-16 10:59:45 -0700797 "direction": sq.direction,
798 "intf-id": sq.intfID,
799 "onu-id": sq.onuID,
800 "uni-id": sq.uniID,
801 "uni-port": sq.uniPort,
802 "tp-id": sq.tpID})
803 }
bseenivaa1622112025-12-11 18:24:02 +0530804 cancel()
Girish Gowdra197acc12021-08-16 10:59:45 -0700805 }
806
bseenivaa1622112025-12-11 18:24:02 +0530807 subCtx, cancel := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
Girish Gowdra197acc12021-08-16 10:59:45 -0700808 // Remove traffic schedulers. Ignore any errors, just log them.
bseenivaa1622112025-12-11 18:24:02 +0530809 if _, err := f.deviceHandler.Client.RemoveTrafficSchedulers(subCtx, &tp_pb.TrafficSchedulers{
Girish Gowdra197acc12021-08-16 10:59:45 -0700810 IntfId: sq.intfID, OnuId: sq.onuID,
811 UniId: sq.uniID, PortNo: sq.uniPort,
812 TrafficScheds: TrafficSched}); err != nil {
813 logger.Warnw(ctx, "error removing traffic scheduler", log.Fields{
814 "direction": sq.direction,
815 "intf-id": sq.intfID,
816 "onu-id": sq.onuID,
817 "uni-id": sq.uniID,
818 "uni-port": sq.uniPort,
819 "tp-id": sq.tpID,
820 "device-id": f.deviceHandler.device.Id,
821 "err": err})
822 } else {
balaji.nagarajan31db4ea2026-01-28 09:13:55 +0530823 logger.Debugw(ctx, "removed-traffic-schedulers-successfully", log.Fields{"device-id": f.deviceHandler.device.Id,
Girish Gowdra197acc12021-08-16 10:59:45 -0700824 "direction": sq.direction,
825 "intf-id": sq.intfID,
826 "onu-id": sq.onuID,
827 "uni-id": sq.uniID,
828 "uni-port": sq.uniPort,
829 "tp-id": sq.tpID})
830 }
bseenivaa1622112025-12-11 18:24:02 +0530831 cancel()
Girish Gowdra197acc12021-08-16 10:59:45 -0700832}
833
Gamze Abakafee36392019-10-03 11:17:24 +0000834// This function allocates tconts and GEM ports for an ONU
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530835func (f *OpenOltFlowMgr) createTcontGemports(ctx context.Context, intfID uint32, nniIntfID uint32, onuID uint32, uniID uint32, uni string, uniPort uint32, TpID uint32, UsMeterID uint32, DsMeterID uint32, flowMetadata *ofp.FlowMetadata) (uint32, []uint32, interface{}) {
Gamze Abakafee36392019-10-03 11:17:24 +0000836 var allocIDs []uint32
837 var allgemPortIDs []uint32
manikkaraj kbf256be2019-03-25 00:13:48 +0530838 var gemPortIDs []uint32
Girish Gowdra3d633032019-12-10 16:37:05 +0530839 tpInstanceExists := false
Girish Kumar8f73fe02019-12-09 13:19:37 +0000840 var err error
yasin saplibddc2d72022-02-08 13:10:17 +0000841 allocIDs = f.resourceMgr.GetCurrentAllocIDsForOnu(ctx, onuID, uniID)
842 allgemPortIDs = f.resourceMgr.GetCurrentGEMPortIDsForOnu(ctx, onuID, uniID)
843 tpPath := f.getTPpath(ctx, uni, TpID)
Girish Gowdra54934262019-11-13 14:19:55 +0530844
Neha Sharma96b7bf22020-06-15 10:37:32 +0000845 logger.Debugw(ctx, "creating-new-tcont-and-gem", log.Fields{
Shrey Baid26912972020-04-16 21:02:31 +0530846 "intf-id": intfID,
847 "onu-id": onuID,
848 "uni-id": uniID,
849 "device-id": f.deviceHandler.device.Id,
850 "tp-id": TpID})
Girish Gowdra54934262019-11-13 14:19:55 +0530851
Manikkaraj kb1d51442019-07-23 10:41:02 -0400852 // Check tech profile instance already exists for derived port name
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700853 techProfileInstance, _ := f.techprofile.GetTPInstance(ctx, tpPath)
salmansiddiqui7ac62132019-08-22 03:58:50 +0000854 if techProfileInstance == nil {
balaji.nagarajan31db4ea2026-01-28 09:13:55 +0530855 logger.Debugw(ctx, "tp-instance-not-found--creating-new",
Shrey Baid26912972020-04-16 21:02:31 +0530856 log.Fields{
857 "path": tpPath,
858 "device-id": f.deviceHandler.device.Id})
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700859 techProfileInstance, err = f.techprofile.CreateTechProfileInstance(ctx, TpID, uni, intfID)
Girish Kumar8f73fe02019-12-09 13:19:37 +0000860 if err != nil {
Girish Gowdra54934262019-11-13 14:19:55 +0530861 // This should not happen, something wrong in KV backend transaction
Neha Sharma96b7bf22020-06-15 10:37:32 +0000862 logger.Errorw(ctx, "tp-instance-create-failed",
Shrey Baid26912972020-04-16 21:02:31 +0530863 log.Fields{
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700864 "err": err,
Shrey Baid26912972020-04-16 21:02:31 +0530865 "tp-id": TpID,
866 "device-id": f.deviceHandler.device.Id})
Gamze Abakafee36392019-10-03 11:17:24 +0000867 return 0, nil, nil
manikkaraj kbf256be2019-03-25 00:13:48 +0530868 }
yasin saplibddc2d72022-02-08 13:10:17 +0000869 if err := f.resourceMgr.UpdateTechProfileIDForOnu(ctx, onuID, uniID, TpID); err != nil {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700870 logger.Warnw(ctx, "failed-to-update-tech-profile-id", log.Fields{"err": err})
Kent Hagermane6ff1012020-07-14 15:07:53 -0400871 }
manikkaraj kbf256be2019-03-25 00:13:48 +0530872 } else {
Neha Sharma96b7bf22020-06-15 10:37:32 +0000873 logger.Debugw(ctx, "tech-profile-instance-already-exist-for-given port-name",
Shrey Baid26912972020-04-16 21:02:31 +0530874 log.Fields{
875 "uni": uni,
876 "device-id": f.deviceHandler.device.Id})
Girish Gowdra3d633032019-12-10 16:37:05 +0530877 tpInstanceExists = true
manikkaraj kbf256be2019-03-25 00:13:48 +0530878 }
Gamze Abakafee36392019-10-03 11:17:24 +0000879
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700880 switch tpInst := techProfileInstance.(type) {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700881 case *tp_pb.TechProfileInstance:
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700882 if UsMeterID != 0 {
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530883 sq := schedQueue{direction: tp_pb.Direction_UPSTREAM, intfID: intfID, nniIntfID: nniIntfID, onuID: onuID, uniID: uniID, tpID: TpID,
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700884 uniPort: uniPort, tpInst: techProfileInstance, meterID: UsMeterID, flowMetadata: flowMetadata}
885 if err := f.CreateSchedulerQueues(ctx, sq); err != nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +0000886 logger.Errorw(ctx, "CreateSchedulerQueues-failed-upstream",
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700887 log.Fields{
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700888 "err": err,
Matteo Scandolo2f6b5bc2020-09-17 13:58:10 -0700889 "onu-id": onuID,
890 "uni-id": uniID,
891 "intf-id": intfID,
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700892 "meter-id": UsMeterID,
893 "device-id": f.deviceHandler.device.Id})
Gamze Abaka411ef2f2021-11-22 08:38:08 +0000894 f.revertTechProfileInstance(ctx, sq)
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700895 return 0, nil, nil
896 }
897 }
898 if DsMeterID != 0 {
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +0530899 sq := schedQueue{direction: tp_pb.Direction_DOWNSTREAM, intfID: intfID, nniIntfID: nniIntfID, onuID: onuID, uniID: uniID, tpID: TpID,
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700900 uniPort: uniPort, tpInst: techProfileInstance, meterID: DsMeterID, flowMetadata: flowMetadata}
901 if err := f.CreateSchedulerQueues(ctx, sq); err != nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +0000902 logger.Errorw(ctx, "CreateSchedulerQueues-failed-downstream",
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700903 log.Fields{
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700904 "err": err,
Matteo Scandolo2f6b5bc2020-09-17 13:58:10 -0700905 "onu-id": onuID,
906 "uni-id": uniID,
907 "intf-id": intfID,
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700908 "meter-id": DsMeterID,
909 "device-id": f.deviceHandler.device.Id})
Gamze Abaka411ef2f2021-11-22 08:38:08 +0000910 f.revertTechProfileInstance(ctx, sq)
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700911 return 0, nil, nil
912 }
913 }
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700914 allocID := tpInst.UsScheduler.AllocId
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700915 for _, gem := range tpInst.UpstreamGemPortAttributeList {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700916 gemPortIDs = append(gemPortIDs, gem.GemportId)
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700917 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700918 allocIDs = appendUnique32bit(allocIDs, allocID)
Gamze Abakafee36392019-10-03 11:17:24 +0000919
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700920 if tpInstanceExists {
921 return allocID, gemPortIDs, techProfileInstance
922 }
923
924 for _, gemPortID := range gemPortIDs {
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700925 allgemPortIDs = appendUnique32bit(allgemPortIDs, gemPortID)
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700926 }
Neha Sharma96b7bf22020-06-15 10:37:32 +0000927 logger.Infow(ctx, "allocated-tcont-and-gem-ports",
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700928 log.Fields{
Matteo Scandolo84585372021-03-18 14:21:22 -0700929 "intf-id": intfID,
930 "onu-id": onuID,
931 "uni-id": uniID,
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700932 "alloc-ids": allocIDs,
933 "gemports": allgemPortIDs,
934 "device-id": f.deviceHandler.device.Id})
935 // Send Tconts and GEM ports to KV store
936 f.storeTcontsGEMPortsIntoKVStore(ctx, intfID, onuID, uniID, allocIDs, allgemPortIDs)
Girish Gowdra3d633032019-12-10 16:37:05 +0530937 return allocID, gemPortIDs, techProfileInstance
khenaidoodc2116e2021-10-19 17:33:19 -0400938 case *tp_pb.EponTechProfileInstance:
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700939 // CreateSchedulerQueues for EPON needs to be implemented here
940 // when voltha-protos for EPON is completed.
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700941 allocID := tpInst.AllocId
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700942 for _, gem := range tpInst.UpstreamQueueAttributeList {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700943 gemPortIDs = append(gemPortIDs, gem.GemportId)
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700944 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700945 allocIDs = appendUnique32bit(allocIDs, allocID)
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700946
947 if tpInstanceExists {
948 return allocID, gemPortIDs, techProfileInstance
949 }
950
951 for _, gemPortID := range gemPortIDs {
Girish Gowdraa09aeab2020-09-14 16:30:52 -0700952 allgemPortIDs = appendUnique32bit(allgemPortIDs, gemPortID)
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700953 }
Neha Sharma96b7bf22020-06-15 10:37:32 +0000954 logger.Infow(ctx, "allocated-tcont-and-gem-ports",
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700955 log.Fields{
956 "alloc-ids": allocIDs,
957 "gemports": allgemPortIDs,
958 "device-id": f.deviceHandler.device.Id})
959 // Send Tconts and GEM ports to KV store
960 f.storeTcontsGEMPortsIntoKVStore(ctx, intfID, onuID, uniID, allocIDs, allgemPortIDs)
961 return allocID, gemPortIDs, techProfileInstance
962 default:
Neha Sharma96b7bf22020-06-15 10:37:32 +0000963 logger.Errorw(ctx, "unknown-tech",
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -0700964 log.Fields{
965 "tpInst": tpInst})
966 return 0, nil, nil
Girish Gowdra3d633032019-12-10 16:37:05 +0530967 }
manikkaraj kbf256be2019-03-25 00:13:48 +0530968}
969
npujarec5762e2020-01-01 14:08:48 +0530970func (f *OpenOltFlowMgr) storeTcontsGEMPortsIntoKVStore(ctx context.Context, intfID uint32, onuID uint32, uniID uint32, allocID []uint32, gemPortIDs []uint32) {
Neha Sharma96b7bf22020-06-15 10:37:32 +0000971 logger.Debugw(ctx, "storing-allocated-tconts-and-gem-ports-into-KV-store",
Shrey Baid26912972020-04-16 21:02:31 +0530972 log.Fields{
973 "intf-id": intfID,
974 "onu-id": onuID,
975 "uni-id": uniID,
976 "alloc-id": allocID,
977 "gemport-ids": gemPortIDs,
978 "device-id": f.deviceHandler.device.Id})
manikkaraj kbf256be2019-03-25 00:13:48 +0530979 /* Update the allocated alloc_id and gem_port_id for the ONU/UNI to KV store */
yasin saplibddc2d72022-02-08 13:10:17 +0000980 if err := f.resourceMgr.UpdateAllocIdsForOnu(ctx, onuID, uniID, allocID); err != nil {
yasin sapli9e4c5092022-02-01 13:52:33 +0000981 logger.Errorw(ctx, "error-while-uploading-allocid-to-kv-store", log.Fields{"device-id": f.deviceHandler.device.Id, "onuID": onuID, "allocID": allocID})
manikkaraj kbf256be2019-03-25 00:13:48 +0530982 }
yasin saplibddc2d72022-02-08 13:10:17 +0000983 if err := f.resourceMgr.UpdateGEMPortIDsForOnu(ctx, onuID, uniID, gemPortIDs); err != nil {
yasin sapli9e4c5092022-02-01 13:52:33 +0000984 logger.Errorw(ctx, "error-while-uploading-gemports-to-kv-store", log.Fields{"device-id": f.deviceHandler.device.Id, "onuID": onuID, "gemPort": gemPortIDs})
manikkaraj kbf256be2019-03-25 00:13:48 +0530985 }
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -0700986
Neha Sharma96b7bf22020-06-15 10:37:32 +0000987 logger.Infow(ctx, "stored-tconts-and-gem-into-kv-store-successfully", log.Fields{"device-id": f.deviceHandler.device.Id})
balaji.nagarajan29be8bc2026-01-26 16:31:41 +0530988 if err := f.resourceMgr.AddGemToOnuGemInfo(ctx, onuID, gemPortIDs); err != nil {
989 logger.Errorw(ctx, "error-while-uploading-onugeminfos-to-kv-store", log.Fields{"device-id": f.deviceHandler.device.Id, "onuID": onuID, "gemPort": gemPortIDs})
manikkaraj k9eb6cac2019-05-09 12:32:03 -0400990 }
manikkaraj kbf256be2019-03-25 00:13:48 +0530991}
992
Girish Gowdra4736e5c2021-08-25 15:19:10 -0700993func (f *OpenOltFlowMgr) populateTechProfileForCurrentPonPort(ctx context.Context) error {
manikkaraj kbf256be2019-03-25 00:13:48 +0530994 for _, techRange := range f.resourceMgr.DevInfo.Ranges {
salmansiddiqui7ac62132019-08-22 03:58:50 +0000995 for _, intfID := range techRange.IntfIds {
Girish Gowdra4736e5c2021-08-25 15:19:10 -0700996 if intfID == f.ponPortIdx { // initialize only for the pon port that this flow manager is managing
997 var err error
pnalmas937a24d2025-01-16 18:48:30 +0530998 f.techprofile, err = tp.NewTechProfile(ctx, intfID, f.resourceMgr.DeviceID, f.resourceMgr.PonRsrMgr, f.resourceMgr.PonRsrMgr.Backend,
Girish Gowdra4736e5c2021-08-25 15:19:10 -0700999 f.resourceMgr.PonRsrMgr.Address, f.deviceHandler.cm.Backend.PathPrefix)
1000 if err != nil || f.techprofile == nil {
1001 logger.Errorw(ctx, "failed-to-allocate-to-techprofile-for-pon-port", log.Fields{"intfID": intfID, "err": err})
1002 return fmt.Errorf("failed-to-allocate-tech-profile-for-pon-port--pon-%v-err-%v", intfID, err)
1003 }
1004 logger.Debugw(ctx, "init-tech-profile-done",
1005 log.Fields{
1006 "intf-id": intfID,
1007 "device-id": f.deviceHandler.device.Id})
1008 return nil
Girish Gowdra4c3d4602021-07-22 16:33:37 -07001009 }
manikkaraj kbf256be2019-03-25 00:13:48 +05301010 }
1011 }
Girish Gowdra4736e5c2021-08-25 15:19:10 -07001012 logger.Errorw(ctx, "pon port not found in the the device pon port range", log.Fields{"intfID": f.ponPortIdx})
1013 return fmt.Errorf("pon-port-idx-not-found-in-the-device-info-pon-port-range-%v", f.ponPortIdx)
manikkaraj kbf256be2019-03-25 00:13:48 +05301014}
1015
Gamze Abaka7650be62021-02-26 10:50:36 +00001016func (f *OpenOltFlowMgr) addUpstreamDataPathFlow(ctx context.Context, flowContext *flowContext) error {
Girish Gowdraffa52e52022-02-16 15:48:10 -08001017 flowContext.classifier[PacketTagType] = SingleTag
1018 // extract the cvid/inner-vid from the write metadata
1019 writeMetadata := flows.GetMetadataFromWriteMetadataAction(ctx, flowContext.logicalFlow)
1020 if writeMetadata != 0 {
1021 // Writemetadata field is 8 bytes
1022 // cvid is on the outer most two bytes of the write metadata
1023 cvid := (writeMetadata & 0xffff000000000000) >> 48
1024 // if the cvid does not match the classifier vlan, then this indicates it is a double tagged packet
1025 if cvid != flowContext.classifier[VlanVid] && cvid != 0 {
1026 flowContext.classifier[PacketTagType] = DoubleTag
1027 }
1028 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001029 logger.Debugw(ctx, "adding-upstream-data-flow",
Shrey Baid26912972020-04-16 21:02:31 +05301030 log.Fields{
Gamze Abaka7650be62021-02-26 10:50:36 +00001031 "uplinkClassifier": flowContext.classifier,
1032 "uplinkAction": flowContext.action})
1033 return f.addSymmetricDataPathFlow(ctx, flowContext, Upstream)
manikkaraj kbf256be2019-03-25 00:13:48 +05301034}
1035
Gamze Abaka7650be62021-02-26 10:50:36 +00001036func (f *OpenOltFlowMgr) addDownstreamDataPathFlow(ctx context.Context, flowContext *flowContext) error {
1037 downlinkClassifier := flowContext.classifier
1038 downlinkAction := flowContext.action
Girish Gowdraffa52e52022-02-16 15:48:10 -08001039 // default to single tag. If we detect an inner cvid from write-metadata, then we mark the PacketTagType as DoubleTag
1040 downlinkClassifier[PacketTagType] = SingleTag
1041 // extract the cvid/inner-vid from the write metadata
1042 writeMetadata := flows.GetMetadataFromWriteMetadataAction(ctx, flowContext.logicalFlow)
1043 if writeMetadata != 0 {
1044 // Writemetadata field is 8 bytes
1045 // cvid is on the outer most two bytes of the write metadata
1046 if cvid := (writeMetadata & 0xffff000000000000) >> 48; cvid != 0 {
1047 downlinkClassifier[PacketTagType] = DoubleTag
1048 }
Andrea Campanellafaa42152021-10-28 11:50:56 +05301049 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001050 logger.Debugw(ctx, "adding-downstream-data-flow",
Shrey Baid26912972020-04-16 21:02:31 +05301051 log.Fields{
1052 "downlinkClassifier": downlinkClassifier,
1053 "downlinkAction": downlinkAction})
Manikkaraj kb1d51442019-07-23 10:41:02 -04001054
Girish Gowdraffa52e52022-02-16 15:48:10 -08001055 // If Pop Vlan action is specified, set the vlan to be popped from the classifier vlan match field.
1056 // The matched vlan is the one that is getting popped.
1057 if val, ok := downlinkAction[PopVlan]; ok && val.(bool) {
1058 // vlan_vid is a uint32. must be type asserted as such or conversion fails
1059 dlClVid, ok := downlinkClassifier[VlanVid].(uint32)
1060 if ok {
1061 downlinkAction[VlanVid] = dlClVid & 0xfff
1062 }
Girish Gowdra26f344b2019-10-23 14:39:13 +05301063 }
1064
Gamze Abaka7650be62021-02-26 10:50:36 +00001065 return f.addSymmetricDataPathFlow(ctx, flowContext, Downstream)
manikkaraj kbf256be2019-03-25 00:13:48 +05301066}
1067
Gamze Abaka7650be62021-02-26 10:50:36 +00001068func (f *OpenOltFlowMgr) addSymmetricDataPathFlow(ctx context.Context, flowContext *flowContext, direction string) error {
Gamze Abaka7650be62021-02-26 10:50:36 +00001069 intfID := flowContext.intfID
1070 onuID := flowContext.onuID
1071 uniID := flowContext.uniID
1072 classifier := flowContext.classifier
1073 action := flowContext.action
1074 allocID := flowContext.allocID
1075 gemPortID := flowContext.gemPortID
1076 tpID := flowContext.tpID
1077 logicalFlow := flowContext.logicalFlow
Neha Sharma96b7bf22020-06-15 10:37:32 +00001078 logger.Infow(ctx, "adding-hsia-flow",
Shrey Baid26912972020-04-16 21:02:31 +05301079 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301080 "intf-id": intfID,
1081 "onu-id": onuID,
1082 "uni-id": uniID,
1083 "device-id": f.deviceHandler.device.Id,
1084 "classifier": classifier,
1085 "action": action,
1086 "direction": direction,
1087 "alloc-id": allocID,
1088 "gemport-id": gemPortID,
1089 "flow-id": logicalFlow.Id})
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001090
yasin saplibddc2d72022-02-08 13:10:17 +00001091 present, err := f.resourceMgr.IsFlowOnKvStore(ctx, int32(onuID), logicalFlow.Id)
yasin saplid0566272021-12-21 09:10:30 +00001092 if present {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001093 logger.Infow(ctx, "flow-already-exists",
Shrey Baid26912972020-04-16 21:02:31 +05301094 log.Fields{
1095 "device-id": f.deviceHandler.device.Id,
1096 "intf-id": intfID,
1097 "onu-id": onuID})
David K. Bainbridge794735f2020-02-11 21:01:37 -08001098 return nil
yasin saplid0566272021-12-21 09:10:30 +00001099 } else if err != nil {
1100 logger.Errorw(ctx, "aborting-addSymmetricDataPathFlow--flow-may-already-exist",
1101 log.Fields{"intf-id": intfID, "onu-id": onuID, "flow-id": logicalFlow.Id})
1102 return err
Girish Gowdra3d633032019-12-10 16:37:05 +05301103 }
yasin saplid0566272021-12-21 09:10:30 +00001104
David K. Bainbridge794735f2020-02-11 21:01:37 -08001105 classifierProto, err := makeOpenOltClassifierField(classifier)
1106 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301107 return olterrors.NewErrInvalidValue(log.Fields{"classifier": classifier, "device-id": f.deviceHandler.device.Id}, err).Log()
Manikkaraj k884c1242019-04-11 16:26:42 +05301108 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001109 logger.Debugw(ctx, "created-classifier-proto",
Shrey Baid26912972020-04-16 21:02:31 +05301110 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301111 "classifier": classifierProto.String(),
Shrey Baid26912972020-04-16 21:02:31 +05301112 "device-id": f.deviceHandler.device.Id})
Gamze Abaka724d0852020-03-18 12:10:24 +00001113 actionProto, err := makeOpenOltActionField(action, classifier)
David K. Bainbridge794735f2020-02-11 21:01:37 -08001114 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301115 return olterrors.NewErrInvalidValue(log.Fields{"action": action, "device-id": f.deviceHandler.device.Id}, err).Log()
Manikkaraj k884c1242019-04-11 16:26:42 +05301116 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001117 logger.Debugw(ctx, "created-action-proto",
Shrey Baid26912972020-04-16 21:02:31 +05301118 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301119 "action": actionProto.String(),
Shrey Baid26912972020-04-16 21:02:31 +05301120 "device-id": f.deviceHandler.device.Id})
Neha Sharma96b7bf22020-06-15 10:37:32 +00001121 networkIntfID, err := getNniIntfID(ctx, classifier, action)
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05301122 if err != nil {
Thomas Lee S94109f12020-03-03 16:39:29 +05301123 return olterrors.NewErrNotFound("nni-interface-id",
David K. Bainbridge794735f2020-02-11 21:01:37 -08001124 log.Fields{
1125 "classifier": classifier,
1126 "action": action,
Shrey Baid26912972020-04-16 21:02:31 +05301127 "device-id": f.deviceHandler.device.Id,
David K. Bainbridge794735f2020-02-11 21:01:37 -08001128 }, err).Log()
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05301129 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001130
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001131 flow := openoltpb2.Flow{AccessIntfId: int32(intfID),
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001132 OnuId: int32(onuID),
1133 UniId: int32(uniID),
1134 FlowId: logicalFlow.Id,
1135 FlowType: direction,
1136 AllocId: int32(allocID),
1137 NetworkIntfId: int32(networkIntfID),
1138 GemportId: int32(gemPortID),
1139 Classifier: classifierProto,
1140 Action: actionProto,
1141 Priority: int32(logicalFlow.Priority),
1142 Cookie: logicalFlow.Cookie,
1143 PortNo: flowContext.portNo,
1144 TechProfileId: tpID,
1145 ReplicateFlow: len(flowContext.pbitToGem) > 0,
1146 PbitToGemport: flowContext.pbitToGem,
1147 GemportToAes: flowContext.gemToAes,
Gamze Abaka78a1d2a2020-04-27 10:17:27 +00001148 }
David K. Bainbridge794735f2020-02-11 21:01:37 -08001149 if err := f.addFlowToDevice(ctx, logicalFlow, &flow); err != nil {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001150 return olterrors.NewErrFlowOp("add", logicalFlow.Id, nil, err).Log()
Manikkaraj k884c1242019-04-11 16:26:42 +05301151 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001152 logger.Infow(ctx, "hsia-flow-added-to-device-successfully",
Shrey Baid26912972020-04-16 21:02:31 +05301153 log.Fields{"direction": direction,
1154 "device-id": f.deviceHandler.device.Id,
bseeniva0b9cbcb2026-02-12 19:11:11 +05301155 "flow-id": flow.FlowId,
Shrey Baid26912972020-04-16 21:02:31 +05301156 "intf-id": intfID,
1157 "onu-id": onuID})
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001158
David K. Bainbridge794735f2020-02-11 21:01:37 -08001159 return nil
Manikkaraj k884c1242019-04-11 16:26:42 +05301160}
Esin Karamanae41e2b2019-12-17 18:13:13 +00001161
Gamze Abaka7650be62021-02-26 10:50:36 +00001162func (f *OpenOltFlowMgr) addDHCPTrapFlow(ctx context.Context, flowContext *flowContext) error {
Gamze Abaka7650be62021-02-26 10:50:36 +00001163 intfID := flowContext.intfID
1164 onuID := flowContext.onuID
1165 uniID := flowContext.uniID
1166 logicalFlow := flowContext.logicalFlow
1167 classifier := flowContext.classifier
1168 action := flowContext.action
Manjunath Vanarajuluadc57d12019-04-23 11:07:21 +05301169
Neha Sharma96b7bf22020-06-15 10:37:32 +00001170 networkIntfID, err := getNniIntfID(ctx, classifier, action)
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05301171 if err != nil {
Thomas Lee S94109f12020-03-03 16:39:29 +05301172 return olterrors.NewErrNotFound("nni-interface-id", log.Fields{
David K. Bainbridge794735f2020-02-11 21:01:37 -08001173 "classifier": classifier,
Shrey Baid26912972020-04-16 21:02:31 +05301174 "action": action,
1175 "device-id": f.deviceHandler.device.Id},
David K. Bainbridge794735f2020-02-11 21:01:37 -08001176 err).Log()
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05301177 }
Manjunath Vanarajuluadc57d12019-04-23 11:07:21 +05301178
1179 // Clear the action map
1180 for k := range action {
1181 delete(action, k)
1182 }
1183
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001184 action[TrapToHost] = true
1185 classifier[UDPSrc] = uint32(68)
1186 classifier[UDPDst] = uint32(67)
1187 classifier[PacketTagType] = SingleTag
Manjunath Vanarajuluadc57d12019-04-23 11:07:21 +05301188
yasin saplibddc2d72022-02-08 13:10:17 +00001189 present, err := f.resourceMgr.IsFlowOnKvStore(ctx, int32(onuID), logicalFlow.Id)
yasin saplid0566272021-12-21 09:10:30 +00001190 if present {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001191 logger.Infow(ctx, "flow-exists--not-re-adding",
Shrey Baid26912972020-04-16 21:02:31 +05301192 log.Fields{
1193 "device-id": f.deviceHandler.device.Id,
1194 "intf-id": intfID,
1195 "onu-id": onuID})
David K. Bainbridge794735f2020-02-11 21:01:37 -08001196 return nil
yasin saplid0566272021-12-21 09:10:30 +00001197 } else if err != nil {
1198 logger.Errorw(ctx, "aborting-addDHCPTrapFlow--flow-may-already-exist",
1199 log.Fields{"intf-id": intfID, "onu-id": onuID, "flow-id": logicalFlow.Id})
1200 return err
Girish Gowdra3d633032019-12-10 16:37:05 +05301201 }
Manjunath Vanarajuluadc57d12019-04-23 11:07:21 +05301202
Neha Sharma96b7bf22020-06-15 10:37:32 +00001203 logger.Debugw(ctx, "creating-ul-dhcp-flow",
Shrey Baid26912972020-04-16 21:02:31 +05301204 log.Fields{
1205 "ul_classifier": classifier,
1206 "ul_action": action,
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001207 "uplinkFlowId": logicalFlow.Id,
Shrey Baid26912972020-04-16 21:02:31 +05301208 "intf-id": intfID,
1209 "onu-id": onuID,
1210 "device-id": f.deviceHandler.device.Id})
Manjunath Vanarajuluadc57d12019-04-23 11:07:21 +05301211
David K. Bainbridge794735f2020-02-11 21:01:37 -08001212 classifierProto, err := makeOpenOltClassifierField(classifier)
1213 if err != nil {
Thomas Lee S94109f12020-03-03 16:39:29 +05301214 return olterrors.NewErrInvalidValue(log.Fields{"classifier": classifier}, err).Log()
Manjunath Vanarajuluadc57d12019-04-23 11:07:21 +05301215 }
bseeniva0b9cbcb2026-02-12 19:11:11 +05301216 logger.Debugw(ctx, "created-classifier-proto", log.Fields{"classifier": classifierProto.String()})
Gamze Abaka724d0852020-03-18 12:10:24 +00001217 actionProto, err := makeOpenOltActionField(action, classifier)
David K. Bainbridge794735f2020-02-11 21:01:37 -08001218 if err != nil {
Thomas Lee S94109f12020-03-03 16:39:29 +05301219 return olterrors.NewErrInvalidValue(log.Fields{"action": action}, err).Log()
Manjunath Vanarajuluadc57d12019-04-23 11:07:21 +05301220 }
1221
David K. Bainbridge794735f2020-02-11 21:01:37 -08001222 dhcpFlow := openoltpb2.Flow{AccessIntfId: int32(intfID),
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001223 OnuId: int32(onuID),
1224 UniId: int32(uniID),
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001225 FlowId: logicalFlow.Id,
David K. Bainbridge82efc492019-09-04 09:57:11 -07001226 FlowType: Upstream,
Gamze Abaka7650be62021-02-26 10:50:36 +00001227 AllocId: int32(flowContext.allocID),
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001228 NetworkIntfId: int32(networkIntfID),
Gamze Abaka7650be62021-02-26 10:50:36 +00001229 GemportId: int32(flowContext.gemPortID),
Manjunath Vanarajuluadc57d12019-04-23 11:07:21 +05301230 Classifier: classifierProto,
1231 Action: actionProto,
1232 Priority: int32(logicalFlow.Priority),
1233 Cookie: logicalFlow.Cookie,
Gamze Abaka7650be62021-02-26 10:50:36 +00001234 PortNo: flowContext.portNo,
1235 TechProfileId: flowContext.tpID,
1236 ReplicateFlow: len(flowContext.pbitToGem) > 0,
1237 PbitToGemport: flowContext.pbitToGem,
1238 GemportToAes: flowContext.gemToAes,
Gamze Abaka78a1d2a2020-04-27 10:17:27 +00001239 }
David K. Bainbridge794735f2020-02-11 21:01:37 -08001240 if err := f.addFlowToDevice(ctx, logicalFlow, &dhcpFlow); err != nil {
bseeniva0b9cbcb2026-02-12 19:11:11 +05301241 return olterrors.NewErrFlowOp("add", logicalFlow.Id, log.Fields{"dhcp-flow-id": dhcpFlow.FlowId}, err).Log()
David K. Bainbridge794735f2020-02-11 21:01:37 -08001242 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001243 logger.Infow(ctx, "dhcp-ul-flow-added-to-device-successfully",
Shrey Baid26912972020-04-16 21:02:31 +05301244 log.Fields{
1245 "device-id": f.deviceHandler.device.Id,
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001246 "flow-id": logicalFlow.Id,
Shrey Baid26912972020-04-16 21:02:31 +05301247 "intf-id": intfID,
1248 "onu-id": onuID})
Manjunath Vanarajuluadc57d12019-04-23 11:07:21 +05301249
David K. Bainbridge794735f2020-02-11 21:01:37 -08001250 return nil
manikkaraj kbf256be2019-03-25 00:13:48 +05301251}
1252
Joey Armstrong3f0e2422023-07-05 18:25:41 -04001253// addIGMPTrapFlow creates IGMP trap-to-host flow
Gamze Abaka7650be62021-02-26 10:50:36 +00001254func (f *OpenOltFlowMgr) addIGMPTrapFlow(ctx context.Context, flowContext *flowContext) error {
1255 delete(flowContext.classifier, VlanVid)
1256 return f.addUpstreamTrapFlow(ctx, flowContext)
Esin Karamanae41e2b2019-12-17 18:13:13 +00001257}
1258
Joey Armstrong3f0e2422023-07-05 18:25:41 -04001259// addUpstreamTrapFlow creates a trap-to-host flow
Gamze Abaka7650be62021-02-26 10:50:36 +00001260func (f *OpenOltFlowMgr) addUpstreamTrapFlow(ctx context.Context, flowContext *flowContext) error {
Gamze Abaka7650be62021-02-26 10:50:36 +00001261 intfID := flowContext.intfID
1262 onuID := flowContext.onuID
1263 uniID := flowContext.uniID
1264 logicalFlow := flowContext.logicalFlow
1265 classifier := flowContext.classifier
1266 action := flowContext.action
Esin Karamanae41e2b2019-12-17 18:13:13 +00001267
Neha Sharma96b7bf22020-06-15 10:37:32 +00001268 networkIntfID, err := getNniIntfID(ctx, classifier, action)
Esin Karamanae41e2b2019-12-17 18:13:13 +00001269 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301270 return olterrors.NewErrNotFound("nni-interface-id",
1271 log.Fields{
1272 "classifier": classifier,
1273 "action": action,
1274 "device-id": f.deviceHandler.device.Id},
David K. Bainbridge794735f2020-02-11 21:01:37 -08001275 err).Log()
Esin Karamanae41e2b2019-12-17 18:13:13 +00001276 }
1277
1278 // Clear the action map
1279 for k := range action {
1280 delete(action, k)
1281 }
1282
1283 action[TrapToHost] = true
1284 classifier[PacketTagType] = SingleTag
Esin Karamanae41e2b2019-12-17 18:13:13 +00001285
yasin saplibddc2d72022-02-08 13:10:17 +00001286 present, err := f.resourceMgr.IsFlowOnKvStore(ctx, int32(onuID), logicalFlow.Id)
yasin saplid0566272021-12-21 09:10:30 +00001287 if present {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001288 logger.Infow(ctx, "flow-exists-not-re-adding", log.Fields{"device-id": f.deviceHandler.device.Id})
David K. Bainbridge794735f2020-02-11 21:01:37 -08001289 return nil
yasin saplid0566272021-12-21 09:10:30 +00001290 } else if err != nil {
1291 logger.Errorw(ctx, "aborting-addUpstreamTrapFlow--flow-may-already-exist",
1292 log.Fields{"intf-id": intfID, "onu-id": onuID, "flow-id": logicalFlow.Id})
1293 return err
Esin Karamanae41e2b2019-12-17 18:13:13 +00001294 }
1295
Neha Sharma96b7bf22020-06-15 10:37:32 +00001296 logger.Debugw(ctx, "creating-upstream-trap-flow",
Shrey Baid26912972020-04-16 21:02:31 +05301297 log.Fields{
1298 "ul_classifier": classifier,
1299 "ul_action": action,
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001300 "uplinkFlowId": logicalFlow.Id,
Shrey Baid26912972020-04-16 21:02:31 +05301301 "device-id": f.deviceHandler.device.Id,
1302 "intf-id": intfID,
1303 "onu-id": onuID})
Esin Karamanae41e2b2019-12-17 18:13:13 +00001304
David K. Bainbridge794735f2020-02-11 21:01:37 -08001305 classifierProto, err := makeOpenOltClassifierField(classifier)
1306 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301307 return olterrors.NewErrInvalidValue(log.Fields{"classifier": classifier, "device-id": f.deviceHandler.device.Id}, err).Log()
Esin Karamanae41e2b2019-12-17 18:13:13 +00001308 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001309 logger.Debugw(ctx, "created-classifier-proto",
Shrey Baid26912972020-04-16 21:02:31 +05301310 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301311 "classifier": classifierProto.String(),
Shrey Baid26912972020-04-16 21:02:31 +05301312 "device-id": f.deviceHandler.device.Id})
Gamze Abaka724d0852020-03-18 12:10:24 +00001313 actionProto, err := makeOpenOltActionField(action, classifier)
David K. Bainbridge794735f2020-02-11 21:01:37 -08001314 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301315 return olterrors.NewErrInvalidValue(log.Fields{"action": action, "device-id": f.deviceHandler.device.Id}, err).Log()
Esin Karamanae41e2b2019-12-17 18:13:13 +00001316 }
1317
David K. Bainbridge794735f2020-02-11 21:01:37 -08001318 flow := openoltpb2.Flow{AccessIntfId: int32(intfID),
Esin Karamanae41e2b2019-12-17 18:13:13 +00001319 OnuId: int32(onuID),
1320 UniId: int32(uniID),
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001321 FlowId: logicalFlow.Id,
Esin Karamanae41e2b2019-12-17 18:13:13 +00001322 FlowType: Upstream,
Gamze Abaka7650be62021-02-26 10:50:36 +00001323 AllocId: int32(flowContext.allocID),
Esin Karamanae41e2b2019-12-17 18:13:13 +00001324 NetworkIntfId: int32(networkIntfID),
Gamze Abaka7650be62021-02-26 10:50:36 +00001325 GemportId: int32(flowContext.gemPortID),
Esin Karamanae41e2b2019-12-17 18:13:13 +00001326 Classifier: classifierProto,
1327 Action: actionProto,
1328 Priority: int32(logicalFlow.Priority),
1329 Cookie: logicalFlow.Cookie,
Gamze Abaka7650be62021-02-26 10:50:36 +00001330 PortNo: flowContext.portNo,
1331 TechProfileId: flowContext.tpID,
1332 ReplicateFlow: len(flowContext.pbitToGem) > 0,
1333 PbitToGemport: flowContext.pbitToGem,
1334 GemportToAes: flowContext.gemToAes,
Gamze Abaka78a1d2a2020-04-27 10:17:27 +00001335 }
Esin Karamanae41e2b2019-12-17 18:13:13 +00001336
David K. Bainbridge794735f2020-02-11 21:01:37 -08001337 if err := f.addFlowToDevice(ctx, logicalFlow, &flow); err != nil {
bseeniva0b9cbcb2026-02-12 19:11:11 +05301338 return olterrors.NewErrFlowOp("add", logicalFlow.Id, log.Fields{"flow-id": flow.FlowId, "device-id": f.deviceHandler.device.Id}, err).Log()
David K. Bainbridge794735f2020-02-11 21:01:37 -08001339 }
Esin Karamanae41e2b2019-12-17 18:13:13 +00001340
David K. Bainbridge794735f2020-02-11 21:01:37 -08001341 return nil
Esin Karamanae41e2b2019-12-17 18:13:13 +00001342}
1343
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03001344// Add EthType flow to device with mac, vlanId as classifier for upstream and downstream
Gamze Abaka7650be62021-02-26 10:50:36 +00001345func (f *OpenOltFlowMgr) addEthTypeBasedFlow(ctx context.Context, flowContext *flowContext, vlanID uint32, ethType uint32) error {
1346 intfID := flowContext.intfID
1347 onuID := flowContext.onuID
1348 uniID := flowContext.uniID
1349 portNo := flowContext.portNo
1350 allocID := flowContext.allocID
1351 gemPortID := flowContext.gemPortID
1352 logicalFlow := flowContext.logicalFlow
1353 classifier := flowContext.classifier
1354 action := flowContext.action
1355
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03001356 logger.Infow(ctx, "adding-ethType-flow-to-device",
Shrey Baid26912972020-04-16 21:02:31 +05301357 log.Fields{
1358 "intf-id": intfID,
1359 "onu-id": onuID,
1360 "port-no": portNo,
1361 "alloc-id": allocID,
1362 "gemport-id": gemPortID,
1363 "vlan-id": vlanID,
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03001364 "flow": logicalFlow,
1365 "ethType": ethType})
manikkaraj kbf256be2019-03-25 00:13:48 +05301366
1367 uplinkClassifier := make(map[string]interface{})
1368 uplinkAction := make(map[string]interface{})
Girish Gowdra3d633032019-12-10 16:37:05 +05301369
manikkaraj kbf256be2019-03-25 00:13:48 +05301370 // Fill Classfier
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001371 uplinkClassifier[EthType] = ethType
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001372 uplinkClassifier[PacketTagType] = SingleTag
1373 uplinkClassifier[VlanVid] = vlanID
Gamze Abaka724d0852020-03-18 12:10:24 +00001374 uplinkClassifier[VlanPcp] = classifier[VlanPcp]
manikkaraj kbf256be2019-03-25 00:13:48 +05301375 // Fill action
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001376 uplinkAction[TrapToHost] = true
yasin saplibddc2d72022-02-08 13:10:17 +00001377 present, err := f.resourceMgr.IsFlowOnKvStore(ctx, int32(onuID), logicalFlow.Id)
yasin saplid0566272021-12-21 09:10:30 +00001378 if present {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001379 logger.Infow(ctx, "flow-exists-not-re-adding", log.Fields{
Shrey Baid26912972020-04-16 21:02:31 +05301380 "device-id": f.deviceHandler.device.Id,
1381 "onu-id": onuID,
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03001382 "intf-id": intfID,
1383 "ethType": ethType})
David K. Bainbridge794735f2020-02-11 21:01:37 -08001384 return nil
yasin saplid0566272021-12-21 09:10:30 +00001385 } else if err != nil {
1386 logger.Errorw(ctx, "aborting-addEthTypeBasedFlow--flow-may-already-exist",
1387 log.Fields{"intf-id": intfID, "onu-id": onuID, "flow-id": logicalFlow.Id})
1388 return err
Girish Gowdra3d633032019-12-10 16:37:05 +05301389 }
Akash Kankanala041a2122024-10-16 15:49:22 +05301390 // Add Uplink EthType Flow
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03001391 logger.Debugw(ctx, "creating-ul-ethType-flow",
Shrey Baid26912972020-04-16 21:02:31 +05301392 log.Fields{
1393 "ul_classifier": uplinkClassifier,
1394 "ul_action": uplinkAction,
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001395 "uplinkFlowId": logicalFlow.Id,
Shrey Baid26912972020-04-16 21:02:31 +05301396 "device-id": f.deviceHandler.device.Id,
1397 "intf-id": intfID,
1398 "onu-id": onuID})
manikkaraj kbf256be2019-03-25 00:13:48 +05301399
David K. Bainbridge794735f2020-02-11 21:01:37 -08001400 classifierProto, err := makeOpenOltClassifierField(uplinkClassifier)
1401 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301402 return olterrors.NewErrInvalidValue(log.Fields{
1403 "classifier": uplinkClassifier,
1404 "device-id": f.deviceHandler.device.Id}, err).Log()
manikkaraj kbf256be2019-03-25 00:13:48 +05301405 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001406 logger.Debugw(ctx, "created-classifier-proto",
Shrey Baid26912972020-04-16 21:02:31 +05301407 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301408 "classifier": classifierProto.String(),
Shrey Baid26912972020-04-16 21:02:31 +05301409 "device-id": f.deviceHandler.device.Id})
Gamze Abaka724d0852020-03-18 12:10:24 +00001410 actionProto, err := makeOpenOltActionField(uplinkAction, uplinkClassifier)
David K. Bainbridge794735f2020-02-11 21:01:37 -08001411 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301412 return olterrors.NewErrInvalidValue(log.Fields{"action": uplinkAction, "device-id": f.deviceHandler.device.Id}, err).Log()
manikkaraj kbf256be2019-03-25 00:13:48 +05301413 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001414 logger.Debugw(ctx, "created-action-proto",
Shrey Baid26912972020-04-16 21:02:31 +05301415 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301416 "action": actionProto.String(),
Shrey Baid26912972020-04-16 21:02:31 +05301417 "device-id": f.deviceHandler.device.Id})
Neha Sharma96b7bf22020-06-15 10:37:32 +00001418 networkIntfID, err := getNniIntfID(ctx, classifier, action)
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05301419 if err != nil {
Thomas Lee S94109f12020-03-03 16:39:29 +05301420 return olterrors.NewErrNotFound("nni-interface-id", log.Fields{
David K. Bainbridge794735f2020-02-11 21:01:37 -08001421 "classifier": classifier,
Shrey Baid26912972020-04-16 21:02:31 +05301422 "action": action,
1423 "device-id": f.deviceHandler.device.Id},
David K. Bainbridge794735f2020-02-11 21:01:37 -08001424 err).Log()
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05301425 }
1426
David K. Bainbridge794735f2020-02-11 21:01:37 -08001427 upstreamFlow := openoltpb2.Flow{AccessIntfId: int32(intfID),
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001428 OnuId: int32(onuID),
1429 UniId: int32(uniID),
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001430 FlowId: logicalFlow.Id,
David K. Bainbridge82efc492019-09-04 09:57:11 -07001431 FlowType: Upstream,
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001432 AllocId: int32(allocID),
1433 NetworkIntfId: int32(networkIntfID),
1434 GemportId: int32(gemPortID),
manikkaraj kbf256be2019-03-25 00:13:48 +05301435 Classifier: classifierProto,
1436 Action: actionProto,
1437 Priority: int32(logicalFlow.Priority),
1438 Cookie: logicalFlow.Cookie,
Gamze Abaka78a1d2a2020-04-27 10:17:27 +00001439 PortNo: portNo,
Gamze Abaka7650be62021-02-26 10:50:36 +00001440 TechProfileId: flowContext.tpID,
1441 ReplicateFlow: len(flowContext.pbitToGem) > 0,
1442 PbitToGemport: flowContext.pbitToGem,
1443 GemportToAes: flowContext.gemToAes,
Gamze Abaka78a1d2a2020-04-27 10:17:27 +00001444 }
David K. Bainbridge794735f2020-02-11 21:01:37 -08001445 if err := f.addFlowToDevice(ctx, logicalFlow, &upstreamFlow); err != nil {
bseeniva0b9cbcb2026-02-12 19:11:11 +05301446 return olterrors.NewErrFlowOp("add", logicalFlow.Id, log.Fields{"flow-id": upstreamFlow.FlowId}, err).Log()
David K. Bainbridge794735f2020-02-11 21:01:37 -08001447 }
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03001448 logger.Infow(ctx, "ethType-ul-flow-added-to-device-successfully",
Shrey Baid26912972020-04-16 21:02:31 +05301449 log.Fields{
1450 "device-id": f.deviceHandler.device.Id,
1451 "onu-id": onuID,
1452 "intf-id": intfID,
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03001453 "ethType": ethType,
Shrey Baid26912972020-04-16 21:02:31 +05301454 })
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001455
David K. Bainbridge794735f2020-02-11 21:01:37 -08001456 return nil
manikkaraj kbf256be2019-03-25 00:13:48 +05301457}
1458
David K. Bainbridge794735f2020-02-11 21:01:37 -08001459func makeOpenOltClassifierField(classifierInfo map[string]interface{}) (*openoltpb2.Classifier, error) {
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001460 var classifier openoltpb2.Classifier
David K. Bainbridge82efc492019-09-04 09:57:11 -07001461
1462 classifier.EthType, _ = classifierInfo[EthType].(uint32)
1463 classifier.IpProto, _ = classifierInfo[IPProto].(uint32)
1464 if vlanID, ok := classifierInfo[VlanVid].(uint32); ok {
Andrea Campanella7acc0b92020-02-14 09:20:49 +01001465 if vlanID != ReservedVlan {
1466 vid := vlanID & VlanvIDMask
Harsh Awasthiea45af72019-08-26 02:39:00 -04001467 classifier.OVid = vid
1468 }
manikkaraj kbf256be2019-03-25 00:13:48 +05301469 }
Girish Gowdrab23f1de2022-03-24 12:01:17 -07001470 // The classifierInfo[Metadata] is set for the following flows
1471 // - In the Downstream datapath flow table0 and table1. From the OLT perspective, only table0 downstream flow is relevant.
1472 // - Mcast flow that points to a group in the treatment
1473 // This value, when present and valid (not 0 and not 4096), is interpreted as below
1474 // - inner vid for a double tagged packet in the datapath flow
1475 // - outer vid for a single tagged packet in the datapath flow
1476 // - inner vid in the mcast flow that points to a group
David K. Bainbridge82efc492019-09-04 09:57:11 -07001477 if metadata, ok := classifierInfo[Metadata].(uint64); ok {
1478 vid := uint32(metadata)
Andrea Campanellafaa42152021-10-28 11:50:56 +05301479 // Set the OVid or IVid classifier based on the whether OLT is using a transparent tag or not
1480 // If OLT is using transparent tag mechanism, then it classifies whatever tag it sees to/from ONU which
Akash Kankanala041a2122024-10-16 15:49:22 +05301481 // is OVid from the perspective of the OLT. When OLT also places or pops the outer tag, then classifierInfo[Metadata]
Andrea Campanellafaa42152021-10-28 11:50:56 +05301482 // becomes the IVid.
1483 if classifier.OVid != 0 && classifier.OVid != ReservedVlan { // This is case when classifier.OVid is not set
1484 if vid != ReservedVlan {
1485 classifier.IVid = vid
1486 }
1487 } else {
1488 if vid != ReservedVlan {
1489 classifier.OVid = vid
1490 }
Harsh Awasthiea45af72019-08-26 02:39:00 -04001491 }
manikkaraj kbf256be2019-03-25 00:13:48 +05301492 }
Girish Gowdrafae935c2020-02-17 19:21:44 +05301493 // Use VlanPCPMask (0xff) to signify NO PCP. Else use valid PCP (0 to 7)
David K. Bainbridge82efc492019-09-04 09:57:11 -07001494 if vlanPcp, ok := classifierInfo[VlanPcp].(uint32); ok {
Girish Gowdrafae935c2020-02-17 19:21:44 +05301495 classifier.OPbits = vlanPcp
1496 } else {
1497 classifier.OPbits = VlanPCPMask
manikkaraj kbf256be2019-03-25 00:13:48 +05301498 }
David K. Bainbridge82efc492019-09-04 09:57:11 -07001499 classifier.SrcPort, _ = classifierInfo[UDPSrc].(uint32)
1500 classifier.DstPort, _ = classifierInfo[UDPDst].(uint32)
1501 classifier.DstIp, _ = classifierInfo[Ipv4Dst].(uint32)
1502 classifier.SrcIp, _ = classifierInfo[Ipv4Src].(uint32)
Esin Karamanccb714b2019-11-29 15:02:06 +00001503 classifier.DstMac, _ = classifierInfo[EthDst].([]uint8)
Girish Gowdraffa52e52022-02-16 15:48:10 -08001504 classifier.SrcMac, _ = classifierInfo[EthSrc].([]uint8)
David K. Bainbridge82efc492019-09-04 09:57:11 -07001505 if pktTagType, ok := classifierInfo[PacketTagType].(string); ok {
1506 classifier.PktTagType = pktTagType
1507
1508 switch pktTagType {
1509 case SingleTag:
1510 case DoubleTag:
1511 case Untagged:
1512 default:
Girish Kumarf26e4882020-03-05 06:49:10 +00001513 return nil, olterrors.NewErrInvalidValue(log.Fields{"packet-tag-type": pktTagType}, nil)
manikkaraj kbf256be2019-03-25 00:13:48 +05301514 }
1515 }
David K. Bainbridge794735f2020-02-11 21:01:37 -08001516 return &classifier, nil
manikkaraj kbf256be2019-03-25 00:13:48 +05301517}
1518
Akash Kankanala041a2122024-10-16 15:49:22 +05301519// nolint: unparam
1520// TODO: Improvise the function & remove the unparam lint, currently it is always returning 'nil' as error.
Gamze Abaka724d0852020-03-18 12:10:24 +00001521func makeOpenOltActionField(actionInfo map[string]interface{}, classifierInfo map[string]interface{}) (*openoltpb2.Action, error) {
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001522 var actionCmd openoltpb2.ActionCmd
1523 var action openoltpb2.Action
manikkaraj kbf256be2019-03-25 00:13:48 +05301524 action.Cmd = &actionCmd
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001525 if _, ok := actionInfo[PopVlan]; ok {
Girish Gowdraffa52e52022-02-16 15:48:10 -08001526 // Pop outer vid
manikkaraj kbf256be2019-03-25 00:13:48 +05301527 action.Cmd.RemoveOuterTag = true
Gamze Abaka724d0852020-03-18 12:10:24 +00001528 if _, ok := actionInfo[VlanPcp]; ok {
Girish Gowdraffa52e52022-02-16 15:48:10 -08001529 // Remark inner pbit
Gamze Abaka724d0852020-03-18 12:10:24 +00001530 action.Cmd.RemarkInnerPbits = true
1531 action.IPbits = actionInfo[VlanPcp].(uint32)
1532 if _, ok := actionInfo[VlanVid]; ok {
Girish Gowdraffa52e52022-02-16 15:48:10 -08001533 // Remark inner vid
Gamze Abaka724d0852020-03-18 12:10:24 +00001534 action.Cmd.TranslateInnerTag = true
1535 action.IVid = actionInfo[VlanVid].(uint32)
1536 }
1537 }
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001538 } else if _, ok := actionInfo[PushVlan]; ok {
Girish Gowdraffa52e52022-02-16 15:48:10 -08001539 // push outer vid
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001540 action.OVid = actionInfo[VlanVid].(uint32)
manikkaraj kbf256be2019-03-25 00:13:48 +05301541 action.Cmd.AddOuterTag = true
Gamze Abaka724d0852020-03-18 12:10:24 +00001542 if _, ok := actionInfo[VlanPcp]; ok {
Girish Gowdraffa52e52022-02-16 15:48:10 -08001543 // translate outer pbit
Gamze Abaka724d0852020-03-18 12:10:24 +00001544 action.OPbits = actionInfo[VlanPcp].(uint32)
1545 action.Cmd.RemarkOuterPbits = true
1546 if _, ok := classifierInfo[VlanVid]; ok {
Girish Gowdraffa52e52022-02-16 15:48:10 -08001547 // translate inner vid
Gamze Abaka724d0852020-03-18 12:10:24 +00001548 action.IVid = classifierInfo[VlanVid].(uint32)
1549 action.Cmd.TranslateInnerTag = true
1550 }
1551 }
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001552 } else if _, ok := actionInfo[TrapToHost]; ok {
1553 action.Cmd.TrapToHost = actionInfo[TrapToHost].(bool)
Girish Gowdraffa52e52022-02-16 15:48:10 -08001554 } else if _, ok := actionInfo[VlanVid]; ok {
1555 // Translate outer vid
1556 action.Cmd.TranslateOuterTag = true
1557 action.OVid = actionInfo[VlanVid].(uint32)
manikkaraj kbf256be2019-03-25 00:13:48 +05301558 }
Andrea Campanellafaa42152021-10-28 11:50:56 +05301559 // When OLT is transparent to vlans no-action is valid.
1560 /*
1561 else {
1562 return nil, olterrors.NewErrInvalidValue(log.Fields{"action-command": actionInfo}, nil)
1563 }
1564 */
David K. Bainbridge794735f2020-02-11 21:01:37 -08001565 return &action, nil
manikkaraj kbf256be2019-03-25 00:13:48 +05301566}
1567
Matteo Scandolod625b4c2020-04-02 16:16:01 -07001568// getTPpath return the ETCD path for a given UNI port
yasin saplibddc2d72022-02-08 13:10:17 +00001569func (f *OpenOltFlowMgr) getTPpath(ctx context.Context, uniPath string, TpID uint32) string {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001570 return f.techprofile.GetTechProfileInstanceKey(ctx, TpID, uniPath)
manikkaraj kbf256be2019-03-25 00:13:48 +05301571}
1572
Gamze Abakafee36392019-10-03 11:17:24 +00001573// DeleteTechProfileInstances removes the tech profile instances from persistent storage
Girish Gowdra197acc12021-08-16 10:59:45 -07001574// We also force release scheduler and queues associated with the tp instance. Theoretically there could be
1575// an issue if the upstream scheduler (DBA) is shared across multiple UNI and we force release it, given that
1576// this function is only meant to clean up TP instances of a given UNI. But in practicality this routine
1577// is only meant to be called when the clean up of resource for the whole ONU is taking place.
1578// The reason for introducing the force cleanup of scheduler and queues (on the OLT) was introduced here
1579// because it was observed that if the ONU device was deleted too soon after the flows were
1580// unprovisioned on that ONU, the scheduler and queue removal pertinent to that ONU would remain
1581// uncleaned on the OLT. So we force clean up here and ignore any error that OLT returns during the
1582// force cleanup (possible if the OLT has already cleared those resources).
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001583func (f *OpenOltFlowMgr) DeleteTechProfileInstances(ctx context.Context, intfID uint32, onuID uint32, uniID uint32) error {
yasin saplibddc2d72022-02-08 13:10:17 +00001584 tpIDList := f.resourceMgr.GetTechProfileIDForOnu(ctx, onuID, uniID)
Matteo Scandolod625b4c2020-04-02 16:16:01 -07001585 uniPortName := getUniPortPath(f.deviceHandler.device.Id, intfID, int32(onuID), int32(uniID))
1586
Gamze Abakafee36392019-10-03 11:17:24 +00001587 for _, tpID := range tpIDList {
Girish Gowdra197acc12021-08-16 10:59:45 -07001588 // Force cleanup scheduler/queues -- start
Mahir Gunyel85f61c12021-10-06 11:53:45 -07001589 uniPortNum := plt.MkUniPortNum(ctx, intfID, onuID, uniID)
Girish Gowdra197acc12021-08-16 10:59:45 -07001590 uni := getUniPortPath(f.deviceHandler.device.Id, intfID, int32(onuID), int32(uniID))
yasin saplibddc2d72022-02-08 13:10:17 +00001591 tpPath := f.getTPpath(ctx, uni, tpID)
Girish Gowdra197acc12021-08-16 10:59:45 -07001592 tpInst, err := f.techprofile.GetTPInstance(ctx, tpPath)
1593 if err != nil || tpInst == nil { // This should not happen, something wrong in KV backend transaction
1594 logger.Warnw(ctx, "tech-profile-not-in-kv-store",
1595 log.Fields{
1596 "tp-id": tpID,
1597 "path": tpPath})
1598 }
1599 switch tpInstance := tpInst.(type) {
1600 case *tp_pb.TechProfileInstance:
1601 f.forceRemoveSchedulerQueues(ctx, schedQueue{direction: tp_pb.Direction_UPSTREAM, intfID: intfID, onuID: onuID, uniID: uniID, tpID: tpID, uniPort: uniPortNum, tpInst: tpInstance})
1602 f.forceRemoveSchedulerQueues(ctx, schedQueue{direction: tp_pb.Direction_DOWNSTREAM, intfID: intfID, onuID: onuID, uniID: uniID, tpID: tpID, uniPort: uniPortNum, tpInst: tpInstance})
1603 }
1604 // Force cleanup scheduler/queues -- end
1605
1606 // Now remove the tp instance
npujarec5762e2020-01-01 14:08:48 +05301607 if err := f.DeleteTechProfileInstance(ctx, intfID, onuID, uniID, uniPortName, tpID); err != nil {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001608 logger.Errorw(ctx, "delete-tech-profile-failed", log.Fields{"err": err, "device-id": f.deviceHandler.device.Id})
Girish Gowdra54934262019-11-13 14:19:55 +05301609 // return err
1610 // We should continue to delete tech-profile instances for other TP IDs
Gamze Abakafee36392019-10-03 11:17:24 +00001611 }
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001612 logger.Debugw(ctx, "tech-profile-instance-deleted", log.Fields{"device-id": f.deviceHandler.device.Id, "uniPortName": uniPortName, "tp-id": tpID})
Gamze Abakafee36392019-10-03 11:17:24 +00001613 }
1614 return nil
1615}
1616
1617// DeleteTechProfileInstance removes the tech profile instance from persistent storage
npujarec5762e2020-01-01 14:08:48 +05301618func (f *OpenOltFlowMgr) DeleteTechProfileInstance(ctx context.Context, intfID uint32, onuID uint32, uniID uint32, uniPortName string, tpID uint32) error {
Gamze Abakafee36392019-10-03 11:17:24 +00001619 if uniPortName == "" {
Matteo Scandolod625b4c2020-04-02 16:16:01 -07001620 uniPortName = getUniPortPath(f.deviceHandler.device.Id, intfID, int32(onuID), int32(uniID))
Gamze Abakafee36392019-10-03 11:17:24 +00001621 }
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001622 if err := f.techprofile.DeleteTechProfileInstance(ctx, tpID, uniPortName); err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301623 return olterrors.NewErrAdapter("failed-to-delete-tp-instance-from-kv-store",
1624 log.Fields{
1625 "tp-id": tpID,
1626 "uni-port-name": uniPortName,
1627 "device-id": f.deviceHandler.device.Id}, err)
Devmalya Paul495b94a2019-08-27 19:42:00 -04001628 }
1629 return nil
1630}
1631
David K. Bainbridge794735f2020-02-11 21:01:37 -08001632func (f *OpenOltFlowMgr) addFlowToDevice(ctx context.Context, logicalFlow *ofp.OfpFlowStats, deviceFlow *openoltpb2.Flow) error {
Daniele Rossi22db98e2019-07-11 11:50:00 +00001633 var intfID uint32
1634 /* For flows which trap out of the NNI, the AccessIntfId is invalid
1635 (set to -1). In such cases, we need to refer to the NetworkIntfId .
1636 */
1637 if deviceFlow.AccessIntfId != -1 {
1638 intfID = uint32(deviceFlow.AccessIntfId)
1639 } else {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001640 // We need to log the valid interface ID.
1641 // For trap-on-nni flows, the access_intf_id is invalid (-1), so choose the network_intf_id.
Daniele Rossi22db98e2019-07-11 11:50:00 +00001642 intfID = uint32(deviceFlow.NetworkIntfId)
1643 }
1644
Neha Sharma96b7bf22020-06-15 10:37:32 +00001645 logger.Debugw(ctx, "sending-flow-to-device-via-grpc", log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301646 "flow-id": deviceFlow.FlowId,
Shrey Baid26912972020-04-16 21:02:31 +05301647 "device-id": f.deviceHandler.device.Id,
1648 "intf-id": intfID})
bseenivaa1622112025-12-11 18:24:02 +05301649 subCtx, cancel := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
1650 _, err := f.deviceHandler.Client.FlowAdd(subCtx, deviceFlow)
1651 cancel()
Daniele Rossi22db98e2019-07-11 11:50:00 +00001652
1653 st, _ := status.FromError(err)
1654 if st.Code() == codes.AlreadyExists {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001655 logger.Debug(ctx, "flow-already-exists", log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301656 "err": err,
1657 "flow-id": deviceFlow.FlowId,
1658 "device-id": f.deviceHandler.device.Id,
1659 "intf-id": intfID})
David K. Bainbridge794735f2020-02-11 21:01:37 -08001660 return nil
manikkaraj kbf256be2019-03-25 00:13:48 +05301661 }
Daniele Rossi22db98e2019-07-11 11:50:00 +00001662
1663 if err != nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001664 logger.Errorw(ctx, "failed-to-add-flow-to-device",
Shrey Baid26912972020-04-16 21:02:31 +05301665 log.Fields{"err": err,
bseeniva0b9cbcb2026-02-12 19:11:11 +05301666 "flow-id": deviceFlow.FlowId,
1667 "device-id": f.deviceHandler.device.Id,
1668 "intf-id": intfID})
David K. Bainbridge794735f2020-02-11 21:01:37 -08001669 return err
Daniele Rossi22db98e2019-07-11 11:50:00 +00001670 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001671 logger.Infow(ctx, "flow-added-to-device-successfully ",
Shrey Baid26912972020-04-16 21:02:31 +05301672 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301673 "flow-id": deviceFlow.FlowId,
Shrey Baid26912972020-04-16 21:02:31 +05301674 "device-id": f.deviceHandler.device.Id,
1675 "intf-id": intfID})
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001676
yasin saplid0566272021-12-21 09:10:30 +00001677 if err := f.registerFlow(ctx, logicalFlow, deviceFlow); err != nil {
1678 logger.Errorw(ctx, "failed-to-register-flow", log.Fields{"err": err})
1679 return err
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001680 }
David K. Bainbridge794735f2020-02-11 21:01:37 -08001681 return nil
Manjunath Vanarajulu28c3e822019-05-16 11:14:28 -04001682}
1683
Neha Sharma96b7bf22020-06-15 10:37:32 +00001684func (f *OpenOltFlowMgr) removeFlowFromDevice(ctx context.Context, deviceFlow *openoltpb2.Flow, ofFlowID uint64) error {
1685 logger.Debugw(ctx, "sending-flow-to-device-via-grpc",
Shrey Baid26912972020-04-16 21:02:31 +05301686 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301687 "flow-id": deviceFlow.FlowId,
Shrey Baid26912972020-04-16 21:02:31 +05301688 "device-id": f.deviceHandler.device.Id})
bseenivaa1622112025-12-11 18:24:02 +05301689 subCtx, cancel := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
1690 _, err := f.deviceHandler.Client.FlowRemove(subCtx, deviceFlow)
1691 cancel()
Manjunath Vanarajulu28c3e822019-05-16 11:14:28 -04001692 if err != nil {
serkant.uluderya245caba2019-09-24 23:15:29 -07001693 if f.deviceHandler.device.ConnectStatus == common.ConnectStatus_UNREACHABLE {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001694 logger.Warnw(ctx, "can-not-remove-flow-from-device--unreachable",
Shrey Baid26912972020-04-16 21:02:31 +05301695 log.Fields{
1696 "err": err,
1697 "deviceFlow": deviceFlow,
1698 "device-id": f.deviceHandler.device.Id})
Akash Kankanala041a2122024-10-16 15:49:22 +05301699 // Assume the flow is removed
David K. Bainbridge794735f2020-02-11 21:01:37 -08001700 return nil
serkant.uluderya245caba2019-09-24 23:15:29 -07001701 }
Girish Kumarf26e4882020-03-05 06:49:10 +00001702 return olterrors.NewErrFlowOp("remove", deviceFlow.FlowId, log.Fields{"deviceFlow": deviceFlow}, err)
Manjunath Vanarajulu28c3e822019-05-16 11:14:28 -04001703 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001704 logger.Infow(ctx, "flow-removed-from-device-successfully", log.Fields{
Matteo Scandolo92186242020-06-12 10:54:18 -07001705 "of-flow-id": ofFlowID,
bseeniva0b9cbcb2026-02-12 19:11:11 +05301706 "flow-id": deviceFlow.FlowId,
Matteo Scandolo92186242020-06-12 10:54:18 -07001707 "device-id": f.deviceHandler.device.Id,
1708 })
David K. Bainbridge794735f2020-02-11 21:01:37 -08001709 return nil
manikkaraj kbf256be2019-03-25 00:13:48 +05301710}
1711
David K. Bainbridge794735f2020-02-11 21:01:37 -08001712func (f *OpenOltFlowMgr) addLLDPFlow(ctx context.Context, flow *ofp.OfpFlowStats, portNo uint32) error {
Humera Kouser94d7a842019-08-25 19:04:32 -04001713 classifierInfo := make(map[string]interface{})
1714 actionInfo := make(map[string]interface{})
1715
1716 classifierInfo[EthType] = uint32(LldpEthType)
1717 classifierInfo[PacketTagType] = Untagged
1718 actionInfo[TrapToHost] = true
1719
1720 // LLDP flow is installed to trap LLDP packets on the NNI port.
1721 // We manage flow_id resource pool on per PON port basis.
1722 // Since this situation is tricky, as a hack, we pass the NNI port
1723 // index (network_intf_id) as PON port Index for the flow_id resource
1724 // pool. Also, there is no ONU Id available for trapping LLDP packets
1725 // on NNI port, use onu_id as -1 (invalid)
1726 // ****************** CAVEAT *******************
1727 // This logic works if the NNI Port Id falls within the same valid
1728 // range of PON Port Ids. If this doesn't work for some OLT Vendor
1729 // we need to have a re-look at this.
1730 // *********************************************
1731
1732 var onuID = -1
1733 var uniID = -1
1734 var gemPortID = -1
1735
Mahir Gunyel85f61c12021-10-06 11:53:45 -07001736 networkInterfaceID, err := plt.IntfIDFromNniPortNum(ctx, portNo)
David K. Bainbridge794735f2020-02-11 21:01:37 -08001737 if err != nil {
Thomas Lee S94109f12020-03-03 16:39:29 +05301738 return olterrors.NewErrInvalidValue(log.Fields{"nni-port-number": portNo}, err).Log()
David K. Bainbridge794735f2020-02-11 21:01:37 -08001739 }
yasin saplibddc2d72022-02-08 13:10:17 +00001740 present, err := f.resourceMgr.IsFlowOnKvStore(ctx, int32(onuID), flow.Id)
yasin saplid0566272021-12-21 09:10:30 +00001741 if present {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001742 logger.Infow(ctx, "flow-exists--not-re-adding", log.Fields{"device-id": f.deviceHandler.device.Id})
David K. Bainbridge794735f2020-02-11 21:01:37 -08001743 return nil
yasin saplid0566272021-12-21 09:10:30 +00001744 } else if err != nil {
1745 logger.Errorw(ctx, "aborting-addLLDPFlow--flow-may-already-exist",
1746 log.Fields{"intf-id": networkInterfaceID, "onu-id": onuID, "flow-id": flow.Id})
1747 return err
Humera Kouser94d7a842019-08-25 19:04:32 -04001748 }
Humera Kouser94d7a842019-08-25 19:04:32 -04001749
David K. Bainbridge794735f2020-02-11 21:01:37 -08001750 classifierProto, err := makeOpenOltClassifierField(classifierInfo)
1751 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301752 return olterrors.NewErrInvalidValue(
1753 log.Fields{
1754 "classifier": classifierInfo,
1755 "device-id": f.deviceHandler.device.Id}, err)
Humera Kouser94d7a842019-08-25 19:04:32 -04001756 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001757 logger.Debugw(ctx, "created-classifier-proto",
Shrey Baid26912972020-04-16 21:02:31 +05301758 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301759 "classifier": classifierProto.String(),
Shrey Baid26912972020-04-16 21:02:31 +05301760 "device-id": f.deviceHandler.device.Id})
Gamze Abaka724d0852020-03-18 12:10:24 +00001761 actionProto, err := makeOpenOltActionField(actionInfo, classifierInfo)
David K. Bainbridge794735f2020-02-11 21:01:37 -08001762 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301763 return olterrors.NewErrInvalidValue(
1764 log.Fields{
1765 "action": actionInfo,
1766 "device-id": f.deviceHandler.device.Id}, err)
Humera Kouser94d7a842019-08-25 19:04:32 -04001767 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001768 logger.Debugw(ctx, "created-action-proto",
Shrey Baid26912972020-04-16 21:02:31 +05301769 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301770 "action": actionProto.String(),
Shrey Baid26912972020-04-16 21:02:31 +05301771 "device-id": f.deviceHandler.device.Id})
Humera Kouser94d7a842019-08-25 19:04:32 -04001772
1773 downstreamflow := openoltpb2.Flow{AccessIntfId: int32(-1), // AccessIntfId not required
1774 OnuId: int32(onuID), // OnuId not required
1775 UniId: int32(uniID), // UniId not used
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001776 FlowId: flow.Id,
Humera Kouser94d7a842019-08-25 19:04:32 -04001777 FlowType: Downstream,
1778 NetworkIntfId: int32(networkInterfaceID),
1779 GemportId: int32(gemPortID),
1780 Classifier: classifierProto,
1781 Action: actionProto,
1782 Priority: int32(flow.Priority),
1783 Cookie: flow.Cookie,
1784 PortNo: portNo}
David K. Bainbridge794735f2020-02-11 21:01:37 -08001785 if err := f.addFlowToDevice(ctx, flow, &downstreamflow); err != nil {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001786 return olterrors.NewErrFlowOp("add", flow.Id,
Shrey Baid26912972020-04-16 21:02:31 +05301787 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301788 "flow-id": downstreamflow.FlowId,
Shrey Baid26912972020-04-16 21:02:31 +05301789 "device-id": f.deviceHandler.device.Id}, err)
Humera Kouser94d7a842019-08-25 19:04:32 -04001790 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001791 logger.Infow(ctx, "lldp-trap-on-nni-flow-added-to-device-successfully",
Shrey Baid26912972020-04-16 21:02:31 +05301792 log.Fields{
1793 "device-id": f.deviceHandler.device.Id,
1794 "onu-id": onuID,
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001795 "flow-id": flow.Id})
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001796
David K. Bainbridge794735f2020-02-11 21:01:37 -08001797 return nil
manikkaraj kbf256be2019-03-25 00:13:48 +05301798}
1799
Matteo Scandolod625b4c2020-04-02 16:16:01 -07001800func getUniPortPath(oltID string, intfID uint32, onuID int32, uniID int32) string {
1801 return fmt.Sprintf("olt-{%s}/pon-{%d}/onu-{%d}/uni-{%d}", oltID, intfID, onuID, uniID)
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07001802}
1803
Joey Armstrong3f0e2422023-07-05 18:25:41 -04001804// getOnuDevice to fetch onu from cache or core.
Neha Sharma96b7bf22020-06-15 10:37:32 +00001805func (f *OpenOltFlowMgr) getOnuDevice(ctx context.Context, intfID uint32, onuID uint32) (*OnuDevice, error) {
Mahir Gunyel0f89fd22020-04-11 18:24:42 -07001806 onuKey := f.deviceHandler.formOnuKey(intfID, onuID)
1807 onuDev, ok := f.deviceHandler.onus.Load(onuKey)
1808 if !ok {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001809 logger.Debugw(ctx, "couldnt-find-onu-in-cache",
Shrey Baid26912972020-04-16 21:02:31 +05301810 log.Fields{
1811 "intf-id": intfID,
1812 "onu-id": onuID,
1813 "device-id": f.deviceHandler.device.Id})
Neha Sharma96b7bf22020-06-15 10:37:32 +00001814 onuDevice, err := f.getChildDevice(ctx, intfID, onuID)
Mahir Gunyel0f89fd22020-04-11 18:24:42 -07001815 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301816 return nil, olterrors.NewErrNotFound("onu-child-device",
1817 log.Fields{
1818 "onu-id": onuID,
1819 "intf-id": intfID,
1820 "device-id": f.deviceHandler.device.Id}, err)
Mahir Gunyel0f89fd22020-04-11 18:24:42 -07001821 }
khenaidoo106c61a2021-08-11 18:05:46 -04001822 onuDev = NewOnuDevice(onuDevice.Id, onuDevice.Type, onuDevice.SerialNumber, onuDevice.ProxyAddress.OnuId, onuDevice.ProxyAddress.ChannelId, onuDevice.ProxyAddress.DeviceId, false, onuDevice.AdapterEndpoint)
Akash Kankanala041a2122024-10-16 15:49:22 +05301823 // better to ad the device to cache here.
Mahir Gunyel0f89fd22020-04-11 18:24:42 -07001824 f.deviceHandler.StoreOnuDevice(onuDev.(*OnuDevice))
1825 } else {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001826 logger.Debugw(ctx, "found-onu-in-cache",
Shrey Baid26912972020-04-16 21:02:31 +05301827 log.Fields{
1828 "intf-id": intfID,
1829 "onu-id": onuID,
1830 "device-id": f.deviceHandler.device.Id})
Mahir Gunyel0f89fd22020-04-11 18:24:42 -07001831 }
1832
1833 return onuDev.(*OnuDevice), nil
1834}
1835
Joey Armstrong3f0e2422023-07-05 18:25:41 -04001836// getChildDevice to fetch onu
Neha Sharma96b7bf22020-06-15 10:37:32 +00001837func (f *OpenOltFlowMgr) getChildDevice(ctx context.Context, intfID uint32, onuID uint32) (*voltha.Device, error) {
1838 logger.Infow(ctx, "GetChildDevice",
Shrey Baid26912972020-04-16 21:02:31 +05301839 log.Fields{
1840 "pon-port": intfID,
1841 "onu-id": onuID,
1842 "device-id": f.deviceHandler.device.Id})
Mahir Gunyel85f61c12021-10-06 11:53:45 -07001843 parentPortNo := plt.IntfIDToPortNo(intfID, voltha.Port_PON_OLT)
Neha Sharma96b7bf22020-06-15 10:37:32 +00001844 onuDevice, err := f.deviceHandler.GetChildDevice(ctx, parentPortNo, onuID)
David K. Bainbridge794735f2020-02-11 21:01:37 -08001845 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301846 return nil, olterrors.NewErrNotFound("onu",
1847 log.Fields{
1848 "interface-id": parentPortNo,
1849 "onu-id": onuID,
1850 "device-id": f.deviceHandler.device.Id},
Girish Kumarf26e4882020-03-05 06:49:10 +00001851 err)
manikkaraj kbf256be2019-03-25 00:13:48 +05301852 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001853 logger.Infow(ctx, "successfully-received-child-device-from-core",
Shrey Baid26912972020-04-16 21:02:31 +05301854 log.Fields{
1855 "device-id": f.deviceHandler.device.Id,
1856 "child_device_id": onuDevice.Id,
1857 "child_device_sn": onuDevice.SerialNumber})
Manikkaraj k884c1242019-04-11 16:26:42 +05301858 return onuDevice, nil
manikkaraj kbf256be2019-03-25 00:13:48 +05301859}
1860
Neha Sharma96b7bf22020-06-15 10:37:32 +00001861func (f *OpenOltFlowMgr) sendDeleteGemPortToChild(ctx context.Context, intfID uint32, onuID uint32, uniID uint32, gemPortID uint32, tpPath string) error {
1862 onuDev, err := f.getOnuDevice(ctx, intfID, onuID)
Girish Gowdra6b130582019-11-20 16:45:20 +05301863 if err != nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001864 logger.Debugw(ctx, "couldnt-find-onu-child-device",
Shrey Baid26912972020-04-16 21:02:31 +05301865 log.Fields{
1866 "intf-id": intfID,
1867 "onu-id": onuID,
1868 "uni-id": uniID,
1869 "device-id": f.deviceHandler.device.Id})
Mahir Gunyel0f89fd22020-04-11 18:24:42 -07001870 return err
Girish Gowdra6b130582019-11-20 16:45:20 +05301871 }
1872
khenaidoodc2116e2021-10-19 17:33:19 -04001873 delGemPortMsg := &ia.DeleteGemPortMessage{
khenaidoo106c61a2021-08-11 18:05:46 -04001874 DeviceId: onuDev.deviceID,
1875 UniId: uniID,
1876 TpInstancePath: tpPath,
1877 GemPortId: gemPortID,
1878 }
bseeniva0b9cbcb2026-02-12 19:11:11 +05301879 logger.Debugw(ctx, "sending-gem-port-delete-to-openonu-adapter", log.Fields{"msg": delGemPortMsg.String(), "child-device-id": onuDev.deviceID})
khenaidoo106c61a2021-08-11 18:05:46 -04001880
1881 if err := f.deviceHandler.sendDeleteGemPortToChildAdapter(ctx, onuDev.adapterEndpoint, delGemPortMsg); err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301882 return olterrors.NewErrCommunication("send-delete-gem-port-to-onu-adapter",
1883 log.Fields{
khenaidoo106c61a2021-08-11 18:05:46 -04001884 "from-adapter": f.deviceHandler.openOLT.config.AdapterEndpoint,
1885 "to-adapter": onuDev.adapterEndpoint,
Shrey Baid26912972020-04-16 21:02:31 +05301886 "onu-id": onuDev.deviceID,
1887 "proxyDeviceID": onuDev.proxyDeviceID,
khenaidoo106c61a2021-08-11 18:05:46 -04001888 "device-id": f.deviceHandler.device.Id}, err)
Girish Gowdra6b130582019-11-20 16:45:20 +05301889 }
khenaidoo106c61a2021-08-11 18:05:46 -04001890
Neha Sharma96b7bf22020-06-15 10:37:32 +00001891 logger.Infow(ctx, "success-sending-del-gem-port-to-onu-adapter",
Shrey Baid26912972020-04-16 21:02:31 +05301892 log.Fields{
khenaidoo106c61a2021-08-11 18:05:46 -04001893 "msg": delGemPortMsg,
1894 "from-adapter": f.deviceHandler.device.Type,
1895 "to-adapter": onuDev.deviceType,
1896 "device-id": f.deviceHandler.device.Id,
1897 "child-device-id": onuDev.deviceID})
Girish Gowdra6b130582019-11-20 16:45:20 +05301898 return nil
1899}
1900
Neha Sharma96b7bf22020-06-15 10:37:32 +00001901func (f *OpenOltFlowMgr) sendDeleteTcontToChild(ctx context.Context, intfID uint32, onuID uint32, uniID uint32, allocID uint32, tpPath string) error {
1902 onuDev, err := f.getOnuDevice(ctx, intfID, onuID)
Girish Gowdra6b130582019-11-20 16:45:20 +05301903 if err != nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +00001904 logger.Warnw(ctx, "couldnt-find-onu-child-device",
Shrey Baid26912972020-04-16 21:02:31 +05301905 log.Fields{
1906 "intf-id": intfID,
1907 "onu-id": onuID,
1908 "uni-id": uniID,
1909 "device-id": f.deviceHandler.device.Id})
Mahir Gunyel0f89fd22020-04-11 18:24:42 -07001910 return err
Girish Gowdra6b130582019-11-20 16:45:20 +05301911 }
1912
khenaidoodc2116e2021-10-19 17:33:19 -04001913 delTcontMsg := &ia.DeleteTcontMessage{
khenaidoo106c61a2021-08-11 18:05:46 -04001914 DeviceId: onuDev.deviceID,
1915 UniId: uniID,
1916 TpInstancePath: tpPath,
1917 AllocId: allocID,
1918 }
1919
Neha Sharma96b7bf22020-06-15 10:37:32 +00001920 logger.Debugw(ctx, "sending-tcont-delete-to-openonu-adapter",
Shrey Baid26912972020-04-16 21:02:31 +05301921 log.Fields{
bseeniva0b9cbcb2026-02-12 19:11:11 +05301922 "msg": delTcontMsg.String(),
Shrey Baid26912972020-04-16 21:02:31 +05301923 "device-id": f.deviceHandler.device.Id})
khenaidoo106c61a2021-08-11 18:05:46 -04001924
1925 if err := f.deviceHandler.sendDeleteTContToChildAdapter(ctx, onuDev.adapterEndpoint, delTcontMsg); err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05301926 return olterrors.NewErrCommunication("send-delete-tcont-to-onu-adapter",
1927 log.Fields{
khenaidoo106c61a2021-08-11 18:05:46 -04001928 "from-adapter": f.deviceHandler.openOLT.config.AdapterEndpoint,
1929 "to-adapter": onuDev.adapterEndpoint,
1930 "onu-id": onuDev.deviceID,
Shrey Baid26912972020-04-16 21:02:31 +05301931 "proxyDeviceID": onuDev.proxyDeviceID,
khenaidoo106c61a2021-08-11 18:05:46 -04001932 "device-id": f.deviceHandler.device.Id}, err)
Girish Gowdra6b130582019-11-20 16:45:20 +05301933 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00001934 logger.Infow(ctx, "success-sending-del-tcont-to-onu-adapter",
Shrey Baid26912972020-04-16 21:02:31 +05301935 log.Fields{
khenaidoo106c61a2021-08-11 18:05:46 -04001936 "msg": delTcontMsg,
1937 "device-id": f.deviceHandler.device.Id,
1938 "child-device-id": onuDev.deviceID})
Girish Gowdra6b130582019-11-20 16:45:20 +05301939 return nil
1940}
1941
Joey Armstrong3f0e2422023-07-05 18:25:41 -04001942// clearResources clears pon resources in kv store and the device
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -07001943// nolint: gocyclo
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001944func (f *OpenOltFlowMgr) clearResources(ctx context.Context, intfID uint32, onuID int32, uniID int32,
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05301945 flowID uint64, portNum uint32, tpID uint32, sendDeleteGemRequest bool, nniIntfID uint32) error {
Gamze Abaka745ccb72021-11-18 11:29:58 +00001946 logger.Debugw(ctx, "clearing-resources", log.Fields{"intfID": intfID, "onuID": onuID, "uniID": uniID, "tpID": tpID})
Gamze Abakafee36392019-10-03 11:17:24 +00001947
Girish Gowdraa482f272021-03-24 23:04:19 -07001948 uni := getUniPortPath(f.deviceHandler.device.Id, intfID, onuID, uniID)
yasin saplibddc2d72022-02-08 13:10:17 +00001949 tpPath := f.getTPpath(ctx, uni, tpID)
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001950 logger.Debugw(ctx, "getting-techprofile-instance-for-subscriber",
1951 log.Fields{
1952 "tpPath": tpPath,
1953 "device-id": f.deviceHandler.device.Id})
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001954
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001955 techprofileInst, err := f.techprofile.GetTPInstance(ctx, tpPath)
Girish Gowdra78fd63d2021-10-18 14:34:53 -07001956 if err != nil || techprofileInst == nil {
1957 // The child device is possibly deleted which in turn had cleaned up all the resources (including tp instances), check..
1958 childDevice, _ := f.getChildDevice(ctx, intfID, uint32(onuID)) // do not care about the error code
1959 if childDevice == nil {
1960 // happens when subscriber un-provision is immediately followed by child device delete
1961 // before all the flow removes are processed, the child device delete has already arrived and cleaned up all the resources
1962 logger.Warnw(ctx, "child device and its associated resources are already cleared", log.Fields{"intfID": intfID, "onuID": onuID, "uniID": uniID})
1963 return nil
1964 }
Elia Battiston2aaf4342022-02-07 15:16:38 +01001965 // If the tech profile is not found, since we want to delete it, there is no need to throw an error
1966 _ = olterrors.NewErrNotFound("tech-profile-in-kv-store",
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001967 log.Fields{
1968 "tp-id": tpID,
Elia Battiston2aaf4342022-02-07 15:16:38 +01001969 "path": tpPath}, err).Log()
1970 return nil
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001971 }
Gamze Abaka745ccb72021-11-18 11:29:58 +00001972
1973 var allGemPortsFree = true
Girish Gowdraa09aeab2020-09-14 16:30:52 -07001974 switch techprofileInst := techprofileInst.(type) {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07001975 case *tp_pb.TechProfileInstance:
Gamze Abaka745ccb72021-11-18 11:29:58 +00001976 for _, gemPort := range techprofileInst.UpstreamGemPortAttributeList {
1977 gemPortID := gemPort.GemportId
Girish Gowdrab4c33302022-03-18 15:07:38 -07001978 flowIDs, err := f.resourceMgr.GetFlowIDsForGem(ctx, gemPortID)
1979 if err != nil {
1980 return err
1981 }
1982 used := false
1983 for _, id := range flowIDs {
1984 if flowID != id {
1985 used = true
1986 break
yasin sapli9e4c5092022-02-01 13:52:33 +00001987 }
Girish Gowdrab4c33302022-03-18 15:07:38 -07001988 }
1989 if used {
Gamze Abaka745ccb72021-11-18 11:29:58 +00001990 for i, flowIDinMap := range flowIDs {
1991 if flowIDinMap == flowID {
1992 flowIDs = append(flowIDs[:i], flowIDs[i+1:]...)
yasin saplibddc2d72022-02-08 13:10:17 +00001993 if err := f.resourceMgr.UpdateFlowIDsForGem(ctx, gemPortID, flowIDs); err != nil {
Gamze Abaka745ccb72021-11-18 11:29:58 +00001994 return err
1995 }
1996 break
1997 }
1998 }
1999 logger.Debugw(ctx, "gem-port-id-is-still-used-by-other-flows",
2000 log.Fields{
2001 "gemport-id": gemPortID,
2002 "usedByFlows": flowIDs,
2003 "currentFlow": flowID,
2004 "device-id": f.deviceHandler.device.Id})
2005 allGemPortsFree = false
2006 }
2007 }
2008 if !allGemPortsFree {
2009 return nil
2010 }
2011 }
2012
2013 logger.Debugw(ctx, "all-gem-ports-are-free-to-be-deleted", log.Fields{"intfID": intfID, "onuID": onuID, "uniID": uniID, "tpID": tpID})
Girish Gowdraf3728b12022-02-02 21:46:51 -08002014
2015 // Free TPInstance, TPID, GemPorts and Traffic Queues. AllocID and Schedulers will be cleared later only if they are not shared across all the UNIs
Gamze Abaka745ccb72021-11-18 11:29:58 +00002016 switch techprofileInst := techprofileInst.(type) {
2017 case *tp_pb.TechProfileInstance:
Girish Gowdraf3728b12022-02-02 21:46:51 -08002018 for _, gemPort := range techprofileInst.UpstreamGemPortAttributeList {
2019 gemPortID := gemPort.GemportId
yasin saplibddc2d72022-02-08 13:10:17 +00002020 _ = f.resourceMgr.RemoveGemFromOnuGemInfo(ctx, uint32(onuID), gemPortID) // ignore error and proceed.
Girish Gowdraf3728b12022-02-02 21:46:51 -08002021
yasin saplibddc2d72022-02-08 13:10:17 +00002022 if err := f.resourceMgr.DeleteFlowIDsForGem(ctx, gemPortID); err != nil {
Girish Gowdraf3728b12022-02-02 21:46:51 -08002023 logger.Errorw(ctx, "error-removing-flow-ids-of-gem-port",
2024 log.Fields{
2025 "err": err,
2026 "intf": intfID,
2027 "onu-id": onuID,
2028 "uni-id": uniID,
2029 "device-id": f.deviceHandler.device.Id,
2030 "gemport-id": gemPortID})
2031 }
Girish Gowdraf3728b12022-02-02 21:46:51 -08002032 }
2033 // Remove queues at OLT in upstream and downstream direction
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302034 schedQ := schedQueue{tpInst: techprofileInst, direction: tp_pb.Direction_UPSTREAM, intfID: intfID, nniIntfID: nniIntfID, onuID: uint32(onuID), uniID: uint32(uniID), tpID: tpID, uniPort: portNum}
Akash Kankanala041a2122024-10-16 15:49:22 +05302035 if err := f.RemoveQueues(ctx, schedQ); err != nil {
Girish Gowdraf3728b12022-02-02 21:46:51 -08002036 logger.Warn(ctx, err)
2037 }
Akash Kankanala041a2122024-10-16 15:49:22 +05302038 schedQ.direction = tp_pb.Direction_DOWNSTREAM
2039 if err := f.RemoveQueues(ctx, schedQ); err != nil {
Girish Gowdraf3728b12022-02-02 21:46:51 -08002040 logger.Warn(ctx, err)
2041 }
2042 }
2043
2044 switch techprofileInst := techprofileInst.(type) {
2045 case *tp_pb.TechProfileInstance:
2046 // Proceed to free allocid and cleanup schedulers (US/DS) if no other references are found for this TP across all the UNIs on the ONU
Akash Kankanala041a2122024-10-16 15:49:22 +05302047 schedQ := schedQueue{direction: tp_pb.Direction_UPSTREAM, intfID: intfID, onuID: uint32(onuID), uniID: uint32(uniID), tpID: tpID, uniPort: portNum, tpInst: techprofileInst}
2048 allocExists := f.isAllocUsedByAnotherUNI(ctx, schedQ)
Andrea Campanella8a0d0502022-01-31 15:31:59 +01002049 if !allocExists {
Girish Gowdraf3728b12022-02-02 21:46:51 -08002050 // all alloc object references removed, remove upstream scheduler
yasin saplibddc2d72022-02-08 13:10:17 +00002051 if KvStoreMeter, _ := f.resourceMgr.GetMeterInfoForOnu(ctx, "upstream", uint32(onuID), uint32(uniID), tpID); KvStoreMeter != nil {
Akash Kankanala041a2122024-10-16 15:49:22 +05302052 if err := f.RemoveScheduler(ctx, schedQ); err != nil {
Andrea Campanella8a0d0502022-01-31 15:31:59 +01002053 logger.Warn(ctx, err)
2054 }
2055 }
Girish Gowdraf3728b12022-02-02 21:46:51 -08002056 // remove alloc id from resource pool by setting the 'freeFromResourcePool' to true
yasin saplibddc2d72022-02-08 13:10:17 +00002057 f.resourceMgr.FreeAllocID(ctx, uint32(onuID), uint32(uniID), techprofileInst.UsScheduler.AllocId, true)
Girish Gowdraf3728b12022-02-02 21:46:51 -08002058 } else {
2059 // just remove meter reference for the upstream direction for the current pon/onu/uni
2060 // The upstream scheduler, alloc id and meter-reference for the last remaining pon/onu/uni will be removed when no other alloc references that TP
Akash Kankanala041a2122024-10-16 15:49:22 +05302061 if err := f.removeMeterReference(ctx, "upstream", schedQ); err != nil {
Girish Gowdraf3728b12022-02-02 21:46:51 -08002062 return err
Andrea Campanella8a0d0502022-01-31 15:31:59 +01002063 }
Girish Gowdraf3728b12022-02-02 21:46:51 -08002064 // setting 'freeFromResourcePool' to false in resourceMgr.FreeAllocID will only remove alloc-id data for the given pon/onu/uni
2065 // but still preserve it on the resource pool.
yasin saplibddc2d72022-02-08 13:10:17 +00002066 f.resourceMgr.FreeAllocID(ctx, uint32(onuID), uint32(uniID), techprofileInst.UsScheduler.AllocId, false)
Girish Gowdraf3728b12022-02-02 21:46:51 -08002067 }
Andrea Campanella8a0d0502022-01-31 15:31:59 +01002068
Girish Gowdraf3728b12022-02-02 21:46:51 -08002069 // Downstream scheduler removal is simple, just invoke RemoveScheduler without all the complex handling we do for the alloc object.
Akash Kankanala041a2122024-10-16 15:49:22 +05302070 schedQ.direction = tp_pb.Direction_DOWNSTREAM
yasin saplibddc2d72022-02-08 13:10:17 +00002071 if KvStoreMeter, _ := f.resourceMgr.GetMeterInfoForOnu(ctx, "downstream", uint32(onuID), uint32(uniID), tpID); KvStoreMeter != nil {
Akash Kankanala041a2122024-10-16 15:49:22 +05302072 if err := f.RemoveScheduler(ctx, schedQ); err != nil {
Girish Gowdraf3728b12022-02-02 21:46:51 -08002073 logger.Warn(ctx, err)
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002074 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002075 }
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002076 case *tp_pb.EponTechProfileInstance:
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002077 // Delete the TCONT on the ONU.
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002078 if err := f.sendDeleteTcontToChild(ctx, intfID, uint32(onuID), uint32(uniID), techprofileInst.AllocId, tpPath); err != nil {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002079 logger.Errorw(ctx, "error-processing-delete-tcont-towards-onu",
Shrey Baid26912972020-04-16 21:02:31 +05302080 log.Fields{
Girish Gowdraa482f272021-03-24 23:04:19 -07002081 "intf": intfID,
Shrey Baid26912972020-04-16 21:02:31 +05302082 "onu-id": onuID,
2083 "uni-id": uniID,
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002084 "device-id": f.deviceHandler.device.Id,
khenaidoo106c61a2021-08-11 18:05:46 -04002085 "alloc-id": techprofileInst.AllocId,
2086 "error": err})
Gamze Abakafee36392019-10-03 11:17:24 +00002087 }
yasin saplibddc2d72022-02-08 13:10:17 +00002088 f.resourceMgr.FreeAllocID(ctx, uint32(onuID), uint32(uniID), techprofileInst.AllocId, true)
Abhilash Laxmeshward0f58cf2022-06-01 12:15:19 +05302089 if err := f.resourceMgr.RemoveTechProfileIDForOnu(ctx, uint32(onuID), uint32(uniID), tpID); err != nil {
2090 logger.Warn(ctx, err)
2091 }
2092 if err := f.DeleteTechProfileInstance(ctx, intfID, uint32(onuID), uint32(uniID), "", tpID); err != nil {
2093 logger.Warn(ctx, err)
2094 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002095 default:
2096 logger.Errorw(ctx, "error-unknown-tech",
2097 log.Fields{
2098 "techprofileInst": techprofileInst})
Manjunath Vanarajulu28c3e822019-05-16 11:14:28 -04002099 }
Girish Gowdraf3728b12022-02-02 21:46:51 -08002100
2101 // Free TPInstance, TPID, GemPorts and Traffic Queues. AllocID and Schedulers will be cleared later only if they are not shared across all the UNIs
2102 switch techprofileInst := techprofileInst.(type) {
2103 case *tp_pb.TechProfileInstance:
2104 for _, gemPort := range techprofileInst.UpstreamGemPortAttributeList {
2105 // Delete the gem port on the ONU.
2106 if sendDeleteGemRequest {
2107 if err := f.sendDeleteGemPortToChild(ctx, intfID, uint32(onuID), uint32(uniID), gemPort.GemportId, tpPath); err != nil {
2108 logger.Errorw(ctx, "error-processing-delete-gem-port-towards-onu",
2109 log.Fields{
2110 "err": err,
2111 "intfID": intfID,
2112 "onu-id": onuID,
2113 "uni-id": uniID,
2114 "device-id": f.deviceHandler.device.Id,
2115 "gemport-id": gemPort.GemportId})
2116 }
yasin saplibddc2d72022-02-08 13:10:17 +00002117 f.resourceMgr.FreeGemPortID(ctx, uint32(onuID), uint32(uniID), gemPort.GemportId)
Girish Gowdraf3728b12022-02-02 21:46:51 -08002118 }
2119 }
Akash Kankanala041a2122024-10-16 15:49:22 +05302120 // Delete the tp instance and the techprofileid for onu at the end
Abhilash Laxmeshward0f58cf2022-06-01 12:15:19 +05302121 if err := f.DeleteTechProfileInstance(ctx, intfID, uint32(onuID), uint32(uniID), "", tpID); err != nil {
2122 logger.Warn(ctx, err)
2123 }
2124 if err := f.resourceMgr.RemoveTechProfileIDForOnu(ctx, uint32(onuID), uint32(uniID), tpID); err != nil {
2125 logger.Warn(ctx, err)
2126 }
Girish Gowdraf3728b12022-02-02 21:46:51 -08002127 }
2128
Abhilash Laxmeshwarb7300fe2019-11-13 03:38:33 +05302129 return nil
2130}
2131
David K. Bainbridge794735f2020-02-11 21:01:37 -08002132// nolint: gocyclo
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302133func (f *OpenOltFlowMgr) clearFlowFromDeviceAndResourceManager(ctx context.Context, flow *ofp.OfpFlowStats, flowDirection string, nni_port uint32) error {
balaji.nagarajan31db4ea2026-01-28 09:13:55 +05302134 logger.Debugw(ctx, "clear-flow-from-resource-manager",
Shrey Baid26912972020-04-16 21:02:31 +05302135 log.Fields{
2136 "flowDirection": flowDirection,
bseeniva0b9cbcb2026-02-12 19:11:11 +05302137 "flow-id": flow.Id,
Shrey Baid26912972020-04-16 21:02:31 +05302138 "device-id": f.deviceHandler.device.Id})
Esin Karamanccb714b2019-11-29 15:02:06 +00002139
2140 if flowDirection == Multicast {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002141 return f.clearMulticastFlowFromResourceManager(ctx, flow)
Esin Karamanccb714b2019-11-29 15:02:06 +00002142 }
2143
Girish Gowdra5c00ef12021-12-01 17:19:41 +05302144 var ethType, ipProto, inPort uint32
2145 for _, field := range flows.GetOfbFields(flow) {
mgouda86543582025-10-29 20:58:16 +05302146 switch field.Type {
2147 case flows.IP_PROTO:
Girish Gowdra5c00ef12021-12-01 17:19:41 +05302148 ipProto = field.GetIpProto()
2149 logger.Debugw(ctx, "field-type-ip-proto", log.Fields{"ipProto": ipProto})
mgouda86543582025-10-29 20:58:16 +05302150 case flows.ETH_TYPE:
Girish Gowdra5c00ef12021-12-01 17:19:41 +05302151 ethType = field.GetEthType()
2152 logger.Debugw(ctx, "field-type-eth-type", log.Fields{"ethType": ethType})
mgouda86543582025-10-29 20:58:16 +05302153 case flows.IN_PORT:
Girish Gowdra5c00ef12021-12-01 17:19:41 +05302154 inPort = field.GetPort()
2155 logger.Debugw(ctx, "field-type-in-port", log.Fields{"inPort": inPort})
2156 }
2157 }
2158 portType := plt.IntfIDToPortTypeName(inPort)
2159 if (ethType == uint32(LldpEthType) || ipProto == uint32(IPProtoDhcp) || ipProto == uint32(IgmpProto)) &&
2160 (portType == voltha.Port_ETHERNET_NNI) {
2161 removeFlowMessage := openoltpb2.Flow{FlowId: flow.Id, AccessIntfId: -1, OnuId: -1, UniId: -1, TechProfileId: 0, FlowType: Downstream}
2162 logger.Debugw(ctx, "nni-trap-flow-to-be-deleted", log.Fields{"flow": flow})
2163 return f.removeFlowFromDevice(ctx, &removeFlowMessage, flow.Id)
2164 // No more processing needed for trap from nni flows.
2165 }
Abhilash Laxmeshwarb7300fe2019-11-13 03:38:33 +05302166
Girish Gowdra5c00ef12021-12-01 17:19:41 +05302167 portNum, Intf, onu, uni, _, _, err := plt.FlowExtractInfo(ctx, flow, flowDirection)
Abhilash Laxmeshwarb7300fe2019-11-13 03:38:33 +05302168 if err != nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002169 logger.Error(ctx, err)
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002170 return err
Abhilash Laxmeshwarb7300fe2019-11-13 03:38:33 +05302171 }
David K. Bainbridge794735f2020-02-11 21:01:37 -08002172 onuID := int32(onu)
2173 uniID := int32(uni)
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002174 tpID, err := getTpIDFromFlow(ctx, flow)
2175 if err != nil {
2176 return olterrors.NewErrNotFound("tp-id",
2177 log.Fields{
2178 "flow": flow,
2179 "intf-id": Intf,
2180 "onu-id": onuID,
2181 "uni-id": uniID,
2182 "device-id": f.deviceHandler.device.Id}, err)
2183 }
Abhilash Laxmeshwarb7300fe2019-11-13 03:38:33 +05302184
Neha Sharma96b7bf22020-06-15 10:37:32 +00002185 logger.Infow(ctx, "extracted-access-info-from-flow-to-be-deleted",
Shrey Baid26912972020-04-16 21:02:31 +05302186 log.Fields{
Matteo Scandolo92186242020-06-12 10:54:18 -07002187 "flow-id": flow.Id,
2188 "intf-id": Intf,
Shrey Baid26912972020-04-16 21:02:31 +05302189 "onu-id": onuID,
2190 "uni-id": uniID})
Abhilash Laxmeshwarb7300fe2019-11-13 03:38:33 +05302191
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002192 removeFlowMessage := openoltpb2.Flow{FlowId: flow.Id, AccessIntfId: int32(Intf), OnuId: onuID, UniId: uniID, TechProfileId: tpID, FlowType: flowDirection}
2193 logger.Debugw(ctx, "flow-to-be-deleted", log.Fields{"flow": flow})
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002194 if err = f.removeFlowFromDevice(ctx, &removeFlowMessage, flow.Id); err != nil {
2195 return err
2196 }
Girish Gowdra82c80982021-03-26 16:22:02 -07002197
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302198 if err = f.clearResources(ctx, Intf, onuID, uniID, flow.Id, portNum, tpID, true, nni_port); err != nil {
Gamze Abaka745ccb72021-11-18 11:29:58 +00002199 logger.Errorw(ctx, "failed-to-clear-resources-for-flow", log.Fields{
2200 "flow-id": flow.Id,
2201 "device-id": f.deviceHandler.device.Id,
2202 "onu-id": onuID,
2203 "intf": Intf,
2204 "err": err,
2205 })
2206 return err
Abhilash Laxmeshwarb7300fe2019-11-13 03:38:33 +05302207 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002208
Girish Gowdra82c80982021-03-26 16:22:02 -07002209 // Decrement reference count for the meter associated with the given <(pon_id, onu_id, uni_id)>/<tp_id>/meter_id/<direction>
balaji.nagarajan29be8bc2026-01-26 16:31:41 +05302210 if err := f.resourceMgr.HandleMeterInfoRefCntUpdate(ctx, nil, flowDirection, uint32(onuID), uint32(uniID), tpID, false); err != nil {
Girish Gowdra82c80982021-03-26 16:22:02 -07002211 return err
2212 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002213 return nil
Manjunath Vanarajulu28c3e822019-05-16 11:14:28 -04002214}
2215
Joey Armstrong3f0e2422023-07-05 18:25:41 -04002216// RemoveFlow removes the flow from the device
Girish Gowdracefae192020-03-19 18:14:10 -07002217func (f *OpenOltFlowMgr) RemoveFlow(ctx context.Context, flow *ofp.OfpFlowStats) error {
bseeniva0b9cbcb2026-02-12 19:11:11 +05302218 logger.Debugw(ctx, "removing-flow", log.Fields{"flow-id": flow.Id})
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302219 var direction string
2220 actionInfo := make(map[string]interface{})
Manjunath Vanarajulu28c3e822019-05-16 11:14:28 -04002221
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302222 for _, action := range flows.GetActions(flow) {
2223 if action.Type == flows.OUTPUT {
2224 if out := action.GetOutput(); out != nil {
2225 actionInfo[Output] = out.GetPort()
Neha Sharma96b7bf22020-06-15 10:37:32 +00002226 logger.Debugw(ctx, "action-type-output", log.Fields{"out_port": actionInfo[Output].(uint32)})
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302227 } else {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002228 logger.Error(ctx, "invalid-output-port-in-action")
Girish Gowdracefae192020-03-19 18:14:10 -07002229 return olterrors.NewErrInvalidValue(log.Fields{"invalid-out-port-action": 0}, nil)
Manjunath Vanarajulu28c3e822019-05-16 11:14:28 -04002230 }
2231 }
2232 }
Esin Karamanccb714b2019-11-29 15:02:06 +00002233
2234 if flows.HasGroup(flow) {
2235 direction = Multicast
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002236 } else if plt.IsUpstream(actionInfo[Output].(uint32)) {
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302237 direction = Upstream
2238 } else {
2239 direction = Downstream
2240 }
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302241
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302242 var nni_port uint32
2243 if direction == Upstream {
2244 if !plt.IsControllerBoundFlow(actionInfo[Output].(uint32)) {
2245 nni_port = actionInfo[Output].(uint32) & 0x1f // convert e.g. 16777220 to port 4
2246 }
2247 } else {
2248 classifierInfo := make(map[string]interface{})
2249 formulateClassifierInfoFromFlow(ctx, classifierInfo, flow)
2250 nni_port = classifierInfo[InPort].(uint32) & 0x1f // convert e.g. 16777220 to port 4
2251 }
2252
Girish Gowdracefae192020-03-19 18:14:10 -07002253 // Serialize flow removes on a per subscriber basis
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302254 err := f.clearFlowFromDeviceAndResourceManager(ctx, flow, direction, nni_port)
Girish Gowdracefae192020-03-19 18:14:10 -07002255
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002256 return err
Manjunath Vanarajulu28c3e822019-05-16 11:14:28 -04002257}
2258
Joey Armstrong3f0e2422023-07-05 18:25:41 -04002259// isIgmpTrapDownstreamFlow return true if the flow is a downsteam IGMP trap-to-host flow; false otherwise
Esin Karamanae41e2b2019-12-17 18:13:13 +00002260func isIgmpTrapDownstreamFlow(classifierInfo map[string]interface{}) bool {
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002261 if portType := plt.IntfIDToPortTypeName(classifierInfo[InPort].(uint32)); portType == voltha.Port_ETHERNET_NNI {
Esin Karamanae41e2b2019-12-17 18:13:13 +00002262 if ethType, ok := classifierInfo[EthType]; ok {
2263 if ethType.(uint32) == IPv4EthType {
2264 if ipProto, ok := classifierInfo[IPProto]; ok {
2265 if ipProto.(uint32) == IgmpProto {
2266 return true
2267 }
2268 }
2269 }
2270 }
2271 }
2272 return false
2273}
2274
Girish Gowdrafb3d6102020-10-16 16:32:36 -07002275// RouteFlowToOnuChannel routes incoming flow to ONU specific channel
khenaidoodc2116e2021-10-19 17:33:19 -04002276func (f *OpenOltFlowMgr) RouteFlowToOnuChannel(ctx context.Context, flow *ofp.OfpFlowStats, addFlow bool, flowMetadata *ofp.FlowMetadata) error {
Girish Gowdra20e3dcd2021-11-18 22:56:49 -08002277 if f.deviceHandler.getDeviceDeletionInProgressFlag() {
2278 // The device itself is going to be reset as part of deletion. So nothing to be done.
2279 logger.Infow(ctx, "device-deletion-in-progress--not-handling-flows-or-groups", log.Fields{"device-id": f.deviceHandler.device.Id})
2280 return nil
2281 }
Girish Gowdrafb3d6102020-10-16 16:32:36 -07002282 // Step1 : Fill flowControlBlock
2283 // Step2 : Push the flowControlBlock to ONU channel
2284 // Step3 : Wait on response channel for response
2285 // Step4 : Return error value
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002286 startTime := time.Now()
2287 logger.Infow(ctx, "process-flow", log.Fields{"flow": flow, "addFlow": addFlow})
Girish Gowdrafb3d6102020-10-16 16:32:36 -07002288 errChan := make(chan error)
2289 flowCb := flowControlBlock{
2290 ctx: ctx,
2291 addFlow: addFlow,
2292 flow: flow,
2293 flowMetadata: flowMetadata,
2294 errChan: &errChan,
2295 }
2296 inPort, outPort := getPorts(flow)
2297 var onuID uint32
2298 if inPort != InvalidPort && outPort != InvalidPort {
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002299 _, _, onuID, _ = plt.ExtractAccessFromFlow(inPort, outPort)
Girish Gowdrafb3d6102020-10-16 16:32:36 -07002300 }
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002301 if f.flowHandlerRoutineActive[onuID] {
2302 // inPort or outPort is InvalidPort for trap-from-nni flows.
2303 // In the that case onuID is 0 which is the reserved index for trap-from-nni flows in the f.incomingFlows slice
2304 // Send the flowCb on the ONU flow channel
2305 f.incomingFlows[onuID] <- flowCb
2306 // Wait on the channel for flow handlers return value
2307 err := <-errChan
balaji.nagarajan31db4ea2026-01-28 09:13:55 +05302308 logger.Debugw(ctx, "process-flow-received-resp", log.Fields{"err": err, "totalTimeSeconds": time.Since(startTime).Seconds()})
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002309 return err
2310 }
2311 logger.Errorw(ctx, "flow handler routine not active for onu", log.Fields{"onuID": onuID, "ponPortIdx": f.ponPortIdx})
2312 return fmt.Errorf("flow-handler-routine-not-active-for-onu-%v-pon-%d", onuID, f.ponPortIdx)
Girish Gowdrafb3d6102020-10-16 16:32:36 -07002313}
2314
2315// This routine is unique per ONU ID and blocks on flowControlBlock channel for incoming flows
2316// Each incoming flow is processed in a synchronous manner, i.e., the flow is processed to completion before picking another
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002317func (f *OpenOltFlowMgr) perOnuFlowHandlerRoutine(handlerRoutineIndex int, subscriberFlowChannel chan flowControlBlock, stopHandler chan bool) {
Girish Gowdra20e3dcd2021-11-18 22:56:49 -08002318 var flowCb flowControlBlock
Girish Gowdrafb3d6102020-10-16 16:32:36 -07002319 for {
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002320 select {
Girish Gowdrafb3d6102020-10-16 16:32:36 -07002321 // block on the channel to receive an incoming flow
2322 // process the flow completely before proceeding to handle the next flow
Girish Gowdra20e3dcd2021-11-18 22:56:49 -08002323 case flowCb = <-subscriberFlowChannel:
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002324 if flowCb.addFlow {
balaji.nagarajan31db4ea2026-01-28 09:13:55 +05302325 logger.Debug(flowCb.ctx, "adding-flow-start")
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002326 startTime := time.Now()
2327 err := f.AddFlow(flowCb.ctx, flowCb.flow, flowCb.flowMetadata)
balaji.nagarajan31db4ea2026-01-28 09:13:55 +05302328 logger.Infow(flowCb.ctx, "adding-flow-complete", log.Fields{"processTimeSecs": time.Since(startTime).Seconds(), "flow": flowCb.flow})
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002329 // Pass the return value over the return channel
2330 *flowCb.errChan <- err
2331 } else {
balaji.nagarajan31db4ea2026-01-28 09:13:55 +05302332 logger.Debug(flowCb.ctx, "removing-flow-start")
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002333 startTime := time.Now()
2334 err := f.RemoveFlow(flowCb.ctx, flowCb.flow)
balaji.nagarajan31db4ea2026-01-28 09:13:55 +05302335 logger.Infow(flowCb.ctx, "removing-flow-complete", log.Fields{"processTimeSecs": time.Since(startTime).Seconds(), "flow": flowCb.flow})
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002336 // Pass the return value over the return channel
2337 *flowCb.errChan <- err
2338 }
2339 case <-stopHandler:
2340 f.flowHandlerRoutineActive[handlerRoutineIndex] = false
2341 return
Girish Gowdrafb3d6102020-10-16 16:32:36 -07002342 }
2343 }
2344}
2345
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002346// StopAllFlowHandlerRoutines stops all flow handler routines. Call this when device is being rebooted or deleted
Girish Gowdra20e3dcd2021-11-18 22:56:49 -08002347func (f *OpenOltFlowMgr) StopAllFlowHandlerRoutines(ctx context.Context, wg *sync.WaitGroup) {
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002348 for i, v := range f.stopFlowHandlerRoutine {
2349 if f.flowHandlerRoutineActive[i] {
Girish Gowdra20e3dcd2021-11-18 22:56:49 -08002350 select {
2351 case v <- true:
2352 case <-time.After(time.Second * 5):
2353 logger.Warnw(ctx, "timeout stopping flow handler routine", log.Fields{"onuID": i, "deviceID": f.deviceHandler.device.Id})
2354 }
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002355 }
Holger Hildebrandt143b5be2023-02-10 08:28:15 +00002356 f.stopFlowHandlerRoutine[i] = nil
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002357 }
Holger Hildebrandt143b5be2023-02-10 08:28:15 +00002358 f.stopFlowHandlerRoutine = nil
2359
2360 if f.incomingFlows != nil {
2361 for k := range f.incomingFlows {
2362 if f.incomingFlows[k] != nil {
2363 f.incomingFlows[k] = nil
2364 }
2365 }
2366 f.incomingFlows = nil
2367 }
2368
Girish Gowdra20e3dcd2021-11-18 22:56:49 -08002369 wg.Done()
Girish Gowdra4736e5c2021-08-25 15:19:10 -07002370 logger.Debugw(ctx, "stopped all flow handler routines", log.Fields{"ponPortIdx": f.ponPortIdx})
2371}
2372
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07002373// AddFlow add flow to device
Naga Manjunathb8438aa2020-01-02 17:52:33 +05302374// nolint: gocyclo
khenaidoodc2116e2021-10-19 17:33:19 -04002375func (f *OpenOltFlowMgr) AddFlow(ctx context.Context, flow *ofp.OfpFlowStats, flowMetadata *ofp.FlowMetadata) error {
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07002376 classifierInfo := make(map[string]interface{})
2377 actionInfo := make(map[string]interface{})
Manikkaraj kb1d51442019-07-23 10:41:02 -04002378 var UsMeterID uint32
2379 var DsMeterID uint32
2380
Neha Sharma96b7bf22020-06-15 10:37:32 +00002381 logger.Infow(ctx, "adding-flow",
Shrey Baid26912972020-04-16 21:02:31 +05302382 log.Fields{
2383 "flow": flow,
Matteo Scandolof16389e2021-05-18 00:47:08 +00002384 "flowmetadata": flowMetadata})
Neha Sharma96b7bf22020-06-15 10:37:32 +00002385 formulateClassifierInfoFromFlow(ctx, classifierInfo, flow)
salmansiddiqui7ac62132019-08-22 03:58:50 +00002386
Neha Sharma96b7bf22020-06-15 10:37:32 +00002387 err := formulateActionInfoFromFlow(ctx, actionInfo, classifierInfo, flow)
salmansiddiqui7ac62132019-08-22 03:58:50 +00002388 if err != nil {
2389 // Error logging is already done in the called function
2390 // So just return in case of error
Andrea Campanellac63bba92020-03-10 17:01:04 +01002391 return err
manikkaraj kbf256be2019-03-25 00:13:48 +05302392 }
salmansiddiqui7ac62132019-08-22 03:58:50 +00002393
Esin Karamanccb714b2019-11-29 15:02:06 +00002394 if flows.HasGroup(flow) {
2395 // handle multicast flow
Andrea Campanellac63bba92020-03-10 17:01:04 +01002396 return f.handleFlowWithGroup(ctx, actionInfo, classifierInfo, flow)
Esin Karamanccb714b2019-11-29 15:02:06 +00002397 }
2398
manikkaraj k17652a72019-05-06 09:06:36 -04002399 /* Controller bound trap flows */
Neha Sharma96b7bf22020-06-15 10:37:32 +00002400 err = formulateControllerBoundTrapFlowInfo(ctx, actionInfo, classifierInfo, flow)
salmansiddiqui7ac62132019-08-22 03:58:50 +00002401 if err != nil {
2402 // error if any, already logged in the called function
Andrea Campanellac63bba92020-03-10 17:01:04 +01002403 return err
manikkaraj k17652a72019-05-06 09:06:36 -04002404 }
salmansiddiqui7ac62132019-08-22 03:58:50 +00002405
Neha Sharma96b7bf22020-06-15 10:37:32 +00002406 logger.Debugw(ctx, "flow-ports",
Shrey Baid26912972020-04-16 21:02:31 +05302407 log.Fields{
2408 "classifierinfo_inport": classifierInfo[InPort],
2409 "action_output": actionInfo[Output]})
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002410 portNo, intfID, onuID, uniID := plt.ExtractAccessFromFlow(classifierInfo[InPort].(uint32), actionInfo[Output].(uint32))
A R Karthick1f85b802019-10-11 05:06:05 +00002411
Humera Kouser94d7a842019-08-25 19:04:32 -04002412 if ethType, ok := classifierInfo[EthType]; ok {
2413 if ethType.(uint32) == LldpEthType {
balaji.nagarajan31db4ea2026-01-28 09:13:55 +05302414 logger.Debug(ctx, "adding-lldp-flow")
Andrea Campanellac63bba92020-03-10 17:01:04 +01002415 return f.addLLDPFlow(ctx, flow, portNo)
Humera Kouser94d7a842019-08-25 19:04:32 -04002416 }
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03002417 if ethType.(uint32) == PPPoEDEthType {
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002418 if voltha.Port_ETHERNET_NNI == plt.IntfIDToPortTypeName(classifierInfo[InPort].(uint32)) {
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03002419 logger.Debug(ctx, "trap-pppoed-from-nni-flow")
2420 return f.addTrapFlowOnNNI(ctx, flow, classifierInfo, portNo)
2421 }
2422 }
Humera Kouser94d7a842019-08-25 19:04:32 -04002423 }
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07002424 if ipProto, ok := classifierInfo[IPProto]; ok {
2425 if ipProto.(uint32) == IPProtoDhcp {
2426 if udpSrc, ok := classifierInfo[UDPSrc]; ok {
Naga Manjunathb8438aa2020-01-02 17:52:33 +05302427 if udpSrc.(uint32) == uint32(67) || udpSrc.(uint32) == uint32(546) {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002428 logger.Debug(ctx, "trap-dhcp-from-nni-flow")
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03002429 return f.addTrapFlowOnNNI(ctx, flow, classifierInfo, portNo)
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002430 }
2431 }
2432 }
2433 }
Esin Karamanae41e2b2019-12-17 18:13:13 +00002434 if isIgmpTrapDownstreamFlow(classifierInfo) {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002435 logger.Debug(ctx, "trap-igmp-from-nni-flow")
Andrea Campanellac63bba92020-03-10 17:01:04 +01002436 return f.addIgmpTrapFlowOnNNI(ctx, flow, classifierInfo, portNo)
Esin Karamanae41e2b2019-12-17 18:13:13 +00002437 }
A R Karthick1f85b802019-10-11 05:06:05 +00002438
yasin saplibddc2d72022-02-08 13:10:17 +00002439 f.resourceMgr.AddUniPortToOnuInfo(ctx, onuID, portNo)
A R Karthick1f85b802019-10-11 05:06:05 +00002440
Girish Gowdra6071f382021-12-14 12:52:04 +05302441 tpID, err := getTpIDFromFlow(ctx, flow)
Chaitrashree G S90a17952019-11-14 21:51:21 -05002442 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05302443 return olterrors.NewErrNotFound("tpid-for-flow",
2444 log.Fields{
2445 "flow": flow,
2446 "intf-id": IntfID,
2447 "onu-id": onuID,
2448 "uni-id": uniID}, err)
Chaitrashree G S90a17952019-11-14 21:51:21 -05002449 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00002450 logger.Debugw(ctx, "tpid-for-this-subcriber",
Shrey Baid26912972020-04-16 21:02:31 +05302451 log.Fields{
Girish Gowdra6071f382021-12-14 12:52:04 +05302452 "tp-id": tpID,
Shrey Baid26912972020-04-16 21:02:31 +05302453 "intf-id": intfID,
2454 "onu-id": onuID,
2455 "uni-id": uniID})
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002456 if plt.IsUpstream(actionInfo[Output].(uint32)) {
Scott Baker355d1742019-10-24 10:57:52 -07002457 UsMeterID = flows.GetMeterIdFromFlow(flow)
Neha Sharma96b7bf22020-06-15 10:37:32 +00002458 logger.Debugw(ctx, "upstream-flow-meter-id", log.Fields{"us-meter-id": UsMeterID})
Girish Gowdra6071f382021-12-14 12:52:04 +05302459 if err := f.validateMeter(ctx, Upstream, UsMeterID, intfID, onuID, uniID, tpID); err != nil {
2460 logger.Errorw(ctx, "meter-validation-failed", log.Fields{"err": err})
2461 return err
2462 }
Manikkaraj kb1d51442019-07-23 10:41:02 -04002463 } else {
Scott Baker355d1742019-10-24 10:57:52 -07002464 DsMeterID = flows.GetMeterIdFromFlow(flow)
Neha Sharma96b7bf22020-06-15 10:37:32 +00002465 logger.Debugw(ctx, "downstream-flow-meter-id", log.Fields{"ds-meter-id": DsMeterID})
Girish Gowdra6071f382021-12-14 12:52:04 +05302466 if err := f.validateMeter(ctx, Downstream, DsMeterID, intfID, onuID, uniID, tpID); err != nil {
2467 logger.Errorw(ctx, "meter-validation-failed", log.Fields{"err": err})
2468 return err
2469 }
Manikkaraj kb1d51442019-07-23 10:41:02 -04002470 }
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302471
2472 var nni_port uint32
2473 if plt.IsUpstream(actionInfo[Output].(uint32)) {
2474 if !plt.IsControllerBoundFlow(actionInfo[Output].(uint32)) {
2475 nni_port = actionInfo[Output].(uint32) & 0x1f // convert e.g. 16777220 to port 4
2476 }
2477 } else {
2478 nni_port = classifierInfo[InPort].(uint32) & 0x1f // convert e.g. 16777220 to port 4
2479 }
2480
2481 return f.processAddFlow(ctx, intfID, nni_port, onuID, uniID, portNo, classifierInfo, actionInfo, flow, tpID, UsMeterID, DsMeterID, flowMetadata)
Girish Gowdra9602eb42020-09-09 15:50:39 -07002482}
Girish Gowdra3d633032019-12-10 16:37:05 +05302483
Esin Karamanccb714b2019-11-29 15:02:06 +00002484// handleFlowWithGroup adds multicast flow to the device.
David K. Bainbridge794735f2020-02-11 21:01:37 -08002485func (f *OpenOltFlowMgr) handleFlowWithGroup(ctx context.Context, actionInfo, classifierInfo map[string]interface{}, flow *ofp.OfpFlowStats) error {
Himani Chawlab6296c42021-10-28 11:50:56 +05302486 classifierInfo[PacketTagType] = getPacketTypeFromClassifiers(classifierInfo)
Neha Sharma96b7bf22020-06-15 10:37:32 +00002487 logger.Debugw(ctx, "add-multicast-flow", log.Fields{
Shrey Baid26912972020-04-16 21:02:31 +05302488 "classifier-info": classifierInfo,
2489 "actionInfo": actionInfo})
Esin Karamanccb714b2019-11-29 15:02:06 +00002490
Esin Karaman65409d82020-03-18 10:58:18 +00002491 networkInterfaceID, err := f.getNNIInterfaceIDOfMulticastFlow(ctx, classifierInfo)
Esin Karamanccb714b2019-11-29 15:02:06 +00002492 if err != nil {
Girish Kumarf26e4882020-03-05 06:49:10 +00002493 return olterrors.NewErrNotFound("multicast-in-port", log.Fields{"classifier": classifierInfo}, err)
Esin Karamanccb714b2019-11-29 15:02:06 +00002494 }
Esin Karamanfcddfcf2020-03-04 13:34:38 +00002495
Esin Karamanfcddfcf2020-03-04 13:34:38 +00002496 delete(classifierInfo, EthType)
Esin Karamanccb714b2019-11-29 15:02:06 +00002497
David K. Bainbridge794735f2020-02-11 21:01:37 -08002498 onuID := NoneOnuID
Esin Karamanccb714b2019-11-29 15:02:06 +00002499
yasin saplibddc2d72022-02-08 13:10:17 +00002500 present, err := f.resourceMgr.IsFlowOnKvStore(ctx, int32(onuID), flow.Id)
yasin saplid0566272021-12-21 09:10:30 +00002501 if present {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002502 logger.Infow(ctx, "multicast-flow-exists-not-re-adding", log.Fields{"classifier-info": classifierInfo})
David K. Bainbridge794735f2020-02-11 21:01:37 -08002503 return nil
yasin saplid0566272021-12-21 09:10:30 +00002504 } else if err != nil {
2505 logger.Errorw(ctx, "aborting-handleFlowWithGroup--flow-may-already-exist",
2506 log.Fields{"intf-id": networkInterfaceID, "onu-id": onuID, "flow-id": flow.Id})
2507 return err
Esin Karamanccb714b2019-11-29 15:02:06 +00002508 }
yasin saplid0566272021-12-21 09:10:30 +00002509
David K. Bainbridge794735f2020-02-11 21:01:37 -08002510 classifierProto, err := makeOpenOltClassifierField(classifierInfo)
2511 if err != nil {
Girish Kumarf26e4882020-03-05 06:49:10 +00002512 return olterrors.NewErrInvalidValue(log.Fields{"classifier": classifierInfo}, err)
Esin Karamanccb714b2019-11-29 15:02:06 +00002513 }
2514 groupID := actionInfo[GroupID].(uint32)
2515 multicastFlow := openoltpb2.Flow{
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002516 FlowId: flow.Id,
Esin Karamanccb714b2019-11-29 15:02:06 +00002517 FlowType: Multicast,
2518 NetworkIntfId: int32(networkInterfaceID),
2519 GroupId: groupID,
2520 Classifier: classifierProto,
2521 Priority: int32(flow.Priority),
2522 Cookie: flow.Cookie}
2523
Kent Hagermane6ff1012020-07-14 15:07:53 -04002524 if err := f.addFlowToDevice(ctx, flow, &multicastFlow); err != nil {
bseeniva0b9cbcb2026-02-12 19:11:11 +05302525 return olterrors.NewErrFlowOp("add", flow.Id, log.Fields{"flow-id": multicastFlow.FlowId}, err)
David K. Bainbridge794735f2020-02-11 21:01:37 -08002526 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00002527 logger.Info(ctx, "multicast-flow-added-to-device-successfully")
Akash Kankanala041a2122024-10-16 15:49:22 +05302528 // get cached group
Girish Gowdra9602eb42020-09-09 15:50:39 -07002529 if group, _, err := f.grpMgr.getFlowGroupFromKVStore(ctx, groupID, true); err == nil {
Akash Kankanala041a2122024-10-16 15:49:22 +05302530 // calling groupAdd to set group members after multicast flow creation
Girish Gowdra9602eb42020-09-09 15:50:39 -07002531 if err := f.grpMgr.ModifyGroup(ctx, group); err != nil {
Girish Kumarf26e4882020-03-05 06:49:10 +00002532 return olterrors.NewErrGroupOp("modify", groupID, log.Fields{"group": group}, err)
Esin Karamanccb714b2019-11-29 15:02:06 +00002533 }
Akash Kankanala041a2122024-10-16 15:49:22 +05302534 // cached group can be removed now
Kent Hagermane6ff1012020-07-14 15:07:53 -04002535 if err := f.resourceMgr.RemoveFlowGroupFromKVStore(ctx, groupID, true); err != nil {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002536 logger.Warnw(ctx, "failed-to-remove-flow-group", log.Fields{"group-id": groupID, "err": err})
Kent Hagermane6ff1012020-07-14 15:07:53 -04002537 }
Esin Karamanccb714b2019-11-29 15:02:06 +00002538 }
David K. Bainbridge794735f2020-02-11 21:01:37 -08002539
David K. Bainbridge794735f2020-02-11 21:01:37 -08002540 return nil
Esin Karamanccb714b2019-11-29 15:02:06 +00002541}
2542
Joey Armstrong3f0e2422023-07-05 18:25:41 -04002543// getNNIInterfaceIDOfMulticastFlow returns associated NNI interface id of the inPort criterion if exists; returns the first NNI interface of the device otherwise
Esin Karaman65409d82020-03-18 10:58:18 +00002544func (f *OpenOltFlowMgr) getNNIInterfaceIDOfMulticastFlow(ctx context.Context, classifierInfo map[string]interface{}) (uint32, error) {
2545 if inPort, ok := classifierInfo[InPort]; ok {
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002546 nniInterfaceID, err := plt.IntfIDFromNniPortNum(ctx, inPort.(uint32))
Esin Karaman65409d82020-03-18 10:58:18 +00002547 if err != nil {
2548 return 0, olterrors.NewErrInvalidValue(log.Fields{"nni-in-port-number": inPort}, err)
2549 }
2550 return nniInterfaceID, nil
Esin Karamanccb714b2019-11-29 15:02:06 +00002551 }
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002552
2553 // TODO: For now we support only one NNI port in VOLTHA. We shall use only the first NNI port, i.e., interface-id 0.
2554 return 0, nil
Esin Karamanccb714b2019-11-29 15:02:06 +00002555}
2556
Joey Armstrong3f0e2422023-07-05 18:25:41 -04002557// sendTPDownloadMsgToChild send payload
bseeniva0b9cbcb2026-02-12 19:11:11 +05302558func (f *OpenOltFlowMgr) sendTPDownloadMsgToChild(ctx context.Context, intfID uint32, onuID uint32, uniID uint32, uni string, TpID uint32, tpInst *tp_pb.TechProfileInstance) error {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002559 onuDev, err := f.getOnuDevice(ctx, intfID, onuID)
Manikkaraj k884c1242019-04-11 16:26:42 +05302560 if err != nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002561 logger.Errorw(ctx, "couldnt-find-onu-child-device",
Shrey Baid26912972020-04-16 21:02:31 +05302562 log.Fields{
2563 "intf-id": intfID,
2564 "onu-id": onuID,
2565 "uni-id": uniID})
Mahir Gunyel0f89fd22020-04-11 18:24:42 -07002566 return err
manikkaraj kbf256be2019-03-25 00:13:48 +05302567 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00002568 logger.Debugw(ctx, "got-child-device-from-olt-device-handler", log.Fields{"onu-id": onuDev.deviceID})
manikkaraj k17652a72019-05-06 09:06:36 -04002569
yasin saplibddc2d72022-02-08 13:10:17 +00002570 tpPath := f.getTPpath(ctx, uni, TpID)
khenaidoodc2116e2021-10-19 17:33:19 -04002571 tpDownloadMsg := &ia.TechProfileDownloadMessage{
khenaidoo106c61a2021-08-11 18:05:46 -04002572 DeviceId: onuDev.deviceID,
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002573 UniId: uniID,
2574 TpInstancePath: tpPath,
bseeniva0b9cbcb2026-02-12 19:11:11 +05302575 TechTpInstance: &ia.TechProfileDownloadMessage_TpInstance{TpInstance: tpInst},
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002576 }
bseeniva0b9cbcb2026-02-12 19:11:11 +05302577 logger.Debugw(ctx, "sending-load-tech-profile-request-to-brcm-onu-adapter", log.Fields{"tpDownloadMsg": tpDownloadMsg.String()})
khenaidoo106c61a2021-08-11 18:05:46 -04002578
2579 err = f.deviceHandler.sendDownloadTechProfileToChildAdapter(ctx, onuDev.adapterEndpoint, tpDownloadMsg)
2580 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05302581 return olterrors.NewErrCommunication("send-techprofile-download-request",
2582 log.Fields{
khenaidoo106c61a2021-08-11 18:05:46 -04002583 "from-adapter": f.deviceHandler.openOLT.config.AdapterEndpoint,
Shrey Baid26912972020-04-16 21:02:31 +05302584 "to-adapter": onuDev.deviceType,
2585 "onu-id": onuDev.deviceID,
khenaidoo106c61a2021-08-11 18:05:46 -04002586 "proxyDeviceID": onuDev.proxyDeviceID}, err)
manikkaraj k17652a72019-05-06 09:06:36 -04002587 }
balaji.nagarajan31db4ea2026-01-28 09:13:55 +05302588 logger.Debugw(ctx, "success-sending-load-tech-profile-request-to-brcm-onu-adapter", log.Fields{"tpid": TpID})
Manikkaraj k884c1242019-04-11 16:26:42 +05302589 return nil
manikkaraj kbf256be2019-03-25 00:13:48 +05302590}
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002591
Joey Armstrong3f0e2422023-07-05 18:25:41 -04002592// GetLogicalPortFromPacketIn function computes logical port UNI/NNI port from packet-in indication and returns the same
npujarec5762e2020-01-01 14:08:48 +05302593func (f *OpenOltFlowMgr) GetLogicalPortFromPacketIn(ctx context.Context, packetIn *openoltpb2.PacketIndication) (uint32, error) {
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002594 var logicalPortNum uint32
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002595
mgouda86543582025-10-29 20:58:16 +05302596 switch packetIn.IntfType {
2597 case "pon":
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002598 // packet indication does not have serial number , so sending as nil
Esin Karamandf392e12020-12-16 13:33:09 +00002599 // get onu and uni ids associated with the given pon and gem ports
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002600 onuID, uniID := packetIn.OnuId, packetIn.UniId
2601 logger.Debugf(ctx, "retrieved ONU and UNI IDs [%d, %d] by interface:%d, gem:%d", packetIn.OnuId, packetIn.UniId, packetIn.GemportId)
Esin Karamandf392e12020-12-16 13:33:09 +00002602
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002603 if packetIn.PortNo != 0 {
2604 logicalPortNum = packetIn.PortNo
2605 } else {
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002606 logicalPortNum = plt.MkUniPortNum(ctx, packetIn.IntfId, onuID, uniID)
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002607 }
2608 // Store the gem port through which the packet_in came. Use the same gem port for packet_out
Esin Karaman7fb80c22020-07-16 14:23:33 +00002609 f.UpdateGemPortForPktIn(ctx, packetIn.IntfId, onuID, logicalPortNum, packetIn.GemportId, packetIn.Pkt)
mgouda86543582025-10-29 20:58:16 +05302610 case "nni":
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002611 logicalPortNum = plt.IntfIDToPortNo(packetIn.IntfId, voltha.Port_ETHERNET_NNI)
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002612 }
Girish Gowdraeb450292020-10-26 10:03:39 -07002613
2614 if logger.V(log.DebugLevel) {
2615 logger.Debugw(ctx, "retrieved-logicalport-from-packet-in",
2616 log.Fields{
2617 "logical-port-num": logicalPortNum,
2618 "intf-type": packetIn.IntfType,
2619 "packet": hex.EncodeToString(packetIn.Pkt),
2620 })
2621 }
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002622 return logicalPortNum, nil
2623}
2624
Joey Armstrong3f0e2422023-07-05 18:25:41 -04002625// GetPacketOutGemPortID returns gemPortId
Esin Karaman7fb80c22020-07-16 14:23:33 +00002626func (f *OpenOltFlowMgr) GetPacketOutGemPortID(ctx context.Context, intfID uint32, onuID uint32, portNum uint32, packet []byte) (uint32, error) {
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07002627 var gemPortID uint32
Esin Karaman7fb80c22020-07-16 14:23:33 +00002628
2629 ctag, priority, err := getCTagFromPacket(ctx, packet)
2630 if err != nil {
2631 return 0, err
2632 }
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302633
Esin Karaman7fb80c22020-07-16 14:23:33 +00002634 pktInkey := rsrcMgr.PacketInInfoKey{IntfID: intfID, OnuID: onuID, LogicalPort: portNum, VlanID: ctag, Priority: priority}
Matteo Scandoloabf9c512020-06-23 19:31:14 -07002635 var ok bool
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002636 f.packetInGemPortLock.RLock()
Matteo Scandoloabf9c512020-06-23 19:31:14 -07002637 gemPortID, ok = f.packetInGemPort[pktInkey]
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002638 f.packetInGemPortLock.RUnlock()
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302639 if ok {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002640 logger.Debugw(ctx, "found-gemport-for-pktin-key",
Shrey Baid26912972020-04-16 21:02:31 +05302641 log.Fields{
2642 "pktinkey": pktInkey,
2643 "gem": gemPortID})
Matteo Scandoloabf9c512020-06-23 19:31:14 -07002644
2645 return gemPortID, nil
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002646 }
Akash Kankanala041a2122024-10-16 15:49:22 +05302647 // If gem is not found in cache try to get it from kv store, if found in kv store, update the cache and return.
Esin Karaman7fb80c22020-07-16 14:23:33 +00002648 gemPortID, err = f.resourceMgr.GetGemPortFromOnuPktIn(ctx, pktInkey)
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302649 if err == nil {
2650 if gemPortID != 0 {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002651 f.packetInGemPortLock.Lock()
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302652 f.packetInGemPort[pktInkey] = gemPortID
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002653 f.packetInGemPortLock.Unlock()
Neha Sharma96b7bf22020-06-15 10:37:32 +00002654 logger.Infow(ctx, "found-gem-port-from-kv-store-and-updating-cache-with-gemport",
Shrey Baid26912972020-04-16 21:02:31 +05302655 log.Fields{
2656 "pktinkey": pktInkey,
2657 "gem": gemPortID})
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302658 return gemPortID, nil
2659 }
2660 }
Shrey Baid26912972020-04-16 21:02:31 +05302661 return uint32(0), olterrors.NewErrNotFound("gem-port",
2662 log.Fields{
2663 "pktinkey": pktInkey,
2664 "gem": gemPortID}, err)
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002665}
2666
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03002667func (f *OpenOltFlowMgr) addTrapFlowOnNNI(ctx context.Context, logicalFlow *ofp.OfpFlowStats, classifier map[string]interface{}, portNo uint32) error {
2668 logger.Debug(ctx, "adding-trap-of-nni-flow")
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002669 action := make(map[string]interface{})
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07002670 classifier[PacketTagType] = DoubleTag
2671 action[TrapToHost] = true
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002672 /* We manage flowId resource pool on per PON port basis.
2673 Since this situation is tricky, as a hack, we pass the NNI port
2674 index (network_intf_id) as PON port Index for the flowId resource
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03002675 pool. Also, there is no ONU Id available for trapping packets
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002676 on NNI port, use onu_id as -1 (invalid)
2677 ****************** CAVEAT *******************
2678 This logic works if the NNI Port Id falls within the same valid
2679 range of PON Port Ids. If this doesn't work for some OLT Vendor
2680 we need to have a re-look at this.
2681 *********************************************
2682 */
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07002683 onuID := -1
2684 uniID := -1
2685 gemPortID := -1
2686 allocID := -1
Neha Sharma96b7bf22020-06-15 10:37:32 +00002687 networkInterfaceID, err := getNniIntfID(ctx, classifier, action)
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302688 if err != nil {
Shrey Baid26912972020-04-16 21:02:31 +05302689 return olterrors.NewErrNotFound("nni-intreface-id",
2690 log.Fields{
2691 "classifier": classifier,
2692 "action": action},
Girish Kumarf26e4882020-03-05 06:49:10 +00002693 err)
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05302694 }
2695
yasin saplibddc2d72022-02-08 13:10:17 +00002696 present, err := f.resourceMgr.IsFlowOnKvStore(ctx, int32(onuID), logicalFlow.Id)
yasin saplid0566272021-12-21 09:10:30 +00002697 if present {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002698 logger.Info(ctx, "flow-exists-not-re-adding")
David K. Bainbridge794735f2020-02-11 21:01:37 -08002699 return nil
yasin saplid0566272021-12-21 09:10:30 +00002700 } else if err != nil {
2701 logger.Errorw(ctx, "aborting-addTrapFlowOnNNI--flow-may-already-exist",
2702 log.Fields{"intf-id": networkInterfaceID, "onu-id": onuID, "flow-id": logicalFlow.Id})
2703 return err
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002704 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002705
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03002706 logger.Debugw(ctx, "creating-trap-of-nni-flow",
2707 log.Fields{
2708 "classifier": classifier,
2709 "action": action,
2710 "flowId": logicalFlow.Id,
2711 "intf-id": networkInterfaceID})
2712
David K. Bainbridge794735f2020-02-11 21:01:37 -08002713 classifierProto, err := makeOpenOltClassifierField(classifier)
2714 if err != nil {
Girish Kumarf26e4882020-03-05 06:49:10 +00002715 return olterrors.NewErrInvalidValue(log.Fields{"classifier": classifier}, err)
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002716 }
bseeniva0b9cbcb2026-02-12 19:11:11 +05302717 logger.Debugw(ctx, "created-classifier-proto", log.Fields{"classifier": classifierProto.String()})
Gamze Abaka724d0852020-03-18 12:10:24 +00002718 actionProto, err := makeOpenOltActionField(action, classifier)
David K. Bainbridge794735f2020-02-11 21:01:37 -08002719 if err != nil {
Girish Kumarf26e4882020-03-05 06:49:10 +00002720 return olterrors.NewErrInvalidValue(log.Fields{"action": action}, err)
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002721 }
bseeniva0b9cbcb2026-02-12 19:11:11 +05302722 logger.Debugw(ctx, "created-action-proto", log.Fields{"action": actionProto.String()})
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07002723 downstreamflow := openoltpb2.Flow{AccessIntfId: int32(-1), // AccessIntfId not required
2724 OnuId: int32(onuID), // OnuId not required
2725 UniId: int32(uniID), // UniId not used
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002726 FlowId: logicalFlow.Id,
David K. Bainbridge82efc492019-09-04 09:57:11 -07002727 FlowType: Downstream,
Girish Gowdru6a80bbd2019-07-02 07:36:09 -07002728 AllocId: int32(allocID), // AllocId not used
2729 NetworkIntfId: int32(networkInterfaceID),
2730 GemportId: int32(gemPortID), // GemportId not used
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002731 Classifier: classifierProto,
2732 Action: actionProto,
2733 Priority: int32(logicalFlow.Priority),
2734 Cookie: logicalFlow.Cookie,
2735 PortNo: portNo}
David K. Bainbridge794735f2020-02-11 21:01:37 -08002736 if err := f.addFlowToDevice(ctx, logicalFlow, &downstreamflow); err != nil {
bseeniva0b9cbcb2026-02-12 19:11:11 +05302737 return olterrors.NewErrFlowOp("add", logicalFlow.Id, log.Fields{"flow-id": downstreamflow.FlowId}, err)
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002738 }
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03002739 logger.Info(ctx, "trap-on-nni-flow-added–to-device-successfully")
David K. Bainbridge794735f2020-02-11 21:01:37 -08002740 return nil
manikkaraj k9eb6cac2019-05-09 12:32:03 -04002741}
salmansiddiqui7ac62132019-08-22 03:58:50 +00002742
Joey Armstrong3f0e2422023-07-05 18:25:41 -04002743// getPacketTypeFromClassifiers finds and returns packet type of a flow by checking flow classifiers
Esin Karamanae41e2b2019-12-17 18:13:13 +00002744func getPacketTypeFromClassifiers(classifierInfo map[string]interface{}) string {
2745 var packetType string
2746 ovid, ivid := false, false
2747 if vlanID, ok := classifierInfo[VlanVid].(uint32); ok {
2748 vid := vlanID & VlanvIDMask
2749 if vid != ReservedVlan {
2750 ovid = true
2751 }
2752 }
Girish Gowdrab23f1de2022-03-24 12:01:17 -07002753
2754 // The classifierInfo[Metadata] is set for the following flows
2755 // - In the Downstream datapath flow table0 and table1. From the OLT perspective, only table0 downstream flow is relevant.
2756 // - Mcast flow that points to a group in the action/treatment
2757 // This value, when present and valid (not 0 and not 4096), is interpreted as below
2758 // - inner vid for a double tagged packet in the datapath flow
2759 // - outer vid for a single tagged packet in the datapath flow
2760 // - inner vid in the mcast flow that points to a group
2761
2762 // It is to be noted though that for DT FTTH downstream table0 flow, the classifierInfo[Metadata] is absent.
2763 // And consequently the ivid is not set to true. This will result in packetType being set to singleTag which is not true
2764 // Interestingly enough, this function `getPacketTypeFromClassifiers` is called only by Mcast flow handlers and
2765 // it is to be used with caution elsewhere as it could result in wrong packetType to be returned to the caller.
Esin Karamanae41e2b2019-12-17 18:13:13 +00002766 if metadata, ok := classifierInfo[Metadata].(uint64); ok {
2767 vid := uint32(metadata)
2768 if vid != ReservedVlan {
2769 ivid = true
2770 }
2771 }
2772 if ovid && ivid {
2773 packetType = DoubleTag
2774 } else if !ovid && !ivid {
2775 packetType = Untagged
2776 } else {
2777 packetType = SingleTag
2778 }
2779 return packetType
2780}
2781
Joey Armstrong3f0e2422023-07-05 18:25:41 -04002782// addIgmpTrapFlowOnNNI adds a trap-to-host flow on NNI
David K. Bainbridge794735f2020-02-11 21:01:37 -08002783func (f *OpenOltFlowMgr) addIgmpTrapFlowOnNNI(ctx context.Context, logicalFlow *ofp.OfpFlowStats, classifier map[string]interface{}, portNo uint32) error {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002784 logger.Infow(ctx, "adding-igmp-trap-of-nni-flow", log.Fields{"classifier-info": classifier})
Esin Karamanae41e2b2019-12-17 18:13:13 +00002785 action := make(map[string]interface{})
2786 classifier[PacketTagType] = getPacketTypeFromClassifiers(classifier)
2787 action[TrapToHost] = true
2788 /* We manage flowId resource pool on per PON port basis.
2789 Since this situation is tricky, as a hack, we pass the NNI port
2790 index (network_intf_id) as PON port Index for the flowId resource
2791 pool. Also, there is no ONU Id available for trapping packets
2792 on NNI port, use onu_id as -1 (invalid)
2793 ****************** CAVEAT *******************
2794 This logic works if the NNI Port Id falls within the same valid
2795 range of PON Port Ids. If this doesn't work for some OLT Vendor
2796 we need to have a re-look at this.
2797 *********************************************
2798 */
2799 onuID := -1
2800 uniID := -1
2801 gemPortID := -1
2802 allocID := -1
Neha Sharma96b7bf22020-06-15 10:37:32 +00002803 networkInterfaceID, err := getNniIntfID(ctx, classifier, action)
Esin Karamanae41e2b2019-12-17 18:13:13 +00002804 if err != nil {
Thomas Lee S94109f12020-03-03 16:39:29 +05302805 return olterrors.NewErrNotFound("nni-interface-id", log.Fields{
David K. Bainbridge794735f2020-02-11 21:01:37 -08002806 "classifier": classifier,
2807 "action": action},
Girish Kumarf26e4882020-03-05 06:49:10 +00002808 err)
Esin Karamanae41e2b2019-12-17 18:13:13 +00002809 }
yasin saplibddc2d72022-02-08 13:10:17 +00002810 present, err := f.resourceMgr.IsFlowOnKvStore(ctx, int32(onuID), logicalFlow.Id)
yasin saplid0566272021-12-21 09:10:30 +00002811 if present {
Neha Sharma96b7bf22020-06-15 10:37:32 +00002812 logger.Info(ctx, "igmp-flow-exists-not-re-adding")
David K. Bainbridge794735f2020-02-11 21:01:37 -08002813 return nil
yasin saplid0566272021-12-21 09:10:30 +00002814 } else if err != nil {
2815 logger.Errorw(ctx, "aborting-addIgmpTrapFlowOnNNI--flow-may-already-exist",
2816 log.Fields{"intf-id": networkInterfaceID, "onu-id": onuID, "flow-id": logicalFlow.Id})
2817 return err
Esin Karamanae41e2b2019-12-17 18:13:13 +00002818 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002819
David K. Bainbridge794735f2020-02-11 21:01:37 -08002820 classifierProto, err := makeOpenOltClassifierField(classifier)
2821 if err != nil {
Girish Kumarf26e4882020-03-05 06:49:10 +00002822 return olterrors.NewErrInvalidValue(log.Fields{"classifier": classifier}, err)
Esin Karamanae41e2b2019-12-17 18:13:13 +00002823 }
bseeniva0b9cbcb2026-02-12 19:11:11 +05302824 logger.Debugw(ctx, "created-classifier-proto-for-the-igmp-flow", log.Fields{"classifier": classifierProto.String()})
Gamze Abaka724d0852020-03-18 12:10:24 +00002825 actionProto, err := makeOpenOltActionField(action, classifier)
David K. Bainbridge794735f2020-02-11 21:01:37 -08002826 if err != nil {
Girish Kumarf26e4882020-03-05 06:49:10 +00002827 return olterrors.NewErrInvalidValue(log.Fields{"action": action}, err)
Esin Karamanae41e2b2019-12-17 18:13:13 +00002828 }
bseeniva0b9cbcb2026-02-12 19:11:11 +05302829 logger.Debugw(ctx, "created-action-proto-for-the-igmp-flow", log.Fields{"action": actionProto.String()})
Esin Karamanae41e2b2019-12-17 18:13:13 +00002830 downstreamflow := openoltpb2.Flow{AccessIntfId: int32(-1), // AccessIntfId not required
2831 OnuId: int32(onuID), // OnuId not required
2832 UniId: int32(uniID), // UniId not used
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002833 FlowId: logicalFlow.Id,
Esin Karamanae41e2b2019-12-17 18:13:13 +00002834 FlowType: Downstream,
2835 AllocId: int32(allocID), // AllocId not used
2836 NetworkIntfId: int32(networkInterfaceID),
2837 GemportId: int32(gemPortID), // GemportId not used
2838 Classifier: classifierProto,
2839 Action: actionProto,
2840 Priority: int32(logicalFlow.Priority),
2841 Cookie: logicalFlow.Cookie,
2842 PortNo: portNo}
David K. Bainbridge794735f2020-02-11 21:01:37 -08002843 if err := f.addFlowToDevice(ctx, logicalFlow, &downstreamflow); err != nil {
bseeniva0b9cbcb2026-02-12 19:11:11 +05302844 return olterrors.NewErrFlowOp("add", logicalFlow.Id, log.Fields{"flow-id": downstreamflow.FlowId}, err)
Esin Karamanae41e2b2019-12-17 18:13:13 +00002845 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00002846 logger.Info(ctx, "igmp-trap-on-nni-flow-added-to-device-successfully")
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002847
David K. Bainbridge794735f2020-02-11 21:01:37 -08002848 return nil
Esin Karamanae41e2b2019-12-17 18:13:13 +00002849}
2850
salmansiddiqui7ac62132019-08-22 03:58:50 +00002851func verifyMeterIDAndGetDirection(MeterID uint32, Dir tp_pb.Direction) (string, error) {
2852 if MeterID == 0 { // This should never happen
Thomas Lee S94109f12020-03-03 16:39:29 +05302853 return "", olterrors.NewErrInvalidValue(log.Fields{"meter-id": MeterID}, nil).Log()
salmansiddiqui7ac62132019-08-22 03:58:50 +00002854 }
mgouda86543582025-10-29 20:58:16 +05302855 switch Dir {
2856 case tp_pb.Direction_UPSTREAM:
salmansiddiqui7ac62132019-08-22 03:58:50 +00002857 return "upstream", nil
mgouda86543582025-10-29 20:58:16 +05302858 case tp_pb.Direction_DOWNSTREAM:
salmansiddiqui7ac62132019-08-22 03:58:50 +00002859 return "downstream", nil
2860 }
2861 return "", nil
2862}
2863
Kent Hagermane6ff1012020-07-14 15:07:53 -04002864// nolint: gocyclo
npujarec5762e2020-01-01 14:08:48 +05302865func (f *OpenOltFlowMgr) checkAndAddFlow(ctx context.Context, args map[string]uint32, classifierInfo map[string]interface{},
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -07002866 actionInfo map[string]interface{}, flow *ofp.OfpFlowStats, TpInst interface{}, gemPorts []uint32,
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00002867 tpID uint32, uni string) error {
Gamze Abaka7650be62021-02-26 10:50:36 +00002868 var gemPortID uint32
Gamze Abakafee36392019-10-03 11:17:24 +00002869 intfID := args[IntfID]
2870 onuID := args[OnuID]
2871 uniID := args[UniID]
2872 portNo := args[PortNo]
Takahiro Suzuki2ba0e0b2020-06-05 14:23:03 -07002873 allocID := args[AllocID]
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002874 pbitToGem := make(map[uint32]uint32)
Gamze Abaka7650be62021-02-26 10:50:36 +00002875 gemToAes := make(map[uint32]bool)
2876
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002877 var attributes []*tp_pb.GemPortAttributes
Gamze Abaka7650be62021-02-26 10:50:36 +00002878 var direction = tp_pb.Direction_UPSTREAM
2879 switch TpInst := TpInst.(type) {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002880 case *tp_pb.TechProfileInstance:
Mahir Gunyel85f61c12021-10-06 11:53:45 -07002881 if plt.IsUpstream(actionInfo[Output].(uint32)) {
Gamze Abaka7650be62021-02-26 10:50:36 +00002882 attributes = TpInst.UpstreamGemPortAttributeList
2883 } else {
2884 attributes = TpInst.DownstreamGemPortAttributeList
2885 direction = tp_pb.Direction_DOWNSTREAM
2886 }
2887 default:
2888 logger.Errorw(ctx, "unsupported-tech", log.Fields{"tpInst": TpInst})
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00002889 return olterrors.NewErrInvalidValue(log.Fields{"tpInst": TpInst}, nil)
Gamze Abaka7650be62021-02-26 10:50:36 +00002890 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002891
2892 if len(gemPorts) == 1 {
2893 // If there is only single gemport use that and do not populate pbitToGem map
Gamze Abaka7650be62021-02-26 10:50:36 +00002894 gemPortID = gemPorts[0]
2895 gemToAes[gemPortID], _ = strconv.ParseBool(attributes[0].AesEncryption)
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002896 } else if pcp, ok := classifierInfo[VlanPcp]; !ok {
2897 for idx, gemID := range gemPorts {
Gamze Abaka7650be62021-02-26 10:50:36 +00002898 pBitMap := attributes[idx].PbitMap
2899 // Trim the bitMapPrefix form the binary string and then iterate each character in the binary string.
2900 // If the character is set to pbit1, extract the pcp value from the position of this character in the string.
2901 // Update the pbitToGem map with key being the pcp bit and the value being the gemPortID that consumes
khenaidoodc2116e2021-10-19 17:33:19 -04002902 // this pcp bit traffca.
Gamze Abaka7650be62021-02-26 10:50:36 +00002903 for pos, pbitSet := range strings.TrimPrefix(pBitMap, bitMapPrefix) {
2904 if pbitSet == pbit1 {
2905 pcp := uint32(len(strings.TrimPrefix(pBitMap, bitMapPrefix))) - 1 - uint32(pos)
2906 pbitToGem[pcp] = gemID
2907 gemToAes[gemID], _ = strconv.ParseBool(attributes[idx].AesEncryption)
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002908 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002909 }
2910 }
2911 } else { // Extract the exact gemport which maps to the PCP classifier in the flow
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07002912 if gem := f.techprofile.GetGemportForPbit(ctx, TpInst, direction, pcp.(uint32)); gem != nil {
2913 gemPortID = gem.(*tp_pb.GemPortAttributes).GemportId
2914 gemToAes[gemPortID], _ = strconv.ParseBool(gem.(*tp_pb.GemPortAttributes).AesEncryption)
Gamze Abaka7650be62021-02-26 10:50:36 +00002915 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002916 }
2917
Akash Kankanala041a2122024-10-16 15:49:22 +05302918 flowContext := &flowContext{classifierInfo, actionInfo, flow, pbitToGem, gemToAes, intfID, onuID, uniID, portNo, allocID, gemPortID, tpID}
Gamze Abaka7650be62021-02-26 10:50:36 +00002919
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302920 var nni_port uint32
2921 if plt.IsUpstream(actionInfo[Output].(uint32)) {
2922 if !plt.IsControllerBoundFlow(actionInfo[Output].(uint32)) {
2923 nni_port = actionInfo[Output].(uint32) & 0x1f // convert e.g. 16777220 to port 4 (starting with 0)
2924 }
2925 } else {
2926 nni_port = classifierInfo[InPort].(uint32) & 0x1f // convert e.g. 16777220 to port 4 (starting with 0)
2927 }
2928
salmansiddiqui7ac62132019-08-22 03:58:50 +00002929 if ipProto, ok := classifierInfo[IPProto]; ok {
mgouda86543582025-10-29 20:58:16 +05302930 switch ipProto.(uint32) {
2931 case IPProtoDhcp:
Neha Sharma96b7bf22020-06-15 10:37:32 +00002932 logger.Infow(ctx, "adding-dhcp-flow", log.Fields{
Matteo Scandolo92186242020-06-12 10:54:18 -07002933 "tp-id": tpID,
2934 "alloc-id": allocID,
2935 "intf-id": intfID,
2936 "onu-id": onuID,
2937 "uni-id": uniID,
2938 })
Akash Kankanala041a2122024-10-16 15:49:22 +05302939 // Adding DHCP upstream flow
Gamze Abaka7650be62021-02-26 10:50:36 +00002940 if err := f.addDHCPTrapFlow(ctx, flowContext); err != nil {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002941 logger.Warn(ctx, err)
Gamze Abaka411ef2f2021-11-22 08:38:08 +00002942 logger.Errorw(ctx, "reverting-scheduler-and-queue-for-onu", log.Fields{"intf-id": intfID, "onu-id": onuID, "uni-id": uniID, "flow-id": flow.Id, "tp-id": tpID})
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302943 _ = f.clearResources(ctx, intfID, int32(onuID), int32(uniID), flow.Id, portNo, tpID, false, nni_port)
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00002944 return err
salmansiddiqui7ac62132019-08-22 03:58:50 +00002945 }
mgouda86543582025-10-29 20:58:16 +05302946 case IgmpProto:
Neha Sharma96b7bf22020-06-15 10:37:32 +00002947 logger.Infow(ctx, "adding-us-igmp-flow",
Shrey Baid26912972020-04-16 21:02:31 +05302948 log.Fields{
2949 "intf-id": intfID,
2950 "onu-id": onuID,
2951 "uni-id": uniID,
2952 "classifier-info:": classifierInfo})
Gamze Abaka7650be62021-02-26 10:50:36 +00002953 if err := f.addIGMPTrapFlow(ctx, flowContext); err != nil {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002954 logger.Warn(ctx, err)
Gamze Abaka411ef2f2021-11-22 08:38:08 +00002955 logger.Errorw(ctx, "reverting-scheduler-and-queue-for-onu", log.Fields{"intf-id": intfID, "onu-id": onuID, "uni-id": uniID, "flow-id": flow.Id, "tp-id": tpID})
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302956 _ = f.clearResources(ctx, intfID, int32(onuID), int32(uniID), flow.Id, portNo, tpID, false, nni_port)
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00002957 return err
Esin Karamanae41e2b2019-12-17 18:13:13 +00002958 }
mgouda86543582025-10-29 20:58:16 +05302959 default:
Neha Sharma96b7bf22020-06-15 10:37:32 +00002960 logger.Errorw(ctx, "invalid-classifier-to-handle", log.Fields{"classifier": classifierInfo, "action": actionInfo})
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00002961 return olterrors.NewErrInvalidValue(log.Fields{"classifier": classifierInfo, "action": actionInfo}, nil)
salmansiddiqui7ac62132019-08-22 03:58:50 +00002962 }
2963 } else if ethType, ok := classifierInfo[EthType]; ok {
mgouda86543582025-10-29 20:58:16 +05302964 switch ethType.(uint32) {
2965 case EapEthType:
Marcos Aurelio Carrero (Furukawa)1dc2bfb2021-02-17 15:10:12 -03002966 logger.Infow(ctx, "adding-eapol-flow", log.Fields{
Matteo Scandolo92186242020-06-12 10:54:18 -07002967 "intf-id": intfID,
2968 "onu-id": onuID,
2969 "uni-id": uniID,
Marcos Aurelio Carrero (Furukawa)388fb0a2021-02-04 18:05:11 -03002970 "ethType": ethType,
Matteo Scandolo92186242020-06-12 10:54:18 -07002971 })
salmansiddiqui7ac62132019-08-22 03:58:50 +00002972 var vlanID uint32
2973 if val, ok := classifierInfo[VlanVid]; ok {
2974 vlanID = (val.(uint32)) & VlanvIDMask
2975 } else {
2976 vlanID = DefaultMgmtVlan
2977 }
Gamze Abaka7650be62021-02-26 10:50:36 +00002978 if err := f.addEthTypeBasedFlow(ctx, flowContext, vlanID, ethType.(uint32)); err != nil {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07002979 logger.Warn(ctx, err)
Gamze Abaka411ef2f2021-11-22 08:38:08 +00002980 logger.Errorw(ctx, "reverting-scheduler-and-queue-for-onu", log.Fields{"intf-id": intfID, "onu-id": onuID, "uni-id": uniID, "flow-id": flow.Id, "tp-id": tpID})
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302981 _ = f.clearResources(ctx, intfID, int32(onuID), int32(uniID), flow.Id, portNo, tpID, false, nni_port)
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00002982 return err
salmansiddiqui7ac62132019-08-22 03:58:50 +00002983 }
mgouda86543582025-10-29 20:58:16 +05302984 case PPPoEDEthType:
Marcos Aurelio Carrero (Furukawa)1dc2bfb2021-02-17 15:10:12 -03002985 logger.Infow(ctx, "adding-pppoed-flow", log.Fields{
2986 "tp-id": tpID,
2987 "alloc-id": allocID,
2988 "intf-id": intfID,
2989 "onu-id": onuID,
2990 "uni-id": uniID,
2991 })
Akash Kankanala041a2122024-10-16 15:49:22 +05302992 // Adding PPPOED upstream flow
Gamze Abaka7650be62021-02-26 10:50:36 +00002993 if err := f.addUpstreamTrapFlow(ctx, flowContext); err != nil {
Marcos Aurelio Carrero (Furukawa)1dc2bfb2021-02-17 15:10:12 -03002994 logger.Warn(ctx, err)
Gamze Abaka411ef2f2021-11-22 08:38:08 +00002995 logger.Errorw(ctx, "reverting-scheduler-and-queue-for-onu", log.Fields{"intf-id": intfID, "onu-id": onuID, "uni-id": uniID, "flow-id": flow.Id, "tp-id": tpID})
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05302996 _ = f.clearResources(ctx, intfID, int32(onuID), int32(uniID), flow.Id, portNo, tpID, false, nni_port)
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00002997 return err
Marcos Aurelio Carrero (Furukawa)1dc2bfb2021-02-17 15:10:12 -03002998 }
salmansiddiqui7ac62132019-08-22 03:58:50 +00002999 }
Gamze Abaka7650be62021-02-26 10:50:36 +00003000 } else if direction == tp_pb.Direction_UPSTREAM {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003001 logger.Infow(ctx, "adding-upstream-data-rule", log.Fields{
Matteo Scandolo92186242020-06-12 10:54:18 -07003002 "intf-id": intfID,
3003 "onu-id": onuID,
3004 "uni-id": uniID,
3005 })
Akash Kankanala041a2122024-10-16 15:49:22 +05303006 // Adding HSIA upstream flow
Gamze Abaka7650be62021-02-26 10:50:36 +00003007 if err := f.addUpstreamDataPathFlow(ctx, flowContext); err != nil {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003008 logger.Warn(ctx, err)
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003009 logger.Errorw(ctx, "reverting-scheduler-and-queue-for-onu", log.Fields{"intf-id": intfID, "onu-id": onuID, "uni-id": uniID, "flow-id": flow.Id, "tp-id": tpID})
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05303010 _ = f.clearResources(ctx, intfID, int32(onuID), int32(uniID), flow.Id, portNo, tpID, false, nni_port)
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00003011 return err
salmansiddiqui7ac62132019-08-22 03:58:50 +00003012 }
Gamze Abaka7650be62021-02-26 10:50:36 +00003013 } else if direction == tp_pb.Direction_DOWNSTREAM {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003014 logger.Infow(ctx, "adding-downstream-data-rule", log.Fields{
Matteo Scandolo92186242020-06-12 10:54:18 -07003015 "intf-id": intfID,
3016 "onu-id": onuID,
3017 "uni-id": uniID,
3018 })
Akash Kankanala041a2122024-10-16 15:49:22 +05303019 // Adding HSIA downstream flow
Gamze Abaka7650be62021-02-26 10:50:36 +00003020 if err := f.addDownstreamDataPathFlow(ctx, flowContext); err != nil {
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003021 logger.Warn(ctx, err)
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003022 logger.Errorw(ctx, "reverting-scheduler-and-queue-for-onu", log.Fields{"intf-id": intfID, "onu-id": onuID, "uni-id": uniID, "flow-id": flow.Id, "tp-id": tpID})
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05303023 _ = f.clearResources(ctx, intfID, int32(onuID), int32(uniID), flow.Id, portNo, tpID, false, nni_port)
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00003024 return err
salmansiddiqui7ac62132019-08-22 03:58:50 +00003025 }
3026 } else {
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00003027 return olterrors.NewErrInvalidValue(log.Fields{
3028 "intf-id": intfID,
3029 "onu-id": onuID,
3030 "uni-id": uniID,
3031 "classifier": classifierInfo,
3032 "action": actionInfo,
3033 "flow": flow},
3034 nil).Log()
salmansiddiqui7ac62132019-08-22 03:58:50 +00003035 }
3036 // Send Techprofile download event to child device in go routine as it takes time
Kent Hagermane6ff1012020-07-14 15:07:53 -04003037 go func() {
bseeniva0b9cbcb2026-02-12 19:11:11 +05303038 if err := f.sendTPDownloadMsgToChild(ctx, intfID, onuID, uniID, uni, tpID, TpInst.(*tp_pb.TechProfileInstance)); err != nil {
Kent Hagermane6ff1012020-07-14 15:07:53 -04003039 logger.Warn(ctx, err)
3040 }
3041 }()
Gamze Abaka6d0a64f2021-11-18 08:08:33 +00003042 return nil
salmansiddiqui7ac62132019-08-22 03:58:50 +00003043}
3044
Gamze Abakacb0e6772021-06-10 08:32:12 +00003045func (f *OpenOltFlowMgr) isAllocUsedByAnotherUNI(ctx context.Context, sq schedQueue) bool {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003046 tpInst := sq.tpInst.(*tp_pb.TechProfileInstance)
3047 if tpInst.InstanceControl.Onu == "single-instance" && sq.direction == tp_pb.Direction_UPSTREAM {
3048 tpInstances := f.techprofile.FindAllTpInstances(ctx, f.deviceHandler.device.Id, sq.tpID, sq.intfID, sq.onuID).([]tp_pb.TechProfileInstance)
Girish Gowdra54934262019-11-13 14:19:55 +05303049 for i := 0; i < len(tpInstances); i++ {
bseeniva0b9cbcb2026-02-12 19:11:11 +05303050 tpI := &tpInstances[i]
Gamze Abakacb0e6772021-06-10 08:32:12 +00003051 if tpI.SubscriberIdentifier != tpInst.SubscriberIdentifier &&
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003052 tpI.UsScheduler.AllocId == tpInst.UsScheduler.AllocId {
Girish Gowdraf3728b12022-02-02 21:46:51 -08003053 logger.Debugw(ctx, "alloc-is-in-use-on-another-uni",
Gamze Abakacb0e6772021-06-10 08:32:12 +00003054 log.Fields{
3055 "device-id": f.deviceHandler.device.Id,
3056 "intfID": sq.intfID,
3057 "onuID": sq.onuID,
3058 "uniID": sq.uniID,
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003059 "allocID": tpI.UsScheduler.AllocId,
Gamze Abakacb0e6772021-06-10 08:32:12 +00003060 })
3061 return true
Girish Gowdra54934262019-11-13 14:19:55 +05303062 }
3063 }
3064 }
Gamze Abakacb0e6772021-06-10 08:32:12 +00003065 return false
Gamze Abakafee36392019-10-03 11:17:24 +00003066}
3067
Neha Sharma96b7bf22020-06-15 10:37:32 +00003068func formulateClassifierInfoFromFlow(ctx context.Context, classifierInfo map[string]interface{}, flow *ofp.OfpFlowStats) {
Scott Baker355d1742019-10-24 10:57:52 -07003069 for _, field := range flows.GetOfbFields(flow) {
mgouda86543582025-10-29 20:58:16 +05303070 switch field.Type {
3071 case flows.ETH_TYPE:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003072 classifierInfo[EthType] = field.GetEthType()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003073 logger.Debug(ctx, "field-type-eth-type", log.Fields{"classifierInfo[ETH_TYPE]": classifierInfo[EthType].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303074 case flows.ETH_DST:
Esin Karamanccb714b2019-11-29 15:02:06 +00003075 classifierInfo[EthDst] = field.GetEthDst()
Girish Gowdraffa52e52022-02-16 15:48:10 -08003076 logger.Debug(ctx, "field-type-eth-dst", log.Fields{"classifierInfo[ETH_DST]": classifierInfo[EthDst].([]uint8)})
mgouda86543582025-10-29 20:58:16 +05303077 case flows.ETH_SRC:
Girish Gowdraffa52e52022-02-16 15:48:10 -08003078 classifierInfo[EthSrc] = field.GetEthSrc()
3079 logger.Debug(ctx, "field-type-eth-src", log.Fields{"classifierInfo[ETH_SRC]": classifierInfo[EthSrc].([]uint8)})
mgouda86543582025-10-29 20:58:16 +05303080 case flows.IP_PROTO:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003081 classifierInfo[IPProto] = field.GetIpProto()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003082 logger.Debug(ctx, "field-type-ip-proto", log.Fields{"classifierInfo[IP_PROTO]": classifierInfo[IPProto].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303083 case flows.IN_PORT:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003084 classifierInfo[InPort] = field.GetPort()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003085 logger.Debug(ctx, "field-type-in-port", log.Fields{"classifierInfo[IN_PORT]": classifierInfo[InPort].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303086 case flows.VLAN_VID:
Andrea Campanellafaa42152021-10-28 11:50:56 +05303087 // The ReservedVlan is used to signify transparent vlan. Do not do any classification when we see ReservedVlan
3088 if field.GetVlanVid() != ReservedVlan {
3089 classifierInfo[VlanVid] = field.GetVlanVid() & 0xfff
3090 logger.Debug(ctx, "field-type-vlan-vid", log.Fields{"classifierInfo[VLAN_VID]": classifierInfo[VlanVid].(uint32)})
3091 }
mgouda86543582025-10-29 20:58:16 +05303092 case flows.VLAN_PCP:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003093 classifierInfo[VlanPcp] = field.GetVlanPcp()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003094 logger.Debug(ctx, "field-type-vlan-pcp", log.Fields{"classifierInfo[VLAN_PCP]": classifierInfo[VlanPcp].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303095 case flows.UDP_DST:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003096 classifierInfo[UDPDst] = field.GetUdpDst()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003097 logger.Debug(ctx, "field-type-udp-dst", log.Fields{"classifierInfo[UDP_DST]": classifierInfo[UDPDst].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303098 case flows.UDP_SRC:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003099 classifierInfo[UDPSrc] = field.GetUdpSrc()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003100 logger.Debug(ctx, "field-type-udp-src", log.Fields{"classifierInfo[UDP_SRC]": classifierInfo[UDPSrc].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303101 case flows.IPV4_DST:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003102 classifierInfo[Ipv4Dst] = field.GetIpv4Dst()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003103 logger.Debug(ctx, "field-type-ipv4-dst", log.Fields{"classifierInfo[IPV4_DST]": classifierInfo[Ipv4Dst].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303104 case flows.IPV4_SRC:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003105 classifierInfo[Ipv4Src] = field.GetIpv4Src()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003106 logger.Debug(ctx, "field-type-ipv4-src", log.Fields{"classifierInfo[IPV4_SRC]": classifierInfo[Ipv4Src].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303107 case flows.METADATA:
David K. Bainbridge82efc492019-09-04 09:57:11 -07003108 classifierInfo[Metadata] = field.GetTableMetadata()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003109 logger.Debug(ctx, "field-type-metadata", log.Fields{"classifierInfo[Metadata]": classifierInfo[Metadata].(uint64)})
mgouda86543582025-10-29 20:58:16 +05303110 case flows.TUNNEL_ID:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003111 classifierInfo[TunnelID] = field.GetTunnelId()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003112 logger.Debug(ctx, "field-type-tunnelId", log.Fields{"classifierInfo[TUNNEL_ID]": classifierInfo[TunnelID].(uint64)})
mgouda86543582025-10-29 20:58:16 +05303113 default:
Neha Sharma96b7bf22020-06-15 10:37:32 +00003114 logger.Errorw(ctx, "un-supported-field-type", log.Fields{"type": field.Type})
salmansiddiqui7ac62132019-08-22 03:58:50 +00003115 return
3116 }
3117 }
3118}
3119
Neha Sharma96b7bf22020-06-15 10:37:32 +00003120func formulateActionInfoFromFlow(ctx context.Context, actionInfo, classifierInfo map[string]interface{}, flow *ofp.OfpFlowStats) error {
Scott Baker355d1742019-10-24 10:57:52 -07003121 for _, action := range flows.GetActions(flow) {
mgouda86543582025-10-29 20:58:16 +05303122 switch action.Type {
3123 case flows.OUTPUT:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003124 if out := action.GetOutput(); out != nil {
David K. Bainbridge82efc492019-09-04 09:57:11 -07003125 actionInfo[Output] = out.GetPort()
Neha Sharma96b7bf22020-06-15 10:37:32 +00003126 logger.Debugw(ctx, "action-type-output", log.Fields{"out-port": actionInfo[Output].(uint32)})
salmansiddiqui7ac62132019-08-22 03:58:50 +00003127 } else {
Girish Kumarf26e4882020-03-05 06:49:10 +00003128 return olterrors.NewErrInvalidValue(log.Fields{"output-port": nil}, nil)
salmansiddiqui7ac62132019-08-22 03:58:50 +00003129 }
mgouda86543582025-10-29 20:58:16 +05303130 case flows.POP_VLAN:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003131 actionInfo[PopVlan] = true
Neha Sharma96b7bf22020-06-15 10:37:32 +00003132 logger.Debugw(ctx, "action-type-pop-vlan", log.Fields{"in_port": classifierInfo[InPort].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303133 case flows.PUSH_VLAN:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003134 if out := action.GetPush(); out != nil {
3135 if tpid := out.GetEthertype(); tpid != 0x8100 {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003136 logger.Errorw(ctx, "invalid ethertype in push action", log.Fields{"ethertype": actionInfo[PushVlan].(int32)})
salmansiddiqui7ac62132019-08-22 03:58:50 +00003137 } else {
3138 actionInfo[PushVlan] = true
3139 actionInfo[TPID] = tpid
Neha Sharma96b7bf22020-06-15 10:37:32 +00003140 logger.Debugw(ctx, "action-type-push-vlan",
Shrey Baid26912972020-04-16 21:02:31 +05303141 log.Fields{
3142 "push-tpid": actionInfo[TPID].(uint32),
3143 "in-port": classifierInfo[InPort].(uint32)})
salmansiddiqui7ac62132019-08-22 03:58:50 +00003144 }
3145 }
mgouda86543582025-10-29 20:58:16 +05303146 case flows.SET_FIELD:
salmansiddiqui7ac62132019-08-22 03:58:50 +00003147 if out := action.GetSetField(); out != nil {
3148 if field := out.GetField(); field != nil {
3149 if ofClass := field.GetOxmClass(); ofClass != ofp.OfpOxmClass_OFPXMC_OPENFLOW_BASIC {
Girish Kumarf26e4882020-03-05 06:49:10 +00003150 return olterrors.NewErrInvalidValue(log.Fields{"openflow-class": ofClass}, nil)
salmansiddiqui7ac62132019-08-22 03:58:50 +00003151 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00003152 /*logger.Debugw(ctx, "action-type-set-field",log.Fields{"field": field, "in_port": classifierInfo[IN_PORT].(uint32)})*/
3153 formulateSetFieldActionInfoFromFlow(ctx, field, actionInfo)
salmansiddiqui7ac62132019-08-22 03:58:50 +00003154 }
3155 }
mgouda86543582025-10-29 20:58:16 +05303156 case flows.GROUP:
Neha Sharma96b7bf22020-06-15 10:37:32 +00003157 formulateGroupActionInfoFromFlow(ctx, action, actionInfo)
mgouda86543582025-10-29 20:58:16 +05303158 default:
Girish Kumarf26e4882020-03-05 06:49:10 +00003159 return olterrors.NewErrInvalidValue(log.Fields{"action-type": action.Type}, nil)
salmansiddiqui7ac62132019-08-22 03:58:50 +00003160 }
3161 }
3162 return nil
3163}
3164
Neha Sharma96b7bf22020-06-15 10:37:32 +00003165func formulateSetFieldActionInfoFromFlow(ctx context.Context, field *ofp.OfpOxmField, actionInfo map[string]interface{}) {
Esin Karamanccb714b2019-11-29 15:02:06 +00003166 if ofbField := field.GetOfbField(); ofbField != nil {
Gamze Abakac43a66e2020-05-11 11:00:42 +00003167 fieldtype := ofbField.GetType()
mgouda86543582025-10-29 20:58:16 +05303168 switch fieldtype {
3169 case ofp.OxmOfbFieldTypes_OFPXMT_OFB_VLAN_VID:
Esin Karamanccb714b2019-11-29 15:02:06 +00003170 if vlan := ofbField.GetVlanVid(); vlan != 0 {
3171 actionInfo[VlanVid] = vlan & 0xfff
Neha Sharma96b7bf22020-06-15 10:37:32 +00003172 logger.Debugw(ctx, "action-set-vlan-vid", log.Fields{"actionInfo[VLAN_VID]": actionInfo[VlanVid].(uint32)})
Esin Karamanccb714b2019-11-29 15:02:06 +00003173 } else {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003174 logger.Error(ctx, "no-invalid-vlan-id-in-set-vlan-vid-action")
Esin Karamanccb714b2019-11-29 15:02:06 +00003175 }
mgouda86543582025-10-29 20:58:16 +05303176 case ofp.OxmOfbFieldTypes_OFPXMT_OFB_VLAN_PCP:
Gamze Abakac43a66e2020-05-11 11:00:42 +00003177 pcp := ofbField.GetVlanPcp()
3178 actionInfo[VlanPcp] = pcp
Girish Kumara1ea2aa2020-08-19 18:14:22 +00003179 logger.Debugw(ctx, "action-set-vlan-pcp", log.Fields{"actionInfo[VLAN_PCP]": actionInfo[VlanPcp].(uint32)})
mgouda86543582025-10-29 20:58:16 +05303180 default:
Neha Sharma96b7bf22020-06-15 10:37:32 +00003181 logger.Errorw(ctx, "unsupported-action-set-field-type", log.Fields{"type": fieldtype})
Esin Karamanccb714b2019-11-29 15:02:06 +00003182 }
3183 }
3184}
3185
Neha Sharma96b7bf22020-06-15 10:37:32 +00003186func formulateGroupActionInfoFromFlow(ctx context.Context, action *ofp.OfpAction, actionInfo map[string]interface{}) {
Esin Karamanccb714b2019-11-29 15:02:06 +00003187 if action.GetGroup() == nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003188 logger.Warn(ctx, "no-group-entry-found-in-the-group-action")
Esin Karamanccb714b2019-11-29 15:02:06 +00003189 } else {
3190 actionInfo[GroupID] = action.GetGroup().GroupId
Neha Sharma96b7bf22020-06-15 10:37:32 +00003191 logger.Debugw(ctx, "action-group-id", log.Fields{"actionInfo[GroupID]": actionInfo[GroupID].(uint32)})
Esin Karamanccb714b2019-11-29 15:02:06 +00003192 }
3193}
3194
Neha Sharma96b7bf22020-06-15 10:37:32 +00003195func formulateControllerBoundTrapFlowInfo(ctx context.Context, actionInfo, classifierInfo map[string]interface{}, flow *ofp.OfpFlowStats) error {
Mahir Gunyel85f61c12021-10-06 11:53:45 -07003196 if isControllerFlow := plt.IsControllerBoundFlow(actionInfo[Output].(uint32)); isControllerFlow {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003197 logger.Debug(ctx, "controller-bound-trap-flows--getting-inport-from-tunnelid")
salmansiddiqui7ac62132019-08-22 03:58:50 +00003198 /* Get UNI port/ IN Port from tunnel ID field for upstream controller bound flows */
Mahir Gunyel85f61c12021-10-06 11:53:45 -07003199 if portType := plt.IntfIDToPortTypeName(classifierInfo[InPort].(uint32)); portType == voltha.Port_PON_OLT {
Scott Baker355d1742019-10-24 10:57:52 -07003200 if uniPort := flows.GetChildPortFromTunnelId(flow); uniPort != 0 {
salmansiddiqui7ac62132019-08-22 03:58:50 +00003201 classifierInfo[InPort] = uniPort
Neha Sharma96b7bf22020-06-15 10:37:32 +00003202 logger.Debugw(ctx, "upstream-pon-to-controller-flow--inport-in-tunnelid",
Shrey Baid26912972020-04-16 21:02:31 +05303203 log.Fields{
3204 "newinport": classifierInfo[InPort].(uint32),
3205 "outport": actionInfo[Output].(uint32)})
salmansiddiqui7ac62132019-08-22 03:58:50 +00003206 } else {
Shrey Baid26912972020-04-16 21:02:31 +05303207 return olterrors.NewErrNotFound("child-in-port",
3208 log.Fields{
3209 "reason": "upstream-pon-to-controller-flow--no-inport-in-tunnelid",
3210 "flow": flow}, nil)
salmansiddiqui7ac62132019-08-22 03:58:50 +00003211 }
3212 }
3213 } else {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003214 logger.Debug(ctx, "non-controller-flows--getting-uniport-from-tunnelid")
salmansiddiqui7ac62132019-08-22 03:58:50 +00003215 // Downstream flow from NNI to PON port , Use tunnel ID as new OUT port / UNI port
Mahir Gunyel85f61c12021-10-06 11:53:45 -07003216 if portType := plt.IntfIDToPortTypeName(actionInfo[Output].(uint32)); portType == voltha.Port_PON_OLT {
Scott Baker355d1742019-10-24 10:57:52 -07003217 if uniPort := flows.GetChildPortFromTunnelId(flow); uniPort != 0 {
David K. Bainbridge82efc492019-09-04 09:57:11 -07003218 actionInfo[Output] = uniPort
Neha Sharma96b7bf22020-06-15 10:37:32 +00003219 logger.Debugw(ctx, "downstream-nni-to-pon-port-flow, outport-in-tunnelid",
Shrey Baid26912972020-04-16 21:02:31 +05303220 log.Fields{
3221 "newoutport": actionInfo[Output].(uint32),
3222 "outport": actionInfo[Output].(uint32)})
salmansiddiqui7ac62132019-08-22 03:58:50 +00003223 } else {
Shrey Baid26912972020-04-16 21:02:31 +05303224 return olterrors.NewErrNotFound("out-port",
3225 log.Fields{
3226 "reason": "downstream-nni-to-pon-port-flow--no-outport-in-tunnelid",
3227 "flow": flow}, nil)
salmansiddiqui7ac62132019-08-22 03:58:50 +00003228 }
3229 // Upstream flow from PON to NNI port , Use tunnel ID as new IN port / UNI port
Mahir Gunyel85f61c12021-10-06 11:53:45 -07003230 } else if portType := plt.IntfIDToPortTypeName(classifierInfo[InPort].(uint32)); portType == voltha.Port_PON_OLT {
Scott Baker355d1742019-10-24 10:57:52 -07003231 if uniPort := flows.GetChildPortFromTunnelId(flow); uniPort != 0 {
salmansiddiqui7ac62132019-08-22 03:58:50 +00003232 classifierInfo[InPort] = uniPort
Neha Sharma96b7bf22020-06-15 10:37:32 +00003233 logger.Debugw(ctx, "upstream-pon-to-nni-port-flow, inport-in-tunnelid",
Shrey Baid26912972020-04-16 21:02:31 +05303234 log.Fields{
3235 "newinport": actionInfo[Output].(uint32),
3236 "outport": actionInfo[Output].(uint32)})
salmansiddiqui7ac62132019-08-22 03:58:50 +00003237 } else {
Shrey Baid26912972020-04-16 21:02:31 +05303238 return olterrors.NewErrNotFound("nni-port",
3239 log.Fields{
3240 "reason": "upstream-pon-to-nni-port-flow--no-inport-in-tunnelid",
3241 "in-port": classifierInfo[InPort].(uint32),
3242 "out-port": actionInfo[Output].(uint32),
3243 "flow": flow}, nil)
salmansiddiqui7ac62132019-08-22 03:58:50 +00003244 }
3245 }
3246 }
3247 return nil
3248}
Gamze Abakafee36392019-10-03 11:17:24 +00003249
Neha Sharma96b7bf22020-06-15 10:37:32 +00003250func getTpIDFromFlow(ctx context.Context, flow *ofp.OfpFlowStats) (uint32, error) {
Gamze Abakafee36392019-10-03 11:17:24 +00003251 /* Metadata 8 bytes:
3252 Most Significant 2 Bytes = Inner VLAN
3253 Next 2 Bytes = Tech Profile ID(TPID)
3254 Least Significant 4 Bytes = Port ID
3255 Flow Metadata carries Tech-Profile (TP) ID and is mandatory in all
3256 subscriber related flows.
3257 */
Neha Sharma96b7bf22020-06-15 10:37:32 +00003258 metadata := flows.GetMetadataFromWriteMetadataAction(ctx, flow)
Gamze Abakafee36392019-10-03 11:17:24 +00003259 if metadata == 0 {
Girish Kumarf26e4882020-03-05 06:49:10 +00003260 return 0, olterrors.NewErrNotFound("metadata", log.Fields{"flow": flow}, nil)
Gamze Abakafee36392019-10-03 11:17:24 +00003261 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00003262 TpID := flows.GetTechProfileIDFromWriteMetaData(ctx, metadata)
Chaitrashree G S90a17952019-11-14 21:51:21 -05003263 return uint32(TpID), nil
Gamze Abakafee36392019-10-03 11:17:24 +00003264}
3265
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003266func appendUnique32bit(slice []uint32, item uint32) []uint32 {
Gamze Abakafee36392019-10-03 11:17:24 +00003267 for _, sliceElement := range slice {
3268 if sliceElement == item {
3269 return slice
3270 }
3271 }
3272 return append(slice, item)
3273}
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05303274
yasin saplie87d4bd2021-12-06 09:04:03 +00003275func appendUnique64bit(slice []uint64, item uint64) []uint64 {
3276 for _, sliceElement := range slice {
3277 if sliceElement == item {
3278 return slice
3279 }
3280 }
3281 return append(slice, item)
3282}
3283
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05303284// getNniIntfID gets nni intf id from the flow classifier/action
Neha Sharma96b7bf22020-06-15 10:37:32 +00003285func getNniIntfID(ctx context.Context, classifier map[string]interface{}, action map[string]interface{}) (uint32, error) {
Mahir Gunyel85f61c12021-10-06 11:53:45 -07003286 portType := plt.IntfIDToPortTypeName(classifier[InPort].(uint32))
mgouda86543582025-10-29 20:58:16 +05303287 switch portType {
3288 case voltha.Port_PON_OLT:
Mahir Gunyel85f61c12021-10-06 11:53:45 -07003289 intfID, err := plt.IntfIDFromNniPortNum(ctx, action[Output].(uint32))
David K. Bainbridge794735f2020-02-11 21:01:37 -08003290 if err != nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003291 logger.Debugw(ctx, "invalid-action-port-number",
David K. Bainbridge794735f2020-02-11 21:01:37 -08003292 log.Fields{
3293 "port-number": action[Output].(uint32),
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003294 "err": err})
David K. Bainbridge794735f2020-02-11 21:01:37 -08003295 return uint32(0), err
3296 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00003297 logger.Infow(ctx, "output-nni-intfId-is", log.Fields{"intf-id": intfID})
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05303298 return intfID, nil
mgouda86543582025-10-29 20:58:16 +05303299 case voltha.Port_ETHERNET_NNI:
Mahir Gunyel85f61c12021-10-06 11:53:45 -07003300 intfID, err := plt.IntfIDFromNniPortNum(ctx, classifier[InPort].(uint32))
David K. Bainbridge794735f2020-02-11 21:01:37 -08003301 if err != nil {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003302 logger.Debugw(ctx, "invalid-classifier-port-number",
David K. Bainbridge794735f2020-02-11 21:01:37 -08003303 log.Fields{
3304 "port-number": action[Output].(uint32),
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003305 "err": err})
David K. Bainbridge794735f2020-02-11 21:01:37 -08003306 return uint32(0), err
3307 }
Neha Sharma96b7bf22020-06-15 10:37:32 +00003308 logger.Infow(ctx, "input-nni-intfId-is", log.Fields{"intf-id": intfID})
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05303309 return intfID, nil
mgouda86543582025-10-29 20:58:16 +05303310 case voltha.Port_ETHERNET_UNI:
Sridhar Ravindrad104e8d2025-06-02 13:04:49 +05303311 if _, ok := action[Output]; ok {
3312 intfID, err := plt.IntfIDFromNniPortNum(ctx, action[Output].(uint32))
3313 if err != nil {
3314 logger.Debugw(ctx, "invalid-action-port-number",
3315 log.Fields{
3316 "port-number": action[Output].(uint32),
3317 "err": err})
3318 return uint32(0), nil
3319 }
3320 logger.Infow(ctx, "output-nni-intfId-is", log.Fields{"intf-id": intfID})
3321 return intfID, nil
3322 } else {
3323 logger.Debugw(ctx, "action-port-number-empty",
3324 log.Fields{
3325 "action": action})
3326 return uint32(0), nil
3327 }
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05303328 }
3329 return uint32(0), nil
3330}
3331
3332// UpdateGemPortForPktIn updates gemport for packet-in in to the cache and to the kv store as well.
Esin Karaman7fb80c22020-07-16 14:23:33 +00003333func (f *OpenOltFlowMgr) UpdateGemPortForPktIn(ctx context.Context, intfID uint32, onuID uint32, logicalPort uint32, gemPort uint32, pkt []byte) {
3334 cTag, priority, err := getCTagFromPacket(ctx, pkt)
3335 if err != nil {
3336 logger.Errorw(ctx, "unable-to-update-gem-port-for-packet-in",
3337 log.Fields{"intfID": intfID, "onuID": onuID, "logicalPort": logicalPort, "gemPort": gemPort, "err": err})
3338 return
3339 }
3340 pktInkey := rsrcMgr.PacketInInfoKey{IntfID: intfID, OnuID: onuID, LogicalPort: logicalPort, VlanID: cTag, Priority: priority}
Matteo Scandoloabf9c512020-06-23 19:31:14 -07003341
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003342 f.packetInGemPortLock.RLock()
Matt Jeanneret1719a072019-12-20 14:50:14 -05003343 lookupGemPort, ok := f.packetInGemPort[pktInkey]
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003344 f.packetInGemPortLock.RUnlock()
3345
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05303346 if ok {
Matt Jeanneret1719a072019-12-20 14:50:14 -05003347 if lookupGemPort == gemPort {
Neha Sharma96b7bf22020-06-15 10:37:32 +00003348 logger.Infow(ctx, "pktin-key/value-found-in-cache--no-need-to-update-kv--assume-both-in-sync",
Shrey Baid26912972020-04-16 21:02:31 +05303349 log.Fields{
3350 "pktinkey": pktInkey,
3351 "gem": gemPort})
Matt Jeanneret1719a072019-12-20 14:50:14 -05003352 return
3353 }
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05303354 }
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003355 f.packetInGemPortLock.Lock()
Matt Jeanneret1719a072019-12-20 14:50:14 -05003356 f.packetInGemPort[pktInkey] = gemPort
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003357 f.packetInGemPortLock.Unlock()
Matt Jeanneret1719a072019-12-20 14:50:14 -05003358
npujarec5762e2020-01-01 14:08:48 +05303359 f.resourceMgr.UpdateGemPortForPktIn(ctx, pktInkey, gemPort)
Neha Sharma96b7bf22020-06-15 10:37:32 +00003360 logger.Infow(ctx, "pktin-key-not-found-in-local-cache-value-is-different--updating-cache-and-kv-store",
Shrey Baid26912972020-04-16 21:02:31 +05303361 log.Fields{
3362 "pktinkey": pktInkey,
3363 "gem": gemPort})
Abhilash Laxmeshwarab0bd522019-10-21 15:05:15 +05303364}
3365
Joey Armstrong3f0e2422023-07-05 18:25:41 -04003366// getCTagFromPacket retrieves and returns c-tag and priority value from a packet.
Esin Karaman7fb80c22020-07-16 14:23:33 +00003367func getCTagFromPacket(ctx context.Context, packet []byte) (uint16, uint8, error) {
mgouda86543582025-10-29 20:58:16 +05303368 if len(packet) < 18 {
Girish Kumara1ea2aa2020-08-19 18:14:22 +00003369 logger.Error(ctx, "unable-get-c-tag-from-the-packet--invalid-packet-length ")
Esin Karaman7fb80c22020-07-16 14:23:33 +00003370 return 0, 0, errors.New("invalid packet length")
3371 }
3372 outerEthType := (uint16(packet[12]) << 8) | uint16(packet[13])
3373 innerEthType := (uint16(packet[16]) << 8) | uint16(packet[17])
3374
3375 var index int8
3376 if outerEthType == 0x8100 {
3377 if innerEthType == 0x8100 {
3378 // q-in-q 802.1ad or 802.1q double tagged packet.
3379 // get the inner vlanId
3380 index = 18
3381 } else {
3382 index = 14
3383 }
3384 priority := (packet[index] >> 5) & 0x7
Akash Kankanala041a2122024-10-16 15:49:22 +05303385 // 13 bits composes vlanId value
Esin Karaman7fb80c22020-07-16 14:23:33 +00003386 vlan := ((uint16(packet[index]) << 8) & 0x0fff) | uint16(packet[index+1])
3387 return vlan, priority, nil
3388 }
3389 logger.Debugf(ctx, "No vlanId found in the packet. Returning zero as c-tag")
3390 return 0, 0, nil
3391}
3392
Joey Armstrong3f0e2422023-07-05 18:25:41 -04003393// clearMulticastFlowFromResourceManager removes a multicast flow from the KV store and
Girish Gowdra9602eb42020-09-09 15:50:39 -07003394// clears resources reserved for this multicast flow
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003395func (f *OpenOltFlowMgr) clearMulticastFlowFromResourceManager(ctx context.Context, flow *ofp.OfpFlowStats) error {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003396 removeFlowMessage := openoltpb2.Flow{FlowId: flow.Id, FlowType: Multicast}
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003397 logger.Debugw(ctx, "multicast-flow-to-be-deleted",
3398 log.Fields{
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003399 "flow": flow,
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003400 "flow-id": flow.Id,
3401 "device-id": f.deviceHandler.device.Id})
3402 // Remove from device
3403 if err := f.removeFlowFromDevice(ctx, &removeFlowMessage, flow.Id); err != nil {
3404 // DKB
3405 logger.Errorw(ctx, "failed-to-remove-multicast-flow",
3406 log.Fields{
3407 "flow-id": flow.Id,
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003408 "err": err})
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003409 return err
3410 }
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003411
3412 return nil
Esin Karamanccb714b2019-11-29 15:02:06 +00003413}
3414
khenaidoodc2116e2021-10-19 17:33:19 -04003415func (f *OpenOltFlowMgr) getTechProfileDownloadMessage(ctx context.Context, tpPath string, uniID uint32, onuDeviceID string) (*ia.TechProfileDownloadMessage, error) {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003416 tpInst, err := f.techprofile.GetTPInstance(ctx, tpPath)
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003417 if err != nil {
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003418 logger.Errorw(ctx, "error-fetching-tp-instance", log.Fields{"tpPath": tpPath})
khenaidoo106c61a2021-08-11 18:05:46 -04003419 return nil, err
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003420 }
3421
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003422 switch tpInst := tpInst.(type) {
3423 case *tp_pb.TechProfileInstance:
khenaidoo106c61a2021-08-11 18:05:46 -04003424 logger.Debugw(ctx, "fetched-tp-instance-successfully-formulating-tp-download-msg", log.Fields{"tpPath": tpPath})
khenaidoodc2116e2021-10-19 17:33:19 -04003425 return &ia.TechProfileDownloadMessage{
khenaidoo106c61a2021-08-11 18:05:46 -04003426 DeviceId: onuDeviceID,
3427 UniId: uniID,
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003428 TpInstancePath: tpPath,
khenaidoodc2116e2021-10-19 17:33:19 -04003429 TechTpInstance: &ia.TechProfileDownloadMessage_TpInstance{TpInstance: tpInst},
khenaidoo106c61a2021-08-11 18:05:46 -04003430 }, nil
khenaidoodc2116e2021-10-19 17:33:19 -04003431 case *tp_pb.EponTechProfileInstance:
3432 return &ia.TechProfileDownloadMessage{
khenaidoo106c61a2021-08-11 18:05:46 -04003433 DeviceId: onuDeviceID,
3434 UniId: uniID,
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003435 TpInstancePath: tpPath,
khenaidoodc2116e2021-10-19 17:33:19 -04003436 TechTpInstance: &ia.TechProfileDownloadMessage_EponTpInstance{EponTpInstance: tpInst},
khenaidoo106c61a2021-08-11 18:05:46 -04003437 }, nil
Girish Gowdra8a0bdcd2021-05-13 12:31:04 -07003438 default:
3439 logger.Errorw(ctx, "unknown-tech", log.Fields{"tpPath": tpPath})
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003440 }
khenaidoodc2116e2021-10-19 17:33:19 -04003441 return &ia.TechProfileDownloadMessage{
khenaidoo106c61a2021-08-11 18:05:46 -04003442 DeviceId: onuDeviceID,
3443 UniId: uniID,
3444 TpInstancePath: tpPath,
3445 TechTpInstance: nil,
3446 }, nil
Girish Gowdraa09aeab2020-09-14 16:30:52 -07003447}
3448
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003449// revertTechProfileInstance is called when CreateScheduler or CreateQueues request fails
3450func (f *OpenOltFlowMgr) revertTechProfileInstance(ctx context.Context, sq schedQueue) {
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003451 intfID := sq.intfID
3452 onuID := sq.onuID
3453 uniID := sq.uniID
3454 tpID := sq.tpID
3455
3456 var reverseDirection string
3457 if sq.direction == tp_pb.Direction_UPSTREAM {
3458 reverseDirection = "downstream"
3459 } else {
3460 reverseDirection = "upstream"
3461 }
3462
3463 // check reverse direction - if reverse meter exists, tech profile instance is in use - do not delete
yasin saplibddc2d72022-02-08 13:10:17 +00003464 if KvStoreMeter, _ := f.resourceMgr.GetMeterInfoForOnu(ctx, reverseDirection, onuID, uniID, tpID); KvStoreMeter != nil {
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003465 return
3466 }
3467
3468 // revert-delete tech-profile instance and delete tech profile id for onu
3469 logger.Warnw(ctx, "reverting-tech-profile-instance-and-tech-profile-id-for-onu", log.Fields{"intf-id": intfID, "onu-id": onuID, "uni-id": uniID, "tp-id": tpID})
3470 uniPortName := getUniPortPath(f.deviceHandler.device.Id, intfID, int32(onuID), int32(uniID))
3471 _ = f.DeleteTechProfileInstance(ctx, intfID, onuID, uniID, uniPortName, tpID)
yasin saplibddc2d72022-02-08 13:10:17 +00003472 _ = f.resourceMgr.RemoveTechProfileIDForOnu(ctx, onuID, uniID, tpID)
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003473
3474 // free gem/alloc
3475 switch techprofileInst := sq.tpInst.(type) {
3476 case *tp_pb.TechProfileInstance:
3477 for _, gem := range techprofileInst.UpstreamGemPortAttributeList {
yasin saplibddc2d72022-02-08 13:10:17 +00003478 f.resourceMgr.FreeGemPortID(ctx, onuID, uniID, gem.GemportId)
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003479 }
yasin saplibddc2d72022-02-08 13:10:17 +00003480 f.resourceMgr.FreeAllocID(ctx, onuID, uniID, techprofileInst.UsScheduler.AllocId, true)
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003481 }
3482}
3483
3484// revertSchduler is called when CreateQueues request fails
3485func (f *OpenOltFlowMgr) revertScheduler(ctx context.Context, sq schedQueue, TrafficSched []*tp_pb.TrafficScheduler) {
3486 // revert scheduler
3487 logger.Warnw(ctx, "reverting-scheduler-for-onu", log.Fields{"intf-id": sq.intfID, "onu-id": sq.onuID, "uni-id": sq.uniID, "tp-id": sq.tpID})
bseenivaa1622112025-12-11 18:24:02 +05303488 subCtx, cancel := context.WithTimeout(log.WithSpanFromContext(context.Background(), ctx), f.deviceHandler.cfg.RPCTimeout)
3489 _, _ = f.deviceHandler.Client.RemoveTrafficSchedulers(subCtx, &tp_pb.TrafficSchedulers{
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003490 IntfId: sq.intfID, OnuId: sq.onuID,
3491 UniId: sq.uniID, PortNo: sq.uniPort,
3492 TrafficScheds: TrafficSched})
bseenivaa1622112025-12-11 18:24:02 +05303493 cancel()
Gamze Abaka411ef2f2021-11-22 08:38:08 +00003494}
Girish Gowdra6071f382021-12-14 12:52:04 +05303495
3496// validateMeter validates if there is a meter mismatch for the given direction. It also clears the stale meter if the reference count is zero
3497func (f *OpenOltFlowMgr) validateMeter(ctx context.Context, direction string, meterID uint32, intfID uint32, onuID uint32, uniID uint32, tpID uint32) error {
yasin saplibddc2d72022-02-08 13:10:17 +00003498 meterInfo, err := f.resourceMgr.GetMeterInfoForOnu(ctx, direction, onuID, uniID, tpID)
Girish Gowdra6071f382021-12-14 12:52:04 +05303499 if err != nil {
3500 return olterrors.NewErrNotFound("meter",
3501 log.Fields{"intf-id": intfID,
3502 "onu-id": onuID,
3503 "uni-id": uniID,
3504 "device-id": f.deviceHandler.device.Id}, err)
3505 }
3506
3507 if meterInfo != nil {
3508 // If RefCnt become 0 clear the meter information from the DB.
3509 if meterInfo.MeterID != meterID && meterInfo.RefCnt == 0 {
yasin saplibddc2d72022-02-08 13:10:17 +00003510 if err := f.resourceMgr.RemoveMeterInfoForOnu(ctx, direction, onuID, uniID, tpID); err != nil {
Girish Gowdra6071f382021-12-14 12:52:04 +05303511 return err
3512 }
3513 } else if meterInfo.MeterID != meterID {
3514 logger.Errorw(ctx, "meter-mismatch-for-direction",
3515 log.Fields{"direction": direction,
3516 "kv-store-meter-id": meterInfo.MeterID,
3517 "meter-id-in-flow": meterID,
3518 "device-id": f.deviceHandler.device.Id})
3519 return olterrors.NewErrInvalidValue(log.Fields{
3520 "unsupported": "meter-id",
3521 "kv-store-meter-id": meterInfo.MeterID,
3522 "meter-id-in-flow": meterID,
3523 "device-id": f.deviceHandler.device.Id}, nil)
3524 }
3525 }
3526 return nil
3527}
Girish Gowdraf3728b12022-02-02 21:46:51 -08003528
3529func (f *OpenOltFlowMgr) removeMeterReference(ctx context.Context, direction string, sq schedQueue) error {
3530 /* After we successfully remove the scheduler configuration on the OLT device,
3531 * delete the meter id on the KV store.
3532 */
yasin saplibddc2d72022-02-08 13:10:17 +00003533 err := f.resourceMgr.RemoveMeterInfoForOnu(ctx, direction, sq.onuID, sq.uniID, sq.tpID)
Girish Gowdraf3728b12022-02-02 21:46:51 -08003534 if err != nil {
3535 return olterrors.NewErrAdapter("unable-to-remove-meter",
3536 log.Fields{
3537 "onu": sq.onuID,
3538 "device-id": f.deviceHandler.device.Id,
3539 "intf-id": sq.intfID,
3540 "onu-id": sq.onuID,
3541 "uni-id": sq.uniID,
3542 "uni-port": sq.uniPort}, err)
3543 }
3544 logger.Debugw(ctx, "removed-meter-from-KV-store-successfully",
3545 log.Fields{
3546 "dir": direction,
3547 "device-id": f.deviceHandler.device.Id,
3548 "intf-id": sq.intfID,
3549 "onu-id": sq.onuID,
3550 "uni-id": sq.uniID,
3551 "uni-port": sq.uniPort})
3552 return err
3553}