usbmgmt/usbmgr/usbman/chargingplugin/src/CUsbBatteryChargingPlugin.cpp
author hgs
Thu, 16 Sep 2010 12:03:29 +0800
changeset 49 93c0009bd947
parent 29 59aa7d6e3e0f
permissions -rw-r--r--
201036_01

/*
* Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of "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:
*
*/

/** @file
@internalComponent
*/
#include <e32debug.h> 
#include <e32def.h>
#include <e32property.h>
#include <usbotgdefs.h>
#include <musbmanextensionpluginobserver.h>
#include <centralrepository.h>

#include "CUsbBatteryChargingPlugin.h"
#include "chargingstates.h"
#include "reenumerator.h"

#ifdef SYMBIAN_ENABLE_USB_OTG_HOST_PRIV          // For host OTG enabled charging plug-in 
#include "idpinwatcher.h"
#include "otgstatewatcher.h"
#endif

#include "vbuswatcher.h"
#include "OstTraceDefinitions.h"
#ifdef OST_TRACE_COMPILER_IN_USE
#include "CUsbBatteryChargingPluginTraces.h"
#endif


static const TInt KUsbBatteryChargingConfigurationDescriptorCurrentOffset = 8; // see bMaxPower in section 9.6.3 of USB Spec 2.0
static const TInt KUsbBatteryChargingCurrentRequestTimeout = 3000000; // 3 seconds
static const TUint8 KMinCurrent     = 8;    // See USB OTG specification , min current
static const TUint8 KbmAttributes   = 7;    // see bmAttributes in section 9.6.3 of USB Spec 2.0
static const TUint8 KSelfPowered    = 0x40; // Bitmask to bmAttributes (01000000)




/**
Factory function.
@return Ownership of a new CUsbBatteryChargingPlugin.
*/
CUsbBatteryChargingPlugin* CUsbBatteryChargingPlugin::NewL(MUsbmanExtensionPluginObserver& aObserver)
    {
    CUsbBatteryChargingPlugin* self = new(ELeave) CUsbBatteryChargingPlugin(aObserver);
    CleanupStack::PushL(self);
    self->ConstructL();
    CleanupStack::Pop(self);
    return self;
    }

/**
Destructor.
*/
CUsbBatteryChargingPlugin::~CUsbBatteryChargingPlugin()
    {
    OstTraceFunctionEntry1( REF_CUSBBATTERYCHARGINGPLUGIN_CUSBBATTERYCHARGINGPLUGIN_DES_ENTRY, this );
    
    iCurrentValues.Close();
    delete iDeviceReEnumerator;
    delete iDeviceStateTimer;
    
// For host OTG enabled charging plug-in
#ifdef SYMBIAN_ENABLE_USB_OTG_HOST_PRIV
    delete iIdPinWatcher;
    delete iOtgStateWatcher;
#endif
    
    delete iVBusWatcher;
    for (TInt index = 0; index < EPluginStateCount; index ++)
        {
        delete iPluginStates[index];
        iPluginStates[index] = NULL;
        }
    }

/**
Constructor.
*/
CUsbBatteryChargingPlugin::CUsbBatteryChargingPlugin(MUsbmanExtensionPluginObserver& aObserver)
:    CUsbmanExtensionPlugin(aObserver) , iLdd(Observer().DevUsbcClient())
    {
    }

/**
2nd-phase construction.
*/
void CUsbBatteryChargingPlugin::ConstructL()
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_CONSTRUCTL_ENTRY );
   
    iUsbSpeedType = EUsbFullSpeed;
	iChargingInfo().iChargingPortType = EUsbChargingPortTypeStandardDownstreamPort;
	iChargingInfo().iUsbConnStatus = EUsbConnectionStatusNone;
	iChargingInfo().iMinAvailableVbusCurrent = 0;
	iChargingInfo().iMaxVbusCurrent = 0;
	iChargingInfo().iMinVbusVoltage = 0;	
#ifdef SYMBIAN_USB_BATTERYCHARGING_V1_1		
	iChargingInfo().iChargingPortType = EUsbChargingPortTypeNone;
	iChargerDetectotCaps.iCapabilities = 0;
#endif
	iLastPublishedInfo = iChargingInfo();

    // Create state objects
    iPluginStates[EPluginStateIdle] = 
        new (ELeave) TUsbBatteryChargingPluginStateIdle(*this);
    iPluginStates[EPluginStateCurrentNegotiating] = 
        new (ELeave) TUsbBatteryChargingPluginStateCurrentNegotiating(*this);
    iPluginStates[EPluginStateNoValidCurrent] = 
        new (ELeave) TUsbBatteryChargingPluginStateNoValidCurrent(*this);
    iPluginStates[EPluginStateIdleNegotiated] = 
        new (ELeave) TUsbBatteryChargingPluginStateIdleNegotiated(*this);
    iPluginStates[EPluginStateBEndedCableNotPresent] = 
        new (ELeave) TUsbBatteryChargingPluginStateBEndedCableNotPresent(*this);

    // Set initial state to idle
    SetState(EPluginStateIdle);
    
    TInt err = RProperty::Define(KPropertyUidUsbBatteryChargingCategory,
                      KPropertyUidUsbBatteryChargingInfo,
                      RProperty::EByteArray,
                      sizeof(TPublishedUsbChargingInfo));

    if(err == KErrNone || err == KErrAlreadyExists)
        {
        err = RProperty::Set(KPropertyUidUsbBatteryChargingCategory,
                      KPropertyUidUsbBatteryChargingInfo,
                      iChargingInfo);
        }
	else
		{		
        User::Leave(err);
		}

    if(err != KErrNone)
        {
        static_cast<void> (RProperty::Delete (
                KPropertyUidUsbBatteryChargingCategory,
                KPropertyUidUsbBatteryChargingInfo ));
        User::Leave(err);
        }
        
    iDeviceStateTimer = CUsbChargingDeviceStateTimer::NewL(*this);

    iDeviceReEnumerator = CUsbChargingReEnumerator::NewL(iLdd);

    iPluginState = EPluginStateIdle;
    ReadCurrentRequestValuesL();
    iVBusWatcher = CVBusWatcher::NewL(this);
    iVBusState = iVBusWatcher->VBusState();
    
// For host OTG enabled charging plug-in
#ifdef SYMBIAN_ENABLE_USB_OTG_HOST_PRIV
    iOtgStateWatcher = COtgStateWatcher::NewL(this);
    iOtgState = iOtgStateWatcher->OtgState();
    iIdPinWatcher = CIdPinWatcher::NewL(this);
    TInt value = iIdPinWatcher->IdPinValue();
    iIdPinState = iIdPinWatcher->IdPinValue();
    if (iIdPinState == EUsbBatteryChargingIdPinBRole)
#else
    if (ETrue)
#endif
        {
#if !defined(__WINS__) && !defined(__CHARGING_PLUGIN_TEST_CODE__)
        SetInitialConfigurationL();
#endif
        }
    else
        {
        iPluginState = EPluginStateBEndedCableNotPresent;
        OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_CONSTRUCTL_DUP3, "CUsbBatteryChargingPlugin::ConstructL;PluginState => EPluginStateADevice(%d)", iPluginState );
        }

    Observer().RegisterStateObserverL(*this);  
#ifdef SYMBIAN_USB_BATTERYCHARGING_V1_1		
	MUsbChargingNotify* charging = static_cast<MUsbChargingNotify*>(this);
    Observer().RegisterChargingObserverL(*charging);  
#endif	
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_CONSTRUCTL_EXIT );
    }

// For host OTG enabled charging plug-in
TBool CUsbBatteryChargingPlugin::IsUsbChargingPossible()
    {
#ifdef SYMBIAN_ENABLE_USB_OTG_HOST_PRIV
    // 'A' end cable connected. I'm the power supplier, with no way to charge myself :-) 
    if (EUsbBatteryChargingIdPinARole == iIdPinState)
        {
        return EFalse;
        }

    // BHost charging is disabled
    if (EUsbOtgStateBHost == iOtgState)
        {
        return EFalse;
        }
        
    return ETrue;
#else
    return ETrue;
#endif
    }
/**
 Set first power to request
*/
void CUsbBatteryChargingPlugin::SetInitialConfigurationL()
    {
    OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;Setting Initial Configuration" );
    if (iCurrentValues.Count() > 0)
        {
        TInt err;
        TInt configDescriptorSize = 0;
        err = iLdd.GetConfigurationDescriptorSize(configDescriptorSize);
        if(err < 0)
            {
            OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL_DUP1, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;iLdd.GetConfigurationDescriptorSize(configDescriptorSize) with error=%d", err );
            User::Leave(err);
            }
        HBufC8* configDescriptor = HBufC8::NewLC(configDescriptorSize);
        TPtr8 ptr(configDescriptor->Des());

        OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL_DUP2, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;Getting Configuration Descriptor (size = %d)", configDescriptorSize );
        err = iLdd.GetConfigurationDescriptor(ptr);
        if(err < 0)
            {
            OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL_DUP3, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;iLdd.GetConfigurationDescriptor(ptr) with error=%d", err );
            User::Leave(err);
            }

// For host OTG enabled charging plug-in
#ifdef SYMBIAN_ENABLE_USB_OTG_HOST_PRIV
        // Get first power to put in configurator
        OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL_DUP4, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;Checking IdPin state:" );
        if(iIdPinState == EUsbBatteryChargingIdPinBRole)
#else
        if (ETrue)
#endif
        	{
            if (iCurrentValues.Count() > 0)
                {
                iCurrentIndexRequested = 0;
                iRequestedCurrentValue = iCurrentValues[iCurrentIndexRequested];
                OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL_DUP5, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;IdPin state is 0, current set to: %d", iRequestedCurrentValue );
                }
            else
                {
                OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL_DUP6, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;No vailable current found !" );
                }
            }
        else
            {
            iRequestedCurrentValue = 0;
            OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL_DUP7, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;IdPin state is 1, current set to 0" );
            }

        TUint oldCurrentValue = ptr[KUsbBatteryChargingConfigurationDescriptorCurrentOffset] << 1;
        ptr[KUsbBatteryChargingConfigurationDescriptorCurrentOffset] = (iRequestedCurrentValue >> 1);

        OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL_DUP8, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;Setting Updated Configuration Descriptor" );
        err = iLdd.SetConfigurationDescriptor(ptr);
        if(err < 0)
            {
            OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETINITIALCONFIGURATIONL_DUP9, "CUsbBatteryChargingPlugin::SetInitialConfigurationL;iLdd.SetConfigurationDescriptor(ptr) with error=%d", err );
            User::Leave(err);
            }

        CleanupStack::PopAndDestroy(configDescriptor); 
        }
    }


TAny* CUsbBatteryChargingPlugin::GetInterface(TUid aUid)
    {
    OstTrace1( TRACE_FLOW, REF_CUSBBATTERYCHARGINGPLUGIN_GETINTERFACE, "CUsbBatteryChargingPlugin::GetInterface;this = [0x%08x]", this );
    OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_GETINTERFACE_DUP1, "CUsbBatteryChargingPlugin::GetInterface;aUid = 0x%08x", aUid.iUid );
    (void)aUid;

    TAny* ret = NULL;
    OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_GETINTERFACE_DUP2, "CUsbBatteryChargingPlugin::GetInterface;ret = [0x%08x]", ret );
    return ret;
    }

void CUsbBatteryChargingPlugin::Panic(TUsbBatteryChargingPanic aPanic)
    {
    OstTrace1( TRACE_FATAL, REF_CUSBBATTERYCHARGINGPLUGIN_PANIC, "CUsbBatteryChargingPlugin::Panic;*** CUsbBatteryChargingPlugin::Panic(%d) ***", aPanic );
    _LIT(KUsbChargingPanic,"USB Charging");
    User::Panic(KUsbChargingPanic, aPanic);
    }

void CUsbBatteryChargingPlugin::UsbServiceStateChange(TInt /*aLastError*/, TUsbServiceState /*aOldState*/, TUsbServiceState /*aNewState*/)
    {
    // not used
    }

TUsbChargingPluginState CUsbBatteryChargingPlugin::SetState(TUsbChargingPluginState aState)
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_SETSTATE_ENTRY );

    switch (aState)
        {
        case EPluginStateIdle:
            {
            ResetPlugin();
            iCurrentState = iPluginStates[aState];
            }
            break;
        case EPluginStateCurrentNegotiating:
        case EPluginStateNoValidCurrent:
        case EPluginStateIdleNegotiated:
        case EPluginStateBEndedCableNotPresent:
            iCurrentState = iPluginStates[aState];
			break;			
        default:
            // Should never happen ...
            iPluginState = EPluginStateIdle;
            iCurrentState = iPluginStates[EPluginStateIdle];
    
            OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETSTATE, "CUsbBatteryChargingPlugin::SetState;Invalid new state: aState = %d", aState );
            
            Panic(EUsbBatteryChargingPanicUnexpectedPluginState);
        }
    iPluginState = aState;
    
    OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_SETSTATE_DUP1, "CUsbBatteryChargingPlugin::SetState;New state: aState=%d", aState );
    
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_SETSTATE_EXIT );
    return iPluginState;
    }

void CUsbBatteryChargingPlugin::NegotiateChargingCurrent()
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_NEGOTIATECHARGINGCURRENT_ENTRY );

    OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_NEGOTIATECHARGINGCURRENT, "CUsbBatteryChargingPlugin::NegotiateChargingCurrent;iDeviceState=%d", iDeviceState );
    TRAPD(result, NegotiateNextCurrentValueL());
    if(result == KErrNone)
        {
        SetState(EPluginStateCurrentNegotiating);
        }
    else
        {
        OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_NEGOTIATECHARGINGCURRENT_DUP1, "CUsbBatteryChargingPlugin::NegotiateChargingCurrent;Negotiation call failed, iVBusState = 1: result = %d", result );
        }
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_NEGOTIATECHARGINGCURRENT_EXIT );
    }

void CUsbBatteryChargingPlugin::UsbDeviceStateChange(TInt aLastError,
        TUsbDeviceState aOldState, TUsbDeviceState aNewState)
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_USBDEVICESTATECHANGE_ENTRY );
    
    iCurrentState->UsbDeviceStateChange(aLastError, aOldState, aNewState);
	UpdateChargingInfo();
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_USBDEVICESTATECHANGE_EXIT );
    }

void CUsbBatteryChargingPlugin::DeviceStateTimeout()
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_DEVICESTATETIMEOUT_ENTRY );
        
    iCurrentState->DeviceStateTimeout();	
	UpdateChargingInfo();
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_DEVICESTATETIMEOUT_EXIT );
    }

void CUsbBatteryChargingPlugin::NegotiateNextCurrentValueL()
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_NEGOTIATENEXTCURRENTVALUEL_ENTRY );

    iDeviceStateTimer->Cancel();
    TUint newCurrent = 0;

    if ((iPluginState == EPluginStateIdle)  && iCurrentValues.Count() > 0)
        {
        // i.e. we haven't requested anything yet, and there are some current values to try
        iCurrentIndexRequested  = 0;
        newCurrent = iCurrentValues[iCurrentIndexRequested];        
        }
    else if (iPluginState == EPluginStateCurrentNegotiating && ( iCurrentIndexRequested + 1) < iCurrentValues.Count())
        {
        // there are more current values left to try
        iCurrentIndexRequested++;
        newCurrent = iCurrentValues[iCurrentIndexRequested];    
        }
    else if(iRequestedCurrentValue != 0 && iRequestedCurrentValue != KMinCurrent)
        {
        newCurrent = KMinCurrent;
        }
    else
        {
        iPluginState = EPluginStateNoValidCurrent;
        User::Leave(EUsbBatteryChargingPanicBadCharingCurrentNegotiation);
        }
    
    RequestCurrentL(newCurrent);
    iRequestedCurrentValue = newCurrent;
    iPluginState = EPluginStateCurrentNegotiating;
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_NEGOTIATENEXTCURRENTVALUEL_EXIT );
    }

void CUsbBatteryChargingPlugin::ResetPlugin()
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_RESETPLUGIN_ENTRY );
    
    if((iPluginState != EPluginStateIdle))
        {
        iDeviceStateTimer->Cancel(); // doesn't matter if not running
        iPluginState = EPluginStateIdle;
        OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_RESETPLUGIN, "CUsbBatteryChargingPlugin::ResetPlugin;PluginState => EPluginStateIdle(%d)", iPluginState );

        iRequestedCurrentValue = 0;
        iCurrentIndexRequested = 0;
        iAvailableMilliAmps = 0;
		iBDevMaxPower = 0;
        UpdateChargingInfo();
        TRAP_IGNORE(SetInitialConfigurationL());
        }
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_RESETPLUGIN_EXIT );
    }

void CUsbBatteryChargingPlugin::RequestCurrentL(TUint aMilliAmps)
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_ENTRY );
    
    OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL, "CUsbBatteryChargingPlugin::RequestCurrentL;aMilliAmps=%u", aMilliAmps );

    if((EPluginStateCurrentNegotiating == iPluginState) && (iRequestedCurrentValue != aMilliAmps))
        {
        TInt err;
        TInt configDescriptorSize = 0;
        err = iLdd.GetConfigurationDescriptorSize(configDescriptorSize);
        if(err < 0)
            {
            OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP1, "CUsbBatteryChargingPlugin::RequestCurrentL;iLdd.GetConfigurationDescriptorSize(configDescriptorSize) with error=%d", err );
            User::Leave(err);
            }
        HBufC8* configDescriptor = HBufC8::NewLC(configDescriptorSize);
        TPtr8 ptr(configDescriptor->Des());

        OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP2, "CUsbBatteryChargingPlugin::RequestCurrentL;Getting Configuration Descriptor (size = %d)", configDescriptorSize );
        err = iLdd.GetConfigurationDescriptor(ptr);
        if(err < 0)
            {
            OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP3, "CUsbBatteryChargingPlugin::RequestCurrentL;iLdd.GetConfigurationDescriptor(ptr) with error=%d", err );
            User::Leave(err);
            }

		// compliant to charging adaptation in modem adaptation
        TUint8 bmAttributes = ptr[KbmAttributes];
        bmAttributes &= ~(KSelfPowered);
        ptr[KbmAttributes] = bmAttributes;

        // set bMaxPower field. One unit = 2mA, so need to halve aMilliAmps.
        OstTraceExt2( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP4, "CUsbBatteryChargingPlugin::RequestCurrentL;Setting bMaxPower to %u mA ( = %u x 2mA units)", aMilliAmps, aMilliAmps >> 1 );
        TUint oldCurrentValue = ptr[KUsbBatteryChargingConfigurationDescriptorCurrentOffset] << 1;
        OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP5, "CUsbBatteryChargingPlugin::RequestCurrentL;(old value was %u mA)", oldCurrentValue );

        //since the device will force reEnumeration if the value is odd
        aMilliAmps = aMilliAmps & 0xFFFE;    
    
        // to negotiate a new current value, ReEnumerate is needed
        OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP6, "CUsbBatteryChargingPlugin::RequestCurrentL;Forcing ReEnumeration" );
        ptr[KUsbBatteryChargingConfigurationDescriptorCurrentOffset] = (aMilliAmps >> 1);
        OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP7, "CUsbBatteryChargingPlugin::RequestCurrentL;Setting Updated Configuration Descriptor" );
        err = iLdd.SetConfigurationDescriptor(ptr);
        if(err < 0)
            {
            OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP8, "CUsbBatteryChargingPlugin::RequestCurrentL;iLdd.SetConfigurationDescriptor(ptr) with err=%d", err );
            User::Leave(err);
            }
        OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP9, "CUsbBatteryChargingPlugin::RequestCurrentL;Triggering Re-enumeration" );
        iDeviceReEnumerator->ReEnumerate();
        
        CleanupStack::PopAndDestroy(configDescriptor); // configDescriptor
        }    
    
    // Always issue a timer as a watchdog to monitor the request progress
    OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_DUP10, "CUsbBatteryChargingPlugin::RequestCurrentL;Starting timer: %d", User::NTickCount() );
    iDeviceStateTimer->Cancel();
    iDeviceStateTimer->Start(TTimeIntervalMicroSeconds32(KUsbBatteryChargingCurrentRequestTimeout));
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_REQUESTCURRENTL_EXIT );
    }

void CUsbBatteryChargingPlugin::ReadCurrentRequestValuesL()
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_READCURRENTREQUESTVALUESL_ENTRY );
    
    CRepository* repository = CRepository::NewLC(KUsbBatteryChargingCentralRepositoryUid);

    TInt numberOfCurrents = 0;
    repository->Get(KUsbBatteryChargingKeyNumberOfCurrentValues, numberOfCurrents);

    TInt i = 0;
    for (i=0; i<numberOfCurrents; i++)
        {
        TInt value;
        repository->Get(KUsbBatteryChargingCurrentValuesOffset + i, value);
        iCurrentValues.Append(static_cast<TUint>(value));
        OstTraceExt2( TRACE_FLOW, REF_CUSBBATTERYCHARGINGPLUGIN_READCURRENTREQUESTVALUESL, "CUsbBatteryChargingPlugin::ReadCurrentRequestValuesL;CurrentValue %d = %dmA", i, value );
        }

    CleanupStack::PopAndDestroy(repository);
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_READCURRENTREQUESTVALUESL_EXIT );
    }

#ifndef _DEBUG
void CUsbBatteryChargingPlugin::LogStateText(TUsbDeviceState /*aState*/)
    {
    }
#else
void CUsbBatteryChargingPlugin::LogStateText(TUsbDeviceState aState)
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_LOGSTATETEXT_ENTRY );
    
    switch (aState)
        {
        case EUsbDeviceStateUndefined:
            OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_LOGSTATETEXT, "CUsbBatteryChargingPlugin::LogStateText; ***** UNDEFINED *****" );
            break;
        case EUsbDeviceStateDefault:
            OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_LOGSTATETEXT_DUP1, "CUsbBatteryChargingPlugin::LogStateText; ***** DEFAULT *****" );
            break;
        case EUsbDeviceStateAttached:
            OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_LOGSTATETEXT_DUP2, "CUsbBatteryChargingPlugin::LogStateText; ***** ATTACHED *****" );
            break;
        case EUsbDeviceStatePowered:
            OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_LOGSTATETEXT_DUP3, "CUsbBatteryChargingPlugin::LogStateText; ***** POWERED *****" );
            break;
        case EUsbDeviceStateConfigured:
            OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_LOGSTATETEXT_DUP4, "CUsbBatteryChargingPlugin::LogStateText; ***** CONFIGURED *****" );
            break;
        case EUsbDeviceStateAddress:
            OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_LOGSTATETEXT_DUP5, "CUsbBatteryChargingPlugin::LogStateText; ***** ADDRESS *****" );
            break;
        case EUsbDeviceStateSuspended:
            OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_LOGSTATETEXT_DUP6, "CUsbBatteryChargingPlugin::LogStateText; ***** SUSPENDED *****" );
            break;
        default:
            break;
        }
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_LOGSTATETEXT_EXIT );
    }
#endif

void CUsbBatteryChargingPlugin::MpsoVBusStateChanged(TInt aNewState)
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_MPSOVBUSSTATECHANGED_ENTRY );
    if (iChargingInfo().iChargingPortType == EUsbChargingPortTypeStandardDownstreamPort)
    	{
    	iCurrentState->MpsoVBusStateChanged(aNewState);
    	}
	iVBusState = aNewState;
	UpdateChargingInfo();
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_MPSOVBUSSTATECHANGED_EXIT );
    }


// For host OTG enabled charging plug-in
#ifdef SYMBIAN_ENABLE_USB_OTG_HOST_PRIV
void CUsbBatteryChargingPlugin::MpsoIdPinStateChanged(TInt aValue)
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_MPSOIDPINSTATECHANGED_ENTRY );
    
    iCurrentState->MpsoIdPinStateChanged(aValue);
	UpdateChargingInfo();
    OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_MPSOIDPINSTATECHANGED_EXIT );
    }

void CUsbBatteryChargingPlugin::MpsoOtgStateChangedL(TUsbOtgState aNewState)
    {
    OstTraceFunctionEntry0( REF_CUSBBATTERYCHARGINGPLUGIN_MPSOOTGSTATECHANGEDL_ENTRY );

    iCurrentState->MpsoOtgStateChangedL(aNewState);
	UpdateChargingInfo();
	OstTraceFunctionExit0( REF_CUSBBATTERYCHARGINGPLUGIN_MPSOOTGSTATECHANGEDL_EXIT );
    }
#endif


#ifdef SYMBIAN_USB_BATTERYCHARGING_V1_1
void CUsbBatteryChargingPlugin::UsbChargingPortType(TUint aChargingPortType)
	{
	OstTraceExt1( TRACE_NORMAL, CUSBBATTERYCHARGINGPLUGIN_USBCHARGINGPORTTYPE, "CUsbBatteryChargingPlugin::UsbChargingPortType;ChargingTypeToString(aChargingPortType)=%s", *ChargingTypeToString(aChargingPortType) );
	ChargingTypeToString(aChargingPortType);
	switch (aChargingPortType)
		{
		case KUsbChargingPortTypeNone:
			iChargingInfo().iChargingPortType = EUsbChargingPortTypeNone;
			break;
		case KUsbChargingPortTypeUnsupported:
			iChargingInfo().iChargingPortType = EUsbChargingPortTypeUnsupported;
			break;
		case KUsbChargingPortTypeDedicatedChargingPort:
			iChargingInfo().iChargingPortType = EUsbChargingPortTypeDedicatedChargingPort;
			break;				
		case KUsbChargingPortTypeChargingDownstreamPort:
			iChargingInfo().iChargingPortType = EUsbChargingPortTypeChargingDownstreamPort;
			break;
		case KUsbChargingPortTypeStandardDownstreamPort:
			iChargingInfo().iChargingPortType = EUsbChargingPortTypeStandardDownstreamPort;
			break;
		case KUsbChargingPortTypeAcaRidA:
			iChargingInfo().iChargingPortType = EUsbChargingPortTypeAcaRidA;
			break;
		case KUsbChargingPortTypeAcaRidB:
			iChargingInfo().iChargingPortType = EUsbChargingPortTypeAcaRidB;
			break;
		case KUsbChargingPortTypeAcaRidC:
			iChargingInfo().iChargingPortType = EUsbChargingPortTypeAcaRidC;
			break;
		case KUsbChargingPortTypeChargingPort:	
			iChargingInfo().iChargingPortType = EUsbChargingPortTypeChargingPort;
			break;
		default:
			break;
		}
	iLdd.ChargerDetectorCaps(iChargerDetectotCaps);
	iCurrentState->UsbChargingPortType(aChargingPortType);
	UpdateChargingInfo();
	}
#endif

_LIT8(KPortTypeNone, "PortTypeNone");
_LIT8(KPortTypeUnsupported, "PortTypeUnsupported");
_LIT8(KPortTypeDedicatedChargingPort, "PortTypeDedicatedChargingPort");
_LIT8(KPortTypeStandardDownstreamPort, "PortTypeStandardDownstreamPort");
_LIT8(KPortTypeChargingDownstreamPort, "PortTypeChargingDownstreamPort");
_LIT8(KPortTypeAcaRidA, "PortTypeAcaRidA");
_LIT8(KPortTypeAcaRidB, "PortTypeAcaRidB");
_LIT8(KPortTypeAcaRidC, "PortTypeAcaRidC");
_LIT8(KPortTypeChargingPort, "PortTypeChargingPort");


const TDesC8* CUsbBatteryChargingPlugin::ChargingTypeToString(TUint aChargingType)
	{
	const TDesC8* str = NULL;
	switch (aChargingType)
   		{
		case EUsbChargingPortTypeNone:
			str = &KPortTypeNone;
			break;
		case EUsbChargingPortTypeUnsupported:
			str = &KPortTypeUnsupported;
			break;
		case EUsbChargingPortTypeDedicatedChargingPort:
			str = &KPortTypeDedicatedChargingPort;
			break;				
		case EUsbChargingPortTypeChargingDownstreamPort:
			str = &KPortTypeChargingDownstreamPort;
			break;
		case EUsbChargingPortTypeStandardDownstreamPort:
			str = &KPortTypeStandardDownstreamPort;
			break;
		case EUsbChargingPortTypeAcaRidA:
			str = &KPortTypeAcaRidA;
			break;
		case EUsbChargingPortTypeAcaRidB:
			str = &KPortTypeAcaRidB;
			break;
		case EUsbChargingPortTypeAcaRidC:
			str = &KPortTypeAcaRidC;
			break;
		case EUsbChargingPortTypeChargingPort:	
			str = &KPortTypeChargingPort;
			break;
		default:
			break;
		}
	return str;
	}

void CUsbBatteryChargingPlugin::QueryCurrentSpeed()
    {
	if (iChargingInfo().iChargingPortType == EUsbChargingPortTypeChargingDownstreamPort)
		{
		if (iLdd.CurrentlyUsingHighSpeed())
			{
		    OstTrace0( TRACE_NORMAL, CUSBBATTERYCHARGINGPLUGIN_QUERYCURRENTSPEED, "CUsbBatteryChargingPlugin::QueryCurrentSpeed is high speed" );
			iUsbSpeedType = EUsbHighSpeed;
			}
		else
			{
			OstTrace0( TRACE_NORMAL, DUP1_CUSBBATTERYCHARGINGPLUGIN_QUERYCURRENTSPEED, "CUsbBatteryChargingPlugin::QueryCurrentSpeed is full speed" );
			iUsbSpeedType = EUsbFullSpeed;
			}
		}
    }

#ifdef SYMBIAN_USB_BATTERYCHARGING_V1_1	
void CUsbBatteryChargingPlugin::UpdateChargingInfo()
	{
	OstTraceFunctionEntry0( CUSBBATTERYCHARGINGPLUGIN_UPDATECHARGINGINFO_ENTRY );
	const TUint16 KIDev_Dchg = 1800;
	const TUint16 KIDev_Chg = 1500;	
	const TUint16 KIDev_Hchg_Lfs = 1500;
	const TUint16 KIDev_Hchg_Hs = 900;
	const TUint16 KIDev_Hchg_Hs_Chirp = 560;
	const TUint16 KIDev_Avaiable = 500;
	const TUint16 KIDev_Suspend = 2;
	const TUint16 KVBus_Chg = 4500;
	const TUint16 KVBus_Dchg = 2000;

		
    QueryCurrentSpeed();
//set iUsbConnStatus field 	
	iChargingInfo().iUsbConnStatus = EUsbConnectionStatusNone;
	if (EUsbChargingPortTypeStandardDownstreamPort == iChargingInfo().iChargingPortType 
		&& IsUsbChargingPossible())
		{
		switch (iDeviceState)
			{
			case EUsbDeviceStateSuspended:
				iChargingInfo().iUsbConnStatus = EUsbConnectionStatusSuspend;
				break;
			case EUsbDeviceStateConfigured:
				iChargingInfo().iUsbConnStatus = EUsbConnectionStatusConfigured;
				break;
			default:
				break;
			}		
		}
//set current and voltage field	
	switch (iChargingInfo().iChargingPortType)
		{
		case EUsbChargingPortTypeDedicatedChargingPort:
			iChargingInfo().iMinAvailableVbusCurrent = KIDev_Avaiable;
			iChargingInfo().iMaxVbusCurrent = KIDev_Dchg;
			iChargingInfo().iMinVbusVoltage = KVBus_Dchg;						
			break;
		case EUsbChargingPortTypeChargingPort:
		case EUsbChargingPortTypeChargingDownstreamPort:	
			iChargingInfo().iMinAvailableVbusCurrent = KIDev_Avaiable;
			iChargingInfo().iMinVbusVoltage = KVBus_Dchg;		
			if (iChargerDetectotCaps.iCapabilities & KChargerDetectorCapChirpCurrentLimiting)
				{			
				if (iUsbSpeedType == EUsbFullSpeed)
					{
					iChargingInfo().iMaxVbusCurrent = KIDev_Hchg_Lfs;
					}
				else
					{
					iChargingInfo().iMaxVbusCurrent = KIDev_Hchg_Hs;					
					}
				}
			else
				{
				if (iDeviceState == EUsbDeviceStateConfigured && iUsbSpeedType == EUsbFullSpeed)
					{
					iChargingInfo().iMaxVbusCurrent = KIDev_Hchg_Lfs;
					}
				else
					{
					iChargingInfo().iMaxVbusCurrent = KIDev_Hchg_Hs_Chirp;					
					}
				}
			break;
		case EUsbChargingPortTypeNone:
		case EUsbChargingPortTypeUnsupported:	
			iChargingInfo().iMinAvailableVbusCurrent = 0;
			iChargingInfo().iMaxVbusCurrent = 0;
			iChargingInfo().iMinVbusVoltage = 0; 					
			break;
		case EUsbChargingPortTypeStandardDownstreamPort:
			if (IsUsbChargingPossible())
				{
				switch (iDeviceState)
					{
					case EUsbDeviceStateConfigured:
						iChargingInfo().iMinAvailableVbusCurrent = iAvailableMilliAmps;
						iChargingInfo().iMaxVbusCurrent = iAvailableMilliAmps;
						iChargingInfo().iMinVbusVoltage = KVBus_Chg; 								
						break;
					case EUsbDeviceStateSuspended:
						iChargingInfo().iMinAvailableVbusCurrent = KIDev_Suspend;
						iChargingInfo().iMaxVbusCurrent = KIDev_Suspend;
						iChargingInfo().iMinVbusVoltage = KVBus_Chg; 						
						break;
					default:
						iChargingInfo().iMinAvailableVbusCurrent = 0;
						iChargingInfo().iMaxVbusCurrent = 0;
						iChargingInfo().iMinVbusVoltage = 0; 						
						break;
					}
				}
			else
				{
				iChargingInfo().iMinAvailableVbusCurrent = 0;
				iChargingInfo().iMaxVbusCurrent = 0;
				iChargingInfo().iMinVbusVoltage = 0;						
				}
			break;
		case EUsbChargingPortTypeAcaRidA:
			iChargingInfo().iMinAvailableVbusCurrent = KIDev_Avaiable - iBDevMaxPower;
			iChargingInfo().iMaxVbusCurrent = KIDev_Chg;
			iChargingInfo().iMinVbusVoltage = KVBus_Dchg; 					
			break;
		case EUsbChargingPortTypeAcaRidB:
			iChargingInfo().iMinAvailableVbusCurrent = KIDev_Avaiable;
			iChargingInfo().iMaxVbusCurrent = KIDev_Chg;
			iChargingInfo().iMinVbusVoltage = KVBus_Dchg; 					
			break;
		case EUsbChargingPortTypeAcaRidC:
			iChargingInfo().iMinAvailableVbusCurrent = KIDev_Avaiable;
			iChargingInfo().iMaxVbusCurrent = KIDev_Chg;
			iChargingInfo().iMinVbusVoltage = KVBus_Dchg; 					
			break;
		default:
			break;
		}
	PublishChargingInfo();
	OstTraceFunctionExit0( CUSBBATTERYCHARGINGPLUGIN_UPDATECHARGINGINFO_EXIT );
	}

#else
void CUsbBatteryChargingPlugin::UpdateChargingInfo()
	{
	OstTraceFunctionEntry0( CUSBBATTERYCHARGINGPLUGIN_UPDATECHARGINGINF1_ENTRY );
	const TUint16 KIDev_Suspend = 2;
	const TUint16 KVBus_Chg = 4500;

		
    QueryCurrentSpeed();
//set iUsbConnStatus field 	
	iChargingInfo().iUsbConnStatus = EUsbConnectionStatusNone;
	if (EUsbChargingPortTypeStandardDownstreamPort == iChargingInfo().iChargingPortType 
		&& IsUsbChargingPossible())
		{
		switch (iDeviceState)
			{
			case EUsbDeviceStateSuspended:
				iChargingInfo().iUsbConnStatus = EUsbConnectionStatusSuspend;
				break;
			case EUsbDeviceStateConfigured:
				iChargingInfo().iUsbConnStatus = EUsbConnectionStatusConfigured;
				break;
			default:
				break;
			}		
		}
//set current and voltage field	
	switch (iChargingInfo().iChargingPortType)
		{
		case EUsbChargingPortTypeStandardDownstreamPort:
			if (IsUsbChargingPossible())
				{
				switch (iDeviceState)
					{
					case EUsbDeviceStateConfigured:
						iChargingInfo().iMinAvailableVbusCurrent = iAvailableMilliAmps;
						iChargingInfo().iMaxVbusCurrent = iAvailableMilliAmps;
						iChargingInfo().iMinVbusVoltage = KVBus_Chg; 								
						break;
					case EUsbDeviceStateSuspended:
						iChargingInfo().iMinAvailableVbusCurrent = KIDev_Suspend;
						iChargingInfo().iMaxVbusCurrent = KIDev_Suspend;
						iChargingInfo().iMinVbusVoltage = KVBus_Chg; 						
						break;
					default:
						iChargingInfo().iMinAvailableVbusCurrent = 0;
						iChargingInfo().iMaxVbusCurrent = 0;
						iChargingInfo().iMinVbusVoltage = 0; 						
						break;
					}
				}
			else
				{
				iChargingInfo().iMinAvailableVbusCurrent = 0;
				iChargingInfo().iMaxVbusCurrent = 0;
				iChargingInfo().iMinVbusVoltage = 0;						
				}
			break;
		default:
			break;
		}
	PublishChargingInfo();
	OstTraceFunctionExit0( CUSBBATTERYCHARGINGPLUGIN_UPDATECHARGINGINF1_EXIT );
	}

#endif


void CUsbBatteryChargingPlugin::PublishChargingInfo()
	{
	OstTraceFunctionEntry0( CUSBBATTERYCHARGINGPLUGIN_PUBLISHCHARGINGINFO_ENTRY );
	OstTraceExt1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_PUBLISHCHARGINGINFO_DUP1, "charging type: %s", 
		*ChargingTypeToString(iChargingInfo().iChargingPortType) );
	OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_PUBLISHCHARGINGINFO_DUP2, "iUsbConnStatus %d", 
		iChargingInfo().iUsbConnStatus );
	OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_PUBLISHCHARGINGINFO_DUP3, "iMinAvailableVbusCurrent %d", 
		iChargingInfo().iMinAvailableVbusCurrent );
	OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_PUBLISHCHARGINGINFO_DUP4, "iMaxVbusCurrent %d", 
		iChargingInfo().iMaxVbusCurrent );
	OstTrace1( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_PUBLISHCHARGINGINFO_DUP5, "iMinVbusVoltage %d", 
		iChargingInfo().iMinVbusVoltage );
	
	if (iLastPublishedInfo.iChargingPortType != iChargingInfo().iChargingPortType
		|| iLastPublishedInfo.iUsbConnStatus != iChargingInfo().iUsbConnStatus
		|| iLastPublishedInfo.iMinAvailableVbusCurrent != iChargingInfo().iMinAvailableVbusCurrent
		|| iLastPublishedInfo.iMaxVbusCurrent != iChargingInfo().iMaxVbusCurrent
		|| iLastPublishedInfo.iMinVbusVoltage != iChargingInfo().iMinVbusVoltage)
		{
		OstTrace0( TRACE_NORMAL, REF_CUSBBATTERYCHARGINGPLUGIN_PUBLISHCHARGINGINFO_DUP6, "info changed, publish it");
		RProperty::Set(KPropertyUidUsbBatteryChargingCategory,
                      KPropertyUidUsbBatteryChargingInfo,
                      iChargingInfo);
		}
	iLastPublishedInfo = iChargingInfo();
	OstTraceFunctionExit0( CUSBBATTERYCHARGINGPLUGIN_PUBLISHCHARGINGINFO_EXIT );
	}

#ifdef SYMBIAN_USB_BATTERYCHARGING_V1_1	

void CUsbBatteryChargingPlugin::PeerDeviceMaxPower(TUint aCurrent)
	{
	OstTraceFunctionEntry0( CUSBBATTERYCHARGINGPLUGIN_PEERDEVICEMAXPOWER_ENTRY );
	const TUint16 KIBDev_Drawn = 500;
	iBDevMaxPower = (aCurrent < KIBDev_Drawn)?aCurrent:KIBDev_Drawn;	
	UpdateChargingInfo();
	OstTraceFunctionExit0( CUSBBATTERYCHARGINGPLUGIN_PEERDEVICEMAXPOWER_EXIT );
	}
#endif