blob: 9718610ac39364f1f7a81ec084c2e1d92fb5937f [file] [log] [blame]
alshabib77b88482015-04-07 15:47:50 -07001/*
Brian O'Connor5ab426f2016-04-09 01:19:45 -07002 * Copyright 2015-present Open Networking Laboratory
alshabib77b88482015-04-07 15:47:50 -07003 *
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 org.onosproject.net.driver.impl;
17
18import com.google.common.collect.ImmutableSet;
19import com.google.common.collect.Maps;
20import com.google.common.collect.Sets;
21import org.apache.felix.scr.annotations.Activate;
22import org.apache.felix.scr.annotations.Component;
23import org.apache.felix.scr.annotations.Deactivate;
Thomas Vachuskaca88bb72015-04-08 19:38:02 -070024import org.apache.felix.scr.annotations.Reference;
25import org.apache.felix.scr.annotations.ReferenceCardinality;
alshabib77b88482015-04-07 15:47:50 -070026import org.apache.felix.scr.annotations.Service;
HIGUCHI Yuta11d16092015-12-04 23:35:43 -080027import org.onlab.util.ItemNotFoundException;
Thomas Vachuska3afbc7f2016-02-01 15:55:38 -080028import org.onosproject.net.AbstractProjectableModel;
Thomas Vachuskaca88bb72015-04-08 19:38:02 -070029import org.onosproject.net.Device;
alshabib77b88482015-04-07 15:47:50 -070030import org.onosproject.net.DeviceId;
Thomas Vachuskaca88bb72015-04-08 19:38:02 -070031import org.onosproject.net.device.DeviceService;
alshabib77b88482015-04-07 15:47:50 -070032import org.onosproject.net.driver.Behaviour;
33import org.onosproject.net.driver.DefaultDriverData;
34import org.onosproject.net.driver.DefaultDriverHandler;
Thomas Vachuska5c2f8132015-04-08 23:09:08 -070035import org.onosproject.net.driver.DefaultDriverProvider;
alshabib77b88482015-04-07 15:47:50 -070036import org.onosproject.net.driver.Driver;
37import org.onosproject.net.driver.DriverAdminService;
alshabib77b88482015-04-07 15:47:50 -070038import org.onosproject.net.driver.DriverHandler;
39import org.onosproject.net.driver.DriverProvider;
40import org.slf4j.Logger;
41import org.slf4j.LoggerFactory;
42
43import java.util.Map;
Thomas Vachuskaca88bb72015-04-08 19:38:02 -070044import java.util.Optional;
alshabib77b88482015-04-07 15:47:50 -070045import java.util.Set;
Thomas Vachuska5c2f8132015-04-08 23:09:08 -070046import java.util.stream.Collectors;
alshabib77b88482015-04-07 15:47:50 -070047
Thomas Vachuskaca88bb72015-04-08 19:38:02 -070048import static org.onlab.util.Tools.nullIsNotFound;
49import static org.onosproject.net.AnnotationKeys.DRIVER;
Changhoon Yoon541ef712015-05-23 17:18:34 +090050import static org.onosproject.security.AppGuard.checkPermission;
Changhoon Yoonb856b812015-08-10 03:47:19 +090051import static org.onosproject.security.AppPermission.Type.*;
52
Changhoon Yoon541ef712015-05-23 17:18:34 +090053
alshabib77b88482015-04-07 15:47:50 -070054
Thomas Vachuskaca88bb72015-04-08 19:38:02 -070055/**
56 * Manages inventory of device drivers.
57 */
alshabib77b88482015-04-07 15:47:50 -070058@Component(immediate = true)
59@Service
Thomas Vachuska5c2f8132015-04-08 23:09:08 -070060public class DriverManager extends DefaultDriverProvider implements DriverAdminService {
alshabib77b88482015-04-07 15:47:50 -070061
62 private final Logger log = LoggerFactory.getLogger(getClass());
63
Thomas Vachuskaca88bb72015-04-08 19:38:02 -070064 private static final String NO_DRIVER = "Driver not found";
65 private static final String NO_DEVICE = "Device not found";
66 private static final String DEFAULT = "default";
67
68 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
69 protected DeviceService deviceService;
70
alshabib77b88482015-04-07 15:47:50 -070071 private Set<DriverProvider> providers = Sets.newConcurrentHashSet();
alshabib77b88482015-04-07 15:47:50 -070072 private Map<String, Driver> driverByKey = Maps.newConcurrentMap();
Thomas Vachuskae7ea6882016-07-22 10:22:46 -070073 private Map<String, Class<? extends Behaviour>> classes = Maps.newConcurrentMap();
alshabib77b88482015-04-07 15:47:50 -070074
75 @Activate
76 protected void activate() {
Thomas Vachuska3afbc7f2016-02-01 15:55:38 -080077 AbstractProjectableModel.setDriverService(null, this);
alshabib77b88482015-04-07 15:47:50 -070078 log.info("Started");
79 }
80
81 @Deactivate
82 protected void deactivate() {
Thomas Vachuska3afbc7f2016-02-01 15:55:38 -080083 AbstractProjectableModel.setDriverService(this, null);
Thomas Vachuskae7ea6882016-07-22 10:22:46 -070084 providers.clear();
85 driverByKey.clear();
86 classes.clear();
alshabib77b88482015-04-07 15:47:50 -070087 log.info("Stopped");
88 }
89
alshabib77b88482015-04-07 15:47:50 -070090 @Override
91 public Set<DriverProvider> getProviders() {
92 return ImmutableSet.copyOf(providers);
93 }
94
95 @Override
96 public void registerProvider(DriverProvider provider) {
97 provider.getDrivers().forEach(driver -> {
Thomas Vachuskabb062ec2016-01-25 16:49:47 -080098 Driver d = addDriver(driver);
alshabib77b88482015-04-07 15:47:50 -070099 driverByKey.put(key(driver.manufacturer(),
100 driver.hwVersion(),
Thomas Vachuskabb062ec2016-01-25 16:49:47 -0800101 driver.swVersion()), d);
Thomas Vachuskae7ea6882016-07-22 10:22:46 -0700102 d.behaviours().forEach(b -> {
103 Class<? extends Behaviour> implementation = d.implementation(b);
104 classes.put(b.getName(), b);
105 classes.put(implementation.getName(), implementation);
106 });
alshabib77b88482015-04-07 15:47:50 -0700107 });
108 providers.add(provider);
109 }
110
111 @Override
112 public void unregisterProvider(DriverProvider provider) {
113 provider.getDrivers().forEach(driver -> {
Thomas Vachuskabb062ec2016-01-25 16:49:47 -0800114 removeDriver(driver);
alshabib77b88482015-04-07 15:47:50 -0700115 driverByKey.remove(key(driver.manufacturer(),
116 driver.hwVersion(),
117 driver.swVersion()));
118 });
119 providers.remove(provider);
120 }
121
122 @Override
Thomas Vachuskae7ea6882016-07-22 10:22:46 -0700123 public Class<? extends Behaviour> getBehaviourClass(String className) {
124 return classes.get(className);
125 }
126
127 @Override
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700128 public Set<Driver> getDrivers() {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900129 checkPermission(DRIVER_READ);
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700130 ImmutableSet.Builder<Driver> builder = ImmutableSet.builder();
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700131 drivers.values().forEach(builder::add);
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700132 return builder.build();
alshabib77b88482015-04-07 15:47:50 -0700133 }
134
135 @Override
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700136 public Set<Driver> getDrivers(Class<? extends Behaviour> withBehaviour) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900137 checkPermission(DRIVER_READ);
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700138 return drivers.values().stream()
139 .filter(d -> d.hasBehaviour(withBehaviour))
140 .collect(Collectors.toSet());
141 }
142
143 @Override
alshabib77b88482015-04-07 15:47:50 -0700144 public Driver getDriver(String driverName) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900145 checkPermission(DRIVER_READ);
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700146 return nullIsNotFound(drivers.get(driverName), NO_DRIVER);
alshabib77b88482015-04-07 15:47:50 -0700147 }
148
149 @Override
150 public Driver getDriver(String mfr, String hw, String sw) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900151 checkPermission(DRIVER_READ);
Changhoon Yoon541ef712015-05-23 17:18:34 +0900152
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700153 // First attempt a literal search.
154 Driver driver = driverByKey.get(key(mfr, hw, sw));
155 if (driver != null) {
156 return driver;
157 }
158
159 // Otherwise, sweep through the key space and attempt to match using
160 // regular expression matching.
161 Optional<Driver> optional = driverByKey.values().stream()
162 .filter(d -> matches(d, mfr, hw, sw)).findFirst();
163
164 // If no matching driver is found, return default.
Sho SHIMIZUef7e2902016-02-12 18:38:29 -0800165 return optional.orElse(drivers.get(DEFAULT));
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700166 }
167
168 // Matches the given driver using ERE matching against the given criteria.
169 private boolean matches(Driver d, String mfr, String hw, String sw) {
170 // TODO: consider pre-compiling the expressions in the future
171 return mfr.matches(d.manufacturer()) &&
172 hw.matches(d.hwVersion()) &&
173 sw.matches(d.swVersion());
alshabib77b88482015-04-07 15:47:50 -0700174 }
175
176 @Override
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700177 public Driver getDriver(DeviceId deviceId) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900178 checkPermission(DRIVER_READ);
Changhoon Yoon541ef712015-05-23 17:18:34 +0900179
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700180 Device device = nullIsNotFound(deviceService.getDevice(deviceId), NO_DEVICE);
181 String driverName = device.annotations().value(DRIVER);
182 if (driverName != null) {
HIGUCHI Yuta11d16092015-12-04 23:35:43 -0800183 try {
184 return getDriver(driverName);
185 } catch (ItemNotFoundException e) {
186 log.warn("Specified driver {} not found, falling back.", driverName);
187 }
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700188 }
HIGUCHI Yuta11d16092015-12-04 23:35:43 -0800189
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700190 return nullIsNotFound(getDriver(device.manufacturer(),
191 device.hwVersion(), device.swVersion()),
192 NO_DRIVER);
193 }
194
195 @Override
196 public DriverHandler createHandler(DeviceId deviceId, String... credentials) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900197 checkPermission(DRIVER_WRITE);
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700198 Driver driver = getDriver(deviceId);
Thomas Vachuska80b0a802015-07-17 08:43:30 -0700199 return new DefaultDriverHandler(new DefaultDriverData(driver, deviceId));
alshabib77b88482015-04-07 15:47:50 -0700200 }
201
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700202 // Produces a composite driver key using the specified components.
alshabib77b88482015-04-07 15:47:50 -0700203 private String key(String mfr, String hw, String sw) {
204 return String.format("%s-%s-%s", mfr, hw, sw);
205 }
alshabib77b88482015-04-07 15:47:50 -0700206}