VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/utils/usb/UsbTestService.cpp@ 61265

Last change on this file since 61265 was 60793, checked in by vboxsync, 9 years ago

ValidationKit/UsbTestService: Updates, implement the protocol bits for passing extended configuration from the client to the gadget, detect multiple busses from dummy_hcd which creates a separate bus for high-speed and super-speed gadgets, some smaller fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 56.0 KB
Line 
1/* $Id: UsbTestService.cpp 60793 2016-05-02 16:12:43Z vboxsync $ */
2/** @file
3 * UsbTestService - Remote USB test configuration and execution server.
4 */
5
6/*
7 * Copyright (C) 2010-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define LOG_GROUP RTLOGGROUP_DEFAULT
32#include <iprt/alloca.h>
33#include <iprt/asm.h>
34#include <iprt/assert.h>
35#include <iprt/critsect.h>
36#include <iprt/crc.h>
37#include <iprt/ctype.h>
38#include <iprt/dir.h>
39#include <iprt/env.h>
40#include <iprt/err.h>
41#include <iprt/getopt.h>
42#include <iprt/handle.h>
43#include <iprt/initterm.h>
44#include <iprt/list.h>
45#include <iprt/log.h>
46#include <iprt/mem.h>
47#include <iprt/message.h>
48#include <iprt/param.h>
49#include <iprt/path.h>
50#include <iprt/pipe.h>
51#include <iprt/poll.h>
52#include <iprt/process.h>
53#include <iprt/stream.h>
54#include <iprt/string.h>
55#include <iprt/thread.h>
56
57#include "UsbTestServiceCfg.h"
58#include "UsbTestServiceInternal.h"
59#include "UsbTestServiceGadget.h"
60#include "UsbTestServicePlatform.h"
61
62
63
64/*********************************************************************************************************************************
65* Structures and Typedefs *
66*********************************************************************************************************************************/
67
68#define UTS_USBIP_PORT_FIRST 3240
69#define UTS_USBIP_PORT_LAST 3340
70
71/**
72 * UTS client state.
73 */
74typedef enum UTSCLIENTSTATE
75{
76 /** Invalid client state. */
77 UTSCLIENTSTATE_INVALID = 0,
78 /** Client is initialising, only the HOWDY and BYE packets are allowed. */
79 UTSCLIENTSTATE_INITIALISING,
80 /** Client is in fully cuntional state and ready to process all requests. */
81 UTSCLIENTSTATE_READY,
82 /** Client is destroying. */
83 UTSCLIENTSTATE_DESTROYING,
84 /** 32bit hack. */
85 UTSCLIENTSTATE_32BIT_HACK = 0x7fffffff
86} UTSCLIENTSTATE;
87
88/**
89 * UTS client instance.
90 */
91typedef struct UTSCLIENT
92{
93 /** List node for new clients. */
94 RTLISTNODE NdLst;
95 /** The current client state. */
96 UTSCLIENTSTATE enmState;
97 /** Transport backend specific data. */
98 PUTSTRANSPORTCLIENT pTransportClient;
99 /** Client hostname. */
100 char *pszHostname;
101 /** Gadget host handle. */
102 UTSGADGETHOST hGadgetHost;
103 /** Handle fo the current configured gadget. */
104 UTSGADGET hGadget;
105} UTSCLIENT;
106/** Pointer to a UTS client instance. */
107typedef UTSCLIENT *PUTSCLIENT;
108
109/*********************************************************************************************************************************
110* Global Variables *
111*********************************************************************************************************************************/
112/**
113 * Transport layers.
114 */
115static const PCUTSTRANSPORT g_apTransports[] =
116{
117 &g_TcpTransport,
118 //&g_SerialTransport,
119 //&g_FileSysTransport,
120 //&g_GuestPropTransport,
121 //&g_TestDevTransport,
122};
123
124/** The select transport layer. */
125static PCUTSTRANSPORT g_pTransport;
126/** The config path. */
127static char g_szCfgPath[RTPATH_MAX];
128/** The scratch path. */
129static char g_szScratchPath[RTPATH_MAX];
130/** The default scratch path. */
131static char g_szDefScratchPath[RTPATH_MAX];
132/** The CD/DVD-ROM path. */
133static char g_szCdRomPath[RTPATH_MAX];
134/** The default CD/DVD-ROM path. */
135static char g_szDefCdRomPath[RTPATH_MAX];
136/** The operating system short name. */
137static char g_szOsShortName[16];
138/** The CPU architecture short name. */
139static char g_szArchShortName[16];
140/** The combined "OS.arch" name. */
141static char g_szOsDotArchShortName[32];
142/** The combined "OS/arch" name. */
143static char g_szOsSlashArchShortName[32];
144/** The executable suffix. */
145static char g_szExeSuff[8];
146/** The shell script suffix. */
147static char g_szScriptSuff[8];
148/** Whether to display the output of the child process or not. */
149static bool g_fDisplayOutput = true;
150/** Whether to terminate or not.
151 * @todo implement signals and stuff. */
152static bool volatile g_fTerminate = false;
153/** Configuration AST. */
154static PCFGAST g_pCfgAst = NULL;
155/** Pipe for communicating with the serving thread about new clients. - read end */
156static RTPIPE g_hPipeR;
157/** Pipe for communicating with the serving thread about new clients. - write end */
158static RTPIPE g_hPipeW;
159/** Thread serving connected clients. */
160static RTTHREAD g_hThreadServing;
161/** Critical section protecting the list of new clients. */
162static RTCRITSECT g_CritSectClients;
163/** List of new clients waiting to be picked up by the client worker thread. */
164static RTLISTANCHOR g_LstClientsNew;
165/** First USB/IP port we can use. */
166static uint16_t g_uUsbIpPortFirst = UTS_USBIP_PORT_FIRST;
167/** Last USB/IP port we can use. */
168static uint16_t g_uUsbIpPortLast = UTS_USBIP_PORT_LAST;
169/** Next free port. */
170static uint16_t g_uUsbIpPortNext = UTS_USBIP_PORT_FIRST;
171
172
173
174/**
175 * Returns the string represenation of the given state.
176 */
177static const char *utsClientStateStringify(UTSCLIENTSTATE enmState)
178{
179 switch (enmState)
180 {
181 case UTSCLIENTSTATE_INVALID:
182 return "INVALID";
183 case UTSCLIENTSTATE_INITIALISING:
184 return "INITIALISING";
185 case UTSCLIENTSTATE_READY:
186 return "READY";
187 case UTSCLIENTSTATE_DESTROYING:
188 return "DESTROYING";
189 case UTSCLIENTSTATE_32BIT_HACK:
190 default:
191 break;
192 }
193
194 AssertMsgFailed(("Unknown state %#x\n", enmState));
195 return "UNKNOWN";
196}
197
198/**
199 * Calculates the checksum value, zero any padding space and send the packet.
200 *
201 * @returns IPRT status code.
202 * @param pClient The UTS client structure.
203 * @param pPkt The packet to send. Must point to a correctly
204 * aligned buffer.
205 */
206static int utsSendPkt(PUTSCLIENT pClient, PUTSPKTHDR pPkt)
207{
208 Assert(pPkt->cb >= sizeof(*pPkt));
209 pPkt->uCrc32 = RTCrc32(pPkt->achOpcode, pPkt->cb - RT_OFFSETOF(UTSPKTHDR, achOpcode));
210 if (pPkt->cb != RT_ALIGN_32(pPkt->cb, UTSPKT_ALIGNMENT))
211 memset((uint8_t *)pPkt + pPkt->cb, '\0', RT_ALIGN_32(pPkt->cb, UTSPKT_ALIGNMENT) - pPkt->cb);
212
213 Log(("utsSendPkt: cb=%#x opcode=%.8s\n", pPkt->cb, pPkt->achOpcode));
214 Log2(("%.*Rhxd\n", RT_MIN(pPkt->cb, 256), pPkt));
215 int rc = g_pTransport->pfnSendPkt(pClient->pTransportClient, pPkt);
216 while (RT_UNLIKELY(rc == VERR_INTERRUPTED) && !g_fTerminate)
217 rc = g_pTransport->pfnSendPkt(pClient->pTransportClient, pPkt);
218 if (RT_FAILURE(rc))
219 Log(("utsSendPkt: rc=%Rrc\n", rc));
220
221 return rc;
222}
223
224/**
225 * Sends a babble reply and disconnects the client (if applicable).
226 *
227 * @param pClient The UTS client structure.
228 * @param pszOpcode The BABBLE opcode.
229 */
230static void utsReplyBabble(PUTSCLIENT pClient, const char *pszOpcode)
231{
232 UTSPKTHDR Reply;
233 Reply.cb = sizeof(Reply);
234 Reply.uCrc32 = 0;
235 memcpy(Reply.achOpcode, pszOpcode, sizeof(Reply.achOpcode));
236
237 g_pTransport->pfnBabble(pClient->pTransportClient, &Reply, 20*1000);
238}
239
240/**
241 * Receive and validate a packet.
242 *
243 * Will send bable responses to malformed packets that results in a error status
244 * code.
245 *
246 * @returns IPRT status code.
247 * @param pClient The UTS client structure.
248 * @param ppPktHdr Where to return the packet on success. Free
249 * with RTMemFree.
250 * @param fAutoRetryOnFailure Whether to retry on error.
251 */
252static int utsRecvPkt(PUTSCLIENT pClient, PPUTSPKTHDR ppPktHdr, bool fAutoRetryOnFailure)
253{
254 for (;;)
255 {
256 PUTSPKTHDR pPktHdr;
257 int rc = g_pTransport->pfnRecvPkt(pClient->pTransportClient, &pPktHdr);
258 if (RT_SUCCESS(rc))
259 {
260 /* validate the packet. */
261 if ( pPktHdr->cb >= sizeof(UTSPKTHDR)
262 && pPktHdr->cb < UTSPKT_MAX_SIZE)
263 {
264 Log2(("utsRecvPkt: pPktHdr=%p cb=%#x crc32=%#x opcode=%.8s\n"
265 "%.*Rhxd\n",
266 pPktHdr, pPktHdr->cb, pPktHdr->uCrc32, pPktHdr->achOpcode, RT_MIN(pPktHdr->cb, 256), pPktHdr));
267 uint32_t uCrc32Calc = pPktHdr->uCrc32 != 0
268 ? RTCrc32(&pPktHdr->achOpcode[0], pPktHdr->cb - RT_OFFSETOF(UTSPKTHDR, achOpcode))
269 : 0;
270 if (pPktHdr->uCrc32 == uCrc32Calc)
271 {
272 AssertCompileMemberSize(UTSPKTHDR, achOpcode, 8);
273 if ( RT_C_IS_UPPER(pPktHdr->achOpcode[0])
274 && RT_C_IS_UPPER(pPktHdr->achOpcode[1])
275 && (RT_C_IS_UPPER(pPktHdr->achOpcode[2]) || pPktHdr->achOpcode[2] == ' ')
276 && (RT_C_IS_PRINT(pPktHdr->achOpcode[3]) || pPktHdr->achOpcode[3] == ' ')
277 && (RT_C_IS_PRINT(pPktHdr->achOpcode[4]) || pPktHdr->achOpcode[4] == ' ')
278 && (RT_C_IS_PRINT(pPktHdr->achOpcode[5]) || pPktHdr->achOpcode[5] == ' ')
279 && (RT_C_IS_PRINT(pPktHdr->achOpcode[6]) || pPktHdr->achOpcode[6] == ' ')
280 && (RT_C_IS_PRINT(pPktHdr->achOpcode[7]) || pPktHdr->achOpcode[7] == ' ')
281 )
282 {
283 Log(("utsRecvPkt: cb=%#x opcode=%.8s\n", pPktHdr->cb, pPktHdr->achOpcode));
284 *ppPktHdr = pPktHdr;
285 return rc;
286 }
287
288 rc = VERR_IO_BAD_COMMAND;
289 }
290 else
291 {
292 Log(("utsRecvPkt: cb=%#x opcode=%.8s crc32=%#x actual=%#x\n",
293 pPktHdr->cb, pPktHdr->achOpcode, pPktHdr->uCrc32, uCrc32Calc));
294 rc = VERR_IO_CRC;
295 }
296 }
297 else
298 rc = VERR_IO_BAD_LENGTH;
299
300 /* Send babble reply and disconnect the client if the transport is
301 connection oriented. */
302 if (rc == VERR_IO_BAD_LENGTH)
303 utsReplyBabble(pClient, "BABBLE L");
304 else if (rc == VERR_IO_CRC)
305 utsReplyBabble(pClient, "BABBLE C");
306 else if (rc == VERR_IO_BAD_COMMAND)
307 utsReplyBabble(pClient, "BABBLE O");
308 else
309 utsReplyBabble(pClient, "BABBLE ");
310 RTMemFree(pPktHdr);
311 }
312
313 /* Try again or return failure? */
314 if ( g_fTerminate
315 || rc != VERR_INTERRUPTED
316 || !fAutoRetryOnFailure
317 )
318 {
319 Log(("utsRecvPkt: rc=%Rrc\n", rc));
320 return rc;
321 }
322 }
323}
324
325/**
326 * Make a simple reply, only status opcode.
327 *
328 * @returns IPRT status code of the send.
329 * @param pClient The UTS client structure.
330 * @param pReply The reply packet.
331 * @param pszOpcode The status opcode. Exactly 8 chars long, padd
332 * with space.
333 * @param cbExtra Bytes in addition to the header.
334 */
335static int utsReplyInternal(PUTSCLIENT pClient, PUTSPKTSTS pReply, const char *pszOpcode, size_t cbExtra)
336{
337 /* copy the opcode, don't be too strict in case of a padding screw up. */
338 size_t cchOpcode = strlen(pszOpcode);
339 if (RT_LIKELY(cchOpcode == sizeof(pReply->Hdr.achOpcode)))
340 memcpy(pReply->Hdr.achOpcode, pszOpcode, sizeof(pReply->Hdr.achOpcode));
341 else
342 {
343 Assert(cchOpcode == sizeof(pReply->Hdr.achOpcode));
344 while (cchOpcode > 0 && pszOpcode[cchOpcode - 1] == ' ')
345 cchOpcode--;
346 AssertMsgReturn(cchOpcode < sizeof(pReply->Hdr.achOpcode), ("%d/'%.8s'\n", cchOpcode, pszOpcode), VERR_INTERNAL_ERROR_4);
347 memcpy(pReply->Hdr.achOpcode, pszOpcode, cchOpcode);
348 memset(&pReply->Hdr.achOpcode[cchOpcode], ' ', sizeof(pReply->Hdr.achOpcode) - cchOpcode);
349 }
350
351 pReply->Hdr.cb = (uint32_t)sizeof(UTSPKTSTS) + (uint32_t)cbExtra;
352 pReply->Hdr.uCrc32 = 0;
353
354 return utsSendPkt(pClient, &pReply->Hdr);
355}
356
357/**
358 * Make a simple reply, only status opcode.
359 *
360 * @returns IPRT status code of the send.
361 * @param pClient The UTS client structure.
362 * @param pPktHdr The original packet (for future use).
363 * @param pszOpcode The status opcode. Exactly 8 chars long, padd
364 * with space.
365 */
366static int utsReplySimple(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr, const char *pszOpcode)
367{
368 UTSPKTSTS Pkt;
369
370 RT_ZERO(Pkt);
371 Pkt.rcReq = VINF_SUCCESS;
372 Pkt.cchStsMsg = 0;
373 NOREF(pPktHdr);
374 return utsReplyInternal(pClient, &Pkt, pszOpcode, 0);
375}
376
377/**
378 * Acknowledges a packet with success.
379 *
380 * @returns IPRT status code of the send.
381 * @param pClient The UTS client structure.
382 * @param pPktHdr The original packet (for future use).
383 */
384static int utsReplyAck(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
385{
386 return utsReplySimple(pClient, pPktHdr, "ACK ");
387}
388
389/**
390 * Replies with a failure.
391 *
392 * @returns IPRT status code of the send.
393 * @param pClient The UTS client structure.
394 * @param pPktHdr The original packet (for future use).
395 * @param rcReq Status code.
396 * @param pszOpcode The status opcode. Exactly 8 chars long, padd
397 * with space.
398 * @param pszDetailsFmt Longer description of the problem (format
399 * string).
400 * @param va Format arguments.
401 */
402static int utsReplyFailureV(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr, const char *pszOpcode, int rcReq, const char *pszDetailFmt, va_list va)
403{
404 NOREF(pPktHdr);
405 union
406 {
407 UTSPKTSTS Hdr;
408 char ach[256];
409 } uPkt;
410
411 RT_ZERO(uPkt);
412 size_t cchDetail = RTStrPrintfV(&uPkt.ach[sizeof(UTSPKTSTS)],
413 sizeof(uPkt) - sizeof(UTSPKTSTS),
414 pszDetailFmt, va);
415 uPkt.Hdr.rcReq = rcReq;
416 uPkt.Hdr.cchStsMsg = cchDetail;
417 return utsReplyInternal(pClient, &uPkt.Hdr, pszOpcode, cchDetail + 1);
418}
419
420/**
421 * Replies with a failure.
422 *
423 * @returns IPRT status code of the send.
424 * @param pClient The UTS client structure.
425 * @param pPktHdr The original packet (for future use).
426 * @param rcReq Status code.
427 * @param pszOpcode The status opcode. Exactly 8 chars long, padd
428 * with space.
429 * @param pszDetails Longer description of the problem (format
430 * string).
431 * @param ... Format arguments.
432 */
433static int utsReplyFailure(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr, const char *pszOpcode, int rcReq, const char *pszDetailFmt, ...)
434{
435 va_list va;
436 va_start(va, pszDetailFmt);
437 int rc = utsReplyFailureV(pClient, pPktHdr, pszOpcode, rcReq, pszDetailFmt, va);
438 va_end(va);
439 return rc;
440}
441
442/**
443 * Replies according to the return code.
444 *
445 * @returns IPRT status code of the send.
446 * @param pClient The UTS client structure.
447 * @param pPktHdr The packet to reply to.
448 * @param rcOperation The status code to report.
449 * @param pszOperationFmt The operation that failed. Typically giving the
450 * function call with important arguments.
451 * @param ... Arguments to the format string.
452 */
453static int utsReplyRC(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr, int rcOperation, const char *pszOperationFmt, ...)
454{
455 if (RT_SUCCESS(rcOperation))
456 return utsReplyAck(pClient, pPktHdr);
457
458 char szOperation[128];
459 va_list va;
460 va_start(va, pszOperationFmt);
461 RTStrPrintfV(szOperation, sizeof(szOperation), pszOperationFmt, va);
462 va_end(va);
463
464 return utsReplyFailure(pClient, pPktHdr, "FAILED ", rcOperation, "%s failed with rc=%Rrc (opcode '%.8s')",
465 szOperation, rcOperation, pPktHdr->achOpcode);
466}
467
468/**
469 * Signal a bad packet minum size.
470 *
471 * @returns IPRT status code of the send.
472 * @param pClient The UTS client structure.
473 * @param pPktHdr The packet to reply to.
474 * @param cbMin The minimum size.
475 */
476static int utsReplyBadMinSize(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr, size_t cbMin)
477{
478 return utsReplyFailure(pClient, pPktHdr, "BAD SIZE", VERR_INVALID_PARAMETER, "Expected at least %zu bytes, got %u (opcode '%.8s')",
479 cbMin, pPktHdr->cb, pPktHdr->achOpcode);
480}
481
482/**
483 * Signal a bad packet exact size.
484 *
485 * @returns IPRT status code of the send.
486 * @param pClient The UTS client structure.
487 * @param pPktHdr The packet to reply to.
488 * @param cb The wanted size.
489 */
490static int utsReplyBadSize(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr, size_t cb)
491{
492 return utsReplyFailure(pClient, pPktHdr, "BAD SIZE", VERR_INVALID_PARAMETER, "Expected at %zu bytes, got %u (opcode '%.8s')",
493 cb, pPktHdr->cb, pPktHdr->achOpcode);
494}
495
496/**
497 * Deals with a command that isn't implemented yet.
498 * @returns IPRT status code of the send.
499 * @param pClient The UTS client structure.
500 * @param pPktHdr The packet which opcode isn't implemented.
501 */
502static int utsReplyNotImplemented(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
503{
504 return utsReplyFailure(pClient, pPktHdr, "NOT IMPL", VERR_NOT_IMPLEMENTED, "Opcode '%.8s' is not implemented", pPktHdr->achOpcode);
505}
506
507/**
508 * Deals with a unknown command.
509 * @returns IPRT status code of the send.
510 * @param pClient The UTS client structure.
511 * @param pPktHdr The packet to reply to.
512 */
513static int utsReplyUnknown(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
514{
515 return utsReplyFailure(pClient, pPktHdr, "UNKNOWN ", VERR_NOT_FOUND, "Opcode '%.8s' is not known", pPktHdr->achOpcode);
516}
517
518/**
519 * Deals with a command which contains an unterminated string.
520 *
521 * @returns IPRT status code of the send.
522 * @param pClient The UTS client structure.
523 * @param pPktHdr The packet containing the unterminated string.
524 */
525static int utsReplyBadStrTermination(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
526{
527 return utsReplyFailure(pClient, pPktHdr, "BAD TERM", VERR_INVALID_PARAMETER, "Opcode '%.8s' contains an unterminated string", pPktHdr->achOpcode);
528}
529
530/**
531 * Deals with a command sent in an invalid client state.
532 *
533 * @returns IPRT status code of the send.
534 * @param pClient The UTS client structure.
535 * @param pPktHdr The packet containing the unterminated string.
536 */
537static int utsReplyInvalidState(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
538{
539 return utsReplyFailure(pClient, pPktHdr, "INVSTATE", VERR_INVALID_STATE, "Opcode '%.8s' is not supported at client state '%s",
540 pPktHdr->achOpcode, utsClientStateStringify(pClient->enmState));
541}
542
543/**
544 * Parses an unsigned integer from the given value string.
545 *
546 * @returns IPRT status code.
547 * @retval VERR_OUT_OF_RANGE if the parsed value exceeds the given maximum.
548 * @param uMax The maximum value.
549 * @param pu64 Where to store the parsed number on success.
550 */
551static int utsDoGadgetCreateCfgParseUInt(const char *pszVal, uint64_t uMax, uint64_t *pu64)
552{
553 int rc = RTStrToUInt64Ex(pszVal, NULL, 0, pu64);
554 if (RT_SUCCESS(rc))
555 {
556 if (*pu64 > uMax)
557 rc = VERR_OUT_OF_RANGE;
558 }
559
560 return rc;
561}
562
563/**
564 * Parses a signed integer from the given value string.
565 *
566 * @returns IPRT status code.
567 * @retval VERR_OUT_OF_RANGE if the parsed value exceeds the given range.
568 * @param iMin The minimum value.
569 * @param iMax The maximum value.
570 * @param pi64 Where to store the parsed number on success.
571 */
572static int utsDoGadgetCreateCfgParseInt(const char *pszVal, int64_t iMin, int64_t iMax, int64_t *pi64)
573{
574 int rc = RTStrToInt64Ex(pszVal, NULL, 0, pi64);
575 if (RT_SUCCESS(rc))
576 {
577 if ( *pi64 < iMin
578 || *pi64 > iMax)
579 rc = VERR_OUT_OF_RANGE;
580 }
581
582 return rc;
583}
584
585/**
586 * Parses the given config item and fills in the value according to the given type.
587 *
588 * @returns IPRT status code.
589 * @param pCfgItem The config item to parse.
590 * @param u32Type The config type.
591 * @param pszVal The value encoded as a string.
592 */
593static int utsDoGadgetCreateCfgParseItem(PUTSGADGETCFGITEM pCfgItem, uint32_t u32Type,
594 const char *pszVal)
595{
596 int rc = VINF_SUCCESS;
597
598 switch (u32Type)
599 {
600 case UTSPKT_GDGT_CFG_ITEM_TYPE_BOOLEAN:
601 {
602 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_BOOLEAN;
603 if ( RTStrICmp(pszVal, "enabled")
604 || RTStrICmp(pszVal, "1")
605 || RTStrICmp(pszVal, "true"))
606 pCfgItem->Val.u.f = true;
607 else if ( RTStrICmp(pszVal, "disabled")
608 || RTStrICmp(pszVal, "0")
609 || RTStrICmp(pszVal, "false"))
610 pCfgItem->Val.u.f = false;
611 else
612 rc = VERR_INVALID_PARAMETER;
613 break;
614 }
615 case UTSPKT_GDGT_CFG_ITEM_TYPE_STRING:
616 {
617 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_STRING;
618 pCfgItem->Val.u.psz = RTStrDup(pszVal);
619 if (!pCfgItem->Val.u.psz)
620 rc = VERR_NO_STR_MEMORY;
621 break;
622 }
623 case UTSPKT_GDGT_CFG_ITEM_TYPE_UINT8:
624 {
625 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_UINT8;
626
627 uint64_t u64;
628 rc = utsDoGadgetCreateCfgParseUInt(pszVal, UINT8_MAX, &u64);
629 if (RT_SUCCESS(rc))
630 pCfgItem->Val.u.u8 = (uint8_t)u64;
631 break;
632 }
633 case UTSPKT_GDGT_CFG_ITEM_TYPE_UINT16:
634 {
635 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_UINT16;
636
637 uint64_t u64;
638 rc = utsDoGadgetCreateCfgParseUInt(pszVal, UINT16_MAX, &u64);
639 if (RT_SUCCESS(rc))
640 pCfgItem->Val.u.u16 = (uint16_t)u64;
641 break;
642 }
643 case UTSPKT_GDGT_CFG_ITEM_TYPE_UINT32:
644 {
645 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_UINT32;
646
647 uint64_t u64;
648 rc = utsDoGadgetCreateCfgParseUInt(pszVal, UINT32_MAX, &u64);
649 if (RT_SUCCESS(rc))
650 pCfgItem->Val.u.u32 = (uint32_t)u64;
651 break;
652 }
653 case UTSPKT_GDGT_CFG_ITEM_TYPE_UINT64:
654 {
655 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_UINT64;
656 rc = utsDoGadgetCreateCfgParseUInt(pszVal, UINT64_MAX, &pCfgItem->Val.u.u64);
657 break;
658 }
659 case UTSPKT_GDGT_CFG_ITEM_TYPE_INT8:
660 {
661 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_INT8;
662
663 int64_t i64;
664 rc = utsDoGadgetCreateCfgParseInt(pszVal, INT8_MIN, INT8_MAX, &i64);
665 if (RT_SUCCESS(rc))
666 pCfgItem->Val.u.i8 = (int8_t)i64;
667 break;
668 }
669 case UTSPKT_GDGT_CFG_ITEM_TYPE_INT16:
670 {
671 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_INT16;
672
673 int64_t i64;
674 rc = utsDoGadgetCreateCfgParseInt(pszVal, INT16_MIN, INT16_MAX, &i64);
675 if (RT_SUCCESS(rc))
676 pCfgItem->Val.u.i16 = (int16_t)i64;
677 break;
678 }
679 case UTSPKT_GDGT_CFG_ITEM_TYPE_INT32:
680 {
681 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_INT32;
682
683 int64_t i64;
684 rc = utsDoGadgetCreateCfgParseInt(pszVal, INT32_MIN, INT32_MAX, &i64);
685 if (RT_SUCCESS(rc))
686 pCfgItem->Val.u.i32 = (int32_t)i64;
687 break;
688 }
689 case UTSPKT_GDGT_CFG_ITEM_TYPE_INT64:
690 {
691 pCfgItem->Val.enmType = UTSGADGETCFGTYPE_INT64;
692 rc = utsDoGadgetCreateCfgParseInt(pszVal, INT64_MIN, INT64_MAX, &pCfgItem->Val.u.i64);
693 break;
694 }
695 default:
696 rc = VERR_INVALID_PARAMETER;
697 }
698
699 return rc;
700}
701
702/**
703 * Creates the configuration from the given GADGET CREATE packet.
704 *
705 * @returns IPRT status code.
706 * @param pCfgItem The first config item header in the request packet.
707 * @param cCfgItems Number of config items in the packet to parse.
708 * @param cbPkt Number of bytes left in the packet for the config data.
709 * @param paCfg The array of configuration items to fill.
710 */
711static int utsDoGadgetCreateFillCfg(PUTSPKTREQGDGTCTORCFGITEM pCfgItem, unsigned cCfgItems,
712 size_t cbPkt, PUTSGADGETCFGITEM paCfg)
713{
714 int rc = VINF_SUCCESS;
715 unsigned idxCfg = 0;
716
717 while ( RT_SUCCESS(rc)
718 && cCfgItems
719 && cbPkt)
720 {
721 if (cbPkt >= sizeof(UTSPKTREQGDGTCTORCFGITEM))
722 {
723 cbPkt -= sizeof(UTSPKTREQGDGTCTORCFGITEM);
724 if (pCfgItem->u32KeySize + pCfgItem->u32ValSize >= cbPkt)
725 {
726 const char *pszKey = (const char *)(pCfgItem + 1);
727 const char *pszVal = pszKey + pCfgItem->u32KeySize;
728
729 /* Validate termination. */
730 if ( *(pszKey + pCfgItem->u32KeySize - 1) != '\0'
731 || *(pszVal + pCfgItem->u32ValSize - 1) != '\0')
732 rc = VERR_INVALID_PARAMETER;
733 else
734 {
735 paCfg[idxCfg].pszKey = RTStrDup(pszKey);
736
737 rc = utsDoGadgetCreateCfgParseItem(&paCfg[idxCfg], pCfgItem->u32Type, pszVal);
738 if (RT_SUCCESS(rc))
739 {
740 cbPkt -= pCfgItem->u32KeySize + pCfgItem->u32ValSize;
741 cCfgItems--;
742 idxCfg++;
743 pCfgItem = (PUTSPKTREQGDGTCTORCFGITEM)(pszVal + pCfgItem->u32ValSize);
744 }
745 }
746 }
747 else
748 rc = VERR_INVALID_PARAMETER;
749 }
750 else
751 rc = VERR_INVALID_PARAMETER;
752 }
753
754 return rc;
755}
756
757/**
758 * Verifies and acknowledges a "BYE" request.
759 *
760 * @returns IPRT status code.
761 * @param pClient The UTS client structure.
762 * @param pPktHdr The howdy packet.
763 */
764static int utsDoBye(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
765{
766 int rc;
767 if (pPktHdr->cb == sizeof(UTSPKTHDR))
768 rc = utsReplyAck(pClient, pPktHdr);
769 else
770 rc = utsReplyBadSize(pClient, pPktHdr, sizeof(UTSPKTHDR));
771 return rc;
772}
773
774/**
775 * Verifies and acknowledges a "HOWDY" request.
776 *
777 * @returns IPRT status code.
778 * @param pClient The UTS client structure.
779 * @param pPktHdr The howdy packet.
780 */
781static int utsDoHowdy(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
782{
783 int rc = VINF_SUCCESS;
784
785 if (pPktHdr->cb != sizeof(UTSPKTREQHOWDY))
786 return utsReplyBadSize(pClient, pPktHdr, sizeof(UTSPKTREQHOWDY));
787
788 if (pClient->enmState != UTSCLIENTSTATE_INITIALISING)
789 return utsReplyInvalidState(pClient, pPktHdr);
790
791 PUTSPKTREQHOWDY pReq = (PUTSPKTREQHOWDY)pPktHdr;
792
793 if (pReq->uVersion != UTS_PROTOCOL_VS)
794 return utsReplyRC(pClient, pPktHdr, VERR_VERSION_MISMATCH, "The given version %#x is not supported", pReq->uVersion);
795
796 /* Verify hostname string. */
797 if (pReq->cchHostname >= sizeof(pReq->achHostname))
798 return utsReplyBadSize(pClient, pPktHdr, sizeof(pReq->achHostname) - 1);
799
800 if (pReq->achHostname[pReq->cchHostname] != '\0')
801 return utsReplyBadStrTermination(pClient, pPktHdr);
802
803 /* Extract string. */
804 pClient->pszHostname = RTStrDup(&pReq->achHostname[0]);
805 if (!pClient->pszHostname)
806 return utsReplyRC(pClient, pPktHdr, VERR_NO_MEMORY, "Failed to allocate memory for the hostname string");
807
808 if (pReq->fUsbConn & UTSPKT_HOWDY_CONN_F_PHYSICAL)
809 return utsReplyRC(pClient, pPktHdr, VERR_NOT_SUPPORTED, "Physical connections are not yet supported");
810
811 if (pReq->fUsbConn & UTSPKT_HOWDY_CONN_F_USBIP)
812 {
813 /* Set up the USB/IP server, find an unused port we can start the server on. */
814 UTSGADGETCFGITEM aCfg[2];
815
816 uint16_t uPort = g_uUsbIpPortNext;
817
818 if (g_uUsbIpPortNext == g_uUsbIpPortLast)
819 g_uUsbIpPortNext = g_uUsbIpPortFirst;
820 else
821 g_uUsbIpPortNext++;
822
823 aCfg[0].pszKey = "UsbIp/Port";
824 aCfg[0].Val.enmType = UTSGADGETCFGTYPE_UINT16;
825 aCfg[0].Val.u.u16 = uPort;
826 aCfg[1].pszKey = NULL;
827
828 rc = utsGadgetHostCreate(UTSGADGETHOSTTYPE_USBIP, &aCfg[0], &pClient->hGadgetHost);
829 if (RT_SUCCESS(rc))
830 {
831 /* Send the reply with the configured USB/IP port. */
832 UTSPKTREPHOWDY Rep;
833
834 RT_ZERO(Rep);
835
836 Rep.uVersion = UTS_PROTOCOL_VS;
837 Rep.fUsbConn = UTSPKT_HOWDY_CONN_F_USBIP;
838 Rep.uUsbIpPort = uPort;
839 Rep.cUsbIpDevices = 1;
840 Rep.cPhysicalDevices = 0;
841
842 rc = utsReplyInternal(pClient, &Rep.Sts, "ACK ", sizeof(Rep) - sizeof(UTSPKTSTS));
843 if (RT_SUCCESS(rc))
844 {
845 g_pTransport->pfnNotifyHowdy(pClient->pTransportClient);
846 pClient->enmState = UTSCLIENTSTATE_READY;
847 RTDirRemoveRecursive(g_szScratchPath, RTDIRRMREC_F_CONTENT_ONLY);
848 }
849 }
850 else
851 return utsReplyRC(pClient, pPktHdr, rc, "Creating the USB/IP gadget host failed");
852 }
853 else
854 return utsReplyRC(pClient, pPktHdr, VERR_INVALID_PARAMETER, "No access method requested");
855
856 return rc;
857}
858
859/**
860 * Verifies and processes a "GADGET CREATE" request.
861 *
862 * @returns IPRT status code.
863 * @param pClient The UTS client structure.
864 * @param pPktHdr The gadget create packet.
865 */
866static int utsDoGadgetCreate(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
867{
868 int rc = VINF_SUCCESS;
869
870 if (pPktHdr->cb < sizeof(UTSPKTREQGDGTCTOR))
871 return utsReplyBadSize(pClient, pPktHdr, sizeof(UTSPKTREQGDGTCTOR));
872
873 if ( pClient->enmState != UTSCLIENTSTATE_READY
874 || pClient->hGadgetHost == NIL_UTSGADGETHOST)
875 return utsReplyInvalidState(pClient, pPktHdr);
876
877 PUTSPKTREQGDGTCTOR pReq = (PUTSPKTREQGDGTCTOR)pPktHdr;
878
879 if (pReq->u32GdgtType != UTSPKT_GDGT_CREATE_TYPE_TEST)
880 return utsReplyRC(pClient, pPktHdr, VERR_INVALID_PARAMETER, "The given gadget type is not supported");
881
882 if (pReq->u32GdgtAccess != UTSPKT_GDGT_CREATE_ACCESS_USBIP)
883 return utsReplyRC(pClient, pPktHdr, VERR_INVALID_PARAMETER, "The given gadget access method is not supported");
884
885 PUTSGADGETCFGITEM paCfg = NULL;
886 if (pReq->u32CfgItems > 0)
887 {
888 paCfg = (PUTSGADGETCFGITEM)RTMemAllocZ((pReq->u32CfgItems + 1) * sizeof(UTSGADGETCFGITEM));
889 if (RT_UNLIKELY(!paCfg))
890 return utsReplyRC(pClient, pPktHdr, VERR_NO_MEMORY, "Failed to allocate memory for configration items");
891
892 rc = utsDoGadgetCreateFillCfg((PUTSPKTREQGDGTCTORCFGITEM)(pReq + 1), pReq->u32CfgItems,
893 pPktHdr->cb - sizeof(UTSPKTREQGDGTCTOR), paCfg);
894 if (RT_FAILURE(rc))
895 {
896 RTMemFree(paCfg);
897 return utsReplyRC(pClient, pPktHdr, rc, "Failed to parse configuration");
898 }
899 }
900
901 rc = utsGadgetCreate(pClient->hGadgetHost, UTSGADGETCLASS_TEST, paCfg, &pClient->hGadget);
902 if (RT_SUCCESS(rc))
903 {
904 UTSPKTREPGDGTCTOR Rep;
905 RT_ZERO(Rep);
906
907 Rep.idGadget = 0;
908 Rep.u32BusId = utsGadgetGetBusId(pClient->hGadget);
909 Rep.u32DevId = utsGadgetGetDevId(pClient->hGadget);
910 rc = utsReplyInternal(pClient, &Rep.Sts, "ACK ", sizeof(Rep) - sizeof(UTSPKTSTS));
911 }
912 else
913 rc = utsReplyRC(pClient, pPktHdr, rc, "Failed to create gadget with %Rrc\n", rc);
914
915 return rc;
916}
917
918/**
919 * Verifies and processes a "GADGET DESTROY" request.
920 *
921 * @returns IPRT status code.
922 * @param pClient The UTS client structure.
923 * @param pPktHdr The gadget destroy packet.
924 */
925static int utsDoGadgetDestroy(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
926{
927 if (pPktHdr->cb != sizeof(UTSPKTREQGDGTDTOR))
928 return utsReplyBadSize(pClient, pPktHdr, sizeof(UTSPKTREQGDGTDTOR));
929
930 if ( pClient->enmState != UTSCLIENTSTATE_READY
931 || pClient->hGadgetHost == NIL_UTSGADGETHOST)
932 return utsReplyInvalidState(pClient, pPktHdr);
933
934 PUTSPKTREQGDGTDTOR pReq = (PUTSPKTREQGDGTDTOR)pPktHdr;
935
936 if (pReq->idGadget != 0)
937 return utsReplyRC(pClient, pPktHdr, VERR_INVALID_HANDLE, "The given gadget handle is invalid");
938 if (pClient->hGadget == NIL_UTSGADGET)
939 return utsReplyRC(pClient, pPktHdr, VERR_INVALID_STATE, "The gadget is not set up");
940
941 utsGadgetRelease(pClient->hGadget);
942 pClient->hGadget = NIL_UTSGADGET;
943
944 return utsReplyAck(pClient, pPktHdr);
945}
946
947/**
948 * Verifies and processes a "GADGET CONNECT" request.
949 *
950 * @returns IPRT status code.
951 * @param pClient The UTS client structure.
952 * @param pPktHdr The gadget connect packet.
953 */
954static int utsDoGadgetConnect(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
955{
956 if (pPktHdr->cb != sizeof(UTSPKTREQGDGTCNCT))
957 return utsReplyBadSize(pClient, pPktHdr, sizeof(UTSPKTREQGDGTCNCT));
958
959 if ( pClient->enmState != UTSCLIENTSTATE_READY
960 || pClient->hGadgetHost == NIL_UTSGADGETHOST)
961 return utsReplyInvalidState(pClient, pPktHdr);
962
963 PUTSPKTREQGDGTCNCT pReq = (PUTSPKTREQGDGTCNCT)pPktHdr;
964
965 if (pReq->idGadget != 0)
966 return utsReplyRC(pClient, pPktHdr, VERR_INVALID_HANDLE, "The given gadget handle is invalid");
967 if (pClient->hGadget == NIL_UTSGADGET)
968 return utsReplyRC(pClient, pPktHdr, VERR_INVALID_STATE, "The gadget is not set up");
969
970 int rc = utsGadgetConnect(pClient->hGadget);
971 if (RT_SUCCESS(rc))
972 rc = utsReplyAck(pClient, pPktHdr);
973 else
974 rc = utsReplyRC(pClient, pPktHdr, rc, "Failed to connect the gadget");
975
976 return rc;
977}
978
979/**
980 * Verifies and processes a "GADGET DISCONNECT" request.
981 *
982 * @returns IPRT status code.
983 * @param pClient The UTS client structure.
984 * @param pPktHdr The gadget disconnect packet.
985 */
986static int utsDoGadgetDisconnect(PUTSCLIENT pClient, PCUTSPKTHDR pPktHdr)
987{
988 if (pPktHdr->cb != sizeof(UTSPKTREQGDGTDCNT))
989 return utsReplyBadSize(pClient, pPktHdr, sizeof(UTSPKTREQGDGTDCNT));
990
991 if ( pClient->enmState != UTSCLIENTSTATE_READY
992 || pClient->hGadgetHost == NIL_UTSGADGETHOST)
993 return utsReplyInvalidState(pClient, pPktHdr);
994
995 PUTSPKTREQGDGTDCNT pReq = (PUTSPKTREQGDGTDCNT)pPktHdr;
996
997 if (pReq->idGadget != 0)
998 return utsReplyRC(pClient, pPktHdr, VERR_INVALID_HANDLE, "The given gadget handle is invalid");
999 if (pClient->hGadget == NIL_UTSGADGET)
1000 return utsReplyRC(pClient, pPktHdr, VERR_INVALID_STATE, "The gadget is not set up");
1001
1002 int rc = utsGadgetDisconnect(pClient->hGadget);
1003 if (RT_SUCCESS(rc))
1004 rc = utsReplyAck(pClient, pPktHdr);
1005 else
1006 rc = utsReplyRC(pClient, pPktHdr, rc, "Failed to disconnect the gadget");
1007
1008 return rc;
1009}
1010
1011/**
1012 * Main request processing routine for each client.
1013 *
1014 * @returns IPRT status code.
1015 * @param pClient The UTS client structure sending the request.
1016 */
1017static int utsClientReqProcess(PUTSCLIENT pClient)
1018{
1019 /*
1020 * Read client command packet and process it.
1021 */
1022 PUTSPKTHDR pPktHdr = NULL;
1023 int rc = utsRecvPkt(pClient, &pPktHdr, true /*fAutoRetryOnFailure*/);
1024 if (RT_FAILURE(rc))
1025 return rc;
1026
1027 /*
1028 * Do a string switch on the opcode bit.
1029 */
1030 /* Connection: */
1031 if ( utsIsSameOpcode(pPktHdr, UTSPKT_OPCODE_HOWDY))
1032 rc = utsDoHowdy(pClient, pPktHdr);
1033 else if (utsIsSameOpcode(pPktHdr, UTSPKT_OPCODE_BYE))
1034 rc = utsDoBye(pClient, pPktHdr);
1035 /* Gadget API. */
1036 else if (utsIsSameOpcode(pPktHdr, UTSPKT_OPCODE_GADGET_CREATE))
1037 rc = utsDoGadgetCreate(pClient, pPktHdr);
1038 else if (utsIsSameOpcode(pPktHdr, UTSPKT_OPCODE_GADGET_DESTROY))
1039 rc = utsDoGadgetDestroy(pClient, pPktHdr);
1040 else if (utsIsSameOpcode(pPktHdr, UTSPKT_OPCODE_GADGET_CONNECT))
1041 rc = utsDoGadgetConnect(pClient, pPktHdr);
1042 else if (utsIsSameOpcode(pPktHdr, UTSPKT_OPCODE_GADGET_DISCONNECT))
1043 rc = utsDoGadgetDisconnect(pClient, pPktHdr);
1044 /* Misc: */
1045 else
1046 rc = utsReplyUnknown(pClient, pPktHdr);
1047
1048 RTMemFree(pPktHdr);
1049
1050 return rc;
1051}
1052
1053/**
1054 * Destroys a client instance.
1055 *
1056 * @returns nothing.
1057 * @param pClient The UTS client structure.
1058 */
1059static void utsClientDestroy(PUTSCLIENT pClient)
1060{
1061 if (pClient->pszHostname)
1062 RTStrFree(pClient->pszHostname);
1063 if (pClient->hGadget != NIL_UTSGADGET)
1064 utsGadgetRelease(pClient->hGadget);
1065 if (pClient->hGadgetHost != NIL_UTSGADGETHOST)
1066 utsGadgetHostRelease(pClient->hGadgetHost);
1067 RTMemFree(pClient);
1068}
1069
1070/**
1071 * The main thread worker serving the clients.
1072 */
1073static DECLCALLBACK(int) utsClientWorker(RTTHREAD hThread, void *pvUser)
1074{
1075 unsigned cClientsMax = 0;
1076 unsigned cClientsCur = 0;
1077 PUTSCLIENT *papClients = NULL;
1078 RTPOLLSET hPollSet;
1079
1080 int rc = RTPollSetCreate(&hPollSet);
1081 if (RT_FAILURE(rc))
1082 return rc;
1083
1084 /* Add the pipe to the poll set. */
1085 rc = RTPollSetAddPipe(hPollSet, g_hPipeR, RTPOLL_EVT_READ | RTPOLL_EVT_ERROR, 0);
1086 if (RT_SUCCESS(rc))
1087 {
1088 while (!g_fTerminate)
1089 {
1090 uint32_t fEvts;
1091 uint32_t uId;
1092 rc = RTPoll(hPollSet, RT_INDEFINITE_WAIT, &fEvts, &uId);
1093 if (RT_SUCCESS(rc))
1094 {
1095 if (uId == 0)
1096 {
1097 if (fEvts & RTPOLL_EVT_ERROR)
1098 break;
1099
1100 /* We got woken up because of a new client. */
1101 Assert(fEvts & RTPOLL_EVT_READ);
1102
1103 uint8_t bRead;
1104 size_t cbRead = 0;
1105 rc = RTPipeRead(g_hPipeR, &bRead, 1, &cbRead);
1106 AssertRC(rc);
1107
1108 RTCritSectEnter(&g_CritSectClients);
1109 /* Walk the list and add all new clients. */
1110 PUTSCLIENT pIt, pItNext;
1111 RTListForEachSafe(&g_LstClientsNew, pIt, pItNext, UTSCLIENT, NdLst)
1112 {
1113 RTListNodeRemove(&pIt->NdLst);
1114 Assert(cClientsCur <= cClientsMax);
1115 if (cClientsCur == cClientsMax)
1116 {
1117 /* Realloc to accommodate for the new clients. */
1118 PUTSCLIENT *papClientsNew = (PUTSCLIENT *)RTMemRealloc(papClients, (cClientsMax + 10) * sizeof(PUTSCLIENT));
1119 if (RT_LIKELY(papClientsNew))
1120 {
1121 cClientsMax += 10;
1122 papClients = papClientsNew;
1123 }
1124 }
1125
1126 if (cClientsCur < cClientsMax)
1127 {
1128 /* Find a free slot in the client array. */
1129 unsigned idxSlt = 0;
1130 while ( idxSlt < cClientsMax
1131 && papClients[idxSlt] != NULL)
1132 idxSlt++;
1133
1134 rc = g_pTransport->pfnPollSetAdd(hPollSet, pIt->pTransportClient, idxSlt + 1);
1135 if (RT_SUCCESS(rc))
1136 {
1137 cClientsCur++;
1138 papClients[idxSlt] = pIt;
1139 }
1140 else
1141 {
1142 g_pTransport->pfnNotifyBye(pIt->pTransportClient);
1143 utsClientDestroy(pIt);
1144 }
1145 }
1146 else
1147 {
1148 g_pTransport->pfnNotifyBye(pIt->pTransportClient);
1149 utsClientDestroy(pIt);
1150 }
1151 }
1152 RTCritSectLeave(&g_CritSectClients);
1153 }
1154 else
1155 {
1156 /* Client sends a request, pick the right client and process it. */
1157 PUTSCLIENT pClient = papClients[uId - 1];
1158 AssertPtr(pClient);
1159 if (fEvts & RTPOLL_EVT_READ)
1160 rc = utsClientReqProcess(pClient);
1161
1162 if ( (fEvts & RTPOLL_EVT_ERROR)
1163 || RT_FAILURE(rc))
1164 {
1165 /* Close connection and remove client from array. */
1166 rc = g_pTransport->pfnPollSetRemove(hPollSet, pClient->pTransportClient, uId);
1167 AssertRC(rc);
1168
1169 g_pTransport->pfnNotifyBye(pClient->pTransportClient);
1170 papClients[uId - 1] = NULL;
1171 cClientsCur--;
1172 utsClientDestroy(pClient);
1173 }
1174 }
1175 }
1176 }
1177 }
1178
1179 RTPollSetDestroy(hPollSet);
1180
1181 return rc;
1182}
1183
1184/**
1185 * The main loop.
1186 *
1187 * @returns exit code.
1188 */
1189static RTEXITCODE utsMainLoop(void)
1190{
1191 RTEXITCODE enmExitCode = RTEXITCODE_SUCCESS;
1192 while (!g_fTerminate)
1193 {
1194 /*
1195 * Wait for new connection and spin off a new thread
1196 * for every new client.
1197 */
1198 PUTSTRANSPORTCLIENT pTransportClient;
1199 int rc = g_pTransport->pfnWaitForConnect(&pTransportClient);
1200 if (RT_FAILURE(rc))
1201 continue;
1202
1203 /*
1204 * New connection, create new client structure and spin of
1205 * the request handling thread.
1206 */
1207 PUTSCLIENT pClient = (PUTSCLIENT)RTMemAllocZ(sizeof(UTSCLIENT));
1208 if (RT_LIKELY(pClient))
1209 {
1210 pClient->enmState = UTSCLIENTSTATE_INITIALISING;
1211 pClient->pTransportClient = pTransportClient;
1212 pClient->pszHostname = NULL;
1213 pClient->hGadgetHost = NIL_UTSGADGETHOST;
1214 pClient->hGadget = NIL_UTSGADGET;
1215
1216 /* Add client to the new list and inform the worker thread. */
1217 RTCritSectEnter(&g_CritSectClients);
1218 RTListAppend(&g_LstClientsNew, &pClient->NdLst);
1219 RTCritSectLeave(&g_CritSectClients);
1220
1221 size_t cbWritten = 0;
1222 rc = RTPipeWrite(g_hPipeW, "", 1, &cbWritten);
1223 if (RT_FAILURE(rc))
1224 RTMsgError("Failed to inform worker thread of a new client");
1225 }
1226 else
1227 {
1228 RTMsgError("Creating new client structure failed with out of memory error\n");
1229 g_pTransport->pfnNotifyBye(pTransportClient);
1230 }
1231
1232
1233 }
1234
1235 return enmExitCode;
1236}
1237
1238/**
1239 * Initializes the global UTS state.
1240 *
1241 * @returns IPRT status code.
1242 */
1243static int utsInit(void)
1244{
1245 int rc = VINF_SUCCESS;
1246 PRTERRINFO pErrInfo = NULL;
1247
1248 RTListInit(&g_LstClientsNew);
1249
1250 rc = utsParseConfig(g_szCfgPath, &g_pCfgAst, &pErrInfo);
1251 if (RT_SUCCESS(rc))
1252 {
1253 rc = utsPlatformInit();
1254 if (RT_SUCCESS(rc))
1255 {
1256 rc = RTCritSectInit(&g_CritSectClients);
1257 if (RT_SUCCESS(rc))
1258 {
1259 rc = RTPipeCreate(&g_hPipeR, &g_hPipeW, 0);
1260 if (RT_SUCCESS(rc))
1261 {
1262 /* Spin off the thread serving connections. */
1263 rc = RTThreadCreate(&g_hThreadServing, utsClientWorker, NULL, 0, RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE,
1264 "USBTSTSRV");
1265 if (RT_SUCCESS(rc))
1266 return VINF_SUCCESS;
1267 else
1268 RTMsgError("Creating the client worker thread failed with %Rrc\n", rc);
1269
1270 RTPipeClose(g_hPipeR);
1271 RTPipeClose(g_hPipeW);
1272 }
1273 else
1274 RTMsgError("Creating communications pipe failed with %Rrc\n", rc);
1275
1276 RTCritSectDelete(&g_CritSectClients);
1277 }
1278 else
1279 RTMsgError("Creating global critical section failed with %Rrc\n", rc);
1280
1281 utsConfigAstDestroy(g_pCfgAst);
1282 }
1283 else
1284 RTMsgError("Initializing the platform failed with %Rrc\n", rc);
1285 }
1286 else
1287 {
1288 if (RTErrInfoIsSet(pErrInfo))
1289 {
1290 RTMsgError("Failed to parse config with detailed error: %s (%Rrc)\n",
1291 pErrInfo->pszMsg, pErrInfo->rc);
1292 RTErrInfoFree(pErrInfo);
1293 }
1294 else
1295 RTMsgError("Failed to parse config with unknown error (%Rrc)\n", rc);
1296 }
1297
1298 return rc;
1299}
1300
1301/**
1302 * Determines the default configuration.
1303 */
1304static void utsSetDefaults(void)
1305{
1306 /*
1307 * OS and ARCH.
1308 */
1309 AssertCompile(sizeof(KBUILD_TARGET) <= sizeof(g_szOsShortName));
1310 strcpy(g_szOsShortName, KBUILD_TARGET);
1311
1312 AssertCompile(sizeof(KBUILD_TARGET_ARCH) <= sizeof(g_szArchShortName));
1313 strcpy(g_szArchShortName, KBUILD_TARGET_ARCH);
1314
1315 AssertCompile(sizeof(KBUILD_TARGET) + sizeof(KBUILD_TARGET_ARCH) <= sizeof(g_szOsDotArchShortName));
1316 strcpy(g_szOsDotArchShortName, KBUILD_TARGET);
1317 g_szOsDotArchShortName[sizeof(KBUILD_TARGET) - 1] = '.';
1318 strcpy(&g_szOsDotArchShortName[sizeof(KBUILD_TARGET)], KBUILD_TARGET_ARCH);
1319
1320 AssertCompile(sizeof(KBUILD_TARGET) + sizeof(KBUILD_TARGET_ARCH) <= sizeof(g_szOsSlashArchShortName));
1321 strcpy(g_szOsSlashArchShortName, KBUILD_TARGET);
1322 g_szOsSlashArchShortName[sizeof(KBUILD_TARGET) - 1] = '/';
1323 strcpy(&g_szOsSlashArchShortName[sizeof(KBUILD_TARGET)], KBUILD_TARGET_ARCH);
1324
1325#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
1326 strcpy(g_szExeSuff, ".exe");
1327 strcpy(g_szScriptSuff, ".cmd");
1328#else
1329 strcpy(g_szExeSuff, "");
1330 strcpy(g_szScriptSuff, ".sh");
1331#endif
1332
1333 /*
1334 * The CD/DVD-ROM location.
1335 */
1336 /** @todo do a better job here :-) */
1337#ifdef RT_OS_WINDOWS
1338 strcpy(g_szDefCdRomPath, "D:/");
1339#elif defined(RT_OS_OS2)
1340 strcpy(g_szDefCdRomPath, "D:/");
1341#else
1342 if (RTDirExists("/media"))
1343 strcpy(g_szDefCdRomPath, "/media/cdrom");
1344 else
1345 strcpy(g_szDefCdRomPath, "/mnt/cdrom");
1346#endif
1347 strcpy(g_szCdRomPath, g_szDefCdRomPath);
1348
1349 /*
1350 * Temporary directory.
1351 */
1352 int rc = RTPathTemp(g_szDefScratchPath, sizeof(g_szDefScratchPath));
1353 if (RT_SUCCESS(rc))
1354#if defined(RT_OS_OS2) || defined(RT_OS_WINDOWS) || defined(RT_OS_DOS)
1355 rc = RTPathAppend(g_szDefScratchPath, sizeof(g_szDefScratchPath), "uts-XXXX.tmp");
1356#else
1357 rc = RTPathAppend(g_szDefScratchPath, sizeof(g_szDefScratchPath), "uts-XXXXXXXXX.tmp");
1358#endif
1359 if (RT_FAILURE(rc))
1360 {
1361 RTMsgError("RTPathTemp/Append failed when constructing scratch path: %Rrc\n", rc);
1362 strcpy(g_szDefScratchPath, "/tmp/uts-XXXX.tmp");
1363 }
1364 strcpy(g_szScratchPath, g_szDefScratchPath);
1365
1366 /*
1367 * Config file location.
1368 */
1369 /** @todo: Improve */
1370#if !defined(RT_OS_WINDOWS)
1371 strcpy(g_szCfgPath, "/etc/uts.conf");
1372#else
1373 strcpy(g_szCfgPath, "");
1374#endif
1375
1376 /*
1377 * The default transporter is the first one.
1378 */
1379 g_pTransport = g_apTransports[0];
1380}
1381
1382/**
1383 * Prints the usage.
1384 *
1385 * @param pStrm Where to print it.
1386 * @param pszArgv0 The program name (argv[0]).
1387 */
1388static void utsUsage(PRTSTREAM pStrm, const char *argv0)
1389{
1390 RTStrmPrintf(pStrm,
1391 "Usage: %Rbn [options]\n"
1392 "\n"
1393 "Options:\n"
1394 " --config <path>\n"
1395 " Where to load the config from\n"
1396 " --cdrom <path>\n"
1397 " Where the CD/DVD-ROM will be mounted.\n"
1398 " Default: %s\n"
1399 " --scratch <path>\n"
1400 " Where to put scratch files.\n"
1401 " Default: %s \n"
1402 ,
1403 argv0,
1404 g_szDefCdRomPath,
1405 g_szDefScratchPath);
1406 RTStrmPrintf(pStrm,
1407 " --transport <name>\n"
1408 " Use the specified transport layer, one of the following:\n");
1409 for (size_t i = 0; i < RT_ELEMENTS(g_apTransports); i++)
1410 RTStrmPrintf(pStrm, " %s - %s\n", g_apTransports[i]->szName, g_apTransports[i]->pszDesc);
1411 RTStrmPrintf(pStrm, " Default: %s\n", g_pTransport->szName);
1412 RTStrmPrintf(pStrm,
1413 " --display-output, --no-display-output\n"
1414 " Display the output and the result of all child processes.\n");
1415 RTStrmPrintf(pStrm,
1416 " --foreground\n"
1417 " Don't daemonize, run in the foreground.\n");
1418 RTStrmPrintf(pStrm,
1419 " --help, -h, -?\n"
1420 " Display this message and exit.\n"
1421 " --version, -V\n"
1422 " Display the version and exit.\n");
1423
1424 for (size_t i = 0; i < RT_ELEMENTS(g_apTransports); i++)
1425 if (g_apTransports[i]->cOpts)
1426 {
1427 RTStrmPrintf(pStrm,
1428 "\n"
1429 "Options for %s:\n", g_apTransports[i]->szName);
1430 g_apTransports[i]->pfnUsage(g_pStdOut);
1431 }
1432}
1433
1434/**
1435 * Parses the arguments.
1436 *
1437 * @returns Exit code. Exit if this is non-zero or @a *pfExit is set.
1438 * @param argc The number of arguments.
1439 * @param argv The argument vector.
1440 * @param pfExit For indicating exit when the exit code is zero.
1441 */
1442static RTEXITCODE utsParseArgv(int argc, char **argv, bool *pfExit)
1443{
1444 *pfExit = false;
1445
1446 /*
1447 * Storage for locally handled options.
1448 */
1449 bool fDaemonize = true;
1450 bool fDaemonized = false;
1451
1452 /*
1453 * Combine the base and transport layer option arrays.
1454 */
1455 static const RTGETOPTDEF s_aBaseOptions[] =
1456 {
1457 { "--config", 'C', RTGETOPT_REQ_STRING },
1458 { "--transport", 't', RTGETOPT_REQ_STRING },
1459 { "--cdrom", 'c', RTGETOPT_REQ_STRING },
1460 { "--scratch", 's', RTGETOPT_REQ_STRING },
1461 { "--display-output", 'd', RTGETOPT_REQ_NOTHING },
1462 { "--no-display-output",'D', RTGETOPT_REQ_NOTHING },
1463 { "--foreground", 'f', RTGETOPT_REQ_NOTHING },
1464 { "--daemonized", 'Z', RTGETOPT_REQ_NOTHING },
1465 };
1466
1467 size_t cOptions = RT_ELEMENTS(s_aBaseOptions);
1468 for (size_t i = 0; i < RT_ELEMENTS(g_apTransports); i++)
1469 cOptions += g_apTransports[i]->cOpts;
1470
1471 PRTGETOPTDEF paOptions = (PRTGETOPTDEF)alloca(cOptions * sizeof(RTGETOPTDEF));
1472 if (!paOptions)
1473 return RTMsgErrorExit(RTEXITCODE_FAILURE, "alloca failed\n");
1474
1475 memcpy(paOptions, s_aBaseOptions, sizeof(s_aBaseOptions));
1476 cOptions = RT_ELEMENTS(s_aBaseOptions);
1477 for (size_t i = 0; i < RT_ELEMENTS(g_apTransports); i++)
1478 {
1479 memcpy(&paOptions[cOptions], g_apTransports[i]->paOpts, g_apTransports[i]->cOpts * sizeof(RTGETOPTDEF));
1480 cOptions += g_apTransports[i]->cOpts;
1481 }
1482
1483 /*
1484 * Parse the arguments.
1485 */
1486 RTGETOPTSTATE GetState;
1487 int rc = RTGetOptInit(&GetState, argc, argv, paOptions, cOptions, 1, 0 /* fFlags */);
1488 AssertRC(rc);
1489
1490 int ch;
1491 RTGETOPTUNION Val;
1492 while ((ch = RTGetOpt(&GetState, &Val)))
1493 {
1494 switch (ch)
1495 {
1496 case 'C':
1497 rc = RTStrCopy(g_szCfgPath, sizeof(g_szCfgPath), Val.psz);
1498 if (RT_FAILURE(rc))
1499 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Config file path is path too long (%Rrc)\n", rc);
1500 break;
1501
1502 case 'c':
1503 rc = RTStrCopy(g_szCdRomPath, sizeof(g_szCdRomPath), Val.psz);
1504 if (RT_FAILURE(rc))
1505 return RTMsgErrorExit(RTEXITCODE_FAILURE, "CD/DVD-ROM is path too long (%Rrc)\n", rc);
1506 break;
1507
1508 case 'd':
1509 g_fDisplayOutput = true;
1510 break;
1511
1512 case 'D':
1513 g_fDisplayOutput = false;
1514 break;
1515
1516 case 'f':
1517 fDaemonize = false;
1518 break;
1519
1520 case 'h':
1521 utsUsage(g_pStdOut, argv[0]);
1522 *pfExit = true;
1523 return RTEXITCODE_SUCCESS;
1524
1525 case 's':
1526 rc = RTStrCopy(g_szScratchPath, sizeof(g_szScratchPath), Val.psz);
1527 if (RT_FAILURE(rc))
1528 return RTMsgErrorExit(RTEXITCODE_FAILURE, "scratch path is too long (%Rrc)\n", rc);
1529 break;
1530
1531 case 't':
1532 {
1533 PCUTSTRANSPORT pTransport = NULL;
1534 for (size_t i = 0; RT_ELEMENTS(g_apTransports); i++)
1535 if (!strcmp(g_apTransports[i]->szName, Val.psz))
1536 {
1537 pTransport = g_apTransports[i];
1538 break;
1539 }
1540 if (!pTransport)
1541 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Unknown transport layer name '%s'\n", Val.psz);
1542 g_pTransport = pTransport;
1543 break;
1544 }
1545
1546 case 'V':
1547 RTPrintf("$Revision: 60793 $\n");
1548 *pfExit = true;
1549 return RTEXITCODE_SUCCESS;
1550
1551 case 'Z':
1552 fDaemonized = true;
1553 fDaemonize = false;
1554 break;
1555
1556 default:
1557 {
1558 rc = VERR_TRY_AGAIN;
1559 for (size_t i = 0; i < RT_ELEMENTS(g_apTransports); i++)
1560 if (g_apTransports[i]->cOpts)
1561 {
1562 rc = g_apTransports[i]->pfnOption(ch, &Val);
1563 if (RT_SUCCESS(rc))
1564 break;
1565 if (rc != VERR_TRY_AGAIN)
1566 {
1567 *pfExit = true;
1568 return RTEXITCODE_SYNTAX;
1569 }
1570 }
1571 if (rc == VERR_TRY_AGAIN)
1572 {
1573 *pfExit = true;
1574 return RTGetOptPrintError(ch, &Val);
1575 }
1576 break;
1577 }
1578 }
1579 }
1580
1581 /*
1582 * Daemonize ourselves if asked to.
1583 */
1584 if (fDaemonize && !*pfExit)
1585 {
1586 rc = RTProcDaemonize(argv, "--daemonized");
1587 if (RT_FAILURE(rc))
1588 return RTMsgErrorExit(RTEXITCODE_FAILURE, "RTProcDaemonize: %Rrc\n", rc);
1589 *pfExit = true;
1590 }
1591
1592 return RTEXITCODE_SUCCESS;
1593}
1594
1595
1596int main(int argc, char **argv)
1597{
1598 /*
1599 * Initialize the runtime.
1600 */
1601 int rc = RTR3InitExe(argc, &argv, 0);
1602 if (RT_FAILURE(rc))
1603 return RTMsgInitFailure(rc);
1604
1605 /*
1606 * Determine defaults and parse the arguments.
1607 */
1608 utsSetDefaults();
1609 bool fExit;
1610 RTEXITCODE rcExit = utsParseArgv(argc, argv, &fExit);
1611 if (rcExit != RTEXITCODE_SUCCESS || fExit)
1612 return rcExit;
1613
1614 /*
1615 * Initialize global state.
1616 */
1617 rc = utsInit();
1618 if (RT_FAILURE(rc))
1619 return RTEXITCODE_FAILURE;
1620
1621 /*
1622 * Initialize the transport layer.
1623 */
1624 rc = g_pTransport->pfnInit();
1625 if (RT_FAILURE(rc))
1626 return RTEXITCODE_FAILURE;
1627
1628 /*
1629 * Ok, start working
1630 */
1631 rcExit = utsMainLoop();
1632
1633 /*
1634 * Cleanup.
1635 */
1636 g_pTransport->pfnTerm();
1637
1638 utsPlatformTerm();
1639
1640 return rcExit;
1641}
1642
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