blob: a3a29f0a9e82dc1a9fdd20d928146089472ffe28 [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"
khenaidoo19d7b632018-10-30 10:49:50 -040024 "github.com/opencord/voltha-go/protos/openflow_13"
khenaidoob9203542018-09-17 22:56:37 -040025 "github.com/opencord/voltha-go/protos/voltha"
26 "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
34 deviceMgr *DeviceManager
Richard Jankowskidbab94a2018-12-06 16:20:25 -050035 grpcNbiHdlr *APIHandler
khenaidoob9203542018-09-17 22:56:37 -040036 adapterProxy *AdapterProxy
khenaidoo43c82122018-11-22 18:38:28 -050037 kafkaICProxy *kafka.InterContainerProxy
khenaidoo9a468962018-09-19 15:33:13 -040038 clusterDataProxy *model.Proxy
khenaidoob9203542018-09-17 22:56:37 -040039 exitChannel chan int
40 lockLogicalDeviceAgentsMap sync.RWMutex
41}
42
khenaidoo43c82122018-11-22 18:38:28 -050043func newLogicalDeviceManager(deviceMgr *DeviceManager, kafkaICProxy *kafka.InterContainerProxy, cdProxy *model.Proxy) *LogicalDeviceManager {
khenaidoob9203542018-09-17 22:56:37 -040044 var logicalDeviceMgr LogicalDeviceManager
45 logicalDeviceMgr.exitChannel = make(chan int, 1)
46 logicalDeviceMgr.logicalDeviceAgents = make(map[string]*LogicalDeviceAgent)
47 logicalDeviceMgr.deviceMgr = deviceMgr
khenaidoo43c82122018-11-22 18:38:28 -050048 logicalDeviceMgr.kafkaICProxy = kafkaICProxy
khenaidoo9a468962018-09-19 15:33:13 -040049 logicalDeviceMgr.clusterDataProxy = cdProxy
khenaidoob9203542018-09-17 22:56:37 -040050 logicalDeviceMgr.lockLogicalDeviceAgentsMap = sync.RWMutex{}
51 return &logicalDeviceMgr
52}
53
Richard Jankowskidbab94a2018-12-06 16:20:25 -050054func (ldMgr *LogicalDeviceManager) setGrpcNbiHandler(grpcNbiHandler *APIHandler) {
55 ldMgr.grpcNbiHdlr = grpcNbiHandler
56}
57
khenaidoo4d4802d2018-10-04 21:59:49 -040058func (ldMgr *LogicalDeviceManager) start(ctx context.Context) {
khenaidoob9203542018-09-17 22:56:37 -040059 log.Info("starting-logical-device-manager")
60 log.Info("logical-device-manager-started")
61}
62
khenaidoo4d4802d2018-10-04 21:59:49 -040063func (ldMgr *LogicalDeviceManager) stop(ctx context.Context) {
khenaidoob9203542018-09-17 22:56:37 -040064 log.Info("stopping-logical-device-manager")
65 ldMgr.exitChannel <- 1
66 log.Info("logical-device-manager-stopped")
67}
68
khenaidoo19d7b632018-10-30 10:49:50 -040069func sendAPIResponse(ctx context.Context, ch chan interface{}, result interface{}) {
70 if ctx.Err() == nil {
71 // Returned response only of the ctx has not been cancelled/timeout/etc
72 // Channel is automatically closed when a context is Done
73 ch <- result
74 log.Debugw("sendResponse", log.Fields{"result": result})
75 } else {
76 // Should the transaction be reverted back?
77 log.Debugw("sendResponse-context-error", log.Fields{"context-error": ctx.Err()})
78 }
79}
80
khenaidoob9203542018-09-17 22:56:37 -040081func (ldMgr *LogicalDeviceManager) addLogicalDeviceAgentToMap(agent *LogicalDeviceAgent) {
82 ldMgr.lockLogicalDeviceAgentsMap.Lock()
83 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
84 if _, exist := ldMgr.logicalDeviceAgents[agent.logicalDeviceId]; !exist {
85 ldMgr.logicalDeviceAgents[agent.logicalDeviceId] = agent
86 }
87}
88
khenaidoo8c3303d2019-02-13 14:59:39 -050089// getLogicalDeviceAgent returns the logical device agent. If the device is not in memory then the device will
90// be loaded from dB and a logical device agent created to managed it.
khenaidoob9203542018-09-17 22:56:37 -040091func (ldMgr *LogicalDeviceManager) getLogicalDeviceAgent(logicalDeviceId string) *LogicalDeviceAgent {
92 ldMgr.lockLogicalDeviceAgentsMap.Lock()
khenaidoob9203542018-09-17 22:56:37 -040093 if agent, ok := ldMgr.logicalDeviceAgents[logicalDeviceId]; ok {
khenaidoo8c3303d2019-02-13 14:59:39 -050094 ldMgr.lockLogicalDeviceAgentsMap.Unlock()
khenaidoob9203542018-09-17 22:56:37 -040095 return agent
khenaidoo8c3303d2019-02-13 14:59:39 -050096 } else {
97 // Try to load into memory - loading will also create the logical device agent
98 ldMgr.lockLogicalDeviceAgentsMap.Unlock()
99 if err := ldMgr.load(logicalDeviceId); err == nil {
100 ldMgr.lockLogicalDeviceAgentsMap.Lock()
101 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
102 if agent, ok = ldMgr.logicalDeviceAgents[logicalDeviceId]; ok {
103 return agent
104 }
105 }
khenaidoob9203542018-09-17 22:56:37 -0400106 }
107 return nil
108}
109
khenaidoo92e62c52018-10-03 14:02:54 -0400110func (ldMgr *LogicalDeviceManager) deleteLogicalDeviceAgent(logicalDeviceId string) {
111 ldMgr.lockLogicalDeviceAgentsMap.Lock()
112 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
113 delete(ldMgr.logicalDeviceAgents, logicalDeviceId)
114}
115
khenaidoo8c3303d2019-02-13 14:59:39 -0500116// GetLogicalDevice provides a cloned most up to date logical device. If device is not in memory
117// it will be fetched from the dB
khenaidoob9203542018-09-17 22:56:37 -0400118func (ldMgr *LogicalDeviceManager) getLogicalDevice(id string) (*voltha.LogicalDevice, error) {
khenaidoo92e62c52018-10-03 14:02:54 -0400119 log.Debugw("getlogicalDevice", log.Fields{"logicaldeviceid": id})
120 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
khenaidoo19d7b632018-10-30 10:49:50 -0400121 return agent.GetLogicalDevice()
khenaidoob9203542018-09-17 22:56:37 -0400122 }
123 return nil, status.Errorf(codes.NotFound, "%s", id)
124}
125
126func (ldMgr *LogicalDeviceManager) listLogicalDevices() (*voltha.LogicalDevices, error) {
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500127 log.Debug("ListAllLogicalDevices")
khenaidoob9203542018-09-17 22:56:37 -0400128 result := &voltha.LogicalDevices{}
khenaidoo297cd252019-02-07 22:10:23 -0500129 if logicalDevices := ldMgr.clusterDataProxy.List("/logical_devices", 0, false, ""); logicalDevices != nil {
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500130 for _, logicalDevice := range logicalDevices.([]interface{}) {
131 if agent := ldMgr.getLogicalDeviceAgent(logicalDevice.(*voltha.LogicalDevice).Id); agent == nil {
132 agent = newLogicalDeviceAgent(
133 logicalDevice.(*voltha.LogicalDevice).Id,
134 logicalDevice.(*voltha.LogicalDevice).RootDeviceId,
135 ldMgr,
136 ldMgr.deviceMgr,
137 ldMgr.clusterDataProxy,
138 )
139 ldMgr.addLogicalDeviceAgentToMap(agent)
khenaidoo297cd252019-02-07 22:10:23 -0500140 go agent.start(nil, true)
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500141 }
142 result.Items = append(result.Items, logicalDevice.(*voltha.LogicalDevice))
khenaidoob9203542018-09-17 22:56:37 -0400143 }
144 }
145 return result, nil
146}
147
khenaidoo8c3303d2019-02-13 14:59:39 -0500148// List only logical devices that are in memory
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500149//func (ldMgr *LogicalDeviceManager) listLogicalDevices() (*voltha.LogicalDevices, error) {
150// log.Debug("listLogicalDevices")
151// result := &voltha.LogicalDevices{}
152// ldMgr.lockLogicalDeviceAgentsMap.Lock()
153// defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
154// for _, agent := range ldMgr.logicalDeviceAgents {
155// if lDevice, err := agent.GetLogicalDevice(); err == nil {
156// result.Items = append(result.Items, lDevice)
157// }
158// }
159// return result, nil
160//}
161
khenaidoo4d4802d2018-10-04 21:59:49 -0400162func (ldMgr *LogicalDeviceManager) createLogicalDevice(ctx context.Context, device *voltha.Device) (*string, error) {
khenaidoo92e62c52018-10-03 14:02:54 -0400163 log.Debugw("creating-logical-device", log.Fields{"deviceId": device.Id})
khenaidoob9203542018-09-17 22:56:37 -0400164 // Sanity check
165 if !device.Root {
166 return nil, errors.New("Device-not-root")
167 }
168
169 // Create a logical device agent - the logical device Id is based on the mac address of the device
170 // For now use the serial number - it may contain any combination of alphabetic characters and numbers,
171 // with length varying from eight characters to a maximum of 14 characters. Mac Address is part of oneof
172 // in the Device model. May need to be moved out.
173 macAddress := device.MacAddress
174 id := strings.Replace(macAddress, ":", "", -1)
khenaidoo92e62c52018-10-03 14:02:54 -0400175 if id == "" {
176 log.Errorw("mac-address-not-set", log.Fields{"deviceId": device.Id})
177 return nil, errors.New("mac-address-not-set")
178 }
179 log.Debugw("logical-device-id", log.Fields{"logicaldeviceId": id})
khenaidoob9203542018-09-17 22:56:37 -0400180
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500181 agent := newLogicalDeviceAgent(id, device.Id, ldMgr, ldMgr.deviceMgr, ldMgr.clusterDataProxy)
khenaidoob9203542018-09-17 22:56:37 -0400182 ldMgr.addLogicalDeviceAgentToMap(agent)
khenaidoo297cd252019-02-07 22:10:23 -0500183 go agent.start(ctx, false)
khenaidoob9203542018-09-17 22:56:37 -0400184
khenaidoo92e62c52018-10-03 14:02:54 -0400185 log.Debug("creating-logical-device-ends")
khenaidoob9203542018-09-17 22:56:37 -0400186 return &id, nil
187}
188
khenaidoo297cd252019-02-07 22:10:23 -0500189// load loads a logical device manager in memory
190func (ldMgr *LogicalDeviceManager) load(lDeviceId string) error {
khenaidoofc1314d2019-03-14 09:34:21 -0400191 //log.Debugw("loading-logical-device", log.Fields{"lDeviceId": lDeviceId})
192 log.Errorw("loading-logical-device", log.Fields{"lDeviceId": lDeviceId})
khenaidoo297cd252019-02-07 22:10:23 -0500193 // To prevent a race condition, let's hold the logical device agent map lock. This will prevent a loading and
194 // a create logical device callback from occurring at the same time.
195 ldMgr.lockLogicalDeviceAgentsMap.Lock()
196 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
197 if ldAgent, _ := ldMgr.logicalDeviceAgents[lDeviceId]; ldAgent == nil {
198 // Logical device not in memory - create a temp logical device Agent and let it load from memory
199 agent := newLogicalDeviceAgent(lDeviceId, "", ldMgr, ldMgr.deviceMgr, ldMgr.clusterDataProxy)
200 if err := agent.start(nil, true); err != nil {
khenaidoo8c3303d2019-02-13 14:59:39 -0500201 //agent.stop(nil)
khenaidoo297cd252019-02-07 22:10:23 -0500202 return err
203 }
204 ldMgr.logicalDeviceAgents[agent.logicalDeviceId] = agent
205 }
206 // TODO: load the child device
207 return nil
208}
209
khenaidoo4d4802d2018-10-04 21:59:49 -0400210func (ldMgr *LogicalDeviceManager) deleteLogicalDevice(ctx context.Context, device *voltha.Device) error {
khenaidoo92e62c52018-10-03 14:02:54 -0400211 log.Debugw("deleting-logical-device", log.Fields{"deviceId": device.Id})
212 // Sanity check
213 if !device.Root {
214 return errors.New("Device-not-root")
215 }
216 logDeviceId := device.ParentId
217 if agent := ldMgr.getLogicalDeviceAgent(logDeviceId); agent != nil {
218 // Stop the logical device agent
khenaidoo4d4802d2018-10-04 21:59:49 -0400219 agent.stop(ctx)
khenaidoo92e62c52018-10-03 14:02:54 -0400220 //Remove the logical device agent from the Map
221 ldMgr.deleteLogicalDeviceAgent(logDeviceId)
222 }
223
224 log.Debug("deleting-logical-device-ends")
225 return nil
226}
227
228func (ldMgr *LogicalDeviceManager) getLogicalDeviceId(device *voltha.Device) (*string, error) {
229 // Device can either be a parent or a child device
230 if device.Root {
231 // Parent device. The ID of a parent device is the logical device ID
232 return &device.ParentId, nil
233 }
234 // Device is child device
235 // retrieve parent device using child device ID
236 if parentDevice := ldMgr.deviceMgr.getParentDevice(device); parentDevice != nil {
237 return &parentDevice.ParentId, nil
238 }
239 return nil, status.Errorf(codes.NotFound, "%s", device.Id)
240}
241
khenaidoo19d7b632018-10-30 10:49:50 -0400242func (ldMgr *LogicalDeviceManager) getLogicalPortId(device *voltha.Device) (*voltha.LogicalPortId, error) {
243 // Get the logical device where this device is attached
244 var lDeviceId *string
245 var err error
246 if lDeviceId, err = ldMgr.getLogicalDeviceId(device); err != nil {
247 return nil, err
248 }
249 var lDevice *voltha.LogicalDevice
250 if lDevice, err = ldMgr.getLogicalDevice(*lDeviceId); err != nil {
251 return nil, err
252 }
253 // Go over list of ports
254 for _, port := range lDevice.Ports {
255 if port.DeviceId == device.Id {
256 return &voltha.LogicalPortId{Id: *lDeviceId, PortId: port.Id}, nil
257 }
258 }
259 return nil, status.Errorf(codes.NotFound, "%s", device.Id)
260}
261
262func (ldMgr *LogicalDeviceManager) ListLogicalDevicePorts(ctx context.Context, id string) (*voltha.LogicalPorts, error) {
263 log.Debugw("ListLogicalDevicePorts", log.Fields{"logicaldeviceid": id})
264 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
265 return agent.ListLogicalDevicePorts()
266 }
267 return nil, status.Errorf(codes.NotFound, "%s", id)
268
269}
270
271func (ldMgr *LogicalDeviceManager) getLogicalPort(lPortId *voltha.LogicalPortId) (*voltha.LogicalPort, error) {
272 // Get the logical device where this device is attached
273 var err error
274 var lDevice *voltha.LogicalDevice
275 if lDevice, err = ldMgr.getLogicalDevice(lPortId.Id); err != nil {
276 return nil, err
277 }
278 // Go over list of ports
279 for _, port := range lDevice.Ports {
280 if port.Id == lPortId.PortId {
281 return port, nil
282 }
283 }
284 return nil, status.Errorf(codes.NotFound, "%s-$s", lPortId.Id, lPortId.PortId)
285}
286
khenaidoofc1314d2019-03-14 09:34:21 -0400287// addLogicalPort sets up a logical port on the logical device based on the device port
288// information.
289func (ldMgr *LogicalDeviceManager) addLogicalPort(device *voltha.Device, port *voltha.Port) error {
290 if ldID, err := ldMgr.getLogicalDeviceId(device); err != nil {
291 // This is not an error as the logical device may not have been created at this time. In such a case,
292 // the ports will be created when the logical device is ready.
293 return nil
294 } else {
295 if agent := ldMgr.getLogicalDeviceAgent(*ldID); agent != nil {
296 if err := agent.addLogicalPort(device, port); err != nil {
297 return err
298 }
299 }
300 }
301 return nil
302}
303
khenaidoo19d7b632018-10-30 10:49:50 -0400304// deleteLogicalPort removes the logical port associated with a child device
305func (ldMgr *LogicalDeviceManager) deleteLogicalPort(ctx context.Context, lPortId *voltha.LogicalPortId) error {
306 log.Debugw("deleting-logical-port", log.Fields{"LDeviceId": lPortId.Id})
307 // Get logical port
308 var logicalPort *voltha.LogicalPort
309 var err error
310 if logicalPort, err = ldMgr.getLogicalPort(lPortId); err != nil {
311 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": lPortId.PortId})
312 return err
313 }
khenaidoo92e62c52018-10-03 14:02:54 -0400314 // Sanity check
khenaidoo19d7b632018-10-30 10:49:50 -0400315 if logicalPort.RootPort {
khenaidoo92e62c52018-10-03 14:02:54 -0400316 return errors.New("Device-root")
317 }
khenaidoo19d7b632018-10-30 10:49:50 -0400318 if agent := ldMgr.getLogicalDeviceAgent(lPortId.Id); agent != nil {
319 agent.deleteLogicalPort(logicalPort)
khenaidoo92e62c52018-10-03 14:02:54 -0400320 }
321
322 log.Debug("deleting-logical-port-ends")
323 return nil
324}
325
khenaidoofc1314d2019-03-14 09:34:21 -0400326func (ldMgr *LogicalDeviceManager) setupUNILogicalPorts(ctx context.Context, childDevice *voltha.Device) error {
327 log.Debugw("setupUNILogicalPorts", log.Fields{"childDeviceId": childDevice.Id, "parentDeviceId": childDevice.ParentId})
khenaidoob9203542018-09-17 22:56:37 -0400328 // Sanity check
329 if childDevice.Root {
330 return errors.New("Device-root")
331 }
332
333 // Get the logical device id parent device
334 parentId := childDevice.ParentId
335 logDeviceId := ldMgr.deviceMgr.GetParentDeviceId(parentId)
336
khenaidoofc1314d2019-03-14 09:34:21 -0400337 log.Debugw("setupUNILogicalPorts", log.Fields{"logDeviceId": logDeviceId, "parentId": parentId})
khenaidoob9203542018-09-17 22:56:37 -0400338
khenaidoo5e677ae2019-02-28 17:26:29 -0500339 if parentId == "" || logDeviceId == nil {
340 return errors.New("Device-in-invalid-state")
341 }
342
khenaidoob9203542018-09-17 22:56:37 -0400343 if agent := ldMgr.getLogicalDeviceAgent(*logDeviceId); agent != nil {
khenaidoofc1314d2019-03-14 09:34:21 -0400344 if err := agent.setupUNILogicalPorts(ctx, childDevice); err != nil {
khenaidoobcf205b2019-01-25 22:21:14 -0500345 return err
346 }
347 // Update the device routes - let it run in its own go routine as it can take time
348 go agent.updateRoutes()
khenaidoob9203542018-09-17 22:56:37 -0400349 }
khenaidoo21d51152019-02-01 13:48:37 -0500350 return nil
khenaidoob9203542018-09-17 22:56:37 -0400351}
khenaidoo19d7b632018-10-30 10:49:50 -0400352
353func (ldMgr *LogicalDeviceManager) updateFlowTable(ctx context.Context, id string, flow *openflow_13.OfpFlowMod, ch chan interface{}) {
354 log.Debugw("updateFlowTable", log.Fields{"logicalDeviceId": id})
355 var res interface{}
356 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
357 res = agent.updateFlowTable(ctx, flow)
358 log.Debugw("updateFlowTable-result", log.Fields{"result": res})
359 } else {
360 res = status.Errorf(codes.NotFound, "%s", id)
361 }
362 sendAPIResponse(ctx, ch, res)
363}
364
365func (ldMgr *LogicalDeviceManager) updateGroupTable(ctx context.Context, id string, groupMod *openflow_13.OfpGroupMod, ch chan interface{}) {
366 log.Debugw("updateGroupTable", log.Fields{"logicalDeviceId": id})
367 var res interface{}
368 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
369 res = agent.updateGroupTable(ctx, groupMod)
370 log.Debugw("updateGroupTable-result", log.Fields{"result": res})
371 } else {
372 res = status.Errorf(codes.NotFound, "%s", id)
373 }
374 sendAPIResponse(ctx, ch, res)
375}
376
377func (ldMgr *LogicalDeviceManager) enableLogicalPort(ctx context.Context, id *voltha.LogicalPortId, ch chan interface{}) {
378 log.Debugw("enableLogicalPort", log.Fields{"logicalDeviceId": id})
379 var res interface{}
380 // Get logical port
381 var logicalPort *voltha.LogicalPort
382 var err error
383 if logicalPort, err = ldMgr.getLogicalPort(id); err != nil {
384 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": id.PortId})
385 res = err
386 }
387 if agent := ldMgr.getLogicalDeviceAgent(id.Id); agent != nil {
388 res = agent.enableLogicalPort(logicalPort)
389 log.Debugw("enableLogicalPort-result", log.Fields{"result": res})
390 } else {
391 res = status.Errorf(codes.NotFound, "%s", id.Id)
392 }
393 sendAPIResponse(ctx, ch, res)
394}
395
396func (ldMgr *LogicalDeviceManager) disableLogicalPort(ctx context.Context, id *voltha.LogicalPortId, ch chan interface{}) {
397 log.Debugw("disableLogicalPort", log.Fields{"logicalDeviceId": id})
398 var res interface{}
399 // Get logical port
400 var logicalPort *voltha.LogicalPort
401 var err error
402 if logicalPort, err = ldMgr.getLogicalPort(id); err != nil {
403 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": id.PortId})
404 res = err
405 }
406 if agent := ldMgr.getLogicalDeviceAgent(id.Id); agent != nil {
407 res = agent.disableLogicalPort(logicalPort)
408 log.Debugw("disableLogicalPort-result", log.Fields{"result": res})
409 } else {
410 res = status.Errorf(codes.NotFound, "%s", id.Id)
411 }
412 sendAPIResponse(ctx, ch, res)
413}
khenaidoofdbad6e2018-11-06 22:26:38 -0500414
khenaidoo43c82122018-11-22 18:38:28 -0500415func (ldMgr *LogicalDeviceManager) packetOut(packetOut *openflow_13.PacketOut) {
khenaidoofdbad6e2018-11-06 22:26:38 -0500416 log.Debugw("packetOut", log.Fields{"logicalDeviceId": packetOut.Id})
417 if agent := ldMgr.getLogicalDeviceAgent(packetOut.Id); agent != nil {
418 agent.packetOut(packetOut.PacketOut)
419 } else {
420 log.Error("logical-device-not-exist", log.Fields{"logicalDeviceId": packetOut.Id})
421 }
422}
423
khenaidoo297cd252019-02-07 22:10:23 -0500424func (ldMgr *LogicalDeviceManager) packetIn(logicalDeviceId string, port uint32, transactionId string, packet []byte) error {
khenaidoofdbad6e2018-11-06 22:26:38 -0500425 log.Debugw("packetIn", log.Fields{"logicalDeviceId": logicalDeviceId, "port": port})
426 if agent := ldMgr.getLogicalDeviceAgent(logicalDeviceId); agent != nil {
khenaidoo297cd252019-02-07 22:10:23 -0500427 agent.packetIn(port, transactionId, packet)
khenaidoofdbad6e2018-11-06 22:26:38 -0500428 } else {
429 log.Error("logical-device-not-exist", log.Fields{"logicalDeviceId": logicalDeviceId})
430 }
431 return nil
432}