VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/tests/usb/usbgadget.py@ 64601

Last change on this file since 64601 was 61831, checked in by vboxsync, 9 years ago

txsclient.py,usbgadget.py: pylint 1.5.5 fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 50.4 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: usbgadget.py 61831 2016-06-22 21:06:11Z vboxsync $
3# pylint: disable=C0302
4
5"""
6UTS (USB Test Service) client.
7"""
8__copyright__ = \
9"""
10Copyright (C) 2010-2016 Oracle Corporation
11
12This file is part of VirtualBox Open Source Edition (OSE), as
13available from http://www.virtualbox.org. This file is free software;
14you can redistribute it and/or modify it under the terms of the GNU
15General Public License (GPL) as published by the Free Software
16Foundation, in version 2 as it comes in the "COPYING" file of the
17VirtualBox OSE distribution. VirtualBox OSE is distributed in the
18hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
19
20The contents of this file may alternatively be used under the terms
21of the Common Development and Distribution License Version 1.0
22(CDDL) only, as it comes in the "COPYING.CDDL" file of the
23VirtualBox OSE distribution, in which case the provisions of the
24CDDL are applicable instead of those of the GPL.
25
26You may elect to license modified versions of this file under the
27terms and conditions of either the GPL or the CDDL or both.
28"""
29__version__ = "$Revision: 61831 $"
30
31# Standard Python imports.
32import array
33import errno
34import select
35import socket
36import threading
37import time
38import types
39import zlib
40
41# Validation Kit imports.
42from common import utils;
43from testdriver import base;
44from testdriver import reporter;
45from testdriver.base import TdTaskBase;
46
47## @name USB gadget impersonation string constants.
48## @{
49g_ksGadgetImpersonationInvalid = 'Invalid';
50g_ksGadgetImpersonationTest = 'Test';
51g_ksGadgetImpersonationMsd = 'Msd';
52g_ksGadgetImpersonationWebcam = 'Webcam';
53g_ksGadgetImpersonationEther = 'Ether';
54## @}
55
56## @name USB gadget type used in the UTS protocol.
57## @{
58g_kiGadgetTypeTest = 1;
59## @}
60
61## @name USB gadget access methods used in the UTS protocol.
62## @{
63g_kiGadgetAccessUsbIp = 1;
64## @}
65
66## @name USB gadget config types.
67## @{
68g_kiGadgetCfgTypeBool = 1;
69g_kiGadgetCfgTypeString = 2;
70g_kiGadgetCfgTypeUInt8 = 3;
71g_kiGadgetCfgTypeUInt16 = 4;
72g_kiGadgetCfgTypeUInt32 = 5;
73g_kiGadgetCfgTypeUInt64 = 6;
74g_kiGadgetCfgTypeInt8 = 7;
75g_kiGadgetCfgTypeInt16 = 8;
76g_kiGadgetCfgTypeInt32 = 9;
77g_kiGadgetCfgTypeInt64 = 10;
78## @}
79
80#
81# Helpers for decoding data received from the UTS.
82# These are used both the Session and Transport classes.
83#
84
85def getU64(abData, off):
86 """Get a U64 field."""
87 return abData[off] \
88 + abData[off + 1] * 256 \
89 + abData[off + 2] * 65536 \
90 + abData[off + 3] * 16777216 \
91 + abData[off + 4] * 4294967296 \
92 + abData[off + 5] * 1099511627776 \
93 + abData[off + 6] * 281474976710656 \
94 + abData[off + 7] * 72057594037927936;
95
96def getU32(abData, off):
97 """Get a U32 field."""
98 return abData[off] \
99 + abData[off + 1] * 256 \
100 + abData[off + 2] * 65536 \
101 + abData[off + 3] * 16777216;
102
103def getU16(abData, off):
104 """Get a U16 field."""
105 return abData[off] \
106 + abData[off + 1] * 256;
107
108def getU8(abData, off):
109 """Get a U8 field."""
110 return abData[off];
111
112def getSZ(abData, off, sDefault = None):
113 """
114 Get a zero-terminated string field.
115 Returns sDefault if the string is invalid.
116 """
117 cchStr = getSZLen(abData, off);
118 if cchStr >= 0:
119 abStr = abData[off:(off + cchStr)];
120 try:
121 return abStr.tostring().decode('utf_8');
122 except:
123 reporter.errorXcpt('getSZ(,%u)' % (off));
124 return sDefault;
125
126def getSZLen(abData, off):
127 """
128 Get the length of a zero-terminated string field, in bytes.
129 Returns -1 if off is beyond the data packet or not properly terminated.
130 """
131 cbData = len(abData);
132 if off >= cbData:
133 return -1;
134
135 offCur = off;
136 while abData[offCur] != 0:
137 offCur = offCur + 1;
138 if offCur >= cbData:
139 return -1;
140
141 return offCur - off;
142
143def isValidOpcodeEncoding(sOpcode):
144 """
145 Checks if the specified opcode is valid or not.
146 Returns True on success.
147 Returns False if it is invalid, details in the log.
148 """
149 sSet1 = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
150 sSet2 = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-_ ";
151 if len(sOpcode) != 8:
152 reporter.error("invalid opcode length: %s" % (len(sOpcode)));
153 return False;
154 for i in range(0, 1):
155 if sSet1.find(sOpcode[i]) < 0:
156 reporter.error("invalid opcode char #%u: %s" % (i, sOpcode));
157 return False;
158 for i in range(2, 7):
159 if sSet2.find(sOpcode[i]) < 0:
160 reporter.error("invalid opcode char #%u: %s" % (i, sOpcode));
161 return False;
162 return True;
163
164#
165# Helper for encoding data sent to the UTS.
166#
167
168def u32ToByteArray(u32):
169 """Encodes the u32 value as a little endian byte (B) array."""
170 return array.array('B', \
171 ( u32 % 256, \
172 (u32 / 256) % 256, \
173 (u32 / 65536) % 256, \
174 (u32 / 16777216) % 256) );
175
176def u16ToByteArray(u16):
177 """Encodes the u16 value as a little endian byte (B) array."""
178 return array.array('B', \
179 ( u16 % 256, \
180 (u16 / 256) % 256) );
181
182def u8ToByteArray(uint8):
183 """Encodes the u8 value as a little endian byte (B) array."""
184 return array.array('B', (uint8 % 256));
185
186def zeroByteArray(cb):
187 """Returns an array with the given size containing 0."""
188 abArray = array.array('B', (0, ));
189 cb = cb - 1;
190 for i in range(cb): # pylint: disable=W0612
191 abArray.append(0);
192 return abArray;
193
194def strToByteArry(sStr):
195 """Encodes the string as a little endian byte (B) array including the terminator."""
196 abArray = array.array('B');
197 sUtf8 = sStr.encode('utf_8');
198 for i in range(0, len(sUtf8)):
199 abArray.append(ord(sUtf8[i]))
200 abArray.append(0);
201 return abArray;
202
203def cfgListToByteArray(lst):
204 """Encodes the given config list as a little endian byte (B) array."""
205 abArray = array.array('B');
206 if lst is not None:
207 for t3Item in lst:
208 # Encode they key size
209 abArray.extend(u32ToByteArray(len(t3Item[0]) + 1)); # Include terminator
210 abArray.extend(u32ToByteArray(t3Item[1])) # Config type
211 abArray.extend(u32ToByteArray(len(t3Item[2]) + 1)); # Value size including temrinator.
212 abArray.extend(u32ToByteArray(0)); # Reserved field.
213
214 abArray.extend(strToByteArry(t3Item[0]));
215 abArray.extend(strToByteArry(t3Item[2]));
216
217 return abArray;
218
219class TransportBase(object):
220 """
221 Base class for the transport layer.
222 """
223
224 def __init__(self, sCaller):
225 self.sDbgCreated = '%s: %s' % (utils.getTimePrefix(), sCaller);
226 self.fDummy = 0;
227 self.abReadAheadHdr = array.array('B');
228
229 def toString(self):
230 """
231 Stringify the instance for logging and debugging.
232 """
233 return '<%s: abReadAheadHdr=%s, sDbgCreated=%s>' % (type(self).__name__, self.abReadAheadHdr, self.sDbgCreated);
234
235 def __str__(self):
236 return self.toString();
237
238 def cancelConnect(self):
239 """
240 Cancels any pending connect() call.
241 Returns None;
242 """
243 return None;
244
245 def connect(self, cMsTimeout):
246 """
247 Quietly attempts to connect to the UTS.
248
249 Returns True on success.
250 Returns False on retryable errors (no logging).
251 Returns None on fatal errors with details in the log.
252
253 Override this method, don't call super.
254 """
255 _ = cMsTimeout;
256 return False;
257
258 def disconnect(self, fQuiet = False):
259 """
260 Disconnect from the UTS.
261
262 Returns True.
263
264 Override this method, don't call super.
265 """
266 _ = fQuiet;
267 return True;
268
269 def sendBytes(self, abBuf, cMsTimeout):
270 """
271 Sends the bytes in the buffer abBuf to the UTS.
272
273 Returns True on success.
274 Returns False on failure and error details in the log.
275
276 Override this method, don't call super.
277
278 Remarks: len(abBuf) is always a multiple of 16.
279 """
280 _ = abBuf; _ = cMsTimeout;
281 return False;
282
283 def recvBytes(self, cb, cMsTimeout, fNoDataOk):
284 """
285 Receive cb number of bytes from the UTS.
286
287 Returns the bytes (array('B')) on success.
288 Returns None on failure and error details in the log.
289
290 Override this method, don't call super.
291
292 Remarks: cb is always a multiple of 16.
293 """
294 _ = cb; _ = cMsTimeout; _ = fNoDataOk;
295 return None;
296
297 def isConnectionOk(self):
298 """
299 Checks if the connection is OK.
300
301 Returns True if it is.
302 Returns False if it isn't (caller should call diconnect).
303
304 Override this method, don't call super.
305 """
306 return True;
307
308 def isRecvPending(self, cMsTimeout = 0):
309 """
310 Checks if there is incoming bytes, optionally waiting cMsTimeout
311 milliseconds for something to arrive.
312
313 Returns True if there is, False if there isn't.
314
315 Override this method, don't call super.
316 """
317 _ = cMsTimeout;
318 return False;
319
320 def sendMsgInt(self, sOpcode, cMsTimeout, abPayload = array.array('B')):
321 """
322 Sends a message (opcode + encoded payload).
323
324 Returns True on success.
325 Returns False on failure and error details in the log.
326 """
327 # Fix + check the opcode.
328 if len(sOpcode) < 2:
329 reporter.fatal('sendMsgInt: invalid opcode length: %d (\"%s\")' % (len(sOpcode), sOpcode));
330 return False;
331 sOpcode = sOpcode.ljust(8);
332 if not isValidOpcodeEncoding(sOpcode):
333 reporter.fatal('sendMsgInt: invalid opcode encoding: \"%s\"' % (sOpcode));
334 return False;
335
336 # Start construct the message.
337 cbMsg = 16 + len(abPayload);
338 abMsg = array.array('B');
339 abMsg.extend(u32ToByteArray(cbMsg));
340 abMsg.extend((0, 0, 0, 0)); # uCrc32
341 try:
342 abMsg.extend(array.array('B', \
343 ( ord(sOpcode[0]), \
344 ord(sOpcode[1]), \
345 ord(sOpcode[2]), \
346 ord(sOpcode[3]), \
347 ord(sOpcode[4]), \
348 ord(sOpcode[5]), \
349 ord(sOpcode[6]), \
350 ord(sOpcode[7]) ) ) );
351 if len(abPayload) > 0:
352 abMsg.extend(abPayload);
353 except:
354 reporter.fatalXcpt('sendMsgInt: packing problem...');
355 return False;
356
357 # checksum it, padd it and send it off.
358 uCrc32 = zlib.crc32(abMsg[8:]);
359 abMsg[4:8] = u32ToByteArray(uCrc32);
360
361 while len(abMsg) % 16:
362 abMsg.append(0);
363
364 reporter.log2('sendMsgInt: op=%s len=%d to=%d' % (sOpcode, len(abMsg), cMsTimeout));
365 return self.sendBytes(abMsg, cMsTimeout);
366
367 def recvMsg(self, cMsTimeout, fNoDataOk = False):
368 """
369 Receives a message from the UTS.
370
371 Returns the message three-tuple: length, opcode, payload.
372 Returns (None, None, None) on failure and error details in the log.
373 """
374
375 # Read the header.
376 if len(self.abReadAheadHdr) > 0:
377 assert(len(self.abReadAheadHdr) == 16);
378 abHdr = self.abReadAheadHdr;
379 self.abReadAheadHdr = array.array('B');
380 else:
381 abHdr = self.recvBytes(16, cMsTimeout, fNoDataOk);
382 if abHdr is None:
383 return (None, None, None);
384 if len(abHdr) != 16:
385 reporter.fatal('recvBytes(16) returns %d bytes!' % (len(abHdr)));
386 return (None, None, None);
387
388 # Unpack and validate the header.
389 cbMsg = getU32(abHdr, 0);
390 uCrc32 = getU32(abHdr, 4);
391 sOpcode = abHdr[8:16].tostring().decode('ascii');
392
393 if cbMsg < 16:
394 reporter.fatal('recvMsg: message length is out of range: %s (min 16 bytes)' % (cbMsg));
395 return (None, None, None);
396 if cbMsg > 1024*1024:
397 reporter.fatal('recvMsg: message length is out of range: %s (max 1MB)' % (cbMsg));
398 return (None, None, None);
399 if not isValidOpcodeEncoding(sOpcode):
400 reporter.fatal('recvMsg: invalid opcode \"%s\"' % (sOpcode));
401 return (None, None, None);
402
403 # Get the payload (if any), dropping the padding.
404 abPayload = array.array('B');
405 if cbMsg > 16:
406 if cbMsg % 16:
407 cbPadding = 16 - (cbMsg % 16);
408 else:
409 cbPadding = 0;
410 abPayload = self.recvBytes(cbMsg - 16 + cbPadding, cMsTimeout, False);
411 if abPayload is None:
412 self.abReadAheadHdr = abHdr;
413 if not fNoDataOk :
414 reporter.log('recvMsg: failed to recv payload bytes!');
415 return (None, None, None);
416
417 while cbPadding > 0:
418 abPayload.pop();
419 cbPadding = cbPadding - 1;
420
421 # Check the CRC-32.
422 if uCrc32 != 0:
423 uActualCrc32 = zlib.crc32(abHdr[8:]);
424 if cbMsg > 16:
425 uActualCrc32 = zlib.crc32(abPayload, uActualCrc32);
426 uActualCrc32 = uActualCrc32 & 0xffffffff;
427 if uCrc32 != uActualCrc32:
428 reporter.fatal('recvMsg: crc error: expected %s, got %s' % (hex(uCrc32), hex(uActualCrc32)));
429 return (None, None, None);
430
431 reporter.log2('recvMsg: op=%s len=%d' % (sOpcode, len(abPayload)));
432 return (cbMsg, sOpcode, abPayload);
433
434 def sendMsg(self, sOpcode, cMsTimeout, aoPayload = ()):
435 """
436 Sends a message (opcode + payload tuple).
437
438 Returns True on success.
439 Returns False on failure and error details in the log.
440 Returns None if you pass the incorrectly typed parameters.
441 """
442 # Encode the payload.
443 abPayload = array.array('B');
444 for o in aoPayload:
445 try:
446 if isinstance(o, basestring):
447 # the primitive approach...
448 sUtf8 = o.encode('utf_8');
449 for i in range(0, len(sUtf8)):
450 abPayload.append(ord(sUtf8[i]))
451 abPayload.append(0);
452 elif isinstance(o, types.LongType):
453 if o < 0 or o > 0xffffffff:
454 reporter.fatal('sendMsg: uint32_t payload is out of range: %s' % (hex(o)));
455 return None;
456 abPayload.extend(u32ToByteArray(o));
457 elif isinstance(o, types.IntType):
458 if o < 0 or o > 0xffffffff:
459 reporter.fatal('sendMsg: uint32_t payload is out of range: %s' % (hex(o)));
460 return None;
461 abPayload.extend(u32ToByteArray(o));
462 elif isinstance(o, array.array):
463 abPayload.extend(o);
464 else:
465 reporter.fatal('sendMsg: unexpected payload type: %s (%s) (aoPayload=%s)' % (type(o), o, aoPayload));
466 return None;
467 except:
468 reporter.fatalXcpt('sendMsg: screwed up the encoding code...');
469 return None;
470 return self.sendMsgInt(sOpcode, cMsTimeout, abPayload);
471
472
473class Session(TdTaskBase):
474 """
475 A USB Test Service (UTS) client session.
476 """
477
478 def __init__(self, oTransport, cMsTimeout, cMsIdleFudge, fTryConnect = False):
479 """
480 Construct a UTS session.
481
482 This starts by connecting to the UTS and will enter the signalled state
483 when connected or the timeout has been reached.
484 """
485 TdTaskBase.__init__(self, utils.getCallerName());
486 self.oTransport = oTransport;
487 self.sStatus = "";
488 self.cMsTimeout = 0;
489 self.fErr = True; # Whether to report errors as error.
490 self.msStart = 0;
491 self.oThread = None;
492 self.fnTask = self.taskDummy;
493 self.aTaskArgs = None;
494 self.oTaskRc = None;
495 self.t3oReply = (None, None, None);
496 self.fScrewedUpMsgState = False;
497 self.fTryConnect = fTryConnect;
498
499 if not self.startTask(cMsTimeout, False, "connecting", self.taskConnect, (cMsIdleFudge,)):
500 raise base.GenError("startTask failed");
501
502 def __del__(self):
503 """Make sure to cancel the task when deleted."""
504 self.cancelTask();
505
506 def toString(self):
507 return '<%s fnTask=%s, aTaskArgs=%s, sStatus=%s, oTaskRc=%s, cMsTimeout=%s,' \
508 ' msStart=%s, fTryConnect=%s, fErr=%s, fScrewedUpMsgState=%s, t3oReply=%s oTransport=%s, oThread=%s>' \
509 % (TdTaskBase.toString(self), self.fnTask, self.aTaskArgs, self.sStatus, self.oTaskRc, self.cMsTimeout,
510 self.msStart, self.fTryConnect, self.fErr, self.fScrewedUpMsgState, self.t3oReply, self.oTransport, self.oThread);
511
512 def taskDummy(self):
513 """Place holder to catch broken state handling."""
514 raise Exception();
515
516 def startTask(self, cMsTimeout, fIgnoreErrors, sStatus, fnTask, aArgs = ()):
517 """
518 Kicks of a new task.
519
520 cMsTimeout: The task timeout in milliseconds. Values less than
521 500 ms will be adjusted to 500 ms. This means it is
522 OK to use negative value.
523 sStatus: The task status.
524 fnTask: The method that'll execute the task.
525 aArgs: Arguments to pass to fnTask.
526
527 Returns True on success, False + error in log on failure.
528 """
529 if not self.cancelTask():
530 reporter.maybeErr(not fIgnoreErrors, 'utsclient.Session.startTask: failed to cancel previous task.');
531 return False;
532
533 # Change status and make sure we're the
534 self.lockTask();
535 if self.sStatus != "":
536 self.unlockTask();
537 reporter.maybeErr(not fIgnoreErrors, 'utsclient.Session.startTask: race.');
538 return False;
539 self.sStatus = "setup";
540 self.oTaskRc = None;
541 self.t3oReply = (None, None, None);
542 self.resetTaskLocked();
543 self.unlockTask();
544
545 self.cMsTimeout = max(cMsTimeout, 500);
546 self.fErr = not fIgnoreErrors;
547 self.fnTask = fnTask;
548 self.aTaskArgs = aArgs;
549 self.oThread = threading.Thread(target=self.taskThread, args=(), name=('UTS-%s' % (sStatus)));
550 self.oThread.setDaemon(True);
551 self.msStart = base.timestampMilli();
552
553 self.lockTask();
554 self.sStatus = sStatus;
555 self.unlockTask();
556 self.oThread.start();
557
558 return True;
559
560 def cancelTask(self, fSync = True):
561 """
562 Attempts to cancel any pending tasks.
563 Returns success indicator (True/False).
564 """
565 self.lockTask();
566
567 if self.sStatus == "":
568 self.unlockTask();
569 return True;
570 if self.sStatus == "setup":
571 self.unlockTask();
572 return False;
573 if self.sStatus == "cancelled":
574 self.unlockTask();
575 return False;
576
577 reporter.log('utsclient: cancelling "%s"...' % (self.sStatus));
578 if self.sStatus == 'connecting':
579 self.oTransport.cancelConnect();
580
581 self.sStatus = "cancelled";
582 oThread = self.oThread;
583 self.unlockTask();
584
585 if not fSync:
586 return False;
587
588 oThread.join(61.0);
589 return oThread.isAlive();
590
591 def taskThread(self):
592 """
593 The task thread function.
594 This does some housekeeping activities around the real task method call.
595 """
596 if not self.isCancelled():
597 try:
598 fnTask = self.fnTask;
599 oTaskRc = fnTask(*self.aTaskArgs);
600 except:
601 reporter.fatalXcpt('taskThread', 15);
602 oTaskRc = None;
603 else:
604 reporter.log('taskThread: cancelled already');
605
606 self.lockTask();
607
608 reporter.log('taskThread: signalling task with status "%s", oTaskRc=%s' % (self.sStatus, oTaskRc));
609 self.oTaskRc = oTaskRc;
610 self.oThread = None;
611 self.sStatus = '';
612 self.signalTaskLocked();
613
614 self.unlockTask();
615 return None;
616
617 def isCancelled(self):
618 """Internal method for checking if the task has been cancelled."""
619 self.lockTask();
620 sStatus = self.sStatus;
621 self.unlockTask();
622 if sStatus == "cancelled":
623 return True;
624 return False;
625
626 def hasTimedOut(self):
627 """Internal method for checking if the task has timed out or not."""
628 cMsLeft = self.getMsLeft();
629 if cMsLeft <= 0:
630 return True;
631 return False;
632
633 def getMsLeft(self, cMsMin = 0, cMsMax = -1):
634 """Gets the time left until the timeout."""
635 cMsElapsed = base.timestampMilli() - self.msStart;
636 if cMsElapsed < 0:
637 return cMsMin;
638 cMsLeft = self.cMsTimeout - cMsElapsed;
639 if cMsLeft <= cMsMin:
640 return cMsMin;
641 if cMsLeft > cMsMax and cMsMax > 0:
642 return cMsMax
643 return cMsLeft;
644
645 def recvReply(self, cMsTimeout = None, fNoDataOk = False):
646 """
647 Wrapper for TransportBase.recvMsg that stashes the response away
648 so the client can inspect it later on.
649 """
650 if cMsTimeout is None:
651 cMsTimeout = self.getMsLeft(500);
652 cbMsg, sOpcode, abPayload = self.oTransport.recvMsg(cMsTimeout, fNoDataOk);
653 self.lockTask();
654 self.t3oReply = (cbMsg, sOpcode, abPayload);
655 self.unlockTask();
656 return (cbMsg, sOpcode, abPayload);
657
658 def recvAck(self, fNoDataOk = False):
659 """
660 Receives an ACK or error response from the UTS.
661
662 Returns True on success.
663 Returns False on timeout or transport error.
664 Returns (sOpcode, sDetails) tuple on failure. The opcode is stripped
665 and there are always details of some sort or another.
666 """
667 cbMsg, sOpcode, abPayload = self.recvReply(None, fNoDataOk);
668 if cbMsg is None:
669 return False;
670 sOpcode = sOpcode.strip()
671 if sOpcode == "ACK":
672 return True;
673 return (sOpcode, getSZ(abPayload, 16, sOpcode));
674
675 def recvAckLogged(self, sCommand, fNoDataOk = False):
676 """
677 Wrapper for recvAck and logging.
678 Returns True on success (ACK).
679 Returns False on time, transport error and errors signalled by UTS.
680 """
681 rc = self.recvAck(fNoDataOk);
682 if rc is not True and not fNoDataOk:
683 if rc is False:
684 reporter.maybeErr(self.fErr, 'recvAckLogged: %s transport error' % (sCommand));
685 else:
686 reporter.maybeErr(self.fErr, 'recvAckLogged: %s response was %s: %s' % (sCommand, rc[0], rc[1]));
687 rc = False;
688 return rc;
689
690 def recvTrueFalse(self, sCommand):
691 """
692 Receives a TRUE/FALSE response from the UTS.
693 Returns True on TRUE, False on FALSE and None on error/other (logged).
694 """
695 cbMsg, sOpcode, abPayload = self.recvReply();
696 if cbMsg is None:
697 reporter.maybeErr(self.fErr, 'recvAckLogged: %s transport error' % (sCommand));
698 return None;
699
700 sOpcode = sOpcode.strip()
701 if sOpcode == "TRUE":
702 return True;
703 if sOpcode == "FALSE":
704 return False;
705 reporter.maybeErr(self.fErr, 'recvAckLogged: %s response was %s: %s' % \
706 (sCommand, sOpcode, getSZ(abPayload, 16, sOpcode)));
707 return None;
708
709 def sendMsg(self, sOpcode, aoPayload = (), cMsTimeout = None):
710 """
711 Wrapper for TransportBase.sendMsg that inserts the correct timeout.
712 """
713 if cMsTimeout is None:
714 cMsTimeout = self.getMsLeft(500);
715 return self.oTransport.sendMsg(sOpcode, cMsTimeout, aoPayload);
716
717 def asyncToSync(self, fnAsync, *aArgs):
718 """
719 Wraps an asynchronous task into a synchronous operation.
720
721 Returns False on failure, task return status on success.
722 """
723 rc = fnAsync(*aArgs);
724 if rc is False:
725 reporter.log2('asyncToSync(%s): returns False (#1)' % (fnAsync));
726 return rc;
727
728 rc = self.waitForTask(self.cMsTimeout + 5000);
729 if rc is False:
730 reporter.maybeErrXcpt(self.fErr, 'asyncToSync: waitForTask failed...');
731 self.cancelTask();
732 #reporter.log2('asyncToSync(%s): returns False (#2)' % (fnAsync, rc));
733 return False;
734
735 rc = self.getResult();
736 #reporter.log2('asyncToSync(%s): returns %s' % (fnAsync, rc));
737 return rc;
738
739 #
740 # Connection tasks.
741 #
742
743 def taskConnect(self, cMsIdleFudge):
744 """Tries to connect to the UTS"""
745 while not self.isCancelled():
746 reporter.log2('taskConnect: connecting ...');
747 rc = self.oTransport.connect(self.getMsLeft(500));
748 if rc is True:
749 reporter.log('taskConnect: succeeded');
750 return self.taskGreet(cMsIdleFudge);
751 if rc is None:
752 reporter.log2('taskConnect: unable to connect');
753 return None;
754 if self.hasTimedOut():
755 reporter.log2('taskConnect: timed out');
756 if not self.fTryConnect:
757 reporter.maybeErr(self.fErr, 'taskConnect: timed out');
758 return False;
759 time.sleep(self.getMsLeft(1, 1000) / 1000.0);
760 if not self.fTryConnect:
761 reporter.maybeErr(self.fErr, 'taskConnect: cancelled');
762 return False;
763
764 def taskGreet(self, cMsIdleFudge):
765 """Greets the UTS"""
766 sHostname = socket.gethostname().lower();
767 cbFill = 68 - len(sHostname) - 1;
768 rc = self.sendMsg("HOWDY", ((1 << 16) | 0, 0x1, len(sHostname), sHostname, zeroByteArray(cbFill)));
769 if rc is True:
770 rc = self.recvAckLogged("HOWDY", self.fTryConnect);
771 if rc is True:
772 while cMsIdleFudge > 0:
773 cMsIdleFudge -= 1000;
774 time.sleep(1);
775 else:
776 self.oTransport.disconnect(self.fTryConnect);
777 return rc;
778
779 def taskBye(self):
780 """Says goodbye to the UTS"""
781 rc = self.sendMsg("BYE");
782 if rc is True:
783 rc = self.recvAckLogged("BYE");
784 self.oTransport.disconnect();
785 return rc;
786
787 #
788 # Gadget tasks.
789 #
790
791 def taskGadgetCreate(self, iGadgetType, iGadgetAccess, lstCfg = None):
792 """Creates a new gadget on UTS"""
793 cCfgItems = 0;
794 if lstCfg is not None:
795 cCfgItems = len(lstCfg);
796 fRc = self.sendMsg("GDGTCRT", (iGadgetType, iGadgetAccess, cCfgItems, 0, cfgListToByteArray(lstCfg)));
797 if fRc is True:
798 fRc = self.recvAckLogged("GDGTCRT");
799 return fRc;
800
801 def taskGadgetDestroy(self, iGadgetId):
802 """Destroys the given gadget handle on UTS"""
803 fRc = self.sendMsg("GDGTDTOR", (iGadgetId, zeroByteArray(12)));
804 if fRc is True:
805 fRc = self.recvAckLogged("GDGTDTOR");
806 return fRc;
807
808 def taskGadgetConnect(self, iGadgetId):
809 """Connects the given gadget handle on UTS"""
810 fRc = self.sendMsg("GDGTCNCT", (iGadgetId, zeroByteArray(12)));
811 if fRc is True:
812 fRc = self.recvAckLogged("GDGTCNCT");
813 return fRc;
814
815 def taskGadgetDisconnect(self, iGadgetId):
816 """Disconnects the given gadget handle from UTS"""
817 fRc = self.sendMsg("GDGTDCNT", (iGadgetId, zeroByteArray(12)));
818 if fRc is True:
819 fRc = self.recvAckLogged("GDGTDCNT");
820 return fRc;
821
822 #
823 # Public methods - generic task queries
824 #
825
826 def isSuccess(self):
827 """Returns True if the task completed successfully, otherwise False."""
828 self.lockTask();
829 sStatus = self.sStatus;
830 oTaskRc = self.oTaskRc;
831 self.unlockTask();
832 if sStatus != "":
833 return False;
834 if oTaskRc is False or oTaskRc is None:
835 return False;
836 return True;
837
838 def getResult(self):
839 """
840 Returns the result of a completed task.
841 Returns None if not completed yet or no previous task.
842 """
843 self.lockTask();
844 sStatus = self.sStatus;
845 oTaskRc = self.oTaskRc;
846 self.unlockTask();
847 if sStatus != "":
848 return None;
849 return oTaskRc;
850
851 def getLastReply(self):
852 """
853 Returns the last reply three-tuple: cbMsg, sOpcode, abPayload.
854 Returns a None, None, None three-tuple if there was no last reply.
855 """
856 self.lockTask();
857 t3oReply = self.t3oReply;
858 self.unlockTask();
859 return t3oReply;
860
861 #
862 # Public methods - connection.
863 #
864
865 def asyncDisconnect(self, cMsTimeout = 30000, fIgnoreErrors = False):
866 """
867 Initiates a disconnect task.
868
869 Returns True on success, False on failure (logged).
870
871 The task returns True on success and False on failure.
872 """
873 return self.startTask(cMsTimeout, fIgnoreErrors, "bye", self.taskBye);
874
875 def syncDisconnect(self, cMsTimeout = 30000, fIgnoreErrors = False):
876 """Synchronous version."""
877 return self.asyncToSync(self.asyncDisconnect, cMsTimeout, fIgnoreErrors);
878
879 #
880 # Public methods - gadget API
881 #
882
883 def asyncGadgetCreate(self, iGadgetType, iGadgetAccess, lstCfg = None, cMsTimeout = 30000, fIgnoreErrors = False):
884 """
885 Initiates a gadget create task.
886
887 Returns True on success, False on failure (logged).
888
889 The task returns True on success and False on failure.
890 """
891 return self.startTask(cMsTimeout, fIgnoreErrors, "GadgetCreate", self.taskGadgetCreate, \
892 (iGadgetType, iGadgetAccess, lstCfg));
893
894 def syncGadgetCreate(self, iGadgetType, iGadgetAccess, lstCfg = None, cMsTimeout = 30000, fIgnoreErrors = False):
895 """Synchronous version."""
896 return self.asyncToSync(self.asyncGadgetCreate, iGadgetType, iGadgetAccess, lstCfg, cMsTimeout, fIgnoreErrors);
897
898 def asyncGadgetDestroy(self, iGadgetId, cMsTimeout = 30000, fIgnoreErrors = False):
899 """
900 Initiates a gadget destroy task.
901
902 Returns True on success, False on failure (logged).
903
904 The task returns True on success and False on failure.
905 """
906 return self.startTask(cMsTimeout, fIgnoreErrors, "GadgetDestroy", self.taskGadgetDestroy, \
907 (iGadgetId, ));
908
909 def syncGadgetDestroy(self, iGadgetId, cMsTimeout = 30000, fIgnoreErrors = False):
910 """Synchronous version."""
911 return self.asyncToSync(self.asyncGadgetDestroy, iGadgetId, cMsTimeout, fIgnoreErrors);
912
913 def asyncGadgetConnect(self, iGadgetId, cMsTimeout = 30000, fIgnoreErrors = False):
914 """
915 Initiates a gadget connect task.
916
917 Returns True on success, False on failure (logged).
918
919 The task returns True on success and False on failure.
920 """
921 return self.startTask(cMsTimeout, fIgnoreErrors, "GadgetConnect", self.taskGadgetConnect, \
922 (iGadgetId, ));
923
924 def syncGadgetConnect(self, iGadgetId, cMsTimeout = 30000, fIgnoreErrors = False):
925 """Synchronous version."""
926 return self.asyncToSync(self.asyncGadgetConnect, iGadgetId, cMsTimeout, fIgnoreErrors);
927
928 def asyncGadgetDisconnect(self, iGadgetId, cMsTimeout = 30000, fIgnoreErrors = False):
929 """
930 Initiates a gadget disconnect task.
931
932 Returns True on success, False on failure (logged).
933
934 The task returns True on success and False on failure.
935 """
936 return self.startTask(cMsTimeout, fIgnoreErrors, "GadgetDisconnect", self.taskGadgetDisconnect, \
937 (iGadgetId, ));
938
939 def syncGadgetDisconnect(self, iGadgetId, cMsTimeout = 30000, fIgnoreErrors = False):
940 """Synchronous version."""
941 return self.asyncToSync(self.asyncGadgetDisconnect, iGadgetId, cMsTimeout, fIgnoreErrors);
942
943
944class TransportTcp(TransportBase):
945 """
946 TCP transport layer for the UTS client session class.
947 """
948
949 def __init__(self, sHostname, uPort):
950 """
951 Save the parameters. The session will call us back to make the
952 connection later on its worker thread.
953 """
954 TransportBase.__init__(self, utils.getCallerName());
955 self.sHostname = sHostname;
956 self.uPort = uPort if uPort is not None else 6042;
957 self.oSocket = None;
958 self.oWakeupW = None;
959 self.oWakeupR = None;
960 self.fConnectCanceled = False;
961 self.fIsConnecting = False;
962 self.oCv = threading.Condition();
963 self.abReadAhead = array.array('B');
964
965 def toString(self):
966 return '<%s sHostname=%s, uPort=%s, oSocket=%s,'\
967 ' fConnectCanceled=%s, fIsConnecting=%s, oCv=%s, abReadAhead=%s>' \
968 % (TransportBase.toString(self), self.sHostname, self.uPort, self.oSocket,
969 self.fConnectCanceled, self.fIsConnecting, self.oCv, self.abReadAhead);
970
971 def __isInProgressXcpt(self, oXcpt):
972 """ In progress exception? """
973 try:
974 if isinstance(oXcpt, socket.error):
975 try:
976 if oXcpt[0] == errno.EINPROGRESS:
977 return True;
978 except: pass;
979 # Windows?
980 try:
981 if oXcpt[0] == errno.EWOULDBLOCK:
982 return True;
983 except: pass;
984 except:
985 pass;
986 return False;
987
988 def __isWouldBlockXcpt(self, oXcpt):
989 """ Would block exception? """
990 try:
991 if isinstance(oXcpt, socket.error):
992 try:
993 if oXcpt[0] == errno.EWOULDBLOCK:
994 return True;
995 except: pass;
996 try:
997 if oXcpt[0] == errno.EAGAIN:
998 return True;
999 except: pass;
1000 except:
1001 pass;
1002 return False;
1003
1004 def __isConnectionReset(self, oXcpt):
1005 """ Connection reset by Peer or others. """
1006 try:
1007 if isinstance(oXcpt, socket.error):
1008 try:
1009 if oXcpt[0] == errno.ECONNRESET:
1010 return True;
1011 except: pass;
1012 try:
1013 if oXcpt[0] == errno.ENETRESET:
1014 return True;
1015 except: pass;
1016 except:
1017 pass;
1018 return False;
1019
1020 def _closeWakeupSockets(self):
1021 """ Closes the wakup sockets. Caller should own the CV. """
1022 oWakeupR = self.oWakeupR;
1023 self.oWakeupR = None;
1024 if oWakeupR is not None:
1025 oWakeupR.close();
1026
1027 oWakeupW = self.oWakeupW;
1028 self.oWakeupW = None;
1029 if oWakeupW is not None:
1030 oWakeupW.close();
1031
1032 return None;
1033
1034 def cancelConnect(self):
1035 # This is bad stuff.
1036 self.oCv.acquire();
1037 reporter.log2('TransportTcp::cancelConnect: fIsConnecting=%s oSocket=%s' % (self.fIsConnecting, self.oSocket));
1038 self.fConnectCanceled = True;
1039 if self.fIsConnecting:
1040 oSocket = self.oSocket;
1041 self.oSocket = None;
1042 if oSocket is not None:
1043 reporter.log2('TransportTcp::cancelConnect: closing the socket');
1044 oSocket.close();
1045
1046 oWakeupW = self.oWakeupW;
1047 self.oWakeupW = None;
1048 if oWakeupW is not None:
1049 reporter.log2('TransportTcp::cancelConnect: wakeup call');
1050 try: oWakeupW.send('cancelled!\n');
1051 except: reporter.logXcpt();
1052 try: oWakeupW.shutdown(socket.SHUT_WR);
1053 except: reporter.logXcpt();
1054 oWakeupW.close();
1055 self.oCv.release();
1056
1057 def _connectAsClient(self, oSocket, oWakeupR, cMsTimeout):
1058 """ Connects to the UTS server as client. """
1059
1060 # Connect w/ timeouts.
1061 rc = None;
1062 try:
1063 oSocket.connect((self.sHostname, self.uPort));
1064 rc = True;
1065 except socket.error, e:
1066 iRc = e[0];
1067 if self.__isInProgressXcpt(e):
1068 # Do the actual waiting.
1069 reporter.log2('TransportTcp::connect: operation in progress (%s)...' % (e,));
1070 try:
1071 ttRc = select.select([oWakeupR], [oSocket], [oSocket, oWakeupR], cMsTimeout / 1000.0);
1072 if len(ttRc[1]) + len(ttRc[2]) == 0:
1073 raise socket.error(errno.ETIMEDOUT, 'select timed out');
1074 iRc = oSocket.getsockopt(socket.SOL_SOCKET, socket.SO_ERROR);
1075 rc = iRc == 0;
1076 except socket.error, e:
1077 iRc = e[0];
1078 except:
1079 iRc = -42;
1080 reporter.fatalXcpt('socket.select() on connect failed');
1081
1082 if rc is True:
1083 pass;
1084 elif iRc == errno.ECONNREFUSED \
1085 or iRc == errno.EHOSTUNREACH \
1086 or iRc == errno.EINTR \
1087 or iRc == errno.ENETDOWN \
1088 or iRc == errno.ENETUNREACH \
1089 or iRc == errno.ETIMEDOUT:
1090 rc = False; # try again.
1091 else:
1092 if iRc != errno.EBADF or not self.fConnectCanceled:
1093 reporter.fatalXcpt('socket.connect((%s,%s)) failed; iRc=%s' % (self.sHostname, self.uPort, iRc));
1094 reporter.log2('TransportTcp::connect: rc=%s iRc=%s' % (rc, iRc));
1095 except:
1096 reporter.fatalXcpt('socket.connect((%s,%s)) failed' % (self.sHostname, self.uPort));
1097 return rc;
1098
1099
1100 def connect(self, cMsTimeout):
1101 # Create a non-blocking socket.
1102 reporter.log2('TransportTcp::connect: cMsTimeout=%s sHostname=%s uPort=%s' % (cMsTimeout, self.sHostname, self.uPort));
1103 try:
1104 oSocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM, 0);
1105 except:
1106 reporter.fatalXcpt('socket.socket() failed');
1107 return None;
1108 try:
1109 oSocket.setblocking(0);
1110 except:
1111 oSocket.close();
1112 reporter.fatalXcpt('socket.socket() failed');
1113 return None;
1114
1115 # Create wakeup socket pair for unix (select doesn't wake up on socket close on Linux).
1116 oWakeupR = None;
1117 oWakeupW = None;
1118 if hasattr(socket, 'socketpair'):
1119 try: (oWakeupR, oWakeupW) = socket.socketpair(); # pylint: disable=E1101
1120 except: reporter.logXcpt('socket.socketpair() failed');
1121
1122 # Update the state.
1123 self.oCv.acquire();
1124 rc = None;
1125 if not self.fConnectCanceled:
1126 self.oSocket = oSocket;
1127 self.oWakeupW = oWakeupW;
1128 self.oWakeupR = oWakeupR;
1129 self.fIsConnecting = True;
1130 self.oCv.release();
1131
1132 # Try connect.
1133 if oWakeupR is None:
1134 oWakeupR = oSocket; # Avoid select failure.
1135 rc = self._connectAsClient(oSocket, oWakeupR, cMsTimeout);
1136 oSocket = None;
1137
1138 # Update the state and cleanup on failure/cancel.
1139 self.oCv.acquire();
1140 if rc is True and self.fConnectCanceled:
1141 rc = False;
1142 self.fIsConnecting = False;
1143
1144 if rc is not True:
1145 if self.oSocket is not None:
1146 self.oSocket.close();
1147 self.oSocket = None;
1148 self._closeWakeupSockets();
1149 self.oCv.release();
1150
1151 reporter.log2('TransportTcp::connect: returning %s' % (rc,));
1152 return rc;
1153
1154 def disconnect(self, fQuiet = False):
1155 if self.oSocket is not None:
1156 self.abReadAhead = array.array('B');
1157
1158 # Try a shutting down the socket gracefully (draining it).
1159 try:
1160 self.oSocket.shutdown(socket.SHUT_WR);
1161 except:
1162 if not fQuiet:
1163 reporter.error('shutdown(SHUT_WR)');
1164 try:
1165 self.oSocket.setblocking(0); # just in case it's not set.
1166 sData = "1";
1167 while len(sData) > 0:
1168 sData = self.oSocket.recv(16384);
1169 except:
1170 pass;
1171
1172 # Close it.
1173 self.oCv.acquire();
1174 try: self.oSocket.setblocking(1);
1175 except: pass;
1176 self.oSocket.close();
1177 self.oSocket = None;
1178 else:
1179 self.oCv.acquire();
1180 self._closeWakeupSockets();
1181 self.oCv.release();
1182
1183 def sendBytes(self, abMsg, cMsTimeout):
1184 if self.oSocket is None:
1185 reporter.error('TransportTcp.sendBytes: No connection.');
1186 return False;
1187
1188 # Try send it all.
1189 try:
1190 cbSent = self.oSocket.send(abMsg);
1191 if cbSent == len(abMsg):
1192 return True;
1193 except Exception, oXcpt:
1194 if not self.__isWouldBlockXcpt(oXcpt):
1195 reporter.errorXcpt('TranportTcp.sendBytes: %s bytes' % (len(abMsg)));
1196 return False;
1197 cbSent = 0;
1198
1199 # Do a timed send.
1200 msStart = base.timestampMilli();
1201 while True:
1202 cMsElapsed = base.timestampMilli() - msStart;
1203 if cMsElapsed > cMsTimeout:
1204 reporter.error('TranportTcp.sendBytes: %s bytes timed out (1)' % (len(abMsg)));
1205 break;
1206
1207 # wait.
1208 try:
1209 ttRc = select.select([], [self.oSocket], [self.oSocket], (cMsTimeout - cMsElapsed) / 1000.0);
1210 if len(ttRc[2]) > 0 and len(ttRc[1]) == 0:
1211 reporter.error('TranportTcp.sendBytes: select returned with exception');
1212 break;
1213 if len(ttRc[1]) == 0:
1214 reporter.error('TranportTcp.sendBytes: %s bytes timed out (2)' % (len(abMsg)));
1215 break;
1216 except:
1217 reporter.errorXcpt('TranportTcp.sendBytes: select failed');
1218 break;
1219
1220 # Try send more.
1221 try:
1222 cbSent += self.oSocket.send(abMsg[cbSent:]);
1223 if cbSent == len(abMsg):
1224 return True;
1225 except Exception, oXcpt:
1226 if not self.__isWouldBlockXcpt(oXcpt):
1227 reporter.errorXcpt('TranportTcp.sendBytes: %s bytes' % (len(abMsg)));
1228 break;
1229
1230 return False;
1231
1232 def __returnReadAheadBytes(self, cb):
1233 """ Internal worker for recvBytes. """
1234 assert(len(self.abReadAhead) >= cb);
1235 abRet = self.abReadAhead[:cb];
1236 self.abReadAhead = self.abReadAhead[cb:];
1237 return abRet;
1238
1239 def recvBytes(self, cb, cMsTimeout, fNoDataOk):
1240 if self.oSocket is None:
1241 reporter.error('TransportTcp.recvBytes(%s,%s): No connection.' % (cb, cMsTimeout));
1242 return None;
1243
1244 # Try read in some more data without bothering with timeout handling first.
1245 if len(self.abReadAhead) < cb:
1246 try:
1247 abBuf = self.oSocket.recv(cb - len(self.abReadAhead));
1248 if len(abBuf) > 0:
1249 self.abReadAhead.extend(array.array('B', abBuf));
1250 except Exception, oXcpt:
1251 if not self.__isWouldBlockXcpt(oXcpt):
1252 reporter.errorXcpt('TranportTcp.recvBytes: 0/%s bytes' % (cb,));
1253 return None;
1254
1255 if len(self.abReadAhead) >= cb:
1256 return self.__returnReadAheadBytes(cb);
1257
1258 # Timeout loop.
1259 msStart = base.timestampMilli();
1260 while True:
1261 cMsElapsed = base.timestampMilli() - msStart;
1262 if cMsElapsed > cMsTimeout:
1263 if not fNoDataOk or len(self.abReadAhead) > 0:
1264 reporter.error('TranportTcp.recvBytes: %s/%s bytes timed out (1)' % (len(self.abReadAhead), cb));
1265 break;
1266
1267 # Wait.
1268 try:
1269 ttRc = select.select([self.oSocket], [], [self.oSocket], (cMsTimeout - cMsElapsed) / 1000.0);
1270 if len(ttRc[2]) > 0 and len(ttRc[0]) == 0:
1271 reporter.error('TranportTcp.recvBytes: select returned with exception');
1272 break;
1273 if len(ttRc[0]) == 0:
1274 if not fNoDataOk or len(self.abReadAhead) > 0:
1275 reporter.error('TranportTcp.recvBytes: %s/%s bytes timed out (2) fNoDataOk=%s'
1276 % (len(self.abReadAhead), cb, fNoDataOk));
1277 break;
1278 except:
1279 reporter.errorXcpt('TranportTcp.recvBytes: select failed');
1280 break;
1281
1282 # Try read more.
1283 try:
1284 abBuf = self.oSocket.recv(cb - len(self.abReadAhead));
1285 if len(abBuf) == 0:
1286 reporter.error('TranportTcp.recvBytes: %s/%s bytes (%s) - connection has been shut down'
1287 % (len(self.abReadAhead), cb, fNoDataOk));
1288 self.disconnect();
1289 return None;
1290
1291 self.abReadAhead.extend(array.array('B', abBuf));
1292
1293 except Exception, oXcpt:
1294 reporter.log('recv => exception %s' % (oXcpt,));
1295 if not self.__isWouldBlockXcpt(oXcpt):
1296 if not fNoDataOk or not self.__isConnectionReset(oXcpt) or len(self.abReadAhead) > 0:
1297 reporter.errorXcpt('TranportTcp.recvBytes: %s/%s bytes (%s)' % (len(self.abReadAhead), cb, fNoDataOk));
1298 break;
1299
1300 # Done?
1301 if len(self.abReadAhead) >= cb:
1302 return self.__returnReadAheadBytes(cb);
1303
1304 #reporter.log('recv => None len(self.abReadAhead) -> %d' % (len(self.abReadAhead), ));
1305 return None;
1306
1307 def isConnectionOk(self):
1308 if self.oSocket is None:
1309 return False;
1310 try:
1311 ttRc = select.select([], [], [self.oSocket], 0.0);
1312 if len(ttRc[2]) > 0:
1313 return False;
1314
1315 self.oSocket.send(array.array('B')); # send zero bytes.
1316 except:
1317 return False;
1318 return True;
1319
1320 def isRecvPending(self, cMsTimeout = 0):
1321 try:
1322 ttRc = select.select([self.oSocket], [], [], cMsTimeout / 1000.0);
1323 if len(ttRc[0]) == 0:
1324 return False;
1325 except:
1326 pass;
1327 return True;
1328
1329
1330class UsbGadget(object):
1331 """
1332 USB Gadget control class using the USBT Test Service to talk to the external
1333 board behaving like a USB device.
1334 """
1335
1336 def __init__(self):
1337 self.oUtsSession = None;
1338 self.sImpersonation = g_ksGadgetImpersonationInvalid;
1339 self.idGadget = None;
1340 self.iBusId = None;
1341 self.iDevId = None;
1342 self.iUsbIpPort = None;
1343
1344 def clearImpersonation(self):
1345 """
1346 Removes the current impersonation of the gadget.
1347 """
1348 fRc = True;
1349
1350 if self.idGadget is not None:
1351 fRc = self.oUtsSession.syncGadgetDestroy(self.idGadget);
1352 self.idGadget = None;
1353 self.iBusId = None;
1354 self.iDevId = None;
1355
1356 return fRc;
1357
1358 def disconnectUsb(self):
1359 """
1360 Disconnects the USB gadget from the host. (USB connection not network
1361 connection used for control)
1362 """
1363 return self.oUtsSession.syncGadgetDisconnect(self.idGadget);
1364
1365 def connectUsb(self):
1366 """
1367 Connect the USB gadget to the host.
1368 """
1369 return self.oUtsSession.syncGadgetConnect(self.idGadget);
1370
1371 def impersonate(self, sImpersonation, fSuperSpeed = False):
1372 """
1373 Impersonate a given device.
1374 """
1375
1376 # Clear any previous impersonation
1377 self.clearImpersonation();
1378 self.sImpersonation = sImpersonation;
1379
1380 fRc = False;
1381 if sImpersonation == g_ksGadgetImpersonationTest:
1382 lstCfg = [];
1383 if fSuperSpeed is True:
1384 lstCfg.append( ('Gadget/SuperSpeed', g_kiGadgetCfgTypeBool, 'true') );
1385 fDone = self.oUtsSession.syncGadgetCreate(g_kiGadgetTypeTest, g_kiGadgetAccessUsbIp, lstCfg);
1386 if fDone is True and self.oUtsSession.isSuccess():
1387 # Get the gadget ID.
1388 _, _, abPayload = self.oUtsSession.getLastReply();
1389
1390 fRc = True;
1391 self.idGadget = getU32(abPayload, 16);
1392 self.iBusId = getU32(abPayload, 20);
1393 self.iDevId = getU32(abPayload, 24);
1394 else:
1395 reporter.log('Invalid or unsupported impersonation');
1396
1397 return fRc;
1398
1399 def getUsbIpPort(self):
1400 """
1401 Returns the port the USB/IP server is listening on if requested,
1402 None if USB/IP is not supported.
1403 """
1404 return self.iUsbIpPort;
1405
1406 def getGadgetBusAndDevId(self):
1407 """
1408 Returns the bus ad device ID of the gadget as a tuple.
1409 """
1410 return (self.iBusId, self.iDevId);
1411
1412 def connectTo(self, cMsTimeout, sHostname, uPort = None, fUsbIpSupport = True, cMsIdleFudge = 0):
1413 """
1414 Connects to the specified target device.
1415 Returns True on Success.
1416 Returns False otherwise.
1417 """
1418 fRc = True;
1419
1420 # @todo
1421 if fUsbIpSupport is False:
1422 return False;
1423
1424 reporter.log2('openTcpSession(%s, %s, %s, %s)' % \
1425 (cMsTimeout, sHostname, uPort, cMsIdleFudge));
1426 try:
1427 oTransport = TransportTcp(sHostname, uPort);
1428 self.oUtsSession = Session(oTransport, cMsTimeout, cMsIdleFudge);
1429
1430 if self.oUtsSession is not None:
1431 fDone = self.oUtsSession.waitForTask(30*1000);
1432 print 'connect: waitForTask -> %s, result %s' % (fDone, self.oUtsSession.getResult());
1433 if fDone is True and self.oUtsSession.isSuccess():
1434 # Parse the reply.
1435 _, _, abPayload = self.oUtsSession.getLastReply();
1436
1437 if getU32(abPayload, 20) is g_kiGadgetAccessUsbIp:
1438 fRc = True;
1439 self.iUsbIpPort = getU32(abPayload, 24);
1440 else:
1441 reporter.log('Gadget doesn\'t support access over USB/IP despite being requested');
1442 fRc = False;
1443 else:
1444 fRc = False;
1445 else:
1446 fRc = False;
1447 except:
1448 reporter.errorXcpt(None, 15);
1449 return False;
1450
1451 return fRc;
1452
1453 def disconnectFrom(self):
1454 """
1455 Disconnects from the target device.
1456 """
1457 fRc = True;
1458
1459 self.clearImpersonation();
1460 if self.oUtsSession is not None:
1461 fRc = self.oUtsSession.syncDisconnect();
1462
1463 return fRc;
1464
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette