blob: 9f88bc7322b2dddfd5eff85905346e777a38867c [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();
73
74 @Activate
75 protected void activate() {
Thomas Vachuska3afbc7f2016-02-01 15:55:38 -080076 AbstractProjectableModel.setDriverService(null, this);
alshabib77b88482015-04-07 15:47:50 -070077 log.info("Started");
78 }
79
80 @Deactivate
81 protected void deactivate() {
Thomas Vachuska3afbc7f2016-02-01 15:55:38 -080082 AbstractProjectableModel.setDriverService(this, null);
alshabib77b88482015-04-07 15:47:50 -070083 log.info("Stopped");
84 }
85
alshabib77b88482015-04-07 15:47:50 -070086 @Override
87 public Set<DriverProvider> getProviders() {
88 return ImmutableSet.copyOf(providers);
89 }
90
91 @Override
92 public void registerProvider(DriverProvider provider) {
93 provider.getDrivers().forEach(driver -> {
Thomas Vachuskabb062ec2016-01-25 16:49:47 -080094 Driver d = addDriver(driver);
alshabib77b88482015-04-07 15:47:50 -070095 driverByKey.put(key(driver.manufacturer(),
96 driver.hwVersion(),
Thomas Vachuskabb062ec2016-01-25 16:49:47 -080097 driver.swVersion()), d);
alshabib77b88482015-04-07 15:47:50 -070098 });
99 providers.add(provider);
100 }
101
102 @Override
103 public void unregisterProvider(DriverProvider provider) {
104 provider.getDrivers().forEach(driver -> {
Thomas Vachuskabb062ec2016-01-25 16:49:47 -0800105 removeDriver(driver);
alshabib77b88482015-04-07 15:47:50 -0700106 driverByKey.remove(key(driver.manufacturer(),
107 driver.hwVersion(),
108 driver.swVersion()));
109 });
110 providers.remove(provider);
111 }
112
113 @Override
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700114 public Set<Driver> getDrivers() {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900115 checkPermission(DRIVER_READ);
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700116 ImmutableSet.Builder<Driver> builder = ImmutableSet.builder();
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700117 drivers.values().forEach(builder::add);
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700118 return builder.build();
alshabib77b88482015-04-07 15:47:50 -0700119 }
120
121 @Override
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700122 public Set<Driver> getDrivers(Class<? extends Behaviour> withBehaviour) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900123 checkPermission(DRIVER_READ);
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700124 return drivers.values().stream()
125 .filter(d -> d.hasBehaviour(withBehaviour))
126 .collect(Collectors.toSet());
127 }
128
129 @Override
alshabib77b88482015-04-07 15:47:50 -0700130 public Driver getDriver(String driverName) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900131 checkPermission(DRIVER_READ);
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700132 return nullIsNotFound(drivers.get(driverName), NO_DRIVER);
alshabib77b88482015-04-07 15:47:50 -0700133 }
134
135 @Override
136 public Driver getDriver(String mfr, String hw, String sw) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900137 checkPermission(DRIVER_READ);
Changhoon Yoon541ef712015-05-23 17:18:34 +0900138
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700139 // First attempt a literal search.
140 Driver driver = driverByKey.get(key(mfr, hw, sw));
141 if (driver != null) {
142 return driver;
143 }
144
145 // Otherwise, sweep through the key space and attempt to match using
146 // regular expression matching.
147 Optional<Driver> optional = driverByKey.values().stream()
148 .filter(d -> matches(d, mfr, hw, sw)).findFirst();
149
150 // If no matching driver is found, return default.
Sho SHIMIZUef7e2902016-02-12 18:38:29 -0800151 return optional.orElse(drivers.get(DEFAULT));
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700152 }
153
154 // Matches the given driver using ERE matching against the given criteria.
155 private boolean matches(Driver d, String mfr, String hw, String sw) {
156 // TODO: consider pre-compiling the expressions in the future
157 return mfr.matches(d.manufacturer()) &&
158 hw.matches(d.hwVersion()) &&
159 sw.matches(d.swVersion());
alshabib77b88482015-04-07 15:47:50 -0700160 }
161
162 @Override
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700163 public Driver getDriver(DeviceId deviceId) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900164 checkPermission(DRIVER_READ);
Changhoon Yoon541ef712015-05-23 17:18:34 +0900165
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700166 Device device = nullIsNotFound(deviceService.getDevice(deviceId), NO_DEVICE);
167 String driverName = device.annotations().value(DRIVER);
168 if (driverName != null) {
HIGUCHI Yuta11d16092015-12-04 23:35:43 -0800169 try {
170 return getDriver(driverName);
171 } catch (ItemNotFoundException e) {
172 log.warn("Specified driver {} not found, falling back.", driverName);
173 }
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700174 }
HIGUCHI Yuta11d16092015-12-04 23:35:43 -0800175
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700176 return nullIsNotFound(getDriver(device.manufacturer(),
177 device.hwVersion(), device.swVersion()),
178 NO_DRIVER);
179 }
180
181 @Override
182 public DriverHandler createHandler(DeviceId deviceId, String... credentials) {
Changhoon Yoonb856b812015-08-10 03:47:19 +0900183 checkPermission(DRIVER_WRITE);
Thomas Vachuskaca88bb72015-04-08 19:38:02 -0700184 Driver driver = getDriver(deviceId);
Thomas Vachuska80b0a802015-07-17 08:43:30 -0700185 return new DefaultDriverHandler(new DefaultDriverData(driver, deviceId));
alshabib77b88482015-04-07 15:47:50 -0700186 }
187
Thomas Vachuska5c2f8132015-04-08 23:09:08 -0700188 // Produces a composite driver key using the specified components.
alshabib77b88482015-04-07 15:47:50 -0700189 private String key(String mfr, String hw, String sw) {
190 return String.format("%s-%s-%s", mfr, hw, sw);
191 }
alshabib77b88482015-04-07 15:47:50 -0700192}