# HG changeset patch # User Tom Sutcliffe # Date 1285149399 -3600 # Node ID f3d01c9dd09907a700e5ab0fe1ec1bf8bde1ef72 # Parent 6a2083f7eeb868f5526e42181bb65cd0b90db6b7# Parent b9edfff731fbd306237260b4b975522c56a5c54e merge diff -r b9edfff731fb -r f3d01c9dd099 build/s60/tb92/vt100usbcons_preamble.script --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/build/s60/tb92/vt100usbcons_preamble.script Wed Sep 22 10:56:39 2010 +0100 @@ -0,0 +1,17 @@ +#!fshell +# vt100usbcons_preamble.script +# +# Copyright (c) 2010 Accenture. All rights reserved. +# This component and the accompanying materials are made available +# under the terms of the "Eclipse Public License v1.0" +# which accompanies this distribution, and is available +# at the URL "http://www.eclipse.org/legal/epl-v10.html". +# +# Initial Contributors: +# Accenture - Initial contribution +# + +# This is needed to prevent TRK from hogging the USB port on DFS. + +kill --all --match trkserver* +sleep 1 # Give USB time to clean up trk's death diff -r b9edfff731fb -r f3d01c9dd099 commands/usb/usb.cpp --- a/commands/usb/usb.cpp Fri Sep 17 20:28:05 2010 +0100 +++ b/commands/usb/usb.cpp Wed Sep 22 10:56:39 2010 +0100 @@ -68,6 +68,9 @@ } #define CASE_LIT(x) case x: { _LIT(KName, #x); return &KName; } +#define CASE_LIT2(x, y) case x: { _LIT(KName, y); return &KName; } + +_LIT(KUnknown, "?"); const TDesC* DeviceState(TUsbDeviceState aState) { @@ -82,7 +85,6 @@ CASE_LIT(EUsbDeviceStateSuspended); default: { - _LIT(KUnknown, "?"); return &KUnknown; } } @@ -98,10 +100,19 @@ CASE_LIT(EUsbServiceStopping); CASE_LIT(EUsbServiceFatalError); default: - { - _LIT(KUnknown, "?"); return &KUnknown; - } + } + } + +const TDesC* ClassName(TUid aClass) + { + switch (aClass.iUid) + { + CASE_LIT2(0x101FBF22, "ACM"); + CASE_LIT2(0x101fbf24, "Obex"); + CASE_LIT2(0x10204bbc, "Mass storage"); + default: + return &KUnknown; } } @@ -118,10 +129,26 @@ Printf(_L("Usb Service state = %S (%d)\r\n"), ServiceState(servicestate), servicestate); RArray ids; + CleanupClosePushL(ids); LeaveIfErr(iUsb.GetPersonalityIds(ids), _L("Couldn't get personalities ids")); for (TInt i = 0; i < ids.Count(); i++) { Printf(_L("Personality %d\r\n"), ids[i]); + RArray classUids; + CleanupClosePushL(classUids); + TInt err = iUsb.GetSupportedClasses(ids[i], classUids); + if (err) + { + PrintWarning(_L("Couldn't get classes for personality %d, err=%d"), ids[i], err); + } + else + { + for (TInt classIdx = 0; classIdx < classUids.Count(); classIdx++) + { + Printf(_L(" 0x%08x (%S)\r\n"), classUids[classIdx], ClassName(classUids[classIdx])); + } + } + CleanupStack::PopAndDestroy(&classUids); } if (iArguments.IsPresent(&iPersonality)) @@ -132,4 +159,5 @@ LeaveIfErr(stat.Int(), _L("Couldn't start USB personality %d"), iPersonality); Printf(_L("USB started ok.\r\n")); } + CleanupStack::PopAndDestroy(&ids); } diff -r b9edfff731fb -r f3d01c9dd099 core/group/autostart.script --- a/core/group/autostart.script Fri Sep 17 20:28:05 2010 +0100 +++ b/core/group/autostart.script Wed Sep 22 10:56:39 2010 +0100 @@ -11,4 +11,4 @@ # Accenture - Initial contribution # -source $SCRIPT_PATH\comm.script +source comm.script diff -r b9edfff731fb -r f3d01c9dd099 core/group/comm.script --- a/core/group/comm.script Fri Sep 17 20:28:05 2010 +0100 +++ b/core/group/comm.script Wed Sep 22 10:56:39 2010 +0100 @@ -31,6 +31,9 @@ # QEMU simulator - assumes you're using 2nd emulated serial port for fshell, ie with options similar to "-serial file:rdebug.txt -serial COMx" variant qemu && export ARGS "--console vt100busdevcons --console-title pdd=eserial,ldd=ecomm,rate=115200,port=1" +# Bridge - terminal keyboard is broken for some reason, but USB now works +variant -u 0x85001100 && export ARGS "--console vt100usbcons --console-title port=ACM::1,personality=1" + # Add new variants here diff -r b9edfff731fb -r f3d01c9dd099 documentation/change_history.pod --- a/documentation/change_history.pod Fri Sep 17 20:28:05 2010 +0100 +++ b/documentation/change_history.pod Wed Sep 22 10:56:39 2010 +0100 @@ -28,6 +28,16 @@ =back +=head2 Release 002 [Not yet released] + +=over 5 + +=item * + +Added support for automatically starting a USB personality to vt100usbcons via a new C key-value pair. See the L documentation for more information. + +=back + =head2 Release 001 =over 5 diff -r b9edfff731fb -r f3d01c9dd099 plugins/consoles/vt100cons/doc/vt100cons.pod --- a/plugins/consoles/vt100cons/doc/vt100cons.pod Fri Sep 17 20:28:05 2010 +0100 +++ b/plugins/consoles/vt100cons/doc/vt100cons.pod Wed Sep 22 10:56:39 2010 +0100 @@ -122,7 +122,27 @@ =head2 USB Variant -Vt100usbcons behaves the same as the serial variant, with the exception that it waits for USB to enumerate into a personality that supports ACM if not already present. In other words it waits for the USB cable to be inserted before trying to open the port. If necessary it also waits for the user to select a config that includes some USB serial ports. It expects the C<--console-title> argument to be of the format C where x is the port number. This number is platform-specific but is usually 0 or 1. Generally this will be taken care of already in the platform-specific build of the toolkit and you can just launch the "fshell (USB)" icon. +Vt100usbcons behaves the same as the serial variant, with the exception that it waits for USB to enumerate into a personality that supports ACM if not already present. In other words it waits for the USB cable to be inserted before trying to open the port. If necessary it also waits for the user to select a config that includes some USB serial ports. + +vt100usbcons.dll is configured via the C<--console-title> string in a similar way to the serial variant. The follow keyword / value pairs are supported: + +=over 5 + +=item * C + +The USB port name. This is platform specific but is usually C or C. Generally this will be taken care of already in the platform-specific configuration and you can just launch the "fshell (USB)" icon or run "comm.script". + +=item * C + +If it is necessary to start a USB personality before connecting, specify the personality number here, for example C. This parameter is not usually needed except on some text-shell builds. + +=back + +Example usage: + + fshell --console vt100usbcons --console-title port=ACM::1,personality=1 + +For compatibility with earlier releases, the C may be ommitted if there are no other configuration parameters. For example "fshell --console vt100usbcons --console-title ACM::1". =head2 Bluetooth Variant diff -r b9edfff731fb -r f3d01c9dd099 plugins/consoles/vt100cons/src/usb/vtc_usb.cpp --- a/plugins/consoles/vt100cons/src/usb/vtc_usb.cpp Fri Sep 17 20:28:05 2010 +0100 +++ b/plugins/consoles/vt100cons/src/usb/vtc_usb.cpp Wed Sep 22 10:56:39 2010 +0100 @@ -20,6 +20,14 @@ return new CUsbConsole; } +TUsbPortConfig::TUsbPortConfig() + :iPort(KNullDesC), iPersonality(-1) + { + } + +#define ErrOrDebug(err) (((err) < 0 && (err) != KErrAlreadyExists) ? EError : EDebug) + + //______________________________________________________________________________ // CUsbConsole CUsbConsole::CUsbConsole() @@ -33,18 +41,19 @@ void CUsbConsole::ConstructL(const TDesC& aTitle) { + TUsbPortConfig portConfig; + User::LeaveIfError(ReadConfig(aTitle, portConfig)); + User::LeaveIfError(iUsb.Connect()); TRequestStatus stat; - // assume USB device already started - /*iUsb.Start(stat); - User::WaitForRequest(stat); - // KErrAccessDenied returned if already started; - if (stat.Int()!=KErrAccessDenied) + if (portConfig.iPersonality >= 0) { - Message(EError, KUsbError, stat.Int(), 1); - User::LeaveIfError(stat.Int()); - }*/ + iUsb.TryStart(portConfig.iPersonality, stat); + User::WaitForRequest(stat); + User::After(500000); // Ugh need to wait for the TryStart to finish. There's probably some other async notification I could wait on but I'm in a hurry to get this working... + Message(ErrOrDebug(stat.Int()), _L("Starting USB personality %d returned %d"), portConfig.iPersonality, stat.Int()); + } // Wait for an enumeration that supports ACM (this is so that if the device defaulted to say mass storage and was then reconfigured to a personality with ACM, we wait for the ACM reconfiguration TBool gotAcm = EFalse; @@ -52,12 +61,15 @@ { TUsbDeviceState usbState; User::LeaveIfError(iUsb.GetDeviceState(usbState)); - if (usbState & EUsbDeviceStateConfigured) + if (usbState & (EUsbDeviceStateConfigured|EUsbDeviceStatePowered)) // We should only need to check for EUsbDeviceStateConfigured, but some HW doesn't go to configured state if the cable is inserted before USB is started { // Check if we have ACM TInt currentPersonality; - User::LeaveIfError(iUsb.GetCurrentPersonalityId(currentPersonality)); - User::LeaveIfError(iUsb.ClassSupported(currentPersonality, KECACMUid, gotAcm)); + TInt err = iUsb.GetCurrentPersonalityId(currentPersonality); + if (!err) + { + err = iUsb.ClassSupported(currentPersonality, KECACMUid, gotAcm); + } _LIT(KGotIt, "Current USB personality has ACM, proceeding"); _LIT(KNotGotIt, "Current USB personality doesn't have ACM, waiting for re-enumeration"); if (gotAcm) Message(EInformation, KGotIt); @@ -69,7 +81,7 @@ // We're not enumerated, or we are but don't have ACM. So wait for a (re-)enumeration _LIT(KWaitingForEnumeration, "Waiting for USB enumeration (please connect USB cable)"); Message(EInformation, KWaitingForEnumeration); - iUsb.DeviceStateNotification(EUsbDeviceStateConfigured, usbState, stat); + iUsb.DeviceStateNotification(EUsbDeviceStateConfigured|EUsbDeviceStatePowered, usbState, stat); User::WaitForRequest(stat); if (stat.Int() != KErrNone) { @@ -77,8 +89,8 @@ Message(EError, KUsbError, stat.Int()); User::Leave(stat.Int()); } - _LIT(KUsbEnumerated, "USB cable connected."); - Message(EInformation, KUsbEnumerated); + _LIT(KUsbEnumerated, "USB state changed to %d."); + Message(EInformation, KUsbEnumerated, usbState); } } @@ -107,7 +119,46 @@ Message(EInformation, _L("Preamble script failed with %d"), err); } - //TODO should we ensure that the port passed in here is an ACM::%s port? - Message(EInformation, _L("Opening %S"), &aTitle); - CVtcSerialConsole::ConstructL(aTitle); + Message(EInformation, _L("Opening %S"), &portConfig.iPort); + CVtcSerialConsole::ConstructL(portConfig.iPort); } + +TInt CUsbConsole::ReadConfig(const TDesC& aConfigDes, TUsbPortConfig& aConfig) + { + _LIT(KKeywordPort, "port"); + _LIT(KKeywordPersonality, "personality"); + + TBool keywordFound(EFalse); + TLex lex(aConfigDes); + while (!lex.Eos()) + { + TPtrC keyword; + TPtrC value; + TInt err = ReadKeywordValuePair(lex, keyword, value); + if (err != KErrNone) + { + break; + } + + if (keyword == KKeywordPort) + { + aConfig.iPort.Set(value); + keywordFound = ETrue; + } + else if (keyword == KKeywordPersonality) + { + TLex lex(value); + TInt err = lex.Val(aConfig.iPersonality); + if (err) return err; + keywordFound = ETrue; + } + } + + if (!keywordFound) + { + // Treat unrecognised string as a port (to preserve backwards compatibility with earlier releases). + aConfig.iPort.Set(aConfigDes); + } + + return KErrNone; + } diff -r b9edfff731fb -r f3d01c9dd099 plugins/consoles/vt100cons/src/usb/vtc_usb.h --- a/plugins/consoles/vt100cons/src/usb/vtc_usb.h Fri Sep 17 20:28:05 2010 +0100 +++ b/plugins/consoles/vt100cons/src/usb/vtc_usb.h Wed Sep 22 10:56:39 2010 +0100 @@ -17,6 +17,15 @@ #include #include "vtc_serial.h" +class TUsbPortConfig + { +public: + TUsbPortConfig(); + + TPtrC iPort; + TInt iPersonality; + }; + NONSHARABLE_CLASS(CUsbConsole) : public CVtcSerialConsole { public: @@ -25,6 +34,9 @@ private: // From CVtcSerialConsole. virtual void ConstructL(const TDesC& aTitle); private: + TInt ReadConfig(const TDesC& aConfigDes, TUsbPortConfig& aConfig); + +private: RUsb iUsb; }; diff -r b9edfff731fb -r f3d01c9dd099 plugins/consoles/vt100cons/src/vtc_busdevcons.cpp --- a/plugins/consoles/vt100cons/src/vtc_busdevcons.cpp Fri Sep 17 20:28:05 2010 +0100 +++ b/plugins/consoles/vt100cons/src/vtc_busdevcons.cpp Wed Sep 22 10:56:39 2010 +0100 @@ -106,7 +106,8 @@ else if (keyword == KKeywordPort) { TLex lex(value); - User::LeaveIfError(lex.Val(aConfig.iPort)); + TInt err = lex.Val(aConfig.iPort); + if (err) return err; keywordFound = ETrue; } else if (keyword == KKeywordRate)