diff -r 000000000000 -r 96e5fb8b040d kerneltest/e32utils/usbmsapp/usbmsapp.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/kerneltest/e32utils/usbmsapp/usbmsapp.cpp Thu Dec 17 09:24:54 2009 +0200 @@ -0,0 +1,1573 @@ +// Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies). +// All rights reserved. +// This component and the accompanying materials are made available +// under the terms of the License "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: +// Nokia Corporation - initial contribution. +// +// Contributors: +// +// Description: +// USB Mass Storage Application - also used as an improvised boot loader mechanism +// +// + +/** + @file +*/ + +#include "usbmsapp.h" + +#include +#include +#include +#include +#include + +#include +#include + +TBool gSharedChunkLdd = EFalse; +#include +#include + +#ifdef BUILD_OTG_USBMSAPP +#include +#endif + +#include +#include + +#ifdef USB_BOOT_LOADER + +#include "usbbootvar.h" +#include +#define KNANDLDRLDD_NAME _L("REBOOT.LDD") +static RReboot* RebootDrv; + + +/// Global number of seconds to delay before reboot +static TInt gRebootDelay = 0; + +#endif + +enum + { + EUsbDeviceStateUndefined = EUsbcDeviceStateUndefined, + EUsbDeviceStateConfigured = EUsbcDeviceStateConfigured, + }; + +static CConsoleBase* console = NULL; +static RFs fs; +static TInt selectedDriveIndex = 0; +static TBuf<0x40> mountList; + +static TFixedArray msfsMountedList; ///< 'true' entry corresponds to the drive with mounted MSFS.FSY +static TFixedArray unmountedFsList; ///< every non-NULL entry corresponds to the unmounted original FS for the drive + + +_LIT(KMsFsy, "MSFS.FSY"); +_LIT(KMsFs, "MassStorageFileSystem"); + +_LIT(KOk,"OK"); +_LIT(KError,"Error"); +_LIT(KBytesTransferredFmt, "%c:%d/%d "); +_LIT(KErrFmt, "Error: %d\r"); + +#ifndef USB_BOOT_LOADER +_LIT(KTxtApp,"USBMSAPP"); +_LIT(KDefPwd,"123"); +#endif + +//-- if defined, some useful information will be printed out via RDebug interface +//#define LOGGING_ENABLED + +//----------------------------------------------------------------------------- +/** + prints a line to the console and copies it to the debug log if LOGGING_ENABLED +*/ +void LogPrint(TRefByValue aFmt,...) + { + VA_LIST list; + VA_START(list, aFmt); + + TBuf<0x100> buf; + // coverity[uninit_use_in_call] + buf.FormatList(aFmt, list); //-- ignore overflows + + if(console) + console->Write(buf); + +#ifdef LOGGING_ENABLED + //-- print out the line via RDebug::Print + const TInt bufLen = buf.Length(); + if(bufLen >0 && buf[bufLen-1] == '\n') + { + buf.Insert(bufLen-1, _L("\r")); + } + else + { + buf.Append(_L("\r\n")); + } + + RDebug::RawPrint(buf); +#endif + } + +//----------------------------------------------------------------------------- +/** + prints a line to the debug log if LOGGING_ENABLED +*/ +void Log(TRefByValue aFmt,...) + { +#ifdef LOGGING_ENABLED + + VA_LIST list; + VA_START(list, aFmt); + + TBuf<0x100> buf; + buf.FormatList(aFmt, list); //-- ignore overflows + + //-- print out the line via RDebug::Print + const TInt bufLen = buf.Length(); + if(bufLen >0 && buf[bufLen-1] == '\n') + { + buf.Insert(bufLen-1, _L("\r")); + } + + RDebug::RawPrint(buf); +#else + (void)aFmt; +#endif + } + + +//----------------------------------------------------------------------------- + +static void Clear(int row, int count=1) + { + _LIT(KBlank," "); + for(TInt i=0; iSetPos(0,row+i); + console->Printf(KBlank); + } + console->SetPos(0,row); + } + + +static void ShowDriveSelection() + { + console->SetPos(0,15); + if(PropertyHandlers::allDrivesStatus.Length()/2 > selectedDriveIndex) + { + LogPrint(_L("Selected Drive: %c"), 'A' + PropertyHandlers::allDrivesStatus[selectedDriveIndex*2]); + } + else + { + LogPrint(_L("Selected Drive: (none)")); + } + } + + +#ifdef USB_BOOT_LOADER + +static void rebootit() + { + TInt r=User::LoadLogicalDevice(KNANDLDRLDD_NAME); + RebootDrv=new RReboot; + if(!RebootDrv) + { + User::Panic(_L("Loading driver"),1); + } + r=RebootDrv->Open(); + if (r!=KErrNone) + { + User::Panic(_L("Opening driver"),r); + } + + if (gRebootDelay>0) + { + _LIT(KMsgRebooting,"*** Reboot in %d secs ***\n"); + TInt delay=gRebootDelay; + console->SetPos(0,20); + do + { + LogPrint(KMsgRebooting, delay); + User::After(1000000); + } while(--delay); + } + r=RebootDrv->VariantCtrl(KVariantUsbmsVariantRebootReason, NULL); + if (r!=KErrNone) + { + User::Panic(_L("Rebooting"),r); + } + } + +#endif + +class CPeriodUpdate : public CActive + { +public: + static CPeriodUpdate* NewLC(); +private: + CPeriodUpdate(); + void ConstructL(); + ~CPeriodUpdate(); + void RunL(); + void DoCancel(); + + RTimer iTimer; + TUint iUpTime; + }; + +CPeriodUpdate* CPeriodUpdate::NewLC() + { + CPeriodUpdate* me=new(ELeave) CPeriodUpdate(); + CleanupStack::PushL(me); + me->ConstructL(); + return me; + } + +CPeriodUpdate::CPeriodUpdate() + : CActive(0), iUpTime(0) + {} + +void CPeriodUpdate::ConstructL() + { + CActiveScheduler::Add(this); + iTimer.CreateLocal(); + RunL(); + } + +CPeriodUpdate::~CPeriodUpdate() + { + Cancel(); + } + +void CPeriodUpdate::DoCancel() + { + } + +void CPeriodUpdate::RunL() + { + SetActive(); + // Print RAM usage & up time + + iUpTime++; + TUint totmins=(iUpTime/60); + TUint tothrs=(totmins/60); + TInt mem=0; + if (HAL::Get(HALData::EMemoryRAMFree, mem)==KErrNone) + { + console->SetPos(0,22); + console->Printf(_L("mem (bytes) : %d\n"), mem); + console->Printf(_L("up time : %dh:%dm:%ds\n"), + tothrs, totmins%60, iUpTime%60); + } + iTimer.After(iStatus, 1000000); + } + +//----------------------------------------------------------------------------- +/** + Dismounts the originally mounted FS and optional primary extension from the drive and stores + this information in the FS descriptor + + @return on success returns a pointer to the instantinated FS descriptor +*/ +static CFileSystemDescriptor* DoDismountOrginalFS(RFs& aFs, TInt aDrive) + { + TInt nRes; + TBuf<128> fsName; + TBuf<128> primaryExtName; + TBool bDrvSync = EFalse; + + Log(_L("# DoDismountOrginalFS drv:%d\n"), aDrive); + + //-- 1. get file system name + nRes = aFs.FileSystemName(fsName, aDrive); + if(nRes != KErrNone) + {//-- probably no file system installed at all + return NULL; + } + + //-- 2. find out if the drive sync/async + TPckgBuf drvSyncBuf; + nRes = aFs.QueryVolumeInfoExt(aDrive, EIsDriveSync, drvSyncBuf); + if(nRes == KErrNone) + { + bDrvSync = drvSyncBuf(); + } + + //-- 3. find out primary extension name if it is present; we will need to add it againt when mounting the FS + //-- other extensions (non-primary) are not supported yet + nRes = aFs.ExtensionName(primaryExtName, aDrive, 0); + if(nRes != KErrNone) + { + primaryExtName.SetLength(0); + } + + //-- 3.1 check if the drive has non-primary extensions, fail in this case, because this FS can't be mounted back normally + nRes = aFs.ExtensionName(primaryExtName, aDrive, 1); + if(nRes == KErrNone) + { + LogPrint(_L("Non-primary extensions are not supported!\n")); + return NULL; + } + + Log(_L("# DoDismountOrginalFS FS:%S, Prim ext:%S, synch:%d\n"), &fsName, &primaryExtName, bDrvSync); + + //-- create FS descriptor and dismount the FS + CFileSystemDescriptor* pFsDesc = NULL; + + TRAP(nRes, pFsDesc = CFileSystemDescriptor::NewL(fsName, primaryExtName, bDrvSync)); + if(nRes != KErrNone) + return NULL; //-- OOM ? + + nRes = aFs.DismountFileSystem(fsName, aDrive); + if(nRes != KErrNone) + { + delete pFsDesc; + pFsDesc = NULL; + Log(_L("# DoDismountOrginalFS Dismounting Err:%d\n"), nRes); + } + + return pFsDesc; +} + +//----------------------------------------------------------------------------- +/** + Tries to restore the original FS on the drive using the FS descriptor provided + @return standard error code. +*/ +static TInt DoRestoreFS(RFs& aFs, TInt aDrive, CFileSystemDescriptor* apFsDesc) + { + TInt nRes; + + Log(_L("# DoRestoreFS drv:%d\n"), aDrive); + + //-- 1. check that there is no FS installed + { + TBuf<128> fsName; + nRes = aFs.FileSystemName(fsName, aDrive); + if(nRes == KErrNone) + {//-- probably no file system installed at all + Log(_L("# This drive already has FS intalled:%S \n"), &fsName); + return KErrAlreadyExists; + } + } + + TPtrC ptrN (apFsDesc->FsName()); + TPtrC ptrExt(apFsDesc->PrimaryExtName()); + Log(_L("# Mounting FS:%S, Prim ext:%S, synch:%d\n"), &ptrN, &ptrExt, apFsDesc->DriveIsSynch()); + + if(ptrExt.Length() >0) + {//-- there is a primary extension to be mounted + nRes = aFs.AddExtension(ptrExt); + if(nRes != KErrNone && nRes != KErrAlreadyExists) + { + return nRes; + } + + nRes = aFs.MountFileSystem(ptrN, ptrExt, aDrive, apFsDesc->DriveIsSynch()); + } + else + { + nRes = aFs.MountFileSystem(ptrN, aDrive, apFsDesc->DriveIsSynch()); + } + + if(nRes != KErrNone) + { + Log(_L("# Mount failed! code:%d\n"),nRes); + } + + return nRes; + } + + +//----------------------------------------------------------------------------- +/** + Dismount the original FS from the drive and mount MsFS instead +*/ +static void MountMsFs(TInt driveNumber) + { + TInt x = console->WhereX(); + TInt y = console->WhereY(); + + //-- 1. try dismounting the original FS + CFileSystemDescriptor* fsDesc = DoDismountOrginalFS(fs, driveNumber); + unmountedFsList[driveNumber] = fsDesc; + + console->SetPos(0, 10); + + if(fsDesc) + { + TPtrC ptrN(fsDesc->FsName()); + LogPrint(_L("drv:%d FS:%S Dismounted OK"),driveNumber, &ptrN); + } + else + { + LogPrint(_L("drv:%d Dismount FS Failed!"),driveNumber); + } + + console->ClearToEndOfLine(); + + //-- 2. try to mount the "MSFS" + TInt error; + error = fs.MountFileSystem(KMsFs, driveNumber); + console->SetPos(0, 11); + LogPrint(_L("MSFS Mount: %S (%d)"), (error?&KError:&KOk), error); + console->ClearToEndOfLine(); + + if (!error) + msfsMountedList[driveNumber] = ETrue; + + // restore console position + console->SetPos(x,y); + } + +//----------------------------------------------------------------------------- +/** + Dismount MsFS and mount the original FS +*/ +static TInt RestoreMount(TInt driveNumber) + { + TInt err = KErrNone; + + TInt x = console->WhereX(); + TInt y = console->WhereY(); + + //-- 1. try dismounting the "MSFS" + if (msfsMountedList[driveNumber]) + { + err = fs.DismountFileSystem(KMsFs, driveNumber); + console->SetPos(0, 11); + LogPrint(_L("MSFS Dismount:%S (%d)"), (err?&KError:&KOk), err); + console->ClearToEndOfLine(); + if (err) + return err; + + msfsMountedList[driveNumber] = EFalse; + } + + //-- 2. try to mount the original FS back + CFileSystemDescriptor* fsDesc = unmountedFsList[driveNumber]; + if(fsDesc) + { + err = DoRestoreFS(fs, driveNumber, fsDesc); + + TPtrC ptrN(fsDesc->FsName()); + console->SetPos(0, 10); + LogPrint(_L("%S Mount: %S (%d)"), &ptrN, (err?&KError:&KOk), err); + console->ClearToEndOfLine(); + + delete fsDesc; + unmountedFsList[driveNumber] = NULL; + } + + + // restore console position + console->SetPos(x,y); + return err; + } + +////////////////////////////////////////////////////////////////////////////// +// +// CPropertyWatch +// An active object that tracks changes to the KUsbMsDriveState properties +// +////////////////////////////////////////////////////////////////////////////// + +CPropertyWatch* CPropertyWatch::NewLC(TUsbMsDriveState_Subkey aSubkey, PropertyHandlers::THandler aHandler) + { + CPropertyWatch* me=new(ELeave) CPropertyWatch(aHandler); + CleanupStack::PushL(me); + me->ConstructL(aSubkey); + return me; + } + +CPropertyWatch::CPropertyWatch(PropertyHandlers::THandler aHandler) + : CActive(0), iHandler(aHandler) + {} + +void CPropertyWatch::ConstructL(TUsbMsDriveState_Subkey aSubkey) + { + User::LeaveIfError(iProperty.Attach(KUsbMsDriveState_Category, aSubkey)); + CActiveScheduler::Add(this); + // initial subscription and process current property value + RunL(); + } + +CPropertyWatch::~CPropertyWatch() + { + Cancel(); + iProperty.Close(); + } + +void CPropertyWatch::DoCancel() + { + iProperty.Cancel(); + } + +void CPropertyWatch::RunL() + { + // resubscribe before processing new value to prevent missing updates + iProperty.Subscribe(iStatus); + SetActive(); + + iHandler(iProperty); + } + +////////////////////////////////////////////////////////////////////////////// +// +// CUsbWatch +// +////////////////////////////////////////////////////////////////////////////// + +CUsbWatch* CUsbWatch::NewLC(TAny* aUsb) + { + CUsbWatch* me=new(ELeave) CUsbWatch(aUsb); + CleanupStack::PushL(me); + me->ConstructL(); + return me; + } + +CUsbWatch::CUsbWatch(TAny* aUsb) + : + CActive(0), + iUsb(aUsb), + iUsbDeviceState(EUsbDeviceStateUndefined), + iWasConfigured(EFalse) + {} + +void CUsbWatch::ConstructL() + { + CActiveScheduler::Add(this); + RunL(); + } + +CUsbWatch::~CUsbWatch() + { + Cancel(); +// iUsb.DeviceStateNotificationCancel(); + if (gSharedChunkLdd) + ((RDevUsbcScClient*)iUsb)->AlternateDeviceStatusNotifyCancel(); + else + ((RDevUsbcClient*)iUsb)->AlternateDeviceStatusNotifyCancel(); + } + +void CUsbWatch::DoCancel() + { +// iUsb.DeviceStateNotificationCancel(); + if (gSharedChunkLdd) + ((RDevUsbcScClient*)iUsb)->AlternateDeviceStatusNotifyCancel(); + else + ((RDevUsbcClient*)iUsb)->AlternateDeviceStatusNotifyCancel(); + } + +static TBool IsDriveConnected(TInt driveStatusIndex) + { + TInt driveStatus = PropertyHandlers::allDrivesStatus[2*driveStatusIndex+1]; + return driveStatus >= EUsbMsDriveState_Connected ? ETrue : EFalse; + } + +static TChar DriveNumberToLetter(TInt driveNumber) + { + TChar driveLetter = '?'; + fs.DriveToChar(driveNumber, driveLetter); + return driveLetter; + } + +static TBool IsDriveInMountList(TUint driveLetter) + { + TUint16 driveLetter16 = static_cast(driveLetter); + return(!mountList.Length() || KErrNotFound != mountList.Find(&driveLetter16, 1)); + } + +void CUsbWatch::RunL() + { +// RDebug::Print(_L(">> CUsbWatch[%d] %d"), iUsbDeviceState, iWasConfigured); + +// const TUint stateMask = 0xFF; +// iUsb.DeviceStateNotification(stateMask, iUsbDeviceState, iStatus); + if (gSharedChunkLdd) + ((RDevUsbcScClient*)iUsb)->AlternateDeviceStatusNotify(iStatus, iUsbDeviceState); + else + ((RDevUsbcClient*)iUsb)->AlternateDeviceStatusNotify(iStatus, iUsbDeviceState); + + SetActive(); + + //RDebug::Print(_L("CUsbWatch DeviceStateNotification: iUsbDeviceState=%d"), iUsbDeviceState); + + // If the cable is disconnected, unmount all the connected drives. + if(iWasConfigured && iUsbDeviceState == EUsbDeviceStateUndefined) + { + for(TInt i=0; i PropertyHandlers::allDrivesStatus; +TUsbMsBytesTransferred PropertyHandlers::iKBytesRead; +TUsbMsBytesTransferred PropertyHandlers::iKBytesWritten; +TInt PropertyHandlers::iMediaError; + +void PropertyHandlers::Read(RProperty& aProperty) + { + Transferred(aProperty, iKBytesRead); + } + +void PropertyHandlers::Written(RProperty& aProperty) + { + Transferred(aProperty, iKBytesWritten); + } + +void PropertyHandlers::Transferred(RProperty& aProperty, TUsbMsBytesTransferred& aReadOrWritten) + { + console->SetPos(0,1); + console->Printf(_L("KB R/W: ")); + TInt err = aProperty.Get(aReadOrWritten); + if(err == KErrNone) + { + for(TInt i = 0; i < allDrivesStatus.Length()/2; i++) + { + console->Printf(KBytesTransferredFmt, (char)DriveNumberToLetter(allDrivesStatus[2*i]), iKBytesRead[i], iKBytesWritten[i]); + } + console->ClearToEndOfLine(); + } + else + { + console->Printf(KErrFmt, err); + } + } + +void PropertyHandlers::DriveStatus(RProperty& aProperty) + { +// RDebug::Print(_L(">> PropertyHandlers::DriveStatus")); + TInt err = aProperty.Get(allDrivesStatus); + console->SetPos(0,0); + if(err == KErrNone) + { + LogPrint(_L("Status: ")); + for(TInt i = 0; i < allDrivesStatus.Length()/2; i++) + { + TInt driveNumber = allDrivesStatus[2*i]; + TInt driveStatus = allDrivesStatus[2*i+1]; + TChar driveLetter = DriveNumberToLetter(driveNumber); + +// RDebug::Print(_L("%c:%d "), (char)driveLetter, driveStatus); + + switch(driveStatus) + { + case EUsbMsDriveState_Disconnected: + { + LogPrint(_L("%c:%d:Disconnected "), (char)driveLetter, driveStatus); + break; + } + case EUsbMsDriveState_Connecting: + { + LogPrint(_L("%c:%d:Connecting "), (char)driveLetter, driveStatus); + break; + } + case EUsbMsDriveState_Connected: + { + LogPrint(_L("%c:%d:Connected "), (char)driveLetter, driveStatus); + break; + } + case EUsbMsDriveState_Disconnecting: + { + LogPrint(_L("%c:%d:Disconnecting"), (char)driveLetter, driveStatus); + break; + } + case EUsbMsDriveState_Active: + { + LogPrint(_L("%c:%d:Active "), (char)driveLetter, driveStatus); + break; + } + case EUsbMsDriveState_Locked: + { + LogPrint(_L("%c:%d:Locked "), (char)driveLetter, driveStatus); + break; + } + case EUsbMsDriveState_MediaNotPresent: + { + LogPrint(_L("%c:%d:Not Present "), (char)driveLetter, driveStatus); + break; + } + case EUsbMsDriveState_Removed: + { + LogPrint(_L("%c:%d:Removed "), (char)driveLetter, driveStatus); + break; + } + case EUsbMsDriveState_Error: + { + LogPrint(_L("%c:%d:Error "), (char)driveLetter, driveStatus); + break; + } + default : + { + LogPrint(_L("%c:%d:Unknown "), (char)driveLetter, driveStatus); + break; + } + } + + if(IsDriveInMountList(driveLetter)) + { +#ifndef USB_BOOT_LOADER + if (driveStatus == EUsbMsDriveState_Connecting) + { + MountMsFs(driveNumber); + } + else if (driveStatus == EUsbMsDriveState_Disconnected) + { + RestoreMount(driveNumber); + } +#else + if (driveStatus == EUsbMsDriveState_Disconnecting) + { + RestoreMount(driveNumber); + } + else if (driveStatus == EUsbMsDriveState_Disconnected) + { + static TBool firstTime = ETrue; + + if (!firstTime) + { + RDebug::Print(_L("Eject...")); + // Exit and reboot the target upon receipt of an eject + CActiveScheduler::Stop(); + rebootit(); + } + + firstTime = EFalse; + } +#endif + else + { + //RDebug::Print(_L("PropertyHandlers::DriveStatus: nothing to do")); + } + } + else + { + //RDebug::Print(_L("PropertyHandlers::DriveStatus: %c: is not in mountList\n"), driveLetter); + } + } + } + else + { + LogPrint(KErrFmt, err); + } + + //RDebug::Print(_L("<< PropertyHandlers::DriveStatus")); + } + +void PropertyHandlers::MediaError(RProperty& aProperty) + { + TInt err = aProperty.Get(iMediaError); + if(err != KErrNone) + { + // RDebug::Printf("RProperty::Get returned %d", err); + return; + } + + //RDebug::Printf("PropertyHandlers::MediaError %x", iMediaError); + + TInt x = console->WhereX(); + TInt y = console->WhereY(); + Clear(27,1); + LogPrint(_L("Media Error %x"), iMediaError); + // restore console position + console->SetPos(x,y); + } + +////////////////////////////////////////////////////////////////////////////// +// +// CMessageKeyProcessor +// +////////////////////////////////////////////////////////////////////////////// +CMessageKeyProcessor::CMessageKeyProcessor(CConsoleBase* aConsole) + : CActive(CActive::EPriorityUserInput), iConsole(aConsole) + { + } + +CMessageKeyProcessor* CMessageKeyProcessor::NewLC(CConsoleBase* aConsole + ) + { + CMessageKeyProcessor* self=new (ELeave) CMessageKeyProcessor(aConsole); + CleanupStack::PushL(self); + self->ConstructL(); + return self; + } + +CMessageKeyProcessor* CMessageKeyProcessor::NewL(CConsoleBase* aConsole + ) + { + CMessageKeyProcessor* self = NewLC(aConsole); + CleanupStack::Pop(); + return self; + } + +void CMessageKeyProcessor::ConstructL() + { + // Add to active scheduler + CActiveScheduler::Add(this); + RequestCharacter(); + } + +#ifndef USB_BOOT_LOADER +void CMessageKeyProcessor::MakePassword(TMediaPassword &aPassword) + { + // Create password with same format as eshell and S60 + TBuf<3> password(KDefPwd); + + // fill aPassword with contents of password, not converting to ASCII + const TInt byteLen = password.Length() * 2; + aPassword.Copy(reinterpret_cast(password.Ptr()), byteLen); + } +#endif + + +CMessageKeyProcessor::~CMessageKeyProcessor() + { + // Make sure we're cancelled + Cancel(); + } + +void CMessageKeyProcessor::DoCancel() + { + iConsole->ReadCancel(); + } + +void CMessageKeyProcessor::RunL() + { + // Handle completed request + ProcessKeyPress(TChar(iConsole->KeyCode())); + } + +void CMessageKeyProcessor::RequestCharacter() + { + // A request is issued to the CConsoleBase to accept a + // character from the keyboard. + iConsole->Read(iStatus); + SetActive(); + } + +void CMessageKeyProcessor::ProcessKeyPress(TChar aChar) + { +#ifndef USB_BOOT_LOADER + TInt error = KErrNone; +#endif +#if defined(_DEBUG) + static TBool tracetoggle=EFalse; +#endif + + switch(aChar) + { + case 'q': + case 'Q': + case EKeyEscape: + { + TInt err = KErrNone; + for(TInt j=0; j= PropertyHandlers::allDrivesStatus.Length()/2) + { + selectedDriveIndex = 0; + } + ShowDriveSelection(); + break; + + case 'm': + case 'M': + if(PropertyHandlers::allDrivesStatus.Length()) + { + MountMsFs(PropertyHandlers::allDrivesStatus[selectedDriveIndex*2]); + } + break; + + case 'u': + case 'U': + if(PropertyHandlers::allDrivesStatus.Length()) + { + RestoreMount(PropertyHandlers::allDrivesStatus[selectedDriveIndex*2]); + } + break; + + case 'l': + { + // lock unprotected drive + TMediaPassword password; + MakePassword(password); + + _LIT(KEmpty, ""); + TMediaPassword nul; + nul.Copy(KEmpty); + error = fs.LockDrive(PropertyHandlers::allDrivesStatus[selectedDriveIndex*2], + nul, password, ETrue); + console->SetPos(0,9); + LogPrint(_L("LockDrive %S (%d)"), (error?&KError:&KOk), error); + break; + } + + case 'L': + { + // lock password protected drive + TMediaPassword password; + MakePassword(password); + error = fs.LockDrive(PropertyHandlers::allDrivesStatus[selectedDriveIndex*2], + password, password, ETrue); + console->SetPos(0,9); + LogPrint(_L("LockDrive %S (%d)"), (error?&KError:&KOk), error); + break; + } + + case 'n': + case 'N': + { + TMediaPassword password; + MakePassword(password); + error = fs.UnlockDrive(PropertyHandlers::allDrivesStatus[selectedDriveIndex*2], + password, ETrue); + Clear(9); + LogPrint(_L("UnlockDrive %S (%d)"), (error?&KError:&KOk), error); + } + break; + + case 'c': + case 'C': + { + TMediaPassword password; + MakePassword(password); + error = fs.ClearPassword(PropertyHandlers::allDrivesStatus[selectedDriveIndex*2], + password); + Clear(9); + LogPrint(_L("ClearPassword %S (%d)"), (error?&KError:&KOk), error); + } + break; +#endif + + default: + break; + } + RequestCharacter(); + } + +#ifdef USB_BOOT_LOADER + +static void RunMode() + { + RFs fs; + + TInt r=fs.Connect(); + if (r!=KErrNone) + { + RDebug::Print(_L("Help\n")); + return; + } + + TFileName sessionpath = _L("?:\\"); + + TDriveList drivelist; + fs.DriveList(drivelist); + for (TInt driveno=EDriveC; driveno<=EDriveZ; driveno++) + { + if (!drivelist[driveno]) + continue; + + sessionpath[0]='A'+driveno; + + /* + If a filename with the format EJECTDELAY.nnn is found, delay any reboot + action by "nnn" seconds + */ + CDir* dir; + TFindFile finder(fs); + r=finder.FindWildByPath(_L("EJECTDELAY.*"),&sessionpath,dir); + if (r == KErrNone) + { // Found one or more files + TEntry entry; + entry=(*dir)[0]; + + TParse parser; + parser.Set(entry.iName, NULL, NULL); + TPtrC tok = parser.Ext(); + TLex lex(tok); + lex.SkipAndMark(1); + tok.Set(lex.NextToken()); + lex.Assign(tok); + + r=lex.Val(gRebootDelay); + if (r!=KErrNone) + continue; + } + } + } + +#endif + +////////////////////////////////////////////////////////////////////////////// +// +// Application entry point +// +////////////////////////////////////////////////////////////////////////////// +static void RunAppL() + { +#ifdef USB_BOOT_LOADER + RunMode(); +#endif + + TInt error = KErrUnknown; + + //RDebug::Print(_L("USBMSAPP: Creating console\n")); + +#ifdef USB_BOOT_LOADER + console = Console::NewL(KVariantUsbmsTitle,TSize(KConsFullScreen,KConsFullScreen)); +#else + console = Console::NewL(KTxtApp,TSize(KConsFullScreen,KConsFullScreen)); +#endif + + CleanupStack::PushL(console); + + console->SetPos(0,2); + console->Printf(_L("========================================")); + + CActiveScheduler* sched = new(ELeave) CActiveScheduler; + CleanupStack::PushL(sched); + CActiveScheduler::Install(sched); + + TBuf<20> KDriverFileName; + // Load the logical device + RDevUsbcClient usb; + RDevUsbcScClient usbsc; + RChunk gChunk; + + fs.Connect(); + CleanupClosePushL(fs); + + _LIT(KMountAllDefault,"(all)"); + console->SetPos(0,3); + LogPrint(_L("Drives to auto-mount: %S"), (mountList.Length() ? &mountList : &KMountAllDefault)); + + // Add MS file system + error = fs.AddFileSystem(KMsFsy); + + if(error != KErrNone && error != KErrAlreadyExists) + { + User::Leave(error); + } + console->SetPos(0,4); + LogPrint(_L("MSFS file system:\tAdded OK\n")); + + if (gSharedChunkLdd) + { + KDriverFileName = _L("EUSBCSC.LDD"); + error = User::LoadLogicalDevice(KDriverFileName); + } + else + { + KDriverFileName = _L("EUSBC.LDD"); + error = User::LoadLogicalDevice(KDriverFileName); + } + + if (error != KErrAlreadyExists) + { + User::LeaveIfError(error); + } + + + if (gSharedChunkLdd) + { + error = usbsc.Open(0); + } + else + { + error = usb.Open(0); + } + + User::LeaveIfError(error); + +#ifdef BUILD_OTG_USBMSAPP + + _LIT(KOtgdiLddFilename, "otgdi"); + // Check for OTG support + TBuf8 otg_desc; + if (gSharedChunkLdd) + { + error = usbsc.GetOtgDescriptor(otg_desc); + } + else + { + error = usb.GetOtgDescriptor(otg_desc); + } + + if (!(error == KErrNotSupported || error == KErrNone)) + { + LogPrint(_L("Error %d while fetching OTG descriptor"), error); + User::Leave(-1); + return; + } + + // On an OTG device we have to start the OTG driver, otherwise the Client + // stack will remain disabled forever. + if (error == KErrNotSupported) + { + if (gSharedChunkLdd) + { + CleanupClosePushL(usbsc); + } + else + { + CleanupClosePushL(usb); + } + + User::Leave(-1); + } + + error = User::LoadLogicalDevice(KOtgdiLddFilename); + + if (error != KErrNone) + { + LogPrint(_L("Error %d on loading OTG LDD"), error); + User::Leave(-1); + return; + } + + RUsbOtgDriver iOtgPort; + + error = iOtgPort.Open(); + + if (error != KErrNone) + { + LogPrint(_L("Error %d on opening OTG port"), error); + User::Leave(-1); + return; + } + error = iOtgPort.StartStacks(); + + if (error != KErrNone) + { + LogPrint(_L("Error %d on starting USB stack"), error); + User::Leave(-1); + return; + } + +#endif + + if (gSharedChunkLdd) + { + CleanupClosePushL(usbsc); + RChunk *tChunk = &gChunk; + usbsc.FinalizeInterface(tChunk); + } + else + { + CleanupClosePushL(usb); + } + + +// RDebug::Print(_L("USBMSAPP: Create active objects\n")); + CMessageKeyProcessor::NewLC(console); + CPropertyWatch::NewLC(EUsbMsDriveState_KBytesRead, PropertyHandlers::Read); + CPropertyWatch::NewLC(EUsbMsDriveState_KBytesWritten, PropertyHandlers::Written); + CPropertyWatch::NewLC(EUsbMsDriveState_DriveStatus, PropertyHandlers::DriveStatus); + CPropertyWatch::NewLC(EUsbMsDriveState_MediaError, PropertyHandlers::MediaError); + if (gSharedChunkLdd) + { + CUsbWatch::NewLC(&usbsc); + } + else + { + CUsbWatch::NewLC(&usb); + } + + CPeriodUpdate::NewLC(); + + RUsbMassStorage UsbMs; + TBuf<8> t_vendorId(_L("vendor")); + TBuf<16> t_productId(_L("product")); + TBuf<4> t_productRev(_L("1.00")); + + TMassStorageConfig msConfig; + msConfig.iVendorId.Copy(t_vendorId); + msConfig.iProductId.Copy(t_productId); + msConfig.iProductRev.Copy(t_productRev); + +// console->Printf(_L("Connect to Mass Storage")); + error = UsbMs.Connect(); + User::LeaveIfError(error); + +// console->Printf(_L("Start Mass Storage")); + error = UsbMs.Start(msConfig); + User::LeaveIfError(error); + + TBuf8 deviceDescriptor; + if (gSharedChunkLdd) + { + error = usbsc.GetDeviceDescriptor(deviceDescriptor); + } + else + { + error = usb.GetDeviceDescriptor(deviceDescriptor); + } + + User::LeaveIfError(error); + + const TInt KUsbSpecOffset = 2; + const TInt KUsbDeviceClassOffset = 4; + const TInt KUsbVendorIdOffset = 8; + const TInt KUsbProductIdOffset = 10; + const TInt KUsbDevReleaseOffset = 12; + //Change the USB spec number to 2.00 + deviceDescriptor[KUsbSpecOffset] = 0x00; + deviceDescriptor[KUsbSpecOffset+1] = 0x02; + //Change the Device Class, Device SubClass and Device Protocol + deviceDescriptor[KUsbDeviceClassOffset] = 0x00; + deviceDescriptor[KUsbDeviceClassOffset+1] = 0x00; + deviceDescriptor[KUsbDeviceClassOffset+2] = 0x00; + //Change the device vendor ID (VID) to 0x0E22 (Symbian) + deviceDescriptor[KUsbVendorIdOffset] = 0x22; // little endian + deviceDescriptor[KUsbVendorIdOffset+1] = 0x0E; + //Change the device product ID (PID) to 0x1111 + deviceDescriptor[KUsbProductIdOffset] = 0x12; + deviceDescriptor[KUsbProductIdOffset+1] = 0x11; + //Change the device release number to 3.05 + deviceDescriptor[KUsbDevReleaseOffset] = 0x05; + deviceDescriptor[KUsbDevReleaseOffset+1] = 0x03; + if (gSharedChunkLdd) + { + error = usbsc.SetDeviceDescriptor(deviceDescriptor); + } + else + { + error = usb.SetDeviceDescriptor(deviceDescriptor); + } + + User::LeaveIfError(error); + + // Remove possible Remote-Wakup support in Configuration descriptor, + // so that we can use the MSC device also easily for Chapter9 testing. + TBuf8 configDescriptor; + if (gSharedChunkLdd) + { + error = usbsc.GetConfigurationDescriptor(configDescriptor); + } + else + { + error = usb.GetConfigurationDescriptor(configDescriptor); + } + + User::LeaveIfError(error); + const TInt KConfDesc_AttribOffset = 7; + configDescriptor[KConfDesc_AttribOffset] &= ~KUsbDevAttr_RemoteWakeup; + if (gSharedChunkLdd) + { + error = usbsc.SetConfigurationDescriptor(configDescriptor); + } + else + { + error = usb.SetConfigurationDescriptor(configDescriptor); + } + + User::LeaveIfError(error); + + _LIT16(productID_L, "Symbian USB Mass Storage Device (Base)"); + TBuf16 productID(productID_L); + if (gSharedChunkLdd) + { + error = usbsc.SetProductStringDescriptor(productID); + } + else + { + error = usb.SetProductStringDescriptor(productID); + } + + User::LeaveIfError(error); + + TRequestStatus enum_status; + console->SetPos(0,5); + LogPrint(_L("Re-enumerating...\n")); + +#ifdef BUILD_OTG_USBMSAPP + // For OTG: The USB stack may not yet in the peripheral role. If it is not, + // then ReEnumerate() will not work here as the stack will ignore the call + // since the stack is not active. Therefore we simulate device connection to + // force the stack into peripheral role by calling DeviceConnectToHost(). + if (gSharedChunkLdd) + { + usbsc.DeviceConnectToHost(); + usbsc.ReEnumerate(enum_status); + + } + else + { + usb.DeviceConnectToHost(); + usb.ReEnumerate(enum_status); + } +#else + if (gSharedChunkLdd) + { + usbsc.ReEnumerate(enum_status); + } + else + { + usb.ReEnumerate(enum_status); + } +#endif + + User::LeaveIfError(error); + console->SetPos(0,5); + User::WaitForRequest(enum_status); + if(enum_status.Int() == KErrNone) + LogPrint(_L("Re-enumeration Done\n")); + else + LogPrint(_L("Re-enumeration not successfully done\n")); + +#ifndef USB_BOOT_LOADER + console->SetPos(0,14); + TBuf<3>password(KDefPwd); + LogPrint(_L("Password: %S"), &password); +#endif + ShowDriveSelection(); + + console->SetPos(0,17); +#ifdef USB_BOOT_LOADER + _LIT(KMsgTitleB,"Menu:\n[Esc,Q]=RESET (boot image)\n"); +#else + _LIT(KMsgTitleB,"Menu: q=quit d=chg drv\n m=mount u=unmount\n l=lock n=unlock\n c=clr pwd"); +#endif + + //RDebug::Print(_L("USBMSAPP: Start CActiveScheduler\n")); + + console->Printf(KMsgTitleB); + +#ifdef USB_BOOT_LOADER + // Mount the mass storage on variant specific drive + MountMsFs(KVariantUsbmsRemoveableDrive); +#endif + + CActiveScheduler::Start(); + + error = UsbMs.Stop(); + User::LeaveIfError(error); + UsbMs.Close(); + error = fs.RemoveFileSystem(KMsFs); + User::LeaveIfError(error); + + //The console object is left on the Cleanup Stack, + //which is used in the delay processing logic of rebootit(). + CleanupStack::PopAndDestroy(10); + +#ifdef BUILD_OTG_USBMSAPP + iOtgPort.StopStacks(); + iOtgPort.Close(); + error = User::FreeLogicalDevice(RUsbOtgDriver::Name()); + User::LeaveIfError(error); +#endif + + // UnLoad the logical device + TBuf<20> KDriverName; + if (gSharedChunkLdd) + { + KDriverName = _L("USBCSC"); + gChunk.Close(); + usbsc.Close(); + User::After(100000); + error = User::FreeLogicalDevice(KDriverName); + } + else + { + KDriverName = _L("USBC"); + error = User::FreeLogicalDevice(KDriverName); + } + + User::LeaveIfError(error); + +#ifdef USB_BOOT_LOADER + rebootit(); +#endif + CleanupStack::PopAndDestroy(1); + } + +TInt ParseCommandLine() + { + TBuf<32> args; + User::CommandLine(args); + TLex lex(args); + TInt err=KErrNone; + FOREVER + { + TPtrC token=lex.NextToken(); + + if(token.Length()!=0) + { + if ((token.MatchF(_L("-sc")) == KErrNone)) + { + gSharedChunkLdd = ETrue; + err = KErrNone; + } + else + { + // Command line: list of drive letters to auto-mount (all if not specified) + mountList.Append(token); + mountList.UpperCase(); + err = KErrNone; + } // endif token + } + else + break; + } + return err; + } + + + +GLDEF_C TInt E32Main() + { + __UHEAP_MARK; + CTrapCleanup* cleanup=CTrapCleanup::New(); + + if (ParseCommandLine()) + return KErrNone; + + msfsMountedList.Reset(); + unmountedFsList.Reset(); + + + TRAPD(error,RunAppL()); +#ifdef USB_BOOT_LOADER + __ASSERT_ALWAYS(!error, User::Panic(KVariantUsbmsTitle, error)); +#else + __ASSERT_ALWAYS(!error, User::Panic(KTxtApp, error)); +#endif + + delete cleanup; + __UHEAP_MARKEND; + return 0; + } + + + + +//----------------------------------------------------------------------------- + +CFileSystemDescriptor::~CFileSystemDescriptor() + { + iFsName.Close(); + iPrimaryExtName.Close(); + } + +//----------------------------------------------------------------------------- +CFileSystemDescriptor* CFileSystemDescriptor::NewL(const TDesC& aFsName, const TDesC& aPrimaryExtName, TBool aDrvSynch) + { + CFileSystemDescriptor* pSelf = new (ELeave) CFileSystemDescriptor; + + CleanupStack::PushL(pSelf); + + pSelf->iFsName.CreateMaxL(aFsName.Length()); + pSelf->iFsName.Copy(aFsName); + + pSelf->iPrimaryExtName.CreateMaxL(aPrimaryExtName.Length()); + pSelf->iPrimaryExtName.Copy(aPrimaryExtName); + + pSelf->iDriveSynch = aDrvSynch; + + CleanupStack::Pop(); + + return pSelf; + } + + + + + + + + + + + + + + + + + + + +