sl@0: /* sl@0: * Copyright (c) 2006 Nokia Corporation and/or its subsidiary(-ies). sl@0: * All rights reserved. sl@0: * This component and the accompanying materials are made available sl@0: * under the terms of "Eclipse Public License v1.0" sl@0: * which accompanies this distribution, and is available sl@0: * at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: * sl@0: * Initial Contributors: sl@0: * Nokia Corporation - initial contribution. sl@0: * sl@0: * Contributors: sl@0: * sl@0: * Description: TelephonyAudioRoutingServer implementation sl@0: * sl@0: */ sl@0: sl@0: sl@0: sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include "TelephonyAudioRoutingServer.h" sl@0: #include "TelephonyAudioRoutingServerSession.h" sl@0: #include "TelephonyAudioRoutingClientServer.h" sl@0: sl@0: sl@0: void PanicClient( sl@0: const RMessage2& aMessage, sl@0: TTelephonyAudioRoutingPanic aPanic) sl@0: { sl@0: _LIT(KPanic,"TelephonyAudioRoutingServer"); sl@0: aMessage.Panic(KPanic,aPanic); sl@0: } sl@0: sl@0: // ========================== OTHER EXPORTED FUNCTIONS ========================= sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // StartThreadL sl@0: // Start the TelephonyAudioRoutingServer thread. sl@0: // Returns: TInt: error code sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: EXPORT_C TInt CTelephonyAudioRoutingServer::StartThreadL( sl@0: TAny* /*aParms*/) sl@0: /** sl@0: Thread entry-point function. sl@0: The TServerStart objects is passed as the thread parameter sl@0: **/ sl@0: { sl@0: sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::StartThreadL ")); sl@0: sl@0: TInt err = KErrNone; sl@0: __UHEAP_MARK; sl@0: sl@0: CTrapCleanup* cleanup = CTrapCleanup::New(); sl@0: sl@0: if (!cleanup) sl@0: err = KErrNoMemory; sl@0: sl@0: if (!err) sl@0: { sl@0: CActiveScheduler* sched=NULL; sl@0: sched=new(ELeave) CActiveScheduler; sl@0: CActiveScheduler::Install(sched); sl@0: CTelephonyAudioRoutingServer* server = NULL; sl@0: TRAPD(err,server = CTelephonyAudioRoutingServer::NewL()); sl@0: sl@0: if(!err) sl@0: { sl@0: // Sync with the client and enter the active scheduler sl@0: RThread::Rendezvous(KErrNone); sl@0: sched->Start(); sl@0: } sl@0: sl@0: delete server; sl@0: delete sched; sl@0: } sl@0: sl@0: delete cleanup; sl@0: sl@0: __UHEAP_MARKEND; sl@0: return err; sl@0: } sl@0: sl@0: sl@0: // ============================ MEMBER FUNCTIONS =============================== sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::CTelephonyAudioRoutingServer sl@0: // C++ default constructor can NOT contain any code, that sl@0: // might leave. sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: CTelephonyAudioRoutingServer::CTelephonyAudioRoutingServer() sl@0: :CServer2(0,EUnsharableSessions), sl@0: iSessionIdsInUse() sl@0: { sl@0: iSessionCount = 0; sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::ConstructL sl@0: // Symbian 2nd phase constructor can leave. sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: void CTelephonyAudioRoutingServer::ConstructL() sl@0: { sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::ConstructL ")); sl@0: TName name(RThread().Name()); sl@0: StartL(name); sl@0: sl@0: // Set default values sl@0: iCurrentAudioOutput = CTelephonyAudioRouting::ENotActive; sl@0: iPreviousAudioOutput = CTelephonyAudioRouting::ENotActive; sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::NewL sl@0: // Two-phased constructor. sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: EXPORT_C CTelephonyAudioRoutingServer* CTelephonyAudioRoutingServer::NewL() sl@0: { sl@0: CTelephonyAudioRoutingServer* self=new(ELeave) CTelephonyAudioRoutingServer(); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(); sl@0: CleanupStack::Pop(); sl@0: return self; sl@0: } sl@0: sl@0: // Destructor sl@0: CTelephonyAudioRoutingServer::~CTelephonyAudioRoutingServer() sl@0: { sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::~CTelephonyAudioRoutingServer ")); sl@0: iSessionIdsInUse.Close(); sl@0: iSetOutputRequests.Close(); sl@0: iAvailableOutputs.Close(); sl@0: } sl@0: sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::NewSessionL sl@0: // Create a new client session sl@0: // (other items were commented in a header). sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: CSession2* CTelephonyAudioRoutingServer::NewSessionL( sl@0: const TVersion&, sl@0: const RMessage2& /*aMessage*/) const sl@0: { sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::NewSessionL ")); sl@0: sl@0: CTelephonyAudioRoutingServer& nonConstThis = *const_cast(this); sl@0: const TInt sessionId = nonConstThis.IdentifyAndAllocateNextFreeSessionIdL(); sl@0: return new (ELeave) CTelephonyAudioRoutingServerSession(sessionId); sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::DoSetOutputL sl@0: // Notify the policy session about a request to change output. sl@0: // (other items were commented in a header). sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: void CTelephonyAudioRoutingServer::DoSetOutputL ( sl@0: TInt aSessionId, sl@0: const RMessage2& aMessage) sl@0: { sl@0: sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL ")); sl@0: sl@0: //Check for multimedia capability: sl@0: RThread clientThread; sl@0: aMessage.ClientL(clientThread); sl@0: RProcess clientProcess; sl@0: User::LeaveIfError(clientThread.Process(clientProcess)); sl@0: sl@0: TProcessId clientProcessID(clientProcess.Id()); sl@0: sl@0: sl@0: TBool clientHasCapabilities = clientProcess.HasCapability(ECapabilityMultimediaDD, KSuppressPlatSecDiagnostic); sl@0: sl@0: if (!clientHasCapabilities) sl@0: { sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL ERROR: Client failed Capability Check")); sl@0: aMessage.Complete(KErrPermissionDenied); sl@0: return; sl@0: } sl@0: sl@0: clientThread.Close(); sl@0: clientProcess.Close(); sl@0: sl@0: TPckgBuf xPackage; sl@0: aMessage.ReadL( 0, xPackage); sl@0: sl@0: CTelephonyAudioRouting::TAudioOutput response = xPackage(); sl@0: TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL Audio Output Value to set on the sessions: %d"), response); sl@0: TPckgBuf yPackage; // space for previous output sl@0: aMessage.ReadL( 1, yPackage); sl@0: sl@0: TPckgBuf zPackage; // space for err sl@0: aMessage.ReadL( 2, zPackage); sl@0: sl@0: TPckgBuf showNotePkg; sl@0: aMessage.ReadL(3, showNotePkg); sl@0: iShowNoteMode = showNotePkg(); sl@0: TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL ShowNoteMode Value Sent to server = %d"), iShowNoteMode); sl@0: sl@0: iSetOutputRequests.AppendL(aSessionId); sl@0: sl@0: // Verify requested audio change exists in available outputs (unless is ENone or ENotActive) sl@0: if ( (response != CTelephonyAudioRouting::ENone) && (response != CTelephonyAudioRouting::ENotActive)) sl@0: { sl@0: TBool found = IsAvailableOutput(response); sl@0: if (!found) sl@0: { sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL: ERROR, Requested output not in availableOutputs!")); sl@0: SetOutputCompleteL(response, KErrPermissionDenied); sl@0: return; sl@0: } sl@0: } sl@0: sl@0: iSessionIter.SetToFirst(); sl@0: sl@0: CTelephonyAudioRoutingServerSession* serverSession = static_cast(iSessionIter++); sl@0: sl@0: while (serverSession != NULL) sl@0: { sl@0: if(serverSession->SessionId() == iPolicySessionId) sl@0: { sl@0: serverSession->OutputChangeRequested(response); sl@0: break; sl@0: } sl@0: serverSession = static_cast(iSessionIter++); sl@0: } sl@0: sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::IdentifyAndAllocateNextFreeSessionIdL sl@0: // Assign a unique session Id to a new session sl@0: // (other items were commented in a header). sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: TInt CTelephonyAudioRoutingServer::IdentifyAndAllocateNextFreeSessionIdL() sl@0: { sl@0: sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::IdentifyAndAllocateNextFreeSessionIdL ")); sl@0: sl@0: // The aim of this method is to locate a session identifier which is not already sl@0: // in use.... sl@0: TInt sessionId = 0; sl@0: TInt errorOrIndex; sl@0: sl@0: // Only 256 (KMaxNumberOfSessions) are allowed sl@0: const TInt numberOfUsedSessionIds = iSessionIdsInUse.Count(); sl@0: if (numberOfUsedSessionIds > KMaxNumberOfSessions) sl@0: { sl@0: User::LeaveIfError(KErrDied); sl@0: } sl@0: sl@0: // Session Id's indexing begins at 1, not 0 sl@0: for (TInt count = 1; count < KMaxNumberOfSessions+1; count++) sl@0: { sl@0: sessionId = count; sl@0: errorOrIndex = iSessionIdsInUse.FindInOrder(sessionId); sl@0: sl@0: // If sessionId=count not currently being used, assign it. sl@0: // Save the session in the array of allocated ids. We use InsertInOrder sl@0: // since it effectively allows a binary search when trying to find sl@0: // free ids: sl@0: if (errorOrIndex == KErrNotFound) sl@0: { sl@0: errorOrIndex = iSessionIdsInUse.InsertInOrder(sessionId); sl@0: if (errorOrIndex < 0) // Handle error sl@0: { sl@0: User::LeaveIfError(KErrDied); sl@0: } sl@0: break; sl@0: } sl@0: sl@0: // Prevent value returned by RArray::FindInOrder from being propagated sl@0: // to the client side in response to a RSessionBase::Connect() request. sl@0: if ((errorOrIndex != KErrNotFound) && (errorOrIndex < 0)) sl@0: { sl@0: User::LeaveIfError(KErrDied); sl@0: } sl@0: sl@0: } // End for sl@0: sl@0: return sessionId; sl@0: sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::AddSession sl@0: // Add a new Session. sl@0: // (other items were commented in a header). sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: void CTelephonyAudioRoutingServer::AddSession() sl@0: { sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::AddSession ")); sl@0: iSessionCount++; sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::RemoveSession sl@0: // Remove an existing session. sl@0: // (other items were commented in a header). sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: void CTelephonyAudioRoutingServer::RemoveSession( sl@0: TInt aSessionId) sl@0: { sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::RemoveSession")); sl@0: FreeSessionId(aSessionId); sl@0: iSessionCount--; sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::FreeSessionId sl@0: // Free the session Id or a delated session for re-use. sl@0: // (other items were commented in a header). sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: void CTelephonyAudioRoutingServer::FreeSessionId( sl@0: TInt aSessionId) sl@0: { sl@0: sl@0: TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::FreeSessionId: ID: %d"), aSessionId); sl@0: sl@0: const TInt indexOrError = iSessionIdsInUse.FindInOrder(aSessionId); sl@0: sl@0: if (indexOrError >= 0) sl@0: { sl@0: // This session id can now be reused... sl@0: iSessionIdsInUse.Remove(indexOrError); sl@0: } sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::SetPolicySessionId sl@0: // Set the Id of the policy session. sl@0: // (other items were commented in a header). sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: void CTelephonyAudioRoutingServer::SetPolicySessionId( sl@0: TInt aSessionId) sl@0: { sl@0: iPolicySessionId = aSessionId; sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::AvailableOutputsChangedL sl@0: // Used by policy session to indicated to all other sessions that available outputs have changed. sl@0: // (other items were commented in a header). sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: void CTelephonyAudioRoutingServer::AvailableOutputsChangedL( sl@0: const TArray& aOutputs) sl@0: { sl@0: sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::AvailableOutputsChangedL ")); sl@0: sl@0: iAvailableOutputs.Reset(); sl@0: TInt count = aOutputs.Count(); sl@0: TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t AvailableOutputsChangedL: Count = %d "),count); sl@0: for(TInt i = 0; i < count; i++) sl@0: { sl@0: TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t AvailableOutputsChangedL: aOutputs[i] = %d "),aOutputs[i]); sl@0: iAvailableOutputs.AppendL(aOutputs[i]); sl@0: } sl@0: sl@0: iSessionIter.SetToFirst(); sl@0: sl@0: CTelephonyAudioRoutingServerSession* serverSession = static_cast(iSessionIter++); sl@0: while (serverSession != NULL) sl@0: { sl@0: if(serverSession->SessionId() != iPolicySessionId) sl@0: serverSession->AvailableOutputsChanged(aOutputs); sl@0: serverSession = static_cast(iSessionIter++); sl@0: } sl@0: sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::SetOutputComplete sl@0: // Used by policy session to indicate the session that requested a SetOutput() that sl@0: // the request is complete, all the other session get OutputChanged() notification sl@0: // (other items were commented in a header). sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: void CTelephonyAudioRoutingServer::SetOutputCompleteL( sl@0: CTelephonyAudioRouting::TAudioOutput aOutput, sl@0: TInt aError) sl@0: { sl@0: sl@0: CTelephonyAudioRouting::TAudioOutput output = aOutput; sl@0: sl@0: TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::SetOutputComplete called with: = %d"), output); sl@0: sl@0: TInt requestCount = iSetOutputRequests.Count(); sl@0: sl@0: if(requestCount>0) sl@0: { sl@0: sl@0: TInt sessionToAlert = iSetOutputRequests[0]; sl@0: iSessionIter.SetToFirst(); sl@0: sl@0: CTelephonyAudioRoutingServerSession* serverSession = static_cast(iSessionIter++); sl@0: while (serverSession != NULL) sl@0: { sl@0: if(serverSession->SessionId() != iPolicySessionId) sl@0: { sl@0: sl@0: if (serverSession->SessionId() == sessionToAlert ) sl@0: { sl@0: TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::SetOutputComplete: Calling serverSession::SetOutputComplete with showNote: %d"), iShowNoteMode); sl@0: serverSession->SetOutputComplete(output, aError, iShowNoteMode); sl@0: } sl@0: else sl@0: { sl@0: if (aError == KErrNone) sl@0: { sl@0: serverSession->OutputChanged(output, iShowNoteMode); sl@0: } sl@0: sl@0: } sl@0: } sl@0: sl@0: serverSession = static_cast(iSessionIter++); sl@0: } sl@0: sl@0: for(TInt i=0;i(iSessionIter++); sl@0: while (serverSession != NULL) sl@0: { sl@0: if(serverSession->SessionId() != iPolicySessionId) sl@0: { sl@0: // Always send ETrue if outputChanged initiated by Policy: sl@0: iShowNoteMode = ETrue; sl@0: serverSession->OutputChanged(output, iShowNoteMode); sl@0: sl@0: } sl@0: serverSession = static_cast(iSessionIter++); sl@0: } sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // CTelephonyAudioRoutingServer::IsAvailableOutput sl@0: // Method used to determine if requested audio output is one of the sl@0: // audio outputs in the available output array. sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: TBool CTelephonyAudioRoutingServer::IsAvailableOutput( sl@0: CTelephonyAudioRouting::TAudioOutput aOutput) sl@0: { sl@0: TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::IsAvailableOutput ")); sl@0: sl@0: TBool found = EFalse; sl@0: CTelephonyAudioRouting::TAudioOutput arrayElement; sl@0: TInt count = iAvailableOutputs.Count(); sl@0: sl@0: for(TInt i=0;i& CTelephonyAudioRoutingServer::AvailableOutputs() sl@0: { sl@0: return iAvailableOutputs; sl@0: } sl@0: sl@0: sl@0: //End of File