blob: 5f9cea2a9316343600c6009012ebb8ac8007e163 [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
35 adapterProxy *AdapterProxy
khenaidoo43c82122018-11-22 18:38:28 -050036 kafkaICProxy *kafka.InterContainerProxy
khenaidoo9a468962018-09-19 15:33:13 -040037 clusterDataProxy *model.Proxy
khenaidoob9203542018-09-17 22:56:37 -040038 exitChannel chan int
39 lockLogicalDeviceAgentsMap sync.RWMutex
40}
41
khenaidoo43c82122018-11-22 18:38:28 -050042func newLogicalDeviceManager(deviceMgr *DeviceManager, kafkaICProxy *kafka.InterContainerProxy, cdProxy *model.Proxy) *LogicalDeviceManager {
khenaidoob9203542018-09-17 22:56:37 -040043 var logicalDeviceMgr LogicalDeviceManager
44 logicalDeviceMgr.exitChannel = make(chan int, 1)
45 logicalDeviceMgr.logicalDeviceAgents = make(map[string]*LogicalDeviceAgent)
46 logicalDeviceMgr.deviceMgr = deviceMgr
khenaidoo43c82122018-11-22 18:38:28 -050047 logicalDeviceMgr.kafkaICProxy = kafkaICProxy
khenaidoo9a468962018-09-19 15:33:13 -040048 logicalDeviceMgr.clusterDataProxy = cdProxy
khenaidoob9203542018-09-17 22:56:37 -040049 logicalDeviceMgr.lockLogicalDeviceAgentsMap = sync.RWMutex{}
50 return &logicalDeviceMgr
51}
52
khenaidoo4d4802d2018-10-04 21:59:49 -040053func (ldMgr *LogicalDeviceManager) start(ctx context.Context) {
khenaidoob9203542018-09-17 22:56:37 -040054 log.Info("starting-logical-device-manager")
55 log.Info("logical-device-manager-started")
56}
57
khenaidoo4d4802d2018-10-04 21:59:49 -040058func (ldMgr *LogicalDeviceManager) stop(ctx context.Context) {
khenaidoob9203542018-09-17 22:56:37 -040059 log.Info("stopping-logical-device-manager")
60 ldMgr.exitChannel <- 1
61 log.Info("logical-device-manager-stopped")
62}
63
khenaidoo19d7b632018-10-30 10:49:50 -040064func sendAPIResponse(ctx context.Context, ch chan interface{}, result interface{}) {
65 if ctx.Err() == nil {
66 // Returned response only of the ctx has not been cancelled/timeout/etc
67 // Channel is automatically closed when a context is Done
68 ch <- result
69 log.Debugw("sendResponse", log.Fields{"result": result})
70 } else {
71 // Should the transaction be reverted back?
72 log.Debugw("sendResponse-context-error", log.Fields{"context-error": ctx.Err()})
73 }
74}
75
khenaidoob9203542018-09-17 22:56:37 -040076func (ldMgr *LogicalDeviceManager) addLogicalDeviceAgentToMap(agent *LogicalDeviceAgent) {
77 ldMgr.lockLogicalDeviceAgentsMap.Lock()
78 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
79 if _, exist := ldMgr.logicalDeviceAgents[agent.logicalDeviceId]; !exist {
80 ldMgr.logicalDeviceAgents[agent.logicalDeviceId] = agent
81 }
82}
83
84func (ldMgr *LogicalDeviceManager) getLogicalDeviceAgent(logicalDeviceId string) *LogicalDeviceAgent {
85 ldMgr.lockLogicalDeviceAgentsMap.Lock()
86 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
87 if agent, ok := ldMgr.logicalDeviceAgents[logicalDeviceId]; ok {
88 return agent
89 }
90 return nil
91}
92
khenaidoo92e62c52018-10-03 14:02:54 -040093func (ldMgr *LogicalDeviceManager) deleteLogicalDeviceAgent(logicalDeviceId string) {
94 ldMgr.lockLogicalDeviceAgentsMap.Lock()
95 defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
96 delete(ldMgr.logicalDeviceAgents, logicalDeviceId)
97}
98
khenaidoo19d7b632018-10-30 10:49:50 -040099// GetLogicalDevice provides a cloned most up to date logical device
khenaidoob9203542018-09-17 22:56:37 -0400100func (ldMgr *LogicalDeviceManager) getLogicalDevice(id string) (*voltha.LogicalDevice, error) {
khenaidoo92e62c52018-10-03 14:02:54 -0400101 log.Debugw("getlogicalDevice", log.Fields{"logicaldeviceid": id})
102 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
khenaidoo19d7b632018-10-30 10:49:50 -0400103 return agent.GetLogicalDevice()
khenaidoob9203542018-09-17 22:56:37 -0400104 }
105 return nil, status.Errorf(codes.NotFound, "%s", id)
106}
107
108func (ldMgr *LogicalDeviceManager) listLogicalDevices() (*voltha.LogicalDevices, error) {
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500109 log.Debug("ListAllLogicalDevices")
khenaidoob9203542018-09-17 22:56:37 -0400110 result := &voltha.LogicalDevices{}
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500111 if logicalDevices := ldMgr.clusterDataProxy.Get("/logical_devices", 0, false, ""); logicalDevices != nil {
112 for _, logicalDevice := range logicalDevices.([]interface{}) {
113 if agent := ldMgr.getLogicalDeviceAgent(logicalDevice.(*voltha.LogicalDevice).Id); agent == nil {
114 agent = newLogicalDeviceAgent(
115 logicalDevice.(*voltha.LogicalDevice).Id,
116 logicalDevice.(*voltha.LogicalDevice).RootDeviceId,
117 ldMgr,
118 ldMgr.deviceMgr,
119 ldMgr.clusterDataProxy,
120 )
121 ldMgr.addLogicalDeviceAgentToMap(agent)
122 go agent.start(nil)
123 }
124 result.Items = append(result.Items, logicalDevice.(*voltha.LogicalDevice))
khenaidoob9203542018-09-17 22:56:37 -0400125 }
126 }
127 return result, nil
128}
129
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500130//func (ldMgr *LogicalDeviceManager) listLogicalDevices() (*voltha.LogicalDevices, error) {
131// log.Debug("listLogicalDevices")
132// result := &voltha.LogicalDevices{}
133// ldMgr.lockLogicalDeviceAgentsMap.Lock()
134// defer ldMgr.lockLogicalDeviceAgentsMap.Unlock()
135// for _, agent := range ldMgr.logicalDeviceAgents {
136// if lDevice, err := agent.GetLogicalDevice(); err == nil {
137// result.Items = append(result.Items, lDevice)
138// }
139// }
140// return result, nil
141//}
142
khenaidoo4d4802d2018-10-04 21:59:49 -0400143func (ldMgr *LogicalDeviceManager) createLogicalDevice(ctx context.Context, device *voltha.Device) (*string, error) {
khenaidoo92e62c52018-10-03 14:02:54 -0400144 log.Debugw("creating-logical-device", log.Fields{"deviceId": device.Id})
khenaidoob9203542018-09-17 22:56:37 -0400145 // Sanity check
146 if !device.Root {
147 return nil, errors.New("Device-not-root")
148 }
149
150 // Create a logical device agent - the logical device Id is based on the mac address of the device
151 // For now use the serial number - it may contain any combination of alphabetic characters and numbers,
152 // with length varying from eight characters to a maximum of 14 characters. Mac Address is part of oneof
153 // in the Device model. May need to be moved out.
154 macAddress := device.MacAddress
155 id := strings.Replace(macAddress, ":", "", -1)
khenaidoo92e62c52018-10-03 14:02:54 -0400156 if id == "" {
157 log.Errorw("mac-address-not-set", log.Fields{"deviceId": device.Id})
158 return nil, errors.New("mac-address-not-set")
159 }
160 log.Debugw("logical-device-id", log.Fields{"logicaldeviceId": id})
khenaidoob9203542018-09-17 22:56:37 -0400161
Stephane Barbarie1ab43272018-12-08 21:42:13 -0500162 agent := newLogicalDeviceAgent(id, device.Id, ldMgr, ldMgr.deviceMgr, ldMgr.clusterDataProxy)
khenaidoob9203542018-09-17 22:56:37 -0400163 ldMgr.addLogicalDeviceAgentToMap(agent)
khenaidoo4d4802d2018-10-04 21:59:49 -0400164 go agent.start(ctx)
khenaidoob9203542018-09-17 22:56:37 -0400165
khenaidoo92e62c52018-10-03 14:02:54 -0400166 log.Debug("creating-logical-device-ends")
khenaidoob9203542018-09-17 22:56:37 -0400167 return &id, nil
168}
169
khenaidoo4d4802d2018-10-04 21:59:49 -0400170func (ldMgr *LogicalDeviceManager) deleteLogicalDevice(ctx context.Context, device *voltha.Device) error {
khenaidoo92e62c52018-10-03 14:02:54 -0400171 log.Debugw("deleting-logical-device", log.Fields{"deviceId": device.Id})
172 // Sanity check
173 if !device.Root {
174 return errors.New("Device-not-root")
175 }
176 logDeviceId := device.ParentId
177 if agent := ldMgr.getLogicalDeviceAgent(logDeviceId); agent != nil {
178 // Stop the logical device agent
khenaidoo4d4802d2018-10-04 21:59:49 -0400179 agent.stop(ctx)
khenaidoo92e62c52018-10-03 14:02:54 -0400180 //Remove the logical device agent from the Map
181 ldMgr.deleteLogicalDeviceAgent(logDeviceId)
182 }
183
184 log.Debug("deleting-logical-device-ends")
185 return nil
186}
187
188func (ldMgr *LogicalDeviceManager) getLogicalDeviceId(device *voltha.Device) (*string, error) {
189 // Device can either be a parent or a child device
190 if device.Root {
191 // Parent device. The ID of a parent device is the logical device ID
192 return &device.ParentId, nil
193 }
194 // Device is child device
195 // retrieve parent device using child device ID
196 if parentDevice := ldMgr.deviceMgr.getParentDevice(device); parentDevice != nil {
197 return &parentDevice.ParentId, nil
198 }
199 return nil, status.Errorf(codes.NotFound, "%s", device.Id)
200}
201
khenaidoo19d7b632018-10-30 10:49:50 -0400202func (ldMgr *LogicalDeviceManager) getLogicalPortId(device *voltha.Device) (*voltha.LogicalPortId, error) {
203 // Get the logical device where this device is attached
204 var lDeviceId *string
205 var err error
206 if lDeviceId, err = ldMgr.getLogicalDeviceId(device); err != nil {
207 return nil, err
208 }
209 var lDevice *voltha.LogicalDevice
210 if lDevice, err = ldMgr.getLogicalDevice(*lDeviceId); err != nil {
211 return nil, err
212 }
213 // Go over list of ports
214 for _, port := range lDevice.Ports {
215 if port.DeviceId == device.Id {
216 return &voltha.LogicalPortId{Id: *lDeviceId, PortId: port.Id}, nil
217 }
218 }
219 return nil, status.Errorf(codes.NotFound, "%s", device.Id)
220}
221
222func (ldMgr *LogicalDeviceManager) ListLogicalDevicePorts(ctx context.Context, id string) (*voltha.LogicalPorts, error) {
223 log.Debugw("ListLogicalDevicePorts", log.Fields{"logicaldeviceid": id})
224 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
225 return agent.ListLogicalDevicePorts()
226 }
227 return nil, status.Errorf(codes.NotFound, "%s", id)
228
229}
230
231func (ldMgr *LogicalDeviceManager) getLogicalPort(lPortId *voltha.LogicalPortId) (*voltha.LogicalPort, error) {
232 // Get the logical device where this device is attached
233 var err error
234 var lDevice *voltha.LogicalDevice
235 if lDevice, err = ldMgr.getLogicalDevice(lPortId.Id); err != nil {
236 return nil, err
237 }
238 // Go over list of ports
239 for _, port := range lDevice.Ports {
240 if port.Id == lPortId.PortId {
241 return port, nil
242 }
243 }
244 return nil, status.Errorf(codes.NotFound, "%s-$s", lPortId.Id, lPortId.PortId)
245}
246
247// deleteLogicalPort removes the logical port associated with a child device
248func (ldMgr *LogicalDeviceManager) deleteLogicalPort(ctx context.Context, lPortId *voltha.LogicalPortId) error {
249 log.Debugw("deleting-logical-port", log.Fields{"LDeviceId": lPortId.Id})
250 // Get logical port
251 var logicalPort *voltha.LogicalPort
252 var err error
253 if logicalPort, err = ldMgr.getLogicalPort(lPortId); err != nil {
254 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": lPortId.PortId})
255 return err
256 }
khenaidoo92e62c52018-10-03 14:02:54 -0400257 // Sanity check
khenaidoo19d7b632018-10-30 10:49:50 -0400258 if logicalPort.RootPort {
khenaidoo92e62c52018-10-03 14:02:54 -0400259 return errors.New("Device-root")
260 }
khenaidoo19d7b632018-10-30 10:49:50 -0400261 if agent := ldMgr.getLogicalDeviceAgent(lPortId.Id); agent != nil {
262 agent.deleteLogicalPort(logicalPort)
khenaidoo92e62c52018-10-03 14:02:54 -0400263 }
264
265 log.Debug("deleting-logical-port-ends")
266 return nil
267}
268
khenaidoo4d4802d2018-10-04 21:59:49 -0400269func (ldMgr *LogicalDeviceManager) addUNILogicalPort(ctx context.Context, childDevice *voltha.Device) error {
khenaidoo92e62c52018-10-03 14:02:54 -0400270 log.Debugw("AddUNILogicalPort", log.Fields{"deviceId": childDevice.Id})
khenaidoob9203542018-09-17 22:56:37 -0400271 // Sanity check
272 if childDevice.Root {
273 return errors.New("Device-root")
274 }
275
276 // Get the logical device id parent device
277 parentId := childDevice.ParentId
278 logDeviceId := ldMgr.deviceMgr.GetParentDeviceId(parentId)
279
khenaidoo92e62c52018-10-03 14:02:54 -0400280 log.Debugw("AddUNILogicalPort", log.Fields{"logDeviceId": logDeviceId, "parentId": parentId})
khenaidoob9203542018-09-17 22:56:37 -0400281
282 if agent := ldMgr.getLogicalDeviceAgent(*logDeviceId); agent != nil {
khenaidoo19d7b632018-10-30 10:49:50 -0400283 return agent.addUNILogicalPort(ctx, childDevice)
khenaidoob9203542018-09-17 22:56:37 -0400284 }
285 return status.Errorf(codes.NotFound, "%s", childDevice.Id)
286}
khenaidoo19d7b632018-10-30 10:49:50 -0400287
288func (ldMgr *LogicalDeviceManager) updateFlowTable(ctx context.Context, id string, flow *openflow_13.OfpFlowMod, ch chan interface{}) {
289 log.Debugw("updateFlowTable", log.Fields{"logicalDeviceId": id})
290 var res interface{}
291 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
292 res = agent.updateFlowTable(ctx, flow)
293 log.Debugw("updateFlowTable-result", log.Fields{"result": res})
294 } else {
295 res = status.Errorf(codes.NotFound, "%s", id)
296 }
297 sendAPIResponse(ctx, ch, res)
298}
299
300func (ldMgr *LogicalDeviceManager) updateGroupTable(ctx context.Context, id string, groupMod *openflow_13.OfpGroupMod, ch chan interface{}) {
301 log.Debugw("updateGroupTable", log.Fields{"logicalDeviceId": id})
302 var res interface{}
303 if agent := ldMgr.getLogicalDeviceAgent(id); agent != nil {
304 res = agent.updateGroupTable(ctx, groupMod)
305 log.Debugw("updateGroupTable-result", log.Fields{"result": res})
306 } else {
307 res = status.Errorf(codes.NotFound, "%s", id)
308 }
309 sendAPIResponse(ctx, ch, res)
310}
311
312func (ldMgr *LogicalDeviceManager) enableLogicalPort(ctx context.Context, id *voltha.LogicalPortId, ch chan interface{}) {
313 log.Debugw("enableLogicalPort", log.Fields{"logicalDeviceId": id})
314 var res interface{}
315 // Get logical port
316 var logicalPort *voltha.LogicalPort
317 var err error
318 if logicalPort, err = ldMgr.getLogicalPort(id); err != nil {
319 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": id.PortId})
320 res = err
321 }
322 if agent := ldMgr.getLogicalDeviceAgent(id.Id); agent != nil {
323 res = agent.enableLogicalPort(logicalPort)
324 log.Debugw("enableLogicalPort-result", log.Fields{"result": res})
325 } else {
326 res = status.Errorf(codes.NotFound, "%s", id.Id)
327 }
328 sendAPIResponse(ctx, ch, res)
329}
330
331func (ldMgr *LogicalDeviceManager) disableLogicalPort(ctx context.Context, id *voltha.LogicalPortId, ch chan interface{}) {
332 log.Debugw("disableLogicalPort", log.Fields{"logicalDeviceId": id})
333 var res interface{}
334 // Get logical port
335 var logicalPort *voltha.LogicalPort
336 var err error
337 if logicalPort, err = ldMgr.getLogicalPort(id); err != nil {
338 log.Debugw("no-logical-device-port-present", log.Fields{"logicalPortId": id.PortId})
339 res = err
340 }
341 if agent := ldMgr.getLogicalDeviceAgent(id.Id); agent != nil {
342 res = agent.disableLogicalPort(logicalPort)
343 log.Debugw("disableLogicalPort-result", log.Fields{"result": res})
344 } else {
345 res = status.Errorf(codes.NotFound, "%s", id.Id)
346 }
347 sendAPIResponse(ctx, ch, res)
348}
khenaidoofdbad6e2018-11-06 22:26:38 -0500349
khenaidoo43c82122018-11-22 18:38:28 -0500350func (ldMgr *LogicalDeviceManager) packetOut(packetOut *openflow_13.PacketOut) {
khenaidoofdbad6e2018-11-06 22:26:38 -0500351 log.Debugw("packetOut", log.Fields{"logicalDeviceId": packetOut.Id})
352 if agent := ldMgr.getLogicalDeviceAgent(packetOut.Id); agent != nil {
353 agent.packetOut(packetOut.PacketOut)
354 } else {
355 log.Error("logical-device-not-exist", log.Fields{"logicalDeviceId": packetOut.Id})
356 }
357}
358
359func (ldMgr *LogicalDeviceManager) packetIn(logicalDeviceId string, port uint32, packet []byte) error {
360 log.Debugw("packetIn", log.Fields{"logicalDeviceId": logicalDeviceId, "port": port})
361 if agent := ldMgr.getLogicalDeviceAgent(logicalDeviceId); agent != nil {
362 agent.packetIn(port, packet)
363 } else {
364 log.Error("logical-device-not-exist", log.Fields{"logicalDeviceId": logicalDeviceId})
365 }
366 return nil
367}