blob: 71843ff74bd307d3ebf0e0a57147d228f311b9ae [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 {
Stephane Barbarieef6650d2019-07-18 12:15:09 -040033 logicalDeviceAgents sync.Map
34 core *Core
35 deviceMgr *DeviceManager
36 grpcNbiHdlr *APIHandler
37 adapterProxy *AdapterProxy
38 kafkaICProxy *kafka.InterContainerProxy
39 clusterDataProxy *model.Proxy
40 exitChannel chan int
41 defaultTimeout int64
khenaidoob9203542018-09-17 22:56:37 -040042}
43
khenaidoo2c6a0992019-04-29 13:46:56 -040044func newLogicalDeviceManager(core *Core, deviceMgr *DeviceManager, kafkaICProxy *kafka.InterContainerProxy, cdProxy *model.Proxy, timeout int64) *LogicalDeviceManager {
khenaidoob9203542018-09-17 22:56:37 -040045 var logicalDeviceMgr LogicalDeviceManager
Richard Jankowski199fd862019-03-18 14:49:51 -040046 logicalDeviceMgr.core = core
khenaidoob9203542018-09-17 22:56:37 -040047 logicalDeviceMgr.exitChannel = make(chan int, 1)
khenaidoob9203542018-09-17 22:56:37 -040048 logicalDeviceMgr.deviceMgr = deviceMgr
khenaidoo43c82122018-11-22 18:38:28 -050049 logicalDeviceMgr.kafkaICProxy = kafkaICProxy
khenaidoo9a468962018-09-19 15:33:13 -040050 logicalDeviceMgr.clusterDataProxy = cdProxy
khenaidoo2c6a0992019-04-29 13:46:56 -040051 logicalDeviceMgr.defaultTimeout = timeout
khenaidoob9203542018-09-17 22:56:37 -040052 return &logicalDeviceMgr
53}
54
Richard Jankowskidbab94a2018-12-06 16:20:25 -050055func (ldMgr *LogicalDeviceManager) setGrpcNbiHandler(grpcNbiHandler *APIHandler) {
56 ldMgr.grpcNbiHdlr = grpcNbiHandler
57}
58
khenaidoo4d4802d2018-10-04 21:59:49 -040059func (ldMgr *LogicalDeviceManager) start(ctx context.Context) {
khenaidoob9203542018-09-17 22:56:37 -040060 log.Info("starting-logical-device-manager")
61 log.Info("logical-device-manager-started")
62}
63
khenaidoo4d4802d2018-10-04 21:59:49 -040064func (ldMgr *LogicalDeviceManager) stop(ctx context.Context) {
khenaidoob9203542018-09-17 22:56:37 -040065 log.Info("stopping-logical-device-manager")
66 ldMgr.exitChannel <- 1
67 log.Info("logical-device-manager-stopped")
68}
69
khenaidoo19d7b632018-10-30 10:49:50 -040070func sendAPIResponse(ctx context.Context, ch chan interface{}, result interface{}) {
71 if ctx.Err() == nil {
72 // Returned response only of the ctx has not been cancelled/timeout/etc
73 // Channel is automatically closed when a context is Done
74 ch <- result
75 log.Debugw("sendResponse", log.Fields{"result": result})
76 } else {
77 // Should the transaction be reverted back?
78 log.Debugw("sendResponse-context-error", log.Fields{"context-error": ctx.Err()})
79 }
80}
81
khenaidoob9203542018-09-17 22:56:37 -040082func (ldMgr *LogicalDeviceManager) addLogicalDeviceAgentToMap(agent *LogicalDeviceAgent) {
Stephane Barbarieef6650d2019-07-18 12:15:09 -040083 if _, exist := ldMgr.logicalDeviceAgents.Load(agent.logicalDeviceId); !exist {
84 ldMgr.logicalDeviceAgents.Store(agent.logicalDeviceId, agent)
khenaidoob9203542018-09-17 22:56:37 -040085 }
86}
87
khenaidoo6d62c002019-05-15 21:57:03 -040088func (ldMgr *LogicalDeviceManager) isLogicalDeviceInCache(logicalDeviceId string) bool {
Stephane Barbarieef6650d2019-07-18 12:15:09 -040089 _, inCache := ldMgr.logicalDeviceAgents.Load(logicalDeviceId)
khenaidoo6d62c002019-05-15 21:57:03 -040090 return inCache
91}
92
khenaidoo8c3303d2019-02-13 14:59:39 -050093// getLogicalDeviceAgent returns the logical device agent. If the device is not in memory then the device will
94// be loaded from dB and a logical device agent created to managed it.
khenaidoob9203542018-09-17 22:56:37 -040095func (ldMgr *LogicalDeviceManager) getLogicalDeviceAgent(logicalDeviceId string) *LogicalDeviceAgent {
Stephane Barbarieef6650d2019-07-18 12:15:09 -040096 if agent, ok := ldMgr.logicalDeviceAgents.Load(logicalDeviceId); ok {
97 return agent.(*LogicalDeviceAgent)
khenaidoo8c3303d2019-02-13 14:59:39 -050098 } else {
99 // Try to load into memory - loading will also create the logical device agent
khenaidoo8c3303d2019-02-13 14:59:39 -0500100 if err := ldMgr.load(logicalDeviceId); err == nil {
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400101 if agent, ok = ldMgr.logicalDeviceAgents.Load(logicalDeviceId); ok {
102 return agent.(*LogicalDeviceAgent)
khenaidoo8c3303d2019-02-13 14:59:39 -0500103 }
104 }
khenaidoob9203542018-09-17 22:56:37 -0400105 }
106 return nil
107}
108
khenaidoo92e62c52018-10-03 14:02:54 -0400109func (ldMgr *LogicalDeviceManager) deleteLogicalDeviceAgent(logicalDeviceId string) {
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400110 ldMgr.logicalDeviceAgents.Delete(logicalDeviceId)
khenaidoo92e62c52018-10-03 14:02:54 -0400111}
112
khenaidoo8c3303d2019-02-13 14:59:39 -0500113// GetLogicalDevice provides a cloned most up to date logical device. If device is not in memory
114// it will be fetched from the dB
khenaidoob9203542018-09-17 22:56:37 -0400115func (ldMgr *LogicalDeviceManager) getLogicalDevice(id string) (*voltha.LogicalDevice, error) {
khenaidoo92e62c52018-10-03 14:02:54 -0400116 log.Debugw("getlogicalDevice", log.Fields{"logicaldeviceid": id})
117 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
khenaidoo19d7b632018-10-30 10:49:50 -0400118 return agent.GetLogicalDevice()
khenaidoob9203542018-09-17 22:56:37 -0400119 }
120 return nil, status.Errorf(codes.NotFound, "%s", id)
121}
122
khenaidoo43aa6bd2019-05-29 13:35:13 -0400123func (ldMgr *LogicalDeviceManager) listManagedLogicalDevices() (*voltha.LogicalDevices, error) {
124 log.Debug("listManagedLogicalDevices")
125 result := &voltha.LogicalDevices{}
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400126 ldMgr.logicalDeviceAgents.Range(func(key, value interface{}) bool {
127 agent := value.(*LogicalDeviceAgent)
khenaidoo43aa6bd2019-05-29 13:35:13 -0400128 if ld, _ := agent.GetLogicalDevice(); ld != nil {
129 result.Items = append(result.Items, ld)
130 }
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400131 return true
132 })
133
khenaidoo43aa6bd2019-05-29 13:35:13 -0400134 return result, nil
135}
136
khenaidooba6b6c42019-08-02 09:11:56 -0400137//listLogicalDevices returns the list of all logical devices
khenaidoob9203542018-09-17 22:56:37 -0400138func (ldMgr *LogicalDeviceManager) listLogicalDevices() (*voltha.LogicalDevices, error) {
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500139 log.Debug("ListAllLogicalDevices")
khenaidoob9203542018-09-17 22:56:37 -0400140 result := &voltha.LogicalDevices{}
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400141 if logicalDevices := ldMgr.clusterDataProxy.List(context.Background(), "/logical_devices", 0, false, ""); logicalDevices != nil {
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500142 for _, logicalDevice := range logicalDevices.([]interface{}) {
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500143 result.Items = append(result.Items, logicalDevice.(*voltha.LogicalDevice))
khenaidoob9203542018-09-17 22:56:37 -0400144 }
145 }
146 return result, nil
147}
148
khenaidoo4d4802d2018-10-04 21:59:49 -0400149func (ldMgr *LogicalDeviceManager) createLogicalDevice(ctx context.Context, device *voltha.Device) (*string, error) {
khenaidoo92e62c52018-10-03 14:02:54 -0400150 log.Debugw("creating-logical-device", log.Fields{"deviceId": device.Id})
khenaidoob9203542018-09-17 22:56:37 -0400151 // Sanity check
152 if !device.Root {
khenaidoo2c6a0992019-04-29 13:46:56 -0400153 return nil, errors.New("device-not-root")
khenaidoob9203542018-09-17 22:56:37 -0400154 }
155
156 // Create a logical device agent - the logical device Id is based on the mac address of the device
157 // For now use the serial number - it may contain any combination of alphabetic characters and numbers,
158 // with length varying from eight characters to a maximum of 14 characters. Mac Address is part of oneof
159 // in the Device model. May need to be moved out.
160 macAddress := device.MacAddress
161 id := strings.Replace(macAddress, ":", "", -1)
khenaidoo92e62c52018-10-03 14:02:54 -0400162 if id == "" {
163 log.Errorw("mac-address-not-set", log.Fields{"deviceId": device.Id})
164 return nil, errors.New("mac-address-not-set")
165 }
166 log.Debugw("logical-device-id", log.Fields{"logicaldeviceId": id})
khenaidoob9203542018-09-17 22:56:37 -0400167
khenaidoo2c6a0992019-04-29 13:46:56 -0400168 agent := newLogicalDeviceAgent(id, device.Id, ldMgr, ldMgr.deviceMgr, ldMgr.clusterDataProxy, ldMgr.defaultTimeout)
khenaidoob9203542018-09-17 22:56:37 -0400169 ldMgr.addLogicalDeviceAgentToMap(agent)
khenaidoo297cd252019-02-07 22:10:23 -0500170 go agent.start(ctx, false)
khenaidoob9203542018-09-17 22:56:37 -0400171
khenaidoo92e62c52018-10-03 14:02:54 -0400172 log.Debug("creating-logical-device-ends")
khenaidoob9203542018-09-17 22:56:37 -0400173 return &id, nil
174}
175
khenaidoo6d62c002019-05-15 21:57:03 -0400176// stopManagingLogicalDeviceWithDeviceId stops the management of the logical device. This implies removal of any
177// reference of this logical device in cache. The device Id is passed as param because the logical device may already
178// have been removed from the model. This function returns the logical device Id if found
179func (ldMgr *LogicalDeviceManager) stopManagingLogicalDeviceWithDeviceId(id string) string {
180 log.Infow("stop-managing-logical-device", log.Fields{"deviceId": id})
181 // Go over the list of logical device agents to find the one which has rootDeviceId as id
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400182 var ldId = ""
183 ldMgr.logicalDeviceAgents.Range(func(key, value interface{}) bool {
184 ldAgent := value.(*LogicalDeviceAgent)
khenaidoo6d62c002019-05-15 21:57:03 -0400185 if ldAgent.rootDeviceId == id {
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400186 log.Infow("stopping-logical-device-agent", log.Fields{"lDeviceId": key})
khenaidoo6d62c002019-05-15 21:57:03 -0400187 ldAgent.stop(nil)
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400188 ldMgr.logicalDeviceAgents.Delete(ldId)
189 ldId = key.(string)
khenaidoo6d62c002019-05-15 21:57:03 -0400190 }
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400191 return true
192 })
193 return ldId
khenaidoo6d62c002019-05-15 21:57:03 -0400194}
195
196//getLogicalDeviceFromModel retrieves the logical device data from the model.
197func (ldMgr *LogicalDeviceManager) getLogicalDeviceFromModel(lDeviceId string) (*voltha.LogicalDevice, error) {
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400198 if logicalDevice := ldMgr.clusterDataProxy.Get(context.Background(), "/logical_devices/"+lDeviceId, 0, false, ""); logicalDevice != nil {
khenaidoo6d62c002019-05-15 21:57:03 -0400199 if lDevice, ok := logicalDevice.(*voltha.LogicalDevice); ok {
200 return lDevice, nil
201 }
202 }
203 return nil, status.Error(codes.NotFound, lDeviceId)
204}
205
khenaidoo297cd252019-02-07 22:10:23 -0500206// load loads a logical device manager in memory
207func (ldMgr *LogicalDeviceManager) load(lDeviceId string) error {
khenaidoo1ce37ad2019-03-24 22:07:24 -0400208 log.Debugw("loading-logical-device", log.Fields{"lDeviceId": lDeviceId})
khenaidoo297cd252019-02-07 22:10:23 -0500209 // To prevent a race condition, let's hold the logical device agent map lock. This will prevent a loading and
210 // a create logical device callback from occurring at the same time.
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400211 if ldAgent, _ := ldMgr.logicalDeviceAgents.Load(lDeviceId); ldAgent == nil {
khenaidoo6d62c002019-05-15 21:57:03 -0400212 // Proceed with the loading only if the logical device exist in the Model (could have been deleted)
213 if _, err := ldMgr.getLogicalDeviceFromModel(lDeviceId); err == nil {
214 // Create a temp logical device Agent and let it load from memory
215 agent := newLogicalDeviceAgent(lDeviceId, "", ldMgr, ldMgr.deviceMgr, ldMgr.clusterDataProxy, ldMgr.defaultTimeout)
216 if err := agent.start(nil, true); err != nil {
217 agent.stop(nil)
218 return err
219 }
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400220 ldMgr.logicalDeviceAgents.Store(agent.logicalDeviceId, agent)
khenaidoo297cd252019-02-07 22:10:23 -0500221 }
khenaidoo297cd252019-02-07 22:10:23 -0500222 }
223 // TODO: load the child device
224 return nil
225}
226
khenaidoo4d4802d2018-10-04 21:59:49 -0400227func (ldMgr *LogicalDeviceManager) deleteLogicalDevice(ctx context.Context, device *voltha.Device) error {
khenaidoo92e62c52018-10-03 14:02:54 -0400228 log.Debugw("deleting-logical-device", log.Fields{"deviceId": device.Id})
229 // Sanity check
230 if !device.Root {
khenaidoo2c6a0992019-04-29 13:46:56 -0400231 return errors.New("device-not-root")
khenaidoo92e62c52018-10-03 14:02:54 -0400232 }
233 logDeviceId := device.ParentId
234 if agent := ldMgr.getLogicalDeviceAgent(logDeviceId); agent != nil {
235 // Stop the logical device agent
khenaidoo4d4802d2018-10-04 21:59:49 -0400236 agent.stop(ctx)
khenaidoo92e62c52018-10-03 14:02:54 -0400237 //Remove the logical device agent from the Map
238 ldMgr.deleteLogicalDeviceAgent(logDeviceId)
Richard Jankowski199fd862019-03-18 14:49:51 -0400239 ldMgr.core.deviceOwnership.AbandonDevice(logDeviceId)
khenaidoo92e62c52018-10-03 14:02:54 -0400240 }
241
242 log.Debug("deleting-logical-device-ends")
243 return nil
244}
245
246func (ldMgr *LogicalDeviceManager) getLogicalDeviceId(device *voltha.Device) (*string, error) {
247 // Device can either be a parent or a child device
248 if device.Root {
249 // Parent device. The ID of a parent device is the logical device ID
250 return &device.ParentId, nil
251 }
252 // Device is child device
253 // retrieve parent device using child device ID
254 if parentDevice := ldMgr.deviceMgr.getParentDevice(device); parentDevice != nil {
255 return &parentDevice.ParentId, nil
256 }
257 return nil, status.Errorf(codes.NotFound, "%s", device.Id)
258}
259
khenaidoo3ab34882019-05-02 21:33:30 -0400260func (ldMgr *LogicalDeviceManager) getLogicalDeviceIdFromDeviceId(deviceId string) (*string, error) {
261 // Get the device
262 var device *voltha.Device
263 var err error
264 if device, err = ldMgr.deviceMgr.GetDevice(deviceId); err != nil {
265 return nil, err
266 }
267 return ldMgr.getLogicalDeviceId(device)
268}
269
khenaidoo19d7b632018-10-30 10:49:50 -0400270func (ldMgr *LogicalDeviceManager) getLogicalPortId(device *voltha.Device) (*voltha.LogicalPortId, error) {
271 // Get the logical device where this device is attached
272 var lDeviceId *string
273 var err error
274 if lDeviceId, err = ldMgr.getLogicalDeviceId(device); err != nil {
275 return nil, err
276 }
277 var lDevice *voltha.LogicalDevice
278 if lDevice, err = ldMgr.getLogicalDevice(*lDeviceId); err != nil {
279 return nil, err
280 }
281 // Go over list of ports
282 for _, port := range lDevice.Ports {
283 if port.DeviceId == device.Id {
284 return &voltha.LogicalPortId{Id: *lDeviceId, PortId: port.Id}, nil
285 }
286 }
287 return nil, status.Errorf(codes.NotFound, "%s", device.Id)
288}
289
khenaidoodd237172019-05-27 16:37:17 -0400290func (ldMgr *LogicalDeviceManager) ListLogicalDeviceFlows(ctx context.Context, id string) (*openflow_13.Flows, error) {
291 log.Debugw("ListLogicalDeviceFlows", log.Fields{"logicaldeviceid": id})
292 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
293 return agent.ListLogicalDeviceFlows()
294 }
295 return nil, status.Errorf(codes.NotFound, "%s", id)
296}
297
298func (ldMgr *LogicalDeviceManager) ListLogicalDeviceFlowGroups(ctx context.Context, id string) (*openflow_13.FlowGroups, error) {
299 log.Debugw("ListLogicalDeviceFlowGroups", log.Fields{"logicaldeviceid": id})
300 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
301 return agent.ListLogicalDeviceFlowGroups()
302 }
303 return nil, status.Errorf(codes.NotFound, "%s", id)
304}
305
khenaidoo19d7b632018-10-30 10:49:50 -0400306func (ldMgr *LogicalDeviceManager) ListLogicalDevicePorts(ctx context.Context, id string) (*voltha.LogicalPorts, error) {
307 log.Debugw("ListLogicalDevicePorts", log.Fields{"logicaldeviceid": id})
308 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
309 return agent.ListLogicalDevicePorts()
310 }
311 return nil, status.Errorf(codes.NotFound, "%s", id)
khenaidoo19d7b632018-10-30 10:49:50 -0400312}
313
314func (ldMgr *LogicalDeviceManager) getLogicalPort(lPortId *voltha.LogicalPortId) (*voltha.LogicalPort, error) {
315 // Get the logical device where this device is attached
316 var err error
317 var lDevice *voltha.LogicalDevice
318 if lDevice, err = ldMgr.getLogicalDevice(lPortId.Id); err != nil {
319 return nil, err
320 }
321 // Go over list of ports
322 for _, port := range lDevice.Ports {
323 if port.Id == lPortId.PortId {
324 return port, nil
325 }
326 }
Kent Hagerman0ab4cb22019-04-24 13:13:35 -0400327 return nil, status.Errorf(codes.NotFound, "%s-%s", lPortId.Id, lPortId.PortId)
khenaidoo19d7b632018-10-30 10:49:50 -0400328}
329
khenaidoo2c6a0992019-04-29 13:46:56 -0400330// updateLogicalPort sets up a logical port on the logical device based on the device port
331// information, if needed
332func (ldMgr *LogicalDeviceManager) updateLogicalPort(device *voltha.Device, port *voltha.Port) error {
333 if ldID, err := ldMgr.getLogicalDeviceId(device); err != nil || *ldID == "" {
334 // This is not an error as the logical device may not have been created at this time. In such a case,
335 // the ports will be created when the logical device is ready.
336 return nil
337 } else {
338 if agent := ldMgr.getLogicalDeviceAgent(*ldID); agent != nil {
339 if err := agent.updateLogicalPort(device, port); err != nil {
340 return err
341 }
342 }
343 }
344 return nil
345}
346
khenaidoofc1314d2019-03-14 09:34:21 -0400347// addLogicalPort sets up a logical port on the logical device based on the device port
348// information.
349func (ldMgr *LogicalDeviceManager) addLogicalPort(device *voltha.Device, port *voltha.Port) error {
Stephane Barbarie40fd3b22019-04-23 21:50:47 -0400350 if ldID, err := ldMgr.getLogicalDeviceId(device); err != nil || *ldID == "" {
khenaidoofc1314d2019-03-14 09:34:21 -0400351 // This is not an error as the logical device may not have been created at this time. In such a case,
352 // the ports will be created when the logical device is ready.
353 return nil
354 } else {
355 if agent := ldMgr.getLogicalDeviceAgent(*ldID); agent != nil {
356 if err := agent.addLogicalPort(device, port); err != nil {
357 return err
358 }
359 }
360 }
361 return nil
362}
363
khenaidoo0a822f92019-05-08 15:15:57 -0400364// deleteLogicalPort removes the logical port associated with a device
khenaidoo19d7b632018-10-30 10:49:50 -0400365func (ldMgr *LogicalDeviceManager) deleteLogicalPort(ctx context.Context, lPortId *voltha.LogicalPortId) error {
366 log.Debugw("deleting-logical-port", log.Fields{"LDeviceId": lPortId.Id})
367 // Get logical port
368 var logicalPort *voltha.LogicalPort
369 var err error
370 if logicalPort, err = ldMgr.getLogicalPort(lPortId); err != nil {
371 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": lPortId.PortId})
372 return err
373 }
khenaidoo92e62c52018-10-03 14:02:54 -0400374 // Sanity check
khenaidoo19d7b632018-10-30 10:49:50 -0400375 if logicalPort.RootPort {
khenaidoo2c6a0992019-04-29 13:46:56 -0400376 return errors.New("device-root")
khenaidoo92e62c52018-10-03 14:02:54 -0400377 }
khenaidoo19d7b632018-10-30 10:49:50 -0400378 if agent := ldMgr.getLogicalDeviceAgent(lPortId.Id); agent != nil {
khenaidoo0a822f92019-05-08 15:15:57 -0400379 if err := agent.deleteLogicalPort(logicalPort); err != nil {
380 log.Warnw("deleting-logicalport-failed", log.Fields{"LDeviceId": lPortId.Id, "error": err})
381 }
khenaidoo92e62c52018-10-03 14:02:54 -0400382 }
383
384 log.Debug("deleting-logical-port-ends")
385 return nil
386}
387
khenaidoo0a822f92019-05-08 15:15:57 -0400388// deleteLogicalPort removes the logical port associated with a child device
389func (ldMgr *LogicalDeviceManager) deleteLogicalPorts(deviceId string) error {
390 log.Debugw("deleting-logical-ports", log.Fields{"deviceId": deviceId})
391 // Get logical port
392 if ldId, err := ldMgr.getLogicalDeviceIdFromDeviceId(deviceId); err != nil {
khenaidoo0a822f92019-05-08 15:15:57 -0400393 return err
394 } else {
395 if agent := ldMgr.getLogicalDeviceAgent(*ldId); agent != nil {
396 if err = agent.deleteLogicalPorts(deviceId); err != nil {
397 log.Warnw("deleteLogicalPorts-failed", log.Fields{"ldeviceId": *ldId})
398 return err
399 }
400 }
401 }
402 log.Debug("deleting-logical-port-ends")
403 return nil
404}
405
khenaidoofc1314d2019-03-14 09:34:21 -0400406func (ldMgr *LogicalDeviceManager) setupUNILogicalPorts(ctx context.Context, childDevice *voltha.Device) error {
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400407 log.Debugw("setupUNILogicalPorts", log.Fields{"childDeviceId": childDevice.Id, "parentDeviceId": childDevice.ParentId, "current-data": childDevice})
khenaidoob9203542018-09-17 22:56:37 -0400408 // Sanity check
409 if childDevice.Root {
410 return errors.New("Device-root")
411 }
412
413 // Get the logical device id parent device
414 parentId := childDevice.ParentId
415 logDeviceId := ldMgr.deviceMgr.GetParentDeviceId(parentId)
416
khenaidoofc1314d2019-03-14 09:34:21 -0400417 log.Debugw("setupUNILogicalPorts", log.Fields{"logDeviceId": logDeviceId, "parentId": parentId})
khenaidoob9203542018-09-17 22:56:37 -0400418
khenaidoo3d3b8c22019-05-22 18:10:39 -0400419 if parentId == "" || logDeviceId == nil || *logDeviceId == "" {
khenaidoo2c6a0992019-04-29 13:46:56 -0400420 return errors.New("device-in-invalid-state")
khenaidoo5e677ae2019-02-28 17:26:29 -0500421 }
422
khenaidoob9203542018-09-17 22:56:37 -0400423 if agent := ldMgr.getLogicalDeviceAgent(*logDeviceId); agent != nil {
khenaidoofc1314d2019-03-14 09:34:21 -0400424 if err := agent.setupUNILogicalPorts(ctx, childDevice); err != nil {
khenaidoobcf205b2019-01-25 22:21:14 -0500425 return err
426 }
khenaidoob9203542018-09-17 22:56:37 -0400427 }
khenaidoo21d51152019-02-01 13:48:37 -0500428 return nil
khenaidoob9203542018-09-17 22:56:37 -0400429}
khenaidoo19d7b632018-10-30 10:49:50 -0400430
khenaidoo0a822f92019-05-08 15:15:57 -0400431func (ldMgr *LogicalDeviceManager) deleteAllLogicalPorts(device *voltha.Device) error {
432 log.Debugw("deleteAllLogicalPorts", log.Fields{"deviceId": device.Id})
433
434 var ldId *string
435 var err error
436 //Get the logical device Id for this device
437 if ldId, err = ldMgr.getLogicalDeviceId(device); err != nil {
438 log.Warnw("no-logical-device-found", log.Fields{"deviceId": device.Id, "error": err})
439 return err
440 }
441 if agent := ldMgr.getLogicalDeviceAgent(*ldId); agent != nil {
442 if err := agent.deleteAllLogicalPorts(device); err != nil {
443 return err
444 }
445 }
446 return nil
447}
448
khenaidoo3ab34882019-05-02 21:33:30 -0400449func (ldMgr *LogicalDeviceManager) updatePortsState(device *voltha.Device, state voltha.AdminState_AdminState) error {
Stephane Barbarieef6650d2019-07-18 12:15:09 -0400450 log.Debugw("updatePortsState", log.Fields{"deviceId": device.Id, "state": state, "current-data": device})
khenaidoo3ab34882019-05-02 21:33:30 -0400451
452 var ldId *string
453 var err error
454 //Get the logical device Id for this device
455 if ldId, err = ldMgr.getLogicalDeviceId(device); err != nil {
456 log.Warnw("no-logical-device-found", log.Fields{"deviceId": device.Id, "error": err})
457 return err
458 }
459 if agent := ldMgr.getLogicalDeviceAgent(*ldId); agent != nil {
460 if err := agent.updatePortsState(device, state); err != nil {
461 return err
462 }
463 }
464 return nil
465}
466
khenaidoo19d7b632018-10-30 10:49:50 -0400467func (ldMgr *LogicalDeviceManager) updateFlowTable(ctx context.Context, id string, flow *openflow_13.OfpFlowMod, ch chan interface{}) {
468 log.Debugw("updateFlowTable", log.Fields{"logicalDeviceId": id})
469 var res interface{}
470 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
471 res = agent.updateFlowTable(ctx, flow)
472 log.Debugw("updateFlowTable-result", log.Fields{"result": res})
473 } else {
474 res = status.Errorf(codes.NotFound, "%s", id)
475 }
476 sendAPIResponse(ctx, ch, res)
477}
478
Manikkaraj kb1a10922019-07-29 12:10:34 -0400479func (ldMgr *LogicalDeviceManager) updateMeterTable(ctx context.Context, id string, meter *openflow_13.OfpMeterMod, ch chan interface{}) {
480 log.Debugw("updateMeterTable", log.Fields{"logicalDeviceId": id})
481 var res interface{}
482 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
483 res = agent.updateMeterTable(ctx, meter)
484 log.Debugw("updateMeterTable-result", log.Fields{"result": res})
485 } else {
486 res = status.Errorf(codes.NotFound, "%s", id)
487 }
488 sendAPIResponse(ctx, ch, res)
489}
490
491func (ldMgr *LogicalDeviceManager) ListLogicalDeviceMeters(ctx context.Context, id string) (*openflow_13.Meters, error) {
492 log.Debugw("ListLogicalDeviceMeters", log.Fields{"logicalDeviceId": id})
493 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
494 return agent.ListLogicalDeviceMeters()
495 }
496 return nil, status.Errorf(codes.NotFound, "%s", id)
497}
khenaidoo19d7b632018-10-30 10:49:50 -0400498func (ldMgr *LogicalDeviceManager) updateGroupTable(ctx context.Context, id string, groupMod *openflow_13.OfpGroupMod, ch chan interface{}) {
499 log.Debugw("updateGroupTable", log.Fields{"logicalDeviceId": id})
500 var res interface{}
501 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
502 res = agent.updateGroupTable(ctx, groupMod)
503 log.Debugw("updateGroupTable-result", log.Fields{"result": res})
504 } else {
505 res = status.Errorf(codes.NotFound, "%s", id)
506 }
507 sendAPIResponse(ctx, ch, res)
508}
509
510func (ldMgr *LogicalDeviceManager) enableLogicalPort(ctx context.Context, id *voltha.LogicalPortId, ch chan interface{}) {
511 log.Debugw("enableLogicalPort", log.Fields{"logicalDeviceId": id})
512 var res interface{}
513 // Get logical port
514 var logicalPort *voltha.LogicalPort
515 var err error
516 if logicalPort, err = ldMgr.getLogicalPort(id); err != nil {
517 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": id.PortId})
518 res = err
519 }
520 if agent := ldMgr.getLogicalDeviceAgent(id.Id); agent != nil {
521 res = agent.enableLogicalPort(logicalPort)
522 log.Debugw("enableLogicalPort-result", log.Fields{"result": res})
523 } else {
524 res = status.Errorf(codes.NotFound, "%s", id.Id)
525 }
526 sendAPIResponse(ctx, ch, res)
527}
528
529func (ldMgr *LogicalDeviceManager) disableLogicalPort(ctx context.Context, id *voltha.LogicalPortId, ch chan interface{}) {
530 log.Debugw("disableLogicalPort", log.Fields{"logicalDeviceId": id})
531 var res interface{}
532 // Get logical port
533 var logicalPort *voltha.LogicalPort
534 var err error
535 if logicalPort, err = ldMgr.getLogicalPort(id); err != nil {
536 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": id.PortId})
537 res = err
538 }
539 if agent := ldMgr.getLogicalDeviceAgent(id.Id); agent != nil {
540 res = agent.disableLogicalPort(logicalPort)
541 log.Debugw("disableLogicalPort-result", log.Fields{"result": res})
542 } else {
543 res = status.Errorf(codes.NotFound, "%s", id.Id)
544 }
545 sendAPIResponse(ctx, ch, res)
546}
khenaidoofdbad6e2018-11-06 22:26:38 -0500547
khenaidoo43c82122018-11-22 18:38:28 -0500548func (ldMgr *LogicalDeviceManager) packetOut(packetOut *openflow_13.PacketOut) {
khenaidoofdbad6e2018-11-06 22:26:38 -0500549 log.Debugw("packetOut", log.Fields{"logicalDeviceId": packetOut.Id})
550 if agent := ldMgr.getLogicalDeviceAgent(packetOut.Id); agent != nil {
551 agent.packetOut(packetOut.PacketOut)
552 } else {
553 log.Error("logical-device-not-exist", log.Fields{"logicalDeviceId": packetOut.Id})
554 }
555}
556
khenaidoo297cd252019-02-07 22:10:23 -0500557func (ldMgr *LogicalDeviceManager) packetIn(logicalDeviceId string, port uint32, transactionId string, packet []byte) error {
khenaidoofdbad6e2018-11-06 22:26:38 -0500558 log.Debugw("packetIn", log.Fields{"logicalDeviceId": logicalDeviceId, "port": port})
559 if agent := ldMgr.getLogicalDeviceAgent(logicalDeviceId); agent != nil {
khenaidoo297cd252019-02-07 22:10:23 -0500560 agent.packetIn(port, transactionId, packet)
khenaidoofdbad6e2018-11-06 22:26:38 -0500561 } else {
562 log.Error("logical-device-not-exist", log.Fields{"logicalDeviceId": logicalDeviceId})
563 }
564 return nil
565}