2 * Copyright 2014-2015 Open Networking Laboratory
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
8 * http://www.apache.org/licenses/LICENSE-2.0
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.
16 package org.onosproject.provider.of.flow.impl;
18 import com.google.common.collect.Lists;
19 import org.onlab.packet.Ip4Address;
20 import org.onlab.packet.Ip4Prefix;
21 import org.onlab.packet.Ip6Address;
22 import org.onlab.packet.Ip6Prefix;
23 import org.onlab.packet.MacAddress;
24 import org.onlab.packet.MplsLabel;
25 import org.onlab.packet.TpPort;
26 import org.onlab.packet.VlanId;
27 import org.onosproject.core.DefaultGroupId;
28 import org.onosproject.net.DeviceId;
29 import org.onosproject.net.Lambda;
30 import org.onosproject.net.PortNumber;
31 import org.onosproject.net.driver.DefaultDriverData;
32 import org.onosproject.net.driver.DefaultDriverHandler;
33 import org.onosproject.net.driver.Driver;
34 import org.onosproject.net.driver.DriverHandler;
35 import org.onosproject.net.driver.DriverService;
36 import org.onosproject.net.flow.DefaultFlowEntry;
37 import org.onosproject.net.flow.DefaultFlowRule;
38 import org.onosproject.net.flow.DefaultTrafficSelector;
39 import org.onosproject.net.flow.DefaultTrafficTreatment;
40 import org.onosproject.net.flow.FlowEntry;
41 import org.onosproject.net.flow.FlowEntry.FlowEntryState;
42 import org.onosproject.net.flow.FlowRule;
43 import org.onosproject.net.flow.TrafficSelector;
44 import org.onosproject.net.flow.TrafficTreatment;
45 import org.onosproject.net.flow.instructions.Instructions;
46 import org.onosproject.openflow.controller.Dpid;
47 import org.onosproject.openflow.controller.ExtensionInterpreter;
48 import org.projectfloodlight.openflow.protocol.OFFlowMod;
49 import org.projectfloodlight.openflow.protocol.OFFlowRemoved;
50 import org.projectfloodlight.openflow.protocol.OFFlowStatsEntry;
51 import org.projectfloodlight.openflow.protocol.action.OFAction;
52 import org.projectfloodlight.openflow.protocol.action.OFActionCircuit;
53 import org.projectfloodlight.openflow.protocol.action.OFActionExperimenter;
54 import org.projectfloodlight.openflow.protocol.action.OFActionGroup;
55 import org.projectfloodlight.openflow.protocol.action.OFActionOutput;
56 import org.projectfloodlight.openflow.protocol.action.OFActionPopMpls;
57 import org.projectfloodlight.openflow.protocol.action.OFActionSetDlDst;
58 import org.projectfloodlight.openflow.protocol.action.OFActionSetDlSrc;
59 import org.projectfloodlight.openflow.protocol.action.OFActionSetField;
60 import org.projectfloodlight.openflow.protocol.action.OFActionSetNwDst;
61 import org.projectfloodlight.openflow.protocol.action.OFActionSetNwSrc;
62 import org.projectfloodlight.openflow.protocol.action.OFActionSetQueue;
63 import org.projectfloodlight.openflow.protocol.action.OFActionSetVlanPcp;
64 import org.projectfloodlight.openflow.protocol.action.OFActionSetVlanVid;
65 import org.projectfloodlight.openflow.protocol.instruction.OFInstruction;
66 import org.projectfloodlight.openflow.protocol.instruction.OFInstructionApplyActions;
67 import org.projectfloodlight.openflow.protocol.instruction.OFInstructionGotoTable;
68 import org.projectfloodlight.openflow.protocol.instruction.OFInstructionWriteActions;
69 import org.projectfloodlight.openflow.protocol.instruction.OFInstructionWriteMetadata;
70 import org.projectfloodlight.openflow.protocol.match.Match;
71 import org.projectfloodlight.openflow.protocol.match.MatchField;
72 import org.projectfloodlight.openflow.protocol.oxm.OFOxm;
73 import org.projectfloodlight.openflow.protocol.oxm.OFOxmOchSigidBasic;
74 import org.projectfloodlight.openflow.protocol.ver13.OFFactoryVer13;
75 import org.projectfloodlight.openflow.types.CircuitSignalID;
76 import org.projectfloodlight.openflow.types.IPv4Address;
77 import org.projectfloodlight.openflow.types.IPv6Address;
78 import org.projectfloodlight.openflow.types.Masked;
79 import org.projectfloodlight.openflow.types.OFVlanVidMatch;
80 import org.projectfloodlight.openflow.types.TransportPort;
81 import org.projectfloodlight.openflow.types.U32;
82 import org.projectfloodlight.openflow.types.U64;
83 import org.projectfloodlight.openflow.types.U8;
84 import org.projectfloodlight.openflow.types.VlanPcp;
85 import org.slf4j.Logger;
87 import java.util.List;
89 import static org.onosproject.net.flow.criteria.Criteria.matchLambda;
90 import static org.onosproject.net.flow.criteria.Criteria.matchOchSignalType;
91 import static org.onosproject.provider.of.flow.impl.OpenFlowValueMapper.lookupChannelSpacing;
92 import static org.onosproject.provider.of.flow.impl.OpenFlowValueMapper.lookupGridType;
93 import static org.onosproject.provider.of.flow.impl.OpenFlowValueMapper.lookupOchSignalType;
94 import static org.slf4j.LoggerFactory.getLogger;
96 public class FlowEntryBuilder {
97 private final Logger log = getLogger(getClass());
99 private final OFFlowStatsEntry stat;
100 private final OFFlowRemoved removed;
101 private final OFFlowMod flowMod;
103 private final Match match;
105 // All actions are contained in an OFInstruction. For OF1.0
106 // the instruction type is apply instruction (immediate set in ONOS speak)
107 private final List<OFInstruction> instructions;
109 private final Dpid dpid;
111 public enum FlowType { STAT, REMOVED, MOD }
113 private final FlowType type;
115 private final DriverService driverService;
117 public FlowEntryBuilder(Dpid dpid, OFFlowStatsEntry entry, DriverService driverService) {
119 this.match = entry.getMatch();
120 this.instructions = getInstructions(entry);
124 this.type = FlowType.STAT;
125 this.driverService = driverService;
128 public FlowEntryBuilder(Dpid dpid, OFFlowRemoved removed, DriverService driverService) {
129 this.match = removed.getMatch();
130 this.removed = removed;
133 this.instructions = null;
136 this.type = FlowType.REMOVED;
137 this.driverService = driverService;
140 public FlowEntryBuilder(Dpid dpid, OFFlowMod fm, DriverService driverService) {
141 this.match = fm.getMatch();
143 this.instructions = getInstructions(fm);
144 this.type = FlowType.MOD;
148 this.driverService = driverService;
151 public FlowEntry build(FlowEntryState... state) {
155 rule = DefaultFlowRule.builder()
156 .forDevice(DeviceId.deviceId(Dpid.uri(dpid)))
157 .withSelector(buildSelector())
158 .withTreatment(buildTreatment())
159 .withPriority(stat.getPriority())
160 .makeTemporary(stat.getIdleTimeout())
161 .withCookie(stat.getCookie().getValue())
162 .forTable(stat.getTableId().getValue())
165 return new DefaultFlowEntry(rule, FlowEntryState.ADDED,
166 stat.getDurationSec(), stat.getPacketCount().getValue(),
167 stat.getByteCount().getValue());
169 rule = DefaultFlowRule.builder()
170 .forDevice(DeviceId.deviceId(Dpid.uri(dpid)))
171 .withSelector(buildSelector())
172 .withPriority(removed.getPriority())
173 .makeTemporary(removed.getIdleTimeout())
174 .withCookie(removed.getCookie().getValue())
175 .forTable(removed.getTableId().getValue())
178 return new DefaultFlowEntry(rule, FlowEntryState.REMOVED, removed.getDurationSec(),
179 removed.getPacketCount().getValue(), removed.getByteCount().getValue());
181 FlowEntryState flowState = state.length > 0 ? state[0] : FlowEntryState.FAILED;
182 rule = DefaultFlowRule.builder()
183 .forDevice(DeviceId.deviceId(Dpid.uri(dpid)))
184 .withSelector(buildSelector())
185 .withTreatment(buildTreatment())
186 .withPriority(flowMod.getPriority())
187 .makeTemporary(flowMod.getIdleTimeout())
188 .withCookie(flowMod.getCookie().getValue())
189 .forTable(flowMod.getTableId().getValue())
192 return new DefaultFlowEntry(rule, flowState, 0, 0, 0);
194 log.error("Unknown flow type : {}", this.type);
200 private List<OFInstruction> getInstructions(OFFlowMod entry) {
201 switch (entry.getVersion()) {
203 return Lists.newArrayList(OFFactoryVer13.INSTANCE.instructions()
205 entry.getActions()));
209 return entry.getInstructions();
211 log.warn("Unknown OF version {}", entry.getVersion());
213 return Lists.newLinkedList();
216 private List<OFInstruction> getInstructions(OFFlowStatsEntry entry) {
217 switch (entry.getVersion()) {
219 return Lists.newArrayList(
220 OFFactoryVer13.INSTANCE.instructions().applyActions(entry.getActions()));
224 return entry.getInstructions();
226 log.warn("Unknown OF version {}", entry.getVersion());
228 return Lists.newLinkedList();
231 private TrafficTreatment buildTreatment() {
232 TrafficTreatment.Builder builder = DefaultTrafficTreatment.builder();
233 for (OFInstruction in : instructions) {
234 switch (in.getType()) {
236 builder.transition(((int) ((OFInstructionGotoTable) in)
237 .getTableId().getValue()));
240 OFInstructionWriteMetadata m = (OFInstructionWriteMetadata) in;
241 builder.writeMetadata(m.getMetadata().getValue(),
242 m.getMetadataMask().getValue());
246 buildActions(((OFInstructionWriteActions) in).getActions(),
251 buildActions(((OFInstructionApplyActions) in).getActions(),
255 builder.wipeDeferred();
262 log.warn("Unknown instructions type {}", in.getType());
266 return builder.build();
269 private TrafficTreatment.Builder buildActions(List<OFAction> actions,
270 TrafficTreatment.Builder builder) {
271 for (OFAction act : actions) {
272 switch (act.getType()) {
274 OFActionOutput out = (OFActionOutput) act;
276 PortNumber.portNumber(out.getPort().getPortNumber()));
279 OFActionSetVlanVid vlan = (OFActionSetVlanVid) act;
280 builder.setVlanId(VlanId.vlanId(vlan.getVlanVid().getVlan()));
283 OFActionSetVlanPcp pcp = (OFActionSetVlanPcp) act;
284 builder.setVlanPcp(pcp.getVlanPcp().getValue());
287 OFActionSetDlDst dldst = (OFActionSetDlDst) act;
289 MacAddress.valueOf(dldst.getDlAddr().getLong()));
292 OFActionSetDlSrc dlsrc = (OFActionSetDlSrc) act;
294 MacAddress.valueOf(dlsrc.getDlAddr().getLong()));
298 OFActionSetNwDst nwdst = (OFActionSetNwDst) act;
299 IPv4Address di = nwdst.getNwAddr();
300 builder.setIpDst(Ip4Address.valueOf(di.getInt()));
303 OFActionSetNwSrc nwsrc = (OFActionSetNwSrc) act;
304 IPv4Address si = nwsrc.getNwAddr();
305 builder.setIpSrc(Ip4Address.valueOf(si.getInt()));
308 OFActionExperimenter exp = (OFActionExperimenter) act;
309 if (exp.getExperimenter() == 0x80005A06 ||
310 exp.getExperimenter() == 0x748771) {
311 OFActionCircuit ct = (OFActionCircuit) exp;
312 short lambda = ((OFOxmOchSigidBasic) ct.getField()).getValue().getChannelNumber();
313 builder.add(Instructions.modL0Lambda(Lambda.indexedLambda(lambda)));
315 log.warn("Unsupported OFActionExperimenter {}", exp.getExperimenter());
319 OFActionSetField setField = (OFActionSetField) act;
320 handleSetField(builder, setField);
323 OFActionPopMpls popMpls = (OFActionPopMpls) act;
324 builder.popMpls((short) popMpls.getEthertype().getValue());
333 builder.copyTtlOut();
336 builder.decMplsTtl();
342 OFActionGroup group = (OFActionGroup) act;
343 builder.group(new DefaultGroupId(group.getGroup().getGroupNumber()));
346 OFActionSetQueue setQueue = (OFActionSetQueue) act;
347 builder.setQueue(setQueue.getQueueId());
369 log.warn("Action type {} not yet implemented.", act.getType());
376 private void handleSetField(TrafficTreatment.Builder builder, OFActionSetField action) {
377 OFOxm<?> oxm = action.getField();
378 switch (oxm.getMatchField().id) {
380 @SuppressWarnings("unchecked")
381 OFOxm<VlanPcp> vlanpcp = (OFOxm<VlanPcp>) oxm;
382 builder.setVlanPcp(vlanpcp.getValue().getValue());
385 @SuppressWarnings("unchecked")
386 OFOxm<OFVlanVidMatch> vlanvid = (OFOxm<OFVlanVidMatch>) oxm;
387 builder.setVlanId(VlanId.vlanId(vlanvid.getValue().getVlan()));
390 @SuppressWarnings("unchecked")
391 OFOxm<org.projectfloodlight.openflow.types.MacAddress> ethdst =
392 (OFOxm<org.projectfloodlight.openflow.types.MacAddress>) oxm;
393 builder.setEthDst(MacAddress.valueOf(ethdst.getValue().getLong()));
396 @SuppressWarnings("unchecked")
397 OFOxm<org.projectfloodlight.openflow.types.MacAddress> ethsrc =
398 (OFOxm<org.projectfloodlight.openflow.types.MacAddress>) oxm;
399 builder.setEthSrc(MacAddress.valueOf(ethsrc.getValue().getLong()));
402 @SuppressWarnings("unchecked")
403 OFOxm<IPv4Address> ip4dst = (OFOxm<IPv4Address>) oxm;
404 builder.setIpDst(Ip4Address.valueOf(ip4dst.getValue().getInt()));
407 @SuppressWarnings("unchecked")
408 OFOxm<IPv4Address> ip4src = (OFOxm<IPv4Address>) oxm;
409 builder.setIpSrc(Ip4Address.valueOf(ip4src.getValue().getInt()));
412 @SuppressWarnings("unchecked")
413 OFOxm<U32> labelId = (OFOxm<U32>) oxm;
414 builder.setMpls(MplsLabel.mplsLabel((int) labelId.getValue().getValue()));
417 @SuppressWarnings("unchecked")
418 OFOxm<U8> mplsBos = (OFOxm<U8>) oxm;
419 builder.setMplsBos(mplsBos.getValue() == U8.ZERO ? false : true);
422 @SuppressWarnings("unchecked")
423 OFOxm<U64> tunnelId = (OFOxm<U64>) oxm;
424 builder.setTunnelId(tunnelId.getValue().getValue());
427 @SuppressWarnings("unchecked")
428 OFOxm<TransportPort> tcpdst = (OFOxm<TransportPort>) oxm;
429 builder.setTcpDst(TpPort.tpPort(tcpdst.getValue().getPort()));
432 @SuppressWarnings("unchecked")
433 OFOxm<TransportPort> tcpsrc = (OFOxm<TransportPort>) oxm;
434 builder.setTcpSrc(TpPort.tpPort(tcpsrc.getValue().getPort()));
437 @SuppressWarnings("unchecked")
438 OFOxm<TransportPort> udpdst = (OFOxm<TransportPort>) oxm;
439 builder.setUdpDst(TpPort.tpPort(udpdst.getValue().getPort()));
442 @SuppressWarnings("unchecked")
443 OFOxm<TransportPort> udpsrc = (OFOxm<TransportPort>) oxm;
444 builder.setUdpSrc(TpPort.tpPort(udpsrc.getValue().getPort()));
446 case TUNNEL_IPV4_DST:
447 DriverHandler driver = getDriver(dpid);
448 ExtensionInterpreter interpreter = driver.behaviour(ExtensionInterpreter.class);
449 if (interpreter != null) {
450 builder.extension(interpreter.mapAction(action), DeviceId.deviceId(Dpid.uri(dpid)));
458 case BSN_EGR_PORT_GROUP_ID:
459 case BSN_GLOBAL_VRF_ALLOWED:
460 case BSN_IN_PORTS_128:
461 case BSN_L3_DST_CLASS_ID:
462 case BSN_L3_INTERFACE_CLASS_ID:
463 case BSN_L3_SRC_CLASS_ID:
474 case BSN_VLAN_XLATE_PORT_GROUP_ID:
495 case OCH_SIGID_BASIC:
497 case OCH_SIGTYPE_BASIC:
501 log.warn("Set field type {} not yet implemented.", oxm.getMatchField().id);
506 // CHECKSTYLE IGNORE MethodLength FOR NEXT 1 LINES
507 private TrafficSelector buildSelector() {
510 Ip6Address ip6Address;
513 TrafficSelector.Builder builder = DefaultTrafficSelector.builder();
514 for (MatchField<?> field : match.getMatchFields()) {
517 builder.matchInPort(PortNumber
518 .portNumber(match.get(MatchField.IN_PORT).getPortNumber()));
521 builder.matchInPhyPort(PortNumber
522 .portNumber(match.get(MatchField.IN_PHY_PORT).getPortNumber()));
526 match.get(MatchField.METADATA).getValue().getValue();
527 builder.matchMetadata(metadata);
530 mac = MacAddress.valueOf(match.get(MatchField.ETH_DST).getLong());
531 builder.matchEthDst(mac);
534 mac = MacAddress.valueOf(match.get(MatchField.ETH_SRC).getLong());
535 builder.matchEthSrc(mac);
538 int ethType = match.get(MatchField.ETH_TYPE).getValue();
539 builder.matchEthType((short) ethType);
542 VlanId vlanId = null;
543 if (match.isPartiallyMasked(MatchField.VLAN_VID)) {
544 Masked<OFVlanVidMatch> masked = match.getMasked(MatchField.VLAN_VID);
545 if (masked.getValue().equals(OFVlanVidMatch.PRESENT)
546 && masked.getMask().equals(OFVlanVidMatch.PRESENT)) {
550 if (!match.get(MatchField.VLAN_VID).isPresentBitSet()) {
551 vlanId = VlanId.NONE;
553 vlanId = VlanId.vlanId(match.get(MatchField.VLAN_VID).getVlan());
556 if (vlanId != null) {
557 builder.matchVlanId(vlanId);
561 byte vlanPcp = match.get(MatchField.VLAN_PCP).getValue();
562 builder.matchVlanPcp(vlanPcp);
565 byte ipDscp = match.get(MatchField.IP_DSCP).getDscpValue();
566 builder.matchIPDscp(ipDscp);
569 byte ipEcn = match.get(MatchField.IP_ECN).getEcnValue();
570 builder.matchIPEcn(ipEcn);
573 short proto = match.get(MatchField.IP_PROTO).getIpProtocolNumber();
574 builder.matchIPProtocol((byte) proto);
577 if (match.isPartiallyMasked(MatchField.IPV4_SRC)) {
578 Masked<IPv4Address> maskedIp = match.getMasked(MatchField.IPV4_SRC);
579 ip4Prefix = Ip4Prefix.valueOf(
580 maskedIp.getValue().getInt(),
581 maskedIp.getMask().asCidrMaskLength());
583 ip4Prefix = Ip4Prefix.valueOf(
584 match.get(MatchField.IPV4_SRC).getInt(),
585 Ip4Prefix.MAX_MASK_LENGTH);
587 builder.matchIPSrc(ip4Prefix);
590 if (match.isPartiallyMasked(MatchField.IPV4_DST)) {
591 Masked<IPv4Address> maskedIp = match.getMasked(MatchField.IPV4_DST);
592 ip4Prefix = Ip4Prefix.valueOf(
593 maskedIp.getValue().getInt(),
594 maskedIp.getMask().asCidrMaskLength());
596 ip4Prefix = Ip4Prefix.valueOf(
597 match.get(MatchField.IPV4_DST).getInt(),
598 Ip4Prefix.MAX_MASK_LENGTH);
600 builder.matchIPDst(ip4Prefix);
603 builder.matchTcpSrc(TpPort.tpPort(match.get(MatchField.TCP_SRC).getPort()));
606 builder.matchTcpDst(TpPort.tpPort(match.get(MatchField.TCP_DST).getPort()));
609 builder.matchUdpSrc(TpPort.tpPort(match.get(MatchField.UDP_SRC).getPort()));
612 builder.matchUdpDst(TpPort.tpPort(match.get(MatchField.UDP_DST).getPort()));
615 builder.matchMplsLabel(MplsLabel.mplsLabel((int) match.get(MatchField.MPLS_LABEL)
619 builder.matchMplsBos(match.get(MatchField.MPLS_BOS).getValue());
622 builder.matchSctpSrc(TpPort.tpPort(match.get(MatchField.SCTP_SRC).getPort()));
625 builder.matchSctpDst(TpPort.tpPort(match.get(MatchField.SCTP_DST).getPort()));
628 byte icmpType = (byte) match.get(MatchField.ICMPV4_TYPE).getType();
629 builder.matchIcmpType(icmpType);
632 byte icmpCode = (byte) match.get(MatchField.ICMPV4_CODE).getCode();
633 builder.matchIcmpCode(icmpCode);
636 if (match.isPartiallyMasked(MatchField.IPV6_SRC)) {
637 Masked<IPv6Address> maskedIp = match.getMasked(MatchField.IPV6_SRC);
638 ip6Prefix = Ip6Prefix.valueOf(
639 maskedIp.getValue().getBytes(),
640 maskedIp.getMask().asCidrMaskLength());
642 ip6Prefix = Ip6Prefix.valueOf(
643 match.get(MatchField.IPV6_SRC).getBytes(),
644 Ip6Prefix.MAX_MASK_LENGTH);
646 builder.matchIPv6Src(ip6Prefix);
649 if (match.isPartiallyMasked(MatchField.IPV6_DST)) {
650 Masked<IPv6Address> maskedIp = match.getMasked(MatchField.IPV6_DST);
651 ip6Prefix = Ip6Prefix.valueOf(
652 maskedIp.getValue().getBytes(),
653 maskedIp.getMask().asCidrMaskLength());
655 ip6Prefix = Ip6Prefix.valueOf(
656 match.get(MatchField.IPV6_DST).getBytes(),
657 Ip6Prefix.MAX_MASK_LENGTH);
659 builder.matchIPv6Dst(ip6Prefix);
663 match.get(MatchField.IPV6_FLABEL).getIPv6FlowLabelValue();
664 builder.matchIPv6FlowLabel(flowLabel);
667 byte icmpv6type = (byte) match.get(MatchField.ICMPV6_TYPE).getValue();
668 builder.matchIcmpv6Type(icmpv6type);
671 byte icmpv6code = (byte) match.get(MatchField.ICMPV6_CODE).getValue();
672 builder.matchIcmpv6Code(icmpv6code);
676 Ip6Address.valueOf(match.get(MatchField.IPV6_ND_TARGET).getBytes());
677 builder.matchIPv6NDTargetAddress(ip6Address);
680 mac = MacAddress.valueOf(match.get(MatchField.IPV6_ND_SLL).getLong());
681 builder.matchIPv6NDSourceLinkLayerAddress(mac);
684 mac = MacAddress.valueOf(match.get(MatchField.IPV6_ND_TLL).getLong());
685 builder.matchIPv6NDTargetLinkLayerAddress(mac);
688 builder.matchIPv6ExthdrFlags((short) match.get(MatchField.IPV6_EXTHDR)
692 CircuitSignalID sigId = match.get(MatchField.OCH_SIGID);
693 builder.add(matchLambda(Lambda.ochSignal(
694 lookupGridType(sigId.getGridType()), lookupChannelSpacing(sigId.getChannelSpacing()),
695 sigId.getChannelNumber(), sigId.getSpectralWidth())
699 U8 sigType = match.get(MatchField.OCH_SIGTYPE);
700 builder.add(matchOchSignalType(lookupOchSignalType((byte) sigType.getValue())));
703 long tunnelId = match.get(MatchField.TUNNEL_ID).getValue();
704 builder.matchTunnelId(tunnelId);
713 log.warn("Match type {} not yet implemented.", field.id);
716 return builder.build();
719 private DriverHandler getDriver(Dpid dpid) {
720 DeviceId deviceId = DeviceId.deviceId(Dpid.uri(dpid));
721 Driver driver = driverService.getDriver(deviceId);
722 DriverHandler handler = new DefaultDriverHandler(new DefaultDriverData(driver, deviceId));