os/mm/devsoundextensions/telephonyaudiorouting/Server/src/TelephonyAudioRoutingServer.cpp
First public contribution.
2 * Copyright (c) 2006 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of "Eclipse Public License v1.0"
6 * which accompanies this distribution, and is available
7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
14 * Description: TelephonyAudioRoutingServer implementation
22 #include <data_caging_path_literals.hrh>
25 #include "TelephonyAudioRoutingServer.h"
26 #include "TelephonyAudioRoutingServerSession.h"
27 #include "TelephonyAudioRoutingClientServer.h"
31 const RMessage2& aMessage,
32 TTelephonyAudioRoutingPanic aPanic)
34 _LIT(KPanic,"TelephonyAudioRoutingServer");
35 aMessage.Panic(KPanic,aPanic);
38 // ========================== OTHER EXPORTED FUNCTIONS =========================
40 // -----------------------------------------------------------------------------
42 // Start the TelephonyAudioRoutingServer thread.
43 // Returns: TInt: error code
44 // -----------------------------------------------------------------------------
46 EXPORT_C TInt CTelephonyAudioRoutingServer::StartThreadL(
49 Thread entry-point function.
50 The TServerStart objects is passed as the thread parameter
54 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::StartThreadL "));
59 CTrapCleanup* cleanup = CTrapCleanup::New();
66 CActiveScheduler* sched=NULL;
67 sched=new(ELeave) CActiveScheduler;
68 CActiveScheduler::Install(sched);
69 CTelephonyAudioRoutingServer* server = NULL;
70 TRAPD(err,server = CTelephonyAudioRoutingServer::NewL());
74 // Sync with the client and enter the active scheduler
75 RThread::Rendezvous(KErrNone);
90 // ============================ MEMBER FUNCTIONS ===============================
92 // -----------------------------------------------------------------------------
93 // CTelephonyAudioRoutingServer::CTelephonyAudioRoutingServer
94 // C++ default constructor can NOT contain any code, that
96 // -----------------------------------------------------------------------------
98 CTelephonyAudioRoutingServer::CTelephonyAudioRoutingServer()
99 :CServer2(0,EUnsharableSessions),
105 // -----------------------------------------------------------------------------
106 // CTelephonyAudioRoutingServer::ConstructL
107 // Symbian 2nd phase constructor can leave.
108 // -----------------------------------------------------------------------------
110 void CTelephonyAudioRoutingServer::ConstructL()
112 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::ConstructL "));
113 TName name(RThread().Name());
116 // Set default values
117 iCurrentAudioOutput = CTelephonyAudioRouting::ENotActive;
118 iPreviousAudioOutput = CTelephonyAudioRouting::ENotActive;
121 // -----------------------------------------------------------------------------
122 // CTelephonyAudioRoutingServer::NewL
123 // Two-phased constructor.
124 // -----------------------------------------------------------------------------
126 EXPORT_C CTelephonyAudioRoutingServer* CTelephonyAudioRoutingServer::NewL()
128 CTelephonyAudioRoutingServer* self=new(ELeave) CTelephonyAudioRoutingServer();
129 CleanupStack::PushL(self);
136 CTelephonyAudioRoutingServer::~CTelephonyAudioRoutingServer()
138 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::~CTelephonyAudioRoutingServer "));
139 iSessionIdsInUse.Close();
140 iSetOutputRequests.Close();
141 iAvailableOutputs.Close();
145 // -----------------------------------------------------------------------------
146 // CTelephonyAudioRoutingServer::NewSessionL
147 // Create a new client session
148 // (other items were commented in a header).
149 // -----------------------------------------------------------------------------
151 CSession2* CTelephonyAudioRoutingServer::NewSessionL(
153 const RMessage2& /*aMessage*/) const
155 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::NewSessionL "));
157 CTelephonyAudioRoutingServer& nonConstThis = *const_cast<CTelephonyAudioRoutingServer*>(this);
158 const TInt sessionId = nonConstThis.IdentifyAndAllocateNextFreeSessionIdL();
159 return new (ELeave) CTelephonyAudioRoutingServerSession(sessionId);
162 // -----------------------------------------------------------------------------
163 // CTelephonyAudioRoutingServer::DoSetOutputL
164 // Notify the policy session about a request to change output.
165 // (other items were commented in a header).
166 // -----------------------------------------------------------------------------
168 void CTelephonyAudioRoutingServer::DoSetOutputL (
170 const RMessage2& aMessage)
173 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL "));
175 //Check for multimedia capability:
176 RThread clientThread;
177 aMessage.ClientL(clientThread);
178 RProcess clientProcess;
179 User::LeaveIfError(clientThread.Process(clientProcess));
181 TProcessId clientProcessID(clientProcess.Id());
184 TBool clientHasCapabilities = clientProcess.HasCapability(ECapabilityMultimediaDD, KSuppressPlatSecDiagnostic);
186 if (!clientHasCapabilities)
188 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL ERROR: Client failed Capability Check"));
189 aMessage.Complete(KErrPermissionDenied);
193 clientThread.Close();
194 clientProcess.Close();
196 TPckgBuf<CTelephonyAudioRouting::TAudioOutput> xPackage;
197 aMessage.ReadL( 0, xPackage);
199 CTelephonyAudioRouting::TAudioOutput response = xPackage();
200 TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL Audio Output Value to set on the sessions: %d"), response);
201 TPckgBuf<CTelephonyAudioRouting::TAudioOutput> yPackage; // space for previous output
202 aMessage.ReadL( 1, yPackage);
204 TPckgBuf<TInt> zPackage; // space for err
205 aMessage.ReadL( 2, zPackage);
207 TPckgBuf<TBool> showNotePkg;
208 aMessage.ReadL(3, showNotePkg);
209 iShowNoteMode = showNotePkg();
210 TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL ShowNoteMode Value Sent to server = %d"), iShowNoteMode);
212 iSetOutputRequests.AppendL(aSessionId);
214 // Verify requested audio change exists in available outputs (unless is ENone or ENotActive)
215 if ( (response != CTelephonyAudioRouting::ENone) && (response != CTelephonyAudioRouting::ENotActive))
217 TBool found = IsAvailableOutput(response);
220 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::DoSetOutputL: ERROR, Requested output not in availableOutputs!"));
221 SetOutputCompleteL(response, KErrPermissionDenied);
226 iSessionIter.SetToFirst();
228 CTelephonyAudioRoutingServerSession* serverSession = static_cast<CTelephonyAudioRoutingServerSession*>(iSessionIter++);
230 while (serverSession != NULL)
232 if(serverSession->SessionId() == iPolicySessionId)
234 serverSession->OutputChangeRequested(response);
237 serverSession = static_cast<CTelephonyAudioRoutingServerSession*>(iSessionIter++);
242 // -----------------------------------------------------------------------------
243 // CTelephonyAudioRoutingServer::IdentifyAndAllocateNextFreeSessionIdL
244 // Assign a unique session Id to a new session
245 // (other items were commented in a header).
246 // -----------------------------------------------------------------------------
248 TInt CTelephonyAudioRoutingServer::IdentifyAndAllocateNextFreeSessionIdL()
251 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::IdentifyAndAllocateNextFreeSessionIdL "));
253 // The aim of this method is to locate a session identifier which is not already
258 // Only 256 (KMaxNumberOfSessions) are allowed
259 const TInt numberOfUsedSessionIds = iSessionIdsInUse.Count();
260 if (numberOfUsedSessionIds > KMaxNumberOfSessions)
262 User::LeaveIfError(KErrDied);
265 // Session Id's indexing begins at 1, not 0
266 for (TInt count = 1; count < KMaxNumberOfSessions+1; count++)
269 errorOrIndex = iSessionIdsInUse.FindInOrder(sessionId);
271 // If sessionId=count not currently being used, assign it.
272 // Save the session in the array of allocated ids. We use InsertInOrder
273 // since it effectively allows a binary search when trying to find
275 if (errorOrIndex == KErrNotFound)
277 errorOrIndex = iSessionIdsInUse.InsertInOrder(sessionId);
278 if (errorOrIndex < 0) // Handle error
280 User::LeaveIfError(KErrDied);
285 // Prevent value returned by RArray<T>::FindInOrder from being propagated
286 // to the client side in response to a RSessionBase::Connect() request.
287 if ((errorOrIndex != KErrNotFound) && (errorOrIndex < 0))
289 User::LeaveIfError(KErrDied);
298 // -----------------------------------------------------------------------------
299 // CTelephonyAudioRoutingServer::AddSession
300 // Add a new Session.
301 // (other items were commented in a header).
302 // -----------------------------------------------------------------------------
304 void CTelephonyAudioRoutingServer::AddSession()
306 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::AddSession "));
310 // -----------------------------------------------------------------------------
311 // CTelephonyAudioRoutingServer::RemoveSession
312 // Remove an existing session.
313 // (other items were commented in a header).
314 // -----------------------------------------------------------------------------
316 void CTelephonyAudioRoutingServer::RemoveSession(
319 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::RemoveSession"));
320 FreeSessionId(aSessionId);
324 // -----------------------------------------------------------------------------
325 // CTelephonyAudioRoutingServer::FreeSessionId
326 // Free the session Id or a delated session for re-use.
327 // (other items were commented in a header).
328 // -----------------------------------------------------------------------------
330 void CTelephonyAudioRoutingServer::FreeSessionId(
334 TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::FreeSessionId: ID: %d"), aSessionId);
336 const TInt indexOrError = iSessionIdsInUse.FindInOrder(aSessionId);
338 if (indexOrError >= 0)
340 // This session id can now be reused...
341 iSessionIdsInUse.Remove(indexOrError);
345 // -----------------------------------------------------------------------------
346 // CTelephonyAudioRoutingServer::SetPolicySessionId
347 // Set the Id of the policy session.
348 // (other items were commented in a header).
349 // -----------------------------------------------------------------------------
351 void CTelephonyAudioRoutingServer::SetPolicySessionId(
354 iPolicySessionId = aSessionId;
357 // -----------------------------------------------------------------------------
358 // CTelephonyAudioRoutingServer::AvailableOutputsChangedL
359 // Used by policy session to indicated to all other sessions that available outputs have changed.
360 // (other items were commented in a header).
361 // -----------------------------------------------------------------------------
363 void CTelephonyAudioRoutingServer::AvailableOutputsChangedL(
364 const TArray<CTelephonyAudioRouting::TAudioOutput>& aOutputs)
367 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::AvailableOutputsChangedL "));
369 iAvailableOutputs.Reset();
370 TInt count = aOutputs.Count();
371 TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t AvailableOutputsChangedL: Count = %d "),count);
372 for(TInt i = 0; i < count; i++)
374 TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t AvailableOutputsChangedL: aOutputs[i] = %d "),aOutputs[i]);
375 iAvailableOutputs.AppendL(aOutputs[i]);
378 iSessionIter.SetToFirst();
380 CTelephonyAudioRoutingServerSession* serverSession = static_cast<CTelephonyAudioRoutingServerSession*>(iSessionIter++);
381 while (serverSession != NULL)
383 if(serverSession->SessionId() != iPolicySessionId)
384 serverSession->AvailableOutputsChanged(aOutputs);
385 serverSession = static_cast<CTelephonyAudioRoutingServerSession*>(iSessionIter++);
390 // -----------------------------------------------------------------------------
391 // CTelephonyAudioRoutingServer::SetOutputComplete
392 // Used by policy session to indicate the session that requested a SetOutput() that
393 // the request is complete, all the other session get OutputChanged() notification
394 // (other items were commented in a header).
395 // -----------------------------------------------------------------------------
397 void CTelephonyAudioRoutingServer::SetOutputCompleteL(
398 CTelephonyAudioRouting::TAudioOutput aOutput,
402 CTelephonyAudioRouting::TAudioOutput output = aOutput;
404 TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::SetOutputComplete called with: = %d"), output);
406 TInt requestCount = iSetOutputRequests.Count();
411 TInt sessionToAlert = iSetOutputRequests[0];
412 iSessionIter.SetToFirst();
414 CTelephonyAudioRoutingServerSession* serverSession = static_cast<CTelephonyAudioRoutingServerSession*>(iSessionIter++);
415 while (serverSession != NULL)
417 if(serverSession->SessionId() != iPolicySessionId)
420 if (serverSession->SessionId() == sessionToAlert )
422 TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::SetOutputComplete: Calling serverSession::SetOutputComplete with showNote: %d"), iShowNoteMode);
423 serverSession->SetOutputComplete(output, aError, iShowNoteMode);
427 if (aError == KErrNone)
429 serverSession->OutputChanged(output, iShowNoteMode);
435 serverSession = static_cast<CTelephonyAudioRoutingServerSession*>(iSessionIter++);
438 for(TInt i=0;i<iSetOutputRequests.Count()-1;i++)
440 iSetOutputRequests[i] = iSetOutputRequests[i+1];
442 iSetOutputRequests.Remove(iSetOutputRequests.Count()-1);
446 // -----------------------------------------------------------------------------
447 // CTelephonyAudioRoutingServer::OutputChanged
448 // Used by policy session to notify all sessions that the output has changed (from the policy side)
449 // (other items were commented in a header).
450 // -----------------------------------------------------------------------------
452 void CTelephonyAudioRoutingServer::OutputChanged(
453 CTelephonyAudioRouting::TAudioOutput aOutput)
456 CTelephonyAudioRouting::TAudioOutput output = aOutput;
458 TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::OutputChanged called with: = %d"), output);
460 iSessionIter.SetToFirst();
462 CTelephonyAudioRoutingServerSession* serverSession = static_cast<CTelephonyAudioRoutingServerSession*>(iSessionIter++);
463 while (serverSession != NULL)
465 if(serverSession->SessionId() != iPolicySessionId)
467 // Always send ETrue if outputChanged initiated by Policy:
468 iShowNoteMode = ETrue;
469 serverSession->OutputChanged(output, iShowNoteMode);
472 serverSession = static_cast<CTelephonyAudioRoutingServerSession*>(iSessionIter++);
476 // -----------------------------------------------------------------------------
477 // CTelephonyAudioRoutingServer::IsAvailableOutput
478 // Method used to determine if requested audio output is one of the
479 // audio outputs in the available output array.
480 // -----------------------------------------------------------------------------
482 TBool CTelephonyAudioRoutingServer::IsAvailableOutput(
483 CTelephonyAudioRouting::TAudioOutput aOutput)
485 TELAUDRTNG_RDEBUG(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::IsAvailableOutput "));
487 TBool found = EFalse;
488 CTelephonyAudioRouting::TAudioOutput arrayElement;
489 TInt count = iAvailableOutputs.Count();
491 for(TInt i=0;i<count;i++)
493 arrayElement = iAvailableOutputs[i];
494 if (aOutput == arrayElement)
502 TELAUDRTNG_RDEBUG1(_L("[TELAUDRTNG]\t CTelephonyAudioRoutingServer::IsAvailableOutput: Found? = %d "), found);
507 // -----------------------------------------------------------------------------
508 // CTelephonyAudioRoutingServer::CurrentAudioOutput
509 // Accessor method returns iCurrentAudioOutput to caller.
510 // (other items were commented in a header).
511 // -----------------------------------------------------------------------------
513 CTelephonyAudioRouting::TAudioOutput& CTelephonyAudioRoutingServer::CurrentAudioOutput()
515 return iCurrentAudioOutput;
518 // -----------------------------------------------------------------------------
519 // CTelephonyAudioRoutingServer::ShowNoteMode
520 // Accessor method returns iShowNoteMode to caller.
521 // (other items were commented in a header).
522 // -----------------------------------------------------------------------------
524 TBool& CTelephonyAudioRoutingServer::ShowNoteMode()
526 return iShowNoteMode;
529 // -----------------------------------------------------------------------------
530 // CTelephonyAudioRoutingServer::SetShowNoteMode
531 // Accessor method allows caller to set iShowNoteMode.
532 // (other items were commented in a header).
533 // -----------------------------------------------------------------------------
535 void CTelephonyAudioRoutingServer::SetShowNoteMode(TBool aShowNoteMode)
537 iShowNoteMode = aShowNoteMode;
541 // -----------------------------------------------------------------------------
542 // CTelephonyAudioRoutingServer::PreviousAudioOutput
543 // Accessor method returns iPreviousAudioOutput to caller.
544 // (other items were commented in a header).
545 // -----------------------------------------------------------------------------
547 CTelephonyAudioRouting::TAudioOutput& CTelephonyAudioRoutingServer::PreviousAudioOutput()
549 return iPreviousAudioOutput;
552 // -----------------------------------------------------------------------------
553 // CTelephonyAudioRoutingServer::AvailableOutputs
554 // Accessor method returns iAvailableOutputs to caller.
555 // (other items were commented in a header).
556 // -----------------------------------------------------------------------------
558 RArray<CTelephonyAudioRouting::TAudioOutput>& CTelephonyAudioRoutingServer::AvailableOutputs()
560 return iAvailableOutputs;