blob: 96e35414e0662707a9df525453d04da83174af58 [file] [log] [blame]
khenaidoob9203542018-09-17 22:56:37 -04001/*
2 * Copyright 2018-present Open Networking Foundation
3
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 */
16package core
17
18import (
19 "context"
20 "errors"
21 "github.com/opencord/voltha-go/common/log"
22 "github.com/opencord/voltha-go/db/model"
23 "github.com/opencord/voltha-go/kafka"
William Kurkiandaa6bb22019-03-07 12:26:28 -050024 "github.com/opencord/voltha-protos/go/openflow_13"
25 "github.com/opencord/voltha-protos/go/voltha"
khenaidoob9203542018-09-17 22:56:37 -040026 "google.golang.org/grpc/codes"
27 "google.golang.org/grpc/status"
khenaidoob9203542018-09-17 22:56:37 -040028 "strings"
29 "sync"
30)
31
32type LogicalDeviceManager struct {
33 logicalDeviceAgents map[string]*LogicalDeviceAgent
Richard Jankowski199fd862019-03-18 14:49:51 -040034 core *Core
khenaidoob9203542018-09-17 22:56:37 -040035 deviceMgr *DeviceManager
Richard Jankowskidbab94a2018-12-06 16:20:25 -050036 grpcNbiHdlr *APIHandler
khenaidoob9203542018-09-17 22:56:37 -040037 adapterProxy *AdapterProxy
khenaidoo43c82122018-11-22 18:38:28 -050038 kafkaICProxy *kafka.InterContainerProxy
khenaidoo9a468962018-09-19 15:33:13 -040039 clusterDataProxy *model.Proxy
khenaidoob9203542018-09-17 22:56:37 -040040 exitChannel chan int
41 lockLogicalDeviceAgentsMap sync.RWMutex
khenaidoo2c6a0992019-04-29 13:46:56 -040042 defaultTimeout int64
khenaidoob9203542018-09-17 22:56:37 -040043}
44
khenaidoo2c6a0992019-04-29 13:46:56 -040045func newLogicalDeviceManager(core *Core, deviceMgr *DeviceManager, kafkaICProxy *kafka.InterContainerProxy, cdProxy *model.Proxy, timeout int64) *LogicalDeviceManager {
khenaidoob9203542018-09-17 22:56:37 -040046 var logicalDeviceMgr LogicalDeviceManager
Richard Jankowski199fd862019-03-18 14:49:51 -040047 logicalDeviceMgr.core = core
khenaidoob9203542018-09-17 22:56:37 -040048 logicalDeviceMgr.exitChannel = make(chan int, 1)
49 logicalDeviceMgr.logicalDeviceAgents = make(map[string]*LogicalDeviceAgent)
50 logicalDeviceMgr.deviceMgr = deviceMgr
khenaidoo43c82122018-11-22 18:38:28 -050051 logicalDeviceMgr.kafkaICProxy = kafkaICProxy
khenaidoo9a468962018-09-19 15:33:13 -040052 logicalDeviceMgr.clusterDataProxy = cdProxy
khenaidoob9203542018-09-17 22:56:37 -040053 logicalDeviceMgr.lockLogicalDeviceAgentsMap = sync.RWMutex{}
khenaidoo2c6a0992019-04-29 13:46:56 -040054 logicalDeviceMgr.defaultTimeout = timeout
khenaidoob9203542018-09-17 22:56:37 -040055 return &logicalDeviceMgr
56}
57
Richard Jankowskidbab94a2018-12-06 16:20:25 -050058func (ldMgr *LogicalDeviceManager) setGrpcNbiHandler(grpcNbiHandler *APIHandler) {
59 ldMgr.grpcNbiHdlr = grpcNbiHandler
60}
61
khenaidoo4d4802d2018-10-04 21:59:49 -040062func (ldMgr *LogicalDeviceManager) start(ctx context.Context) {
khenaidoob9203542018-09-17 22:56:37 -040063 log.Info("starting-logical-device-manager")
64 log.Info("logical-device-manager-started")
65}
66
khenaidoo4d4802d2018-10-04 21:59:49 -040067func (ldMgr *LogicalDeviceManager) stop(ctx context.Context) {
khenaidoob9203542018-09-17 22:56:37 -040068 log.Info("stopping-logical-device-manager")
69 ldMgr.exitChannel <- 1
70 log.Info("logical-device-manager-stopped")
71}
72
khenaidoo19d7b632018-10-30 10:49:50 -040073func sendAPIResponse(ctx context.Context, ch chan interface{}, result interface{}) {
74 if ctx.Err() == nil {
75 // Returned response only of the ctx has not been cancelled/timeout/etc
76 // Channel is automatically closed when a context is Done
77 ch <- result
78 log.Debugw("sendResponse", log.Fields{"result": result})
79 } else {
80 // Should the transaction be reverted back?
81 log.Debugw("sendResponse-context-error", log.Fields{"context-error": ctx.Err()})
82 }
83}
84
khenaidoob9203542018-09-17 22:56:37 -040085func (ldMgr *LogicalDeviceManager) addLogicalDeviceAgentToMap(agent *LogicalDeviceAgent) {
86 ldMgr.lockLogicalDeviceAgentsMap.Lock()
87 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
88 if _, exist := ldMgr.logicalDeviceAgents[agent.logicalDeviceId]; !exist {
89 ldMgr.logicalDeviceAgents[agent.logicalDeviceId] = agent
90 }
91}
92
khenaidoo6d62c002019-05-15 21:57:03 -040093func (ldMgr *LogicalDeviceManager) isLogicalDeviceInCache(logicalDeviceId string) bool {
94 ldMgr.lockLogicalDeviceAgentsMap.RLock()
95 defer ldMgr.lockLogicalDeviceAgentsMap.RUnlock()
96 _, inCache := ldMgr.logicalDeviceAgents[logicalDeviceId]
97 return inCache
98}
99
khenaidoo8c3303d2019-02-13 14:59:39 -0500100// getLogicalDeviceAgent returns the logical device agent. If the device is not in memory then the device will
101// be loaded from dB and a logical device agent created to managed it.
khenaidoob9203542018-09-17 22:56:37 -0400102func (ldMgr *LogicalDeviceManager) getLogicalDeviceAgent(logicalDeviceId string) *LogicalDeviceAgent {
khenaidoo1ce37ad2019-03-24 22:07:24 -0400103 ldMgr.lockLogicalDeviceAgentsMap.RLock()
khenaidoob9203542018-09-17 22:56:37 -0400104 if agent, ok := ldMgr.logicalDeviceAgents[logicalDeviceId]; ok {
khenaidoo1ce37ad2019-03-24 22:07:24 -0400105 ldMgr.lockLogicalDeviceAgentsMap.RUnlock()
khenaidoob9203542018-09-17 22:56:37 -0400106 return agent
khenaidoo8c3303d2019-02-13 14:59:39 -0500107 } else {
108 // Try to load into memory - loading will also create the logical device agent
khenaidoo1ce37ad2019-03-24 22:07:24 -0400109 ldMgr.lockLogicalDeviceAgentsMap.RUnlock()
khenaidoo8c3303d2019-02-13 14:59:39 -0500110 if err := ldMgr.load(logicalDeviceId); err == nil {
khenaidoo1ce37ad2019-03-24 22:07:24 -0400111 ldMgr.lockLogicalDeviceAgentsMap.RLock()
112 defer ldMgr.lockLogicalDeviceAgentsMap.RUnlock()
khenaidoo8c3303d2019-02-13 14:59:39 -0500113 if agent, ok = ldMgr.logicalDeviceAgents[logicalDeviceId]; ok {
114 return agent
115 }
116 }
khenaidoob9203542018-09-17 22:56:37 -0400117 }
118 return nil
119}
120
khenaidoo92e62c52018-10-03 14:02:54 -0400121func (ldMgr *LogicalDeviceManager) deleteLogicalDeviceAgent(logicalDeviceId string) {
122 ldMgr.lockLogicalDeviceAgentsMap.Lock()
123 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
124 delete(ldMgr.logicalDeviceAgents, logicalDeviceId)
125}
126
khenaidoo8c3303d2019-02-13 14:59:39 -0500127// GetLogicalDevice provides a cloned most up to date logical device. If device is not in memory
128// it will be fetched from the dB
khenaidoob9203542018-09-17 22:56:37 -0400129func (ldMgr *LogicalDeviceManager) getLogicalDevice(id string) (*voltha.LogicalDevice, error) {
khenaidoo92e62c52018-10-03 14:02:54 -0400130 log.Debugw("getlogicalDevice", log.Fields{"logicaldeviceid": id})
131 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
khenaidoo19d7b632018-10-30 10:49:50 -0400132 return agent.GetLogicalDevice()
khenaidoob9203542018-09-17 22:56:37 -0400133 }
134 return nil, status.Errorf(codes.NotFound, "%s", id)
135}
136
khenaidoo43aa6bd2019-05-29 13:35:13 -0400137func (ldMgr *LogicalDeviceManager) listManagedLogicalDevices() (*voltha.LogicalDevices, error) {
138 log.Debug("listManagedLogicalDevices")
139 result := &voltha.LogicalDevices{}
140 ldMgr.lockLogicalDeviceAgentsMap.RLock()
141 defer ldMgr.lockLogicalDeviceAgentsMap.RUnlock()
142 for _, agent := range ldMgr.logicalDeviceAgents {
143 if ld, _ := agent.GetLogicalDevice(); ld != nil {
144 result.Items = append(result.Items, ld)
145 }
146 }
147 return result, nil
148}
149
khenaidoob9203542018-09-17 22:56:37 -0400150func (ldMgr *LogicalDeviceManager) listLogicalDevices() (*voltha.LogicalDevices, error) {
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500151 log.Debug("ListAllLogicalDevices")
khenaidoob9203542018-09-17 22:56:37 -0400152 result := &voltha.LogicalDevices{}
khenaidoo297cd252019-02-07 22:10:23 -0500153 if logicalDevices := ldMgr.clusterDataProxy.List("/logical_devices", 0, false, ""); logicalDevices != nil {
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500154 for _, logicalDevice := range logicalDevices.([]interface{}) {
155 if agent := ldMgr.getLogicalDeviceAgent(logicalDevice.(*voltha.LogicalDevice).Id); agent == nil {
156 agent = newLogicalDeviceAgent(
157 logicalDevice.(*voltha.LogicalDevice).Id,
158 logicalDevice.(*voltha.LogicalDevice).RootDeviceId,
159 ldMgr,
160 ldMgr.deviceMgr,
161 ldMgr.clusterDataProxy,
khenaidoo2c6a0992019-04-29 13:46:56 -0400162 ldMgr.defaultTimeout,
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500163 )
164 ldMgr.addLogicalDeviceAgentToMap(agent)
khenaidoo297cd252019-02-07 22:10:23 -0500165 go agent.start(nil, true)
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500166 }
167 result.Items = append(result.Items, logicalDevice.(*voltha.LogicalDevice))
khenaidoob9203542018-09-17 22:56:37 -0400168 }
169 }
170 return result, nil
171}
172
khenaidoo4d4802d2018-10-04 21:59:49 -0400173func (ldMgr *LogicalDeviceManager) createLogicalDevice(ctx context.Context, device *voltha.Device) (*string, error) {
khenaidoo92e62c52018-10-03 14:02:54 -0400174 log.Debugw("creating-logical-device", log.Fields{"deviceId": device.Id})
khenaidoob9203542018-09-17 22:56:37 -0400175 // Sanity check
176 if !device.Root {
khenaidoo2c6a0992019-04-29 13:46:56 -0400177 return nil, errors.New("device-not-root")
khenaidoob9203542018-09-17 22:56:37 -0400178 }
179
180 // Create a logical device agent - the logical device Id is based on the mac address of the device
181 // For now use the serial number - it may contain any combination of alphabetic characters and numbers,
182 // with length varying from eight characters to a maximum of 14 characters. Mac Address is part of oneof
183 // in the Device model. May need to be moved out.
184 macAddress := device.MacAddress
185 id := strings.Replace(macAddress, ":", "", -1)
khenaidoo92e62c52018-10-03 14:02:54 -0400186 if id == "" {
187 log.Errorw("mac-address-not-set", log.Fields{"deviceId": device.Id})
188 return nil, errors.New("mac-address-not-set")
189 }
190 log.Debugw("logical-device-id", log.Fields{"logicaldeviceId": id})
khenaidoob9203542018-09-17 22:56:37 -0400191
khenaidoo2c6a0992019-04-29 13:46:56 -0400192 agent := newLogicalDeviceAgent(id, device.Id, ldMgr, ldMgr.deviceMgr, ldMgr.clusterDataProxy, ldMgr.defaultTimeout)
khenaidoob9203542018-09-17 22:56:37 -0400193 ldMgr.addLogicalDeviceAgentToMap(agent)
khenaidoo297cd252019-02-07 22:10:23 -0500194 go agent.start(ctx, false)
khenaidoob9203542018-09-17 22:56:37 -0400195
khenaidoo92e62c52018-10-03 14:02:54 -0400196 log.Debug("creating-logical-device-ends")
khenaidoob9203542018-09-17 22:56:37 -0400197 return &id, nil
198}
199
khenaidoo6d62c002019-05-15 21:57:03 -0400200// stopManagingLogicalDeviceWithDeviceId stops the management of the logical device. This implies removal of any
201// reference of this logical device in cache. The device Id is passed as param because the logical device may already
202// have been removed from the model. This function returns the logical device Id if found
203func (ldMgr *LogicalDeviceManager) stopManagingLogicalDeviceWithDeviceId(id string) string {
204 log.Infow("stop-managing-logical-device", log.Fields{"deviceId": id})
205 // Go over the list of logical device agents to find the one which has rootDeviceId as id
206 ldMgr.lockLogicalDeviceAgentsMap.RLock()
207 defer ldMgr.lockLogicalDeviceAgentsMap.RUnlock()
208 for ldId, ldAgent := range ldMgr.logicalDeviceAgents {
209 if ldAgent.rootDeviceId == id {
210 log.Infow("stopping-logical-device-agent", log.Fields{"lDeviceId": ldId})
211 ldAgent.stop(nil)
212 delete(ldMgr.logicalDeviceAgents, ldId)
213 return ldId
214 }
215 }
216 return ""
217}
218
219//getLogicalDeviceFromModel retrieves the logical device data from the model.
220func (ldMgr *LogicalDeviceManager) getLogicalDeviceFromModel(lDeviceId string) (*voltha.LogicalDevice, error) {
221
222 if logicalDevice := ldMgr.clusterDataProxy.Get("/logical_devices/"+lDeviceId, 0, false, ""); logicalDevice != nil {
223 if lDevice, ok := logicalDevice.(*voltha.LogicalDevice); ok {
224 return lDevice, nil
225 }
226 }
227 return nil, status.Error(codes.NotFound, lDeviceId)
228}
229
khenaidoo297cd252019-02-07 22:10:23 -0500230// load loads a logical device manager in memory
231func (ldMgr *LogicalDeviceManager) load(lDeviceId string) error {
khenaidoo1ce37ad2019-03-24 22:07:24 -0400232 log.Debugw("loading-logical-device", log.Fields{"lDeviceId": lDeviceId})
khenaidoo297cd252019-02-07 22:10:23 -0500233 // To prevent a race condition, let's hold the logical device agent map lock. This will prevent a loading and
234 // a create logical device callback from occurring at the same time.
235 ldMgr.lockLogicalDeviceAgentsMap.Lock()
236 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
237 if ldAgent, _ := ldMgr.logicalDeviceAgents[lDeviceId]; ldAgent == nil {
khenaidoo6d62c002019-05-15 21:57:03 -0400238 // Proceed with the loading only if the logical device exist in the Model (could have been deleted)
239 if _, err := ldMgr.getLogicalDeviceFromModel(lDeviceId); err == nil {
240 // Create a temp logical device Agent and let it load from memory
241 agent := newLogicalDeviceAgent(lDeviceId, "", ldMgr, ldMgr.deviceMgr, ldMgr.clusterDataProxy, ldMgr.defaultTimeout)
242 if err := agent.start(nil, true); err != nil {
243 agent.stop(nil)
244 return err
245 }
246 ldMgr.logicalDeviceAgents[agent.logicalDeviceId] = agent
khenaidoo297cd252019-02-07 22:10:23 -0500247 }
khenaidoo297cd252019-02-07 22:10:23 -0500248 }
249 // TODO: load the child device
250 return nil
251}
252
khenaidoo4d4802d2018-10-04 21:59:49 -0400253func (ldMgr *LogicalDeviceManager) deleteLogicalDevice(ctx context.Context, device *voltha.Device) error {
khenaidoo92e62c52018-10-03 14:02:54 -0400254 log.Debugw("deleting-logical-device", log.Fields{"deviceId": device.Id})
255 // Sanity check
256 if !device.Root {
khenaidoo2c6a0992019-04-29 13:46:56 -0400257 return errors.New("device-not-root")
khenaidoo92e62c52018-10-03 14:02:54 -0400258 }
259 logDeviceId := device.ParentId
260 if agent := ldMgr.getLogicalDeviceAgent(logDeviceId); agent != nil {
261 // Stop the logical device agent
khenaidoo4d4802d2018-10-04 21:59:49 -0400262 agent.stop(ctx)
khenaidoo92e62c52018-10-03 14:02:54 -0400263 //Remove the logical device agent from the Map
264 ldMgr.deleteLogicalDeviceAgent(logDeviceId)
Richard Jankowski199fd862019-03-18 14:49:51 -0400265 ldMgr.core.deviceOwnership.AbandonDevice(logDeviceId)
khenaidoo92e62c52018-10-03 14:02:54 -0400266 }
267
268 log.Debug("deleting-logical-device-ends")
269 return nil
270}
271
272func (ldMgr *LogicalDeviceManager) getLogicalDeviceId(device *voltha.Device) (*string, error) {
273 // Device can either be a parent or a child device
274 if device.Root {
275 // Parent device. The ID of a parent device is the logical device ID
276 return &device.ParentId, nil
277 }
278 // Device is child device
279 // retrieve parent device using child device ID
280 if parentDevice := ldMgr.deviceMgr.getParentDevice(device); parentDevice != nil {
281 return &parentDevice.ParentId, nil
282 }
283 return nil, status.Errorf(codes.NotFound, "%s", device.Id)
284}
285
khenaidoo3ab34882019-05-02 21:33:30 -0400286func (ldMgr *LogicalDeviceManager) getLogicalDeviceIdFromDeviceId(deviceId string) (*string, error) {
287 // Get the device
288 var device *voltha.Device
289 var err error
290 if device, err = ldMgr.deviceMgr.GetDevice(deviceId); err != nil {
291 return nil, err
292 }
293 return ldMgr.getLogicalDeviceId(device)
294}
295
khenaidoo19d7b632018-10-30 10:49:50 -0400296func (ldMgr *LogicalDeviceManager) getLogicalPortId(device *voltha.Device) (*voltha.LogicalPortId, error) {
297 // Get the logical device where this device is attached
298 var lDeviceId *string
299 var err error
300 if lDeviceId, err = ldMgr.getLogicalDeviceId(device); err != nil {
301 return nil, err
302 }
303 var lDevice *voltha.LogicalDevice
304 if lDevice, err = ldMgr.getLogicalDevice(*lDeviceId); err != nil {
305 return nil, err
306 }
307 // Go over list of ports
308 for _, port := range lDevice.Ports {
309 if port.DeviceId == device.Id {
310 return &voltha.LogicalPortId{Id: *lDeviceId, PortId: port.Id}, nil
311 }
312 }
313 return nil, status.Errorf(codes.NotFound, "%s", device.Id)
314}
315
khenaidoodd237172019-05-27 16:37:17 -0400316func (ldMgr *LogicalDeviceManager) ListLogicalDeviceFlows(ctx context.Context, id string) (*openflow_13.Flows, error) {
317 log.Debugw("ListLogicalDeviceFlows", log.Fields{"logicaldeviceid": id})
318 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
319 return agent.ListLogicalDeviceFlows()
320 }
321 return nil, status.Errorf(codes.NotFound, "%s", id)
322}
323
324func (ldMgr *LogicalDeviceManager) ListLogicalDeviceFlowGroups(ctx context.Context, id string) (*openflow_13.FlowGroups, error) {
325 log.Debugw("ListLogicalDeviceFlowGroups", log.Fields{"logicaldeviceid": id})
326 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
327 return agent.ListLogicalDeviceFlowGroups()
328 }
329 return nil, status.Errorf(codes.NotFound, "%s", id)
330}
331
khenaidoo19d7b632018-10-30 10:49:50 -0400332func (ldMgr *LogicalDeviceManager) ListLogicalDevicePorts(ctx context.Context, id string) (*voltha.LogicalPorts, error) {
333 log.Debugw("ListLogicalDevicePorts", log.Fields{"logicaldeviceid": id})
334 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
335 return agent.ListLogicalDevicePorts()
336 }
337 return nil, status.Errorf(codes.NotFound, "%s", id)
khenaidoo19d7b632018-10-30 10:49:50 -0400338}
339
340func (ldMgr *LogicalDeviceManager) getLogicalPort(lPortId *voltha.LogicalPortId) (*voltha.LogicalPort, error) {
341 // Get the logical device where this device is attached
342 var err error
343 var lDevice *voltha.LogicalDevice
344 if lDevice, err = ldMgr.getLogicalDevice(lPortId.Id); err != nil {
345 return nil, err
346 }
347 // Go over list of ports
348 for _, port := range lDevice.Ports {
349 if port.Id == lPortId.PortId {
350 return port, nil
351 }
352 }
Kent Hagerman0ab4cb22019-04-24 13:13:35 -0400353 return nil, status.Errorf(codes.NotFound, "%s-%s", lPortId.Id, lPortId.PortId)
khenaidoo19d7b632018-10-30 10:49:50 -0400354}
355
khenaidoo2c6a0992019-04-29 13:46:56 -0400356// updateLogicalPort sets up a logical port on the logical device based on the device port
357// information, if needed
358func (ldMgr *LogicalDeviceManager) updateLogicalPort(device *voltha.Device, port *voltha.Port) error {
359 if ldID, err := ldMgr.getLogicalDeviceId(device); err != nil || *ldID == "" {
360 // This is not an error as the logical device may not have been created at this time. In such a case,
361 // the ports will be created when the logical device is ready.
362 return nil
363 } else {
364 if agent := ldMgr.getLogicalDeviceAgent(*ldID); agent != nil {
365 if err := agent.updateLogicalPort(device, port); err != nil {
366 return err
367 }
368 }
369 }
370 return nil
371}
372
khenaidoofc1314d2019-03-14 09:34:21 -0400373// addLogicalPort sets up a logical port on the logical device based on the device port
374// information.
375func (ldMgr *LogicalDeviceManager) addLogicalPort(device *voltha.Device, port *voltha.Port) error {
Stephane Barbarie40fd3b22019-04-23 21:50:47 -0400376 if ldID, err := ldMgr.getLogicalDeviceId(device); err != nil || *ldID == "" {
khenaidoofc1314d2019-03-14 09:34:21 -0400377 // This is not an error as the logical device may not have been created at this time. In such a case,
378 // the ports will be created when the logical device is ready.
379 return nil
380 } else {
381 if agent := ldMgr.getLogicalDeviceAgent(*ldID); agent != nil {
382 if err := agent.addLogicalPort(device, port); err != nil {
383 return err
384 }
385 }
386 }
387 return nil
388}
389
khenaidoo0a822f92019-05-08 15:15:57 -0400390// deleteLogicalPort removes the logical port associated with a device
khenaidoo19d7b632018-10-30 10:49:50 -0400391func (ldMgr *LogicalDeviceManager) deleteLogicalPort(ctx context.Context, lPortId *voltha.LogicalPortId) error {
392 log.Debugw("deleting-logical-port", log.Fields{"LDeviceId": lPortId.Id})
393 // Get logical port
394 var logicalPort *voltha.LogicalPort
395 var err error
396 if logicalPort, err = ldMgr.getLogicalPort(lPortId); err != nil {
397 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": lPortId.PortId})
398 return err
399 }
khenaidoo92e62c52018-10-03 14:02:54 -0400400 // Sanity check
khenaidoo19d7b632018-10-30 10:49:50 -0400401 if logicalPort.RootPort {
khenaidoo2c6a0992019-04-29 13:46:56 -0400402 return errors.New("device-root")
khenaidoo92e62c52018-10-03 14:02:54 -0400403 }
khenaidoo19d7b632018-10-30 10:49:50 -0400404 if agent := ldMgr.getLogicalDeviceAgent(lPortId.Id); agent != nil {
khenaidoo0a822f92019-05-08 15:15:57 -0400405 if err := agent.deleteLogicalPort(logicalPort); err != nil {
406 log.Warnw("deleting-logicalport-failed", log.Fields{"LDeviceId": lPortId.Id, "error": err})
407 }
khenaidoo92e62c52018-10-03 14:02:54 -0400408 }
409
410 log.Debug("deleting-logical-port-ends")
411 return nil
412}
413
khenaidoo0a822f92019-05-08 15:15:57 -0400414// deleteLogicalPort removes the logical port associated with a child device
415func (ldMgr *LogicalDeviceManager) deleteLogicalPorts(deviceId string) error {
416 log.Debugw("deleting-logical-ports", log.Fields{"deviceId": deviceId})
417 // Get logical port
418 if ldId, err := ldMgr.getLogicalDeviceIdFromDeviceId(deviceId); err != nil {
khenaidoo0a822f92019-05-08 15:15:57 -0400419 return err
420 } else {
421 if agent := ldMgr.getLogicalDeviceAgent(*ldId); agent != nil {
422 if err = agent.deleteLogicalPorts(deviceId); err != nil {
423 log.Warnw("deleteLogicalPorts-failed", log.Fields{"ldeviceId": *ldId})
424 return err
425 }
426 }
427 }
428 log.Debug("deleting-logical-port-ends")
429 return nil
430}
431
khenaidoofc1314d2019-03-14 09:34:21 -0400432func (ldMgr *LogicalDeviceManager) setupUNILogicalPorts(ctx context.Context, childDevice *voltha.Device) error {
433 log.Debugw("setupUNILogicalPorts", log.Fields{"childDeviceId": childDevice.Id, "parentDeviceId": childDevice.ParentId})
khenaidoob9203542018-09-17 22:56:37 -0400434 // Sanity check
435 if childDevice.Root {
436 return errors.New("Device-root")
437 }
438
439 // Get the logical device id parent device
440 parentId := childDevice.ParentId
441 logDeviceId := ldMgr.deviceMgr.GetParentDeviceId(parentId)
442
khenaidoofc1314d2019-03-14 09:34:21 -0400443 log.Debugw("setupUNILogicalPorts", log.Fields{"logDeviceId": logDeviceId, "parentId": parentId})
khenaidoob9203542018-09-17 22:56:37 -0400444
khenaidoo3d3b8c22019-05-22 18:10:39 -0400445 if parentId == "" || logDeviceId == nil || *logDeviceId == "" {
khenaidoo2c6a0992019-04-29 13:46:56 -0400446 return errors.New("device-in-invalid-state")
khenaidoo5e677ae2019-02-28 17:26:29 -0500447 }
448
khenaidoob9203542018-09-17 22:56:37 -0400449 if agent := ldMgr.getLogicalDeviceAgent(*logDeviceId); agent != nil {
khenaidoofc1314d2019-03-14 09:34:21 -0400450 if err := agent.setupUNILogicalPorts(ctx, childDevice); err != nil {
khenaidoobcf205b2019-01-25 22:21:14 -0500451 return err
452 }
khenaidoob9203542018-09-17 22:56:37 -0400453 }
khenaidoo21d51152019-02-01 13:48:37 -0500454 return nil
khenaidoob9203542018-09-17 22:56:37 -0400455}
khenaidoo19d7b632018-10-30 10:49:50 -0400456
khenaidoo0a822f92019-05-08 15:15:57 -0400457func (ldMgr *LogicalDeviceManager) deleteAllLogicalPorts(device *voltha.Device) error {
458 log.Debugw("deleteAllLogicalPorts", log.Fields{"deviceId": device.Id})
459
460 var ldId *string
461 var err error
462 //Get the logical device Id for this device
463 if ldId, err = ldMgr.getLogicalDeviceId(device); err != nil {
464 log.Warnw("no-logical-device-found", log.Fields{"deviceId": device.Id, "error": err})
465 return err
466 }
467 if agent := ldMgr.getLogicalDeviceAgent(*ldId); agent != nil {
468 if err := agent.deleteAllLogicalPorts(device); err != nil {
469 return err
470 }
471 }
472 return nil
473}
474
khenaidoo3ab34882019-05-02 21:33:30 -0400475func (ldMgr *LogicalDeviceManager) updatePortsState(device *voltha.Device, state voltha.AdminState_AdminState) error {
476 log.Debugw("updatePortsState", log.Fields{"deviceId": device.Id, "state": state})
477
478 var ldId *string
479 var err error
480 //Get the logical device Id for this device
481 if ldId, err = ldMgr.getLogicalDeviceId(device); err != nil {
482 log.Warnw("no-logical-device-found", log.Fields{"deviceId": device.Id, "error": err})
483 return err
484 }
485 if agent := ldMgr.getLogicalDeviceAgent(*ldId); agent != nil {
486 if err := agent.updatePortsState(device, state); err != nil {
487 return err
488 }
489 }
490 return nil
491}
492
khenaidoo19d7b632018-10-30 10:49:50 -0400493func (ldMgr *LogicalDeviceManager) updateFlowTable(ctx context.Context, id string, flow *openflow_13.OfpFlowMod, ch chan interface{}) {
494 log.Debugw("updateFlowTable", log.Fields{"logicalDeviceId": id})
495 var res interface{}
496 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
497 res = agent.updateFlowTable(ctx, flow)
498 log.Debugw("updateFlowTable-result", log.Fields{"result": res})
499 } else {
500 res = status.Errorf(codes.NotFound, "%s", id)
501 }
502 sendAPIResponse(ctx, ch, res)
503}
504
505func (ldMgr *LogicalDeviceManager) updateGroupTable(ctx context.Context, id string, groupMod *openflow_13.OfpGroupMod, ch chan interface{}) {
506 log.Debugw("updateGroupTable", log.Fields{"logicalDeviceId": id})
507 var res interface{}
508 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
509 res = agent.updateGroupTable(ctx, groupMod)
510 log.Debugw("updateGroupTable-result", log.Fields{"result": res})
511 } else {
512 res = status.Errorf(codes.NotFound, "%s", id)
513 }
514 sendAPIResponse(ctx, ch, res)
515}
516
517func (ldMgr *LogicalDeviceManager) enableLogicalPort(ctx context.Context, id *voltha.LogicalPortId, ch chan interface{}) {
518 log.Debugw("enableLogicalPort", log.Fields{"logicalDeviceId": id})
519 var res interface{}
520 // Get logical port
521 var logicalPort *voltha.LogicalPort
522 var err error
523 if logicalPort, err = ldMgr.getLogicalPort(id); err != nil {
524 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": id.PortId})
525 res = err
526 }
527 if agent := ldMgr.getLogicalDeviceAgent(id.Id); agent != nil {
528 res = agent.enableLogicalPort(logicalPort)
529 log.Debugw("enableLogicalPort-result", log.Fields{"result": res})
530 } else {
531 res = status.Errorf(codes.NotFound, "%s", id.Id)
532 }
533 sendAPIResponse(ctx, ch, res)
534}
535
536func (ldMgr *LogicalDeviceManager) disableLogicalPort(ctx context.Context, id *voltha.LogicalPortId, ch chan interface{}) {
537 log.Debugw("disableLogicalPort", log.Fields{"logicalDeviceId": id})
538 var res interface{}
539 // Get logical port
540 var logicalPort *voltha.LogicalPort
541 var err error
542 if logicalPort, err = ldMgr.getLogicalPort(id); err != nil {
543 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": id.PortId})
544 res = err
545 }
546 if agent := ldMgr.getLogicalDeviceAgent(id.Id); agent != nil {
547 res = agent.disableLogicalPort(logicalPort)
548 log.Debugw("disableLogicalPort-result", log.Fields{"result": res})
549 } else {
550 res = status.Errorf(codes.NotFound, "%s", id.Id)
551 }
552 sendAPIResponse(ctx, ch, res)
553}
khenaidoofdbad6e2018-11-06 22:26:38 -0500554
khenaidoo43c82122018-11-22 18:38:28 -0500555func (ldMgr *LogicalDeviceManager) packetOut(packetOut *openflow_13.PacketOut) {
khenaidoofdbad6e2018-11-06 22:26:38 -0500556 log.Debugw("packetOut", log.Fields{"logicalDeviceId": packetOut.Id})
557 if agent := ldMgr.getLogicalDeviceAgent(packetOut.Id); agent != nil {
558 agent.packetOut(packetOut.PacketOut)
559 } else {
560 log.Error("logical-device-not-exist", log.Fields{"logicalDeviceId": packetOut.Id})
561 }
562}
563
khenaidoo297cd252019-02-07 22:10:23 -0500564func (ldMgr *LogicalDeviceManager) packetIn(logicalDeviceId string, port uint32, transactionId string, packet []byte) error {
khenaidoofdbad6e2018-11-06 22:26:38 -0500565 log.Debugw("packetIn", log.Fields{"logicalDeviceId": logicalDeviceId, "port": port})
566 if agent := ldMgr.getLogicalDeviceAgent(logicalDeviceId); agent != nil {
khenaidoo297cd252019-02-07 22:10:23 -0500567 agent.packetIn(port, transactionId, packet)
khenaidoofdbad6e2018-11-06 22:26:38 -0500568 } else {
569 log.Error("logical-device-not-exist", log.Fields{"logicalDeviceId": logicalDeviceId})
570 }
571 return nil
572}