alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 1 | /* |
Brian O'Connor | 5ab426f | 2016-04-09 01:19:45 -0700 | [diff] [blame] | 2 | * Copyright 2015-present Open Networking Laboratory |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 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 | */ |
| 16 | package org.onosproject.net.driver.impl; |
| 17 | |
| 18 | import com.google.common.collect.ImmutableSet; |
| 19 | import com.google.common.collect.Maps; |
| 20 | import com.google.common.collect.Sets; |
| 21 | import org.apache.felix.scr.annotations.Activate; |
| 22 | import org.apache.felix.scr.annotations.Component; |
| 23 | import org.apache.felix.scr.annotations.Deactivate; |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 24 | import org.apache.felix.scr.annotations.Reference; |
| 25 | import org.apache.felix.scr.annotations.ReferenceCardinality; |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 26 | import org.apache.felix.scr.annotations.Service; |
HIGUCHI Yuta | 11d1609 | 2015-12-04 23:35:43 -0800 | [diff] [blame] | 27 | import org.onlab.util.ItemNotFoundException; |
Thomas Vachuska | 3afbc7f | 2016-02-01 15:55:38 -0800 | [diff] [blame] | 28 | import org.onosproject.net.AbstractProjectableModel; |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 29 | import org.onosproject.net.Device; |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 30 | import org.onosproject.net.DeviceId; |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 31 | import org.onosproject.net.device.DeviceService; |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 32 | import org.onosproject.net.driver.Behaviour; |
| 33 | import org.onosproject.net.driver.DefaultDriverData; |
| 34 | import org.onosproject.net.driver.DefaultDriverHandler; |
Thomas Vachuska | 5c2f813 | 2015-04-08 23:09:08 -0700 | [diff] [blame] | 35 | import org.onosproject.net.driver.DefaultDriverProvider; |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 36 | import org.onosproject.net.driver.Driver; |
| 37 | import org.onosproject.net.driver.DriverAdminService; |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 38 | import org.onosproject.net.driver.DriverHandler; |
| 39 | import org.onosproject.net.driver.DriverProvider; |
| 40 | import org.slf4j.Logger; |
| 41 | import org.slf4j.LoggerFactory; |
| 42 | |
| 43 | import java.util.Map; |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 44 | import java.util.Optional; |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 45 | import java.util.Set; |
Thomas Vachuska | 5c2f813 | 2015-04-08 23:09:08 -0700 | [diff] [blame] | 46 | import java.util.stream.Collectors; |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 47 | |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 48 | import static org.onlab.util.Tools.nullIsNotFound; |
| 49 | import static org.onosproject.net.AnnotationKeys.DRIVER; |
Changhoon Yoon | 541ef71 | 2015-05-23 17:18:34 +0900 | [diff] [blame] | 50 | import static org.onosproject.security.AppGuard.checkPermission; |
Changhoon Yoon | b856b81 | 2015-08-10 03:47:19 +0900 | [diff] [blame] | 51 | import static org.onosproject.security.AppPermission.Type.*; |
| 52 | |
Changhoon Yoon | 541ef71 | 2015-05-23 17:18:34 +0900 | [diff] [blame] | 53 | |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 54 | |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 55 | /** |
| 56 | * Manages inventory of device drivers. |
| 57 | */ |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 58 | @Component(immediate = true) |
| 59 | @Service |
Thomas Vachuska | 5c2f813 | 2015-04-08 23:09:08 -0700 | [diff] [blame] | 60 | public class DriverManager extends DefaultDriverProvider implements DriverAdminService { |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 61 | |
| 62 | private final Logger log = LoggerFactory.getLogger(getClass()); |
| 63 | |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 64 | 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 | |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 71 | private Set<DriverProvider> providers = Sets.newConcurrentHashSet(); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 72 | private Map<String, Driver> driverByKey = Maps.newConcurrentMap(); |
Thomas Vachuska | e7ea688 | 2016-07-22 10:22:46 -0700 | [diff] [blame] | 73 | private Map<String, Class<? extends Behaviour>> classes = Maps.newConcurrentMap(); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 74 | |
| 75 | @Activate |
| 76 | protected void activate() { |
Thomas Vachuska | 3afbc7f | 2016-02-01 15:55:38 -0800 | [diff] [blame] | 77 | AbstractProjectableModel.setDriverService(null, this); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 78 | log.info("Started"); |
| 79 | } |
| 80 | |
| 81 | @Deactivate |
| 82 | protected void deactivate() { |
Thomas Vachuska | 3afbc7f | 2016-02-01 15:55:38 -0800 | [diff] [blame] | 83 | AbstractProjectableModel.setDriverService(this, null); |
Thomas Vachuska | e7ea688 | 2016-07-22 10:22:46 -0700 | [diff] [blame] | 84 | providers.clear(); |
| 85 | driverByKey.clear(); |
| 86 | classes.clear(); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 87 | log.info("Stopped"); |
| 88 | } |
| 89 | |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 90 | @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 Vachuska | bb062ec | 2016-01-25 16:49:47 -0800 | [diff] [blame] | 98 | Driver d = addDriver(driver); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 99 | driverByKey.put(key(driver.manufacturer(), |
| 100 | driver.hwVersion(), |
Thomas Vachuska | bb062ec | 2016-01-25 16:49:47 -0800 | [diff] [blame] | 101 | driver.swVersion()), d); |
Thomas Vachuska | e7ea688 | 2016-07-22 10:22:46 -0700 | [diff] [blame] | 102 | 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 | }); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 107 | }); |
| 108 | providers.add(provider); |
| 109 | } |
| 110 | |
| 111 | @Override |
| 112 | public void unregisterProvider(DriverProvider provider) { |
| 113 | provider.getDrivers().forEach(driver -> { |
Thomas Vachuska | bb062ec | 2016-01-25 16:49:47 -0800 | [diff] [blame] | 114 | removeDriver(driver); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 115 | driverByKey.remove(key(driver.manufacturer(), |
| 116 | driver.hwVersion(), |
| 117 | driver.swVersion())); |
| 118 | }); |
| 119 | providers.remove(provider); |
| 120 | } |
| 121 | |
| 122 | @Override |
Thomas Vachuska | e7ea688 | 2016-07-22 10:22:46 -0700 | [diff] [blame] | 123 | public Class<? extends Behaviour> getBehaviourClass(String className) { |
| 124 | return classes.get(className); |
| 125 | } |
| 126 | |
| 127 | @Override |
Thomas Vachuska | 5c2f813 | 2015-04-08 23:09:08 -0700 | [diff] [blame] | 128 | public Set<Driver> getDrivers() { |
Changhoon Yoon | b856b81 | 2015-08-10 03:47:19 +0900 | [diff] [blame] | 129 | checkPermission(DRIVER_READ); |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 130 | ImmutableSet.Builder<Driver> builder = ImmutableSet.builder(); |
Thomas Vachuska | 5c2f813 | 2015-04-08 23:09:08 -0700 | [diff] [blame] | 131 | drivers.values().forEach(builder::add); |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 132 | return builder.build(); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 133 | } |
| 134 | |
| 135 | @Override |
Thomas Vachuska | 5c2f813 | 2015-04-08 23:09:08 -0700 | [diff] [blame] | 136 | public Set<Driver> getDrivers(Class<? extends Behaviour> withBehaviour) { |
Changhoon Yoon | b856b81 | 2015-08-10 03:47:19 +0900 | [diff] [blame] | 137 | checkPermission(DRIVER_READ); |
Thomas Vachuska | 5c2f813 | 2015-04-08 23:09:08 -0700 | [diff] [blame] | 138 | return drivers.values().stream() |
| 139 | .filter(d -> d.hasBehaviour(withBehaviour)) |
| 140 | .collect(Collectors.toSet()); |
| 141 | } |
| 142 | |
| 143 | @Override |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 144 | public Driver getDriver(String driverName) { |
Changhoon Yoon | b856b81 | 2015-08-10 03:47:19 +0900 | [diff] [blame] | 145 | checkPermission(DRIVER_READ); |
Thomas Vachuska | 5c2f813 | 2015-04-08 23:09:08 -0700 | [diff] [blame] | 146 | return nullIsNotFound(drivers.get(driverName), NO_DRIVER); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 147 | } |
| 148 | |
| 149 | @Override |
| 150 | public Driver getDriver(String mfr, String hw, String sw) { |
Changhoon Yoon | b856b81 | 2015-08-10 03:47:19 +0900 | [diff] [blame] | 151 | checkPermission(DRIVER_READ); |
Changhoon Yoon | 541ef71 | 2015-05-23 17:18:34 +0900 | [diff] [blame] | 152 | |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 153 | // 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 SHIMIZU | ef7e290 | 2016-02-12 18:38:29 -0800 | [diff] [blame] | 165 | return optional.orElse(drivers.get(DEFAULT)); |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 166 | } |
| 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()); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 174 | } |
| 175 | |
| 176 | @Override |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 177 | public Driver getDriver(DeviceId deviceId) { |
Changhoon Yoon | b856b81 | 2015-08-10 03:47:19 +0900 | [diff] [blame] | 178 | checkPermission(DRIVER_READ); |
Changhoon Yoon | 541ef71 | 2015-05-23 17:18:34 +0900 | [diff] [blame] | 179 | |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 180 | Device device = nullIsNotFound(deviceService.getDevice(deviceId), NO_DEVICE); |
| 181 | String driverName = device.annotations().value(DRIVER); |
| 182 | if (driverName != null) { |
HIGUCHI Yuta | 11d1609 | 2015-12-04 23:35:43 -0800 | [diff] [blame] | 183 | try { |
| 184 | return getDriver(driverName); |
| 185 | } catch (ItemNotFoundException e) { |
| 186 | log.warn("Specified driver {} not found, falling back.", driverName); |
| 187 | } |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 188 | } |
HIGUCHI Yuta | 11d1609 | 2015-12-04 23:35:43 -0800 | [diff] [blame] | 189 | |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 190 | 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 Yoon | b856b81 | 2015-08-10 03:47:19 +0900 | [diff] [blame] | 197 | checkPermission(DRIVER_WRITE); |
Thomas Vachuska | ca88bb7 | 2015-04-08 19:38:02 -0700 | [diff] [blame] | 198 | Driver driver = getDriver(deviceId); |
Thomas Vachuska | 80b0a80 | 2015-07-17 08:43:30 -0700 | [diff] [blame] | 199 | return new DefaultDriverHandler(new DefaultDriverData(driver, deviceId)); |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 200 | } |
| 201 | |
Thomas Vachuska | 5c2f813 | 2015-04-08 23:09:08 -0700 | [diff] [blame] | 202 | // Produces a composite driver key using the specified components. |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 203 | private String key(String mfr, String hw, String sw) { |
| 204 | return String.format("%s-%s-%s", mfr, hw, sw); |
| 205 | } |
alshabib | 77b8848 | 2015-04-07 15:47:50 -0700 | [diff] [blame] | 206 | } |