| 0 |      1 | // Copyright (c) 2004-2009 Nokia Corporation and/or its subsidiary(-ies).
 | 
|  |      2 | // All rights reserved.
 | 
|  |      3 | // This component and the accompanying materials are made available
 | 
|  |      4 | // under the terms of the License "Eclipse Public License v1.0"
 | 
|  |      5 | // which accompanies this distribution, and is available
 | 
|  |      6 | // at the URL "http://www.eclipse.org/legal/epl-v10.html".
 | 
|  |      7 | //
 | 
|  |      8 | // Initial Contributors:
 | 
|  |      9 | // Nokia Corporation - initial contribution.
 | 
|  |     10 | //
 | 
|  |     11 | // Contributors:
 | 
|  |     12 | //
 | 
|  |     13 | // Description:
 | 
|  |     14 | // e32test\cppexceptions\t_unmap.cpp
 | 
|  |     15 | // 
 | 
|  |     16 | //
 | 
|  |     17 | 
 | 
|  |     18 | #include <e32std.h>
 | 
|  |     19 | #include <e32std_private.h>
 | 
|  |     20 | #include <e32base.h>
 | 
|  |     21 | #include <e32base_private.h>
 | 
|  |     22 | #include <e32test.h>
 | 
|  |     23 | #include <e32svr.h>
 | 
|  |     24 | #include <f32dbg.h>
 | 
|  |     25 | #include <e32def.h>
 | 
|  |     26 | #include <e32def_private.h>
 | 
|  |     27 | 
 | 
|  |     28 | #include "d_unmap.h"
 | 
|  |     29 | 
 | 
|  |     30 | _LIT(KTestName, "t_unmap");
 | 
|  |     31 | _LIT(KTestThreadName, "t_unmap test thread");
 | 
|  |     32 | _LIT(KNopThreadName, "nop [DLL unload checking] thread");
 | 
|  |     33 | _LIT(KTUnmapPanic, "t_unmap");
 | 
|  |     34 | _LIT(KThread, "Thread");
 | 
|  |     35 | 
 | 
|  |     36 | _LIT(KUnhandledExcCategory, "KERN-EXEC");
 | 
|  |     37 | const TInt KUnhandledExcReason = 3;
 | 
|  |     38 | 
 | 
|  |     39 | enum TUnmapPanic
 | 
|  |     40 | 	{
 | 
|  |     41 | 	EPanickingThread = 123456789
 | 
|  |     42 | 	};
 | 
|  |     43 | 
 | 
|  |     44 | RTest test(KTestName);
 | 
|  |     45 | 
 | 
|  |     46 | RTest testThreadA(KTestThreadName);
 | 
|  |     47 | RTest testThreadB(KTestThreadName);
 | 
|  |     48 | RTest testThreadC(KTestThreadName);
 | 
|  |     49 | RTest testThreadD(KTestThreadName);
 | 
|  |     50 | RTest testThreadE(KTestThreadName);
 | 
|  |     51 | RTest testThreadF(KTestThreadName);
 | 
|  |     52 | RTest testThreadG(KTestThreadName);
 | 
|  |     53 | RTest testThreadH(KTestThreadName);
 | 
|  |     54 | 
 | 
|  |     55 | RTest testThreadI(KTestThreadName);
 | 
|  |     56 | RTest testThreadJ(KTestThreadName);
 | 
|  |     57 | RTest testThreadK(KTestThreadName);
 | 
|  |     58 | 
 | 
|  |     59 | RSemaphore Thread1Semaphore;
 | 
|  |     60 | RSemaphore Thread2Semaphore;
 | 
|  |     61 | 
 | 
|  |     62 | RSemaphore FinishedOpSemaphore;
 | 
|  |     63 | 
 | 
|  |     64 | RLibrary ThreadALibraryHandle;
 | 
|  |     65 | RLibrary ThreadBLibraryHandle;
 | 
|  |     66 | RLibrary ThreadCLibraryHandle;
 | 
|  |     67 | RLibrary ThreadDLibraryHandle;
 | 
|  |     68 | RLibrary ThreadELibraryHandle;
 | 
|  |     69 | RLibrary ThreadFLibraryHandle;
 | 
|  |     70 | RLibrary ThreadGLibraryHandle;
 | 
|  |     71 | RLibrary ThreadHLibraryHandle;
 | 
|  |     72 | RLibrary ThreadILibraryHandle;
 | 
|  |     73 | RLibrary ThreadJLibraryHandle;
 | 
|  |     74 | RLibrary ThreadKLibraryHandle;
 | 
|  |     75 | 
 | 
|  |     76 | TBool CheckKernelHeap;
 | 
|  |     77 | 
 | 
|  |     78 | void TestThreads();
 | 
|  |     79 | 
 | 
|  |     80 | TInt ThreadA(TAny*);
 | 
|  |     81 | TInt ThreadB(TAny*);
 | 
|  |     82 | TInt ThreadC(TAny*);
 | 
|  |     83 | TInt ThreadD(TAny*);
 | 
|  |     84 | TInt ThreadE(TAny*);
 | 
|  |     85 | TInt ThreadF(TAny*);
 | 
|  |     86 | TInt ThreadG(TAny*);
 | 
|  |     87 | TInt ThreadH(TAny*);
 | 
|  |     88 | TInt ThreadI(TAny*);
 | 
|  |     89 | TInt ThreadJ(TAny*);
 | 
|  |     90 | TInt ThreadK(TAny*);
 | 
|  |     91 | 
 | 
|  |     92 | TInt DoThreadAL(TAny*);
 | 
|  |     93 | TInt DoThreadBL(TAny*);
 | 
|  |     94 | TInt DoThreadCL(TAny*);
 | 
|  |     95 | TInt DoThreadDL(TAny*);
 | 
|  |     96 | TInt DoThreadEL(TAny*);
 | 
|  |     97 | TInt DoThreadFL(TAny*);
 | 
|  |     98 | TInt DoThreadGL(TAny*);
 | 
|  |     99 | TInt DoThreadHL(TAny*);
 | 
|  |    100 | TInt DoThreadIL(TAny*);
 | 
|  |    101 | TInt DoThreadJL(TAny*);
 | 
|  |    102 | TInt DoThreadKL(TAny*);
 | 
|  |    103 | 
 | 
|  |    104 | struct STestThreadInfo
 | 
|  |    105 | 	{
 | 
|  |    106 | 	TThreadFunction	iThreadFn;
 | 
|  |    107 | 	TExitType		iExitType;
 | 
|  |    108 | 	TInt			iMappedSignals;
 | 
|  |    109 | 	TInt			iLeaveSignals;
 | 
|  |    110 | 	};
 | 
|  |    111 | 
 | 
|  |    112 | static STestThreadInfo const TheThreadArray[] =
 | 
|  |    113 | 	{
 | 
|  |    114 | 		{ &ThreadA, EExitPanic, 0, 0 },
 | 
|  |    115 | 		{ &ThreadB, EExitKill, 0, 0 },
 | 
|  |    116 | 		{ &ThreadC, EExitPanic, 1, 0 },
 | 
|  |    117 | 		{ &ThreadD, EExitPanic, 2, 0 },
 | 
|  |    118 | 		{ &ThreadE, EExitKill, 2, 2 },
 | 
|  |    119 | 		{ &ThreadF, EExitPanic, 2, 1 },
 | 
|  |    120 | 		{ &ThreadG, EExitKill, 3, 1 },
 | 
|  |    121 | 		{ &ThreadH, EExitKill, 1, 1 },
 | 
|  |    122 | 		{ &ThreadI, EExitKill, 1, 3 },
 | 
|  |    123 | 		{ &ThreadJ, EExitPanic, 1, 2 },
 | 
|  |    124 | 		{ &ThreadK, EExitPanic, 1, 1 }
 | 
|  |    125 | 	};
 | 
|  |    126 | 
 | 
|  |    127 | struct SNopThreadInfo
 | 
|  |    128 | 	{
 | 
|  |    129 | 	TLibraryFunction iFunc;
 | 
|  |    130 | #ifdef __WINS__
 | 
|  |    131 | 	TInt32 iOriginalContents;
 | 
|  |    132 | #endif
 | 
|  |    133 | 	};
 | 
|  |    134 | 
 | 
|  |    135 | static SNopThreadInfo NopThreadInfo;
 | 
|  |    136 | 
 | 
|  |    137 | static const TInt TheThreadCount = (sizeof(TheThreadArray) / sizeof(STestThreadInfo));
 | 
|  |    138 | 
 | 
|  |    139 | static const TInt KHeapSize = 0x2000;
 | 
|  |    140 | 
 | 
|  |    141 | TInt E32Main()
 | 
|  |    142 |    	{
 | 
|  |    143 | 	// Turn off lazy dll unloading
 | 
|  |    144 | 	RLoader l;
 | 
|  |    145 | 	test(l.Connect()==KErrNone);
 | 
|  |    146 | 	test(l.CancelLazyDllUnload()==KErrNone);
 | 
|  |    147 | 	l.Close();
 | 
|  |    148 | 	
 | 
|  |    149 | 	test.Start(_L("Check code seg unmapping over User::Leave()/C++ exceptions."));
 | 
|  |    150 | 
 | 
|  |    151 | 	__UHEAP_MARK;
 | 
|  |    152 |    	//
 | 
|  |    153 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |    154 |    	TInt r = KErrNoMemory;
 | 
|  |    155 |    	if (cleanup)
 | 
|  |    156 |    		{
 | 
|  |    157 | 		TRAP(r, TestThreads());
 | 
|  |    158 | 
 | 
|  |    159 | 		test.Printf(_L("Returned %d, expected %d\n"), r, KErrNone);
 | 
|  |    160 | 		test(r == KErrNone);
 | 
|  |    161 |    		}
 | 
|  |    162 | 
 | 
|  |    163 | 	delete cleanup;
 | 
|  |    164 |    	//
 | 
|  |    165 |    	__UHEAP_MARKEND;
 | 
|  |    166 | 
 | 
|  |    167 | 	test.End();
 | 
|  |    168 | 
 | 
|  |    169 |    	return r;
 | 
|  |    170 |    	}
 | 
|  |    171 | 
 | 
|  |    172 | TInt NopThread(TAny*)
 | 
|  |    173 | 	{
 | 
|  |    174 | #ifdef __WINS__
 | 
|  |    175 | 	TInt32 current = *(TInt*)NopThreadInfo.iFunc;
 | 
|  |    176 | 	if (current != NopThreadInfo.iOriginalContents)
 | 
|  |    177 | 		current = *(TInt32*)NULL; // cause panic
 | 
|  |    178 | #endif
 | 
|  |    179 | 	TInt r = NopThreadInfo.iFunc();
 | 
|  |    180 | 	if (r)
 | 
|  |    181 | 		return KErrNone;
 | 
|  |    182 | 	else
 | 
|  |    183 | 		return KErrGeneral;
 | 
|  |    184 | 	}
 | 
|  |    185 | 
 | 
|  |    186 | void TestLoadWhileUnload();
 | 
|  |    187 | 
 | 
|  |    188 | void TestThreads()
 | 
|  |    189 | 	{
 | 
|  |    190 | 	__KHEAP_MARK;
 | 
|  |    191 | 
 | 
|  |    192 | 	test.Next(_L("Create synchronisation semaphores"));
 | 
|  |    193 | 	TInt r = Thread1Semaphore.CreateLocal(0);
 | 
|  |    194 | 	test(r == KErrNone);
 | 
|  |    195 | 
 | 
|  |    196 | 	r = Thread2Semaphore.CreateLocal(0);
 | 
|  |    197 | 	test(r == KErrNone);
 | 
|  |    198 | 
 | 
|  |    199 | 	r = FinishedOpSemaphore.CreateLocal(0);
 | 
|  |    200 | 	test(r == KErrNone);
 | 
|  |    201 | 
 | 
|  |    202 | 	// Turn off JIT [threads can panic to test exit cleanup]
 | 
|  |    203 | 	TBool jit = User::JustInTime();
 | 
|  |    204 | 	User::SetJustInTime(EFalse);
 | 
|  |    205 | 
 | 
|  |    206 | 	TInt count, count2;
 | 
|  |    207 | 	
 | 
|  |    208 | 	// Do kernel heap checking
 | 
|  |    209 | 	CheckKernelHeap = ETrue;
 | 
|  |    210 | 
 | 
|  |    211 | 	test.Next(_L("Run threads on their own"));
 | 
|  |    212 | 	for (count = 0; count < TheThreadCount ; ++count)
 | 
|  |    213 | 		{
 | 
|  |    214 | 		// Set up descriptor for thread's name
 | 
|  |    215 | 		TBuf16<7> name(KThread);
 | 
|  |    216 | 		name.Append('A' + count);
 | 
|  |    217 | 
 | 
|  |    218 | 		// Create thread
 | 
|  |    219 | 		RThread thread;
 | 
|  |    220 | 		TInt r = thread.Create(
 | 
|  |    221 | 			name,
 | 
|  |    222 | 			TheThreadArray[count].iThreadFn,
 | 
|  |    223 | 			KDefaultStackSize,
 | 
|  |    224 | 			KHeapSize,
 | 
|  |    225 | 			KHeapSize,
 | 
|  |    226 | 			&Thread1Semaphore
 | 
|  |    227 | 		);
 | 
|  |    228 | 		test(r == KErrNone);
 | 
|  |    229 | 
 | 
|  |    230 | 		// Set up notification of thread's death
 | 
|  |    231 | 		TRequestStatus status;
 | 
|  |    232 | 		thread.Logon(status);
 | 
|  |    233 | 
 | 
|  |    234 | 		// Load library
 | 
|  |    235 | 		RLibrary library;
 | 
|  |    236 | 		r = library.Load(KLeavingDll);
 | 
|  |    237 | 		test(r == KErrNone);
 | 
|  |    238 | 
 | 
|  |    239 | 		// Remember the address of the NOP function
 | 
|  |    240 | 		NopThreadInfo.iFunc = library.Lookup(2);
 | 
|  |    241 | #ifdef __WINS__
 | 
|  |    242 | 		NopThreadInfo.iOriginalContents = *(TInt32*)NopThreadInfo.iFunc;
 | 
|  |    243 | #endif
 | 
|  |    244 | 
 | 
|  |    245 | 		// Run thread
 | 
|  |    246 | 		thread.Resume();
 | 
|  |    247 | 
 | 
|  |    248 | 		// Wait until it has an open handle to the library
 | 
|  |    249 | 		FinishedOpSemaphore.Wait();
 | 
|  |    250 | 
 | 
|  |    251 | 		// Close our handle to the library
 | 
|  |    252 | 		library.Close();
 | 
|  |    253 | 
 | 
|  |    254 | 		// Check library is still loaded
 | 
|  |    255 | 		for (count2 = 0; count2 < TheThreadArray[count].iMappedSignals; ++count2)
 | 
|  |    256 | 			{
 | 
|  |    257 | 			// Tell it we're ready to go
 | 
|  |    258 | 			Thread1Semaphore.Signal();
 | 
|  |    259 | 
 | 
|  |    260 | 			// Wait for it to finish next step
 | 
|  |    261 | 			FinishedOpSemaphore.Wait();
 | 
|  |    262 | 
 | 
|  |    263 | 			// Create NOP thread to call NOP function to check DLL still loaded
 | 
|  |    264 | 			RThread nopThread;
 | 
|  |    265 | 			r = nopThread.Create(
 | 
|  |    266 | 				KNopThreadName,
 | 
|  |    267 | 				NopThread,
 | 
|  |    268 | 				KDefaultStackSize,
 | 
|  |    269 | 				KHeapSize,
 | 
|  |    270 | 				KHeapSize,
 | 
|  |    271 | 				NULL
 | 
|  |    272 | 			);
 | 
|  |    273 | 			test(r == KErrNone);
 | 
|  |    274 | 
 | 
|  |    275 | 			// Set up notification of thread's death
 | 
|  |    276 | 			TRequestStatus nopStatus;
 | 
|  |    277 | 			nopThread.Logon(nopStatus);
 | 
|  |    278 | 
 | 
|  |    279 | 			// Run thread
 | 
|  |    280 | 			nopThread.Resume();
 | 
|  |    281 | 
 | 
|  |    282 | 			// Wait for it to die
 | 
|  |    283 | 			User::WaitForRequest(nopStatus);
 | 
|  |    284 | 
 | 
|  |    285 | 			// Check the exit info
 | 
|  |    286 | 			test(nopThread.ExitType() == EExitKill);
 | 
|  |    287 | 			test(nopThread.ExitReason() == KErrNone);
 | 
|  |    288 | 
 | 
|  |    289 | 			// Close thread handle
 | 
|  |    290 | 			CLOSE_AND_WAIT(nopThread);
 | 
|  |    291 | 			}
 | 
|  |    292 | 
 | 
|  |    293 | 		// Check User::Leave() library unloading behaviour
 | 
|  |    294 | 		for (count2 = 0; count2 < TheThreadArray[count].iLeaveSignals; ++count2)
 | 
|  |    295 | 			{
 | 
|  |    296 | 			// Tell it we're ready to go
 | 
|  |    297 | 			Thread1Semaphore.Signal();
 | 
|  |    298 | 
 | 
|  |    299 | 			// Wait for it to finish next step
 | 
|  |    300 | 			FinishedOpSemaphore.Wait();
 | 
|  |    301 | 
 | 
|  |    302 | 			// Create NOP thread to call NOP function to check whether DLL is still loaded
 | 
|  |    303 | 			RThread nopThread;
 | 
|  |    304 | 			r = nopThread.Create(
 | 
|  |    305 | 				KNopThreadName,
 | 
|  |    306 | 				NopThread,
 | 
|  |    307 | 				KDefaultStackSize,
 | 
|  |    308 | 				KHeapSize,
 | 
|  |    309 | 				KHeapSize,
 | 
|  |    310 | 				NULL
 | 
|  |    311 | 			);
 | 
|  |    312 | 			test(r == KErrNone);
 | 
|  |    313 | 
 | 
|  |    314 | 			// Set up notification of thread's death
 | 
|  |    315 | 			TRequestStatus nopStatus;
 | 
|  |    316 | 			nopThread.Logon(nopStatus);
 | 
|  |    317 | 
 | 
|  |    318 | 			// Run thread
 | 
|  |    319 | 			nopThread.Resume();
 | 
|  |    320 | 
 | 
|  |    321 | 			// Wait for it to die
 | 
|  |    322 | 			User::WaitForRequest(nopStatus);
 | 
|  |    323 | 
 | 
|  |    324 | 			// Check the exit info
 | 
|  |    325 | #ifdef __LEAVE_EQUALS_THROW__
 | 
|  |    326 | 			test(nopThread.ExitType() == EExitKill);
 | 
|  |    327 | 			test(nopThread.ExitReason() == KErrGeneral);
 | 
|  |    328 | #else //!__LEAVE_EQUALS_THROW__
 | 
|  |    329 | 			test(nopThread.ExitType() == EExitPanic);
 | 
|  |    330 | 			test(nopThread.ExitCategory() == KUnhandledExcCategory);
 | 
|  |    331 | 			test(nopThread.ExitReason() == KUnhandledExcReason);
 | 
|  |    332 | #endif //__LEAVE_EQUALS_THROW__
 | 
|  |    333 | 
 | 
|  |    334 | 			// Close thread handle
 | 
|  |    335 | 			CLOSE_AND_WAIT(nopThread);
 | 
|  |    336 | 			}
 | 
|  |    337 | 
 | 
|  |    338 | 		// Tell it we're ready to go again
 | 
|  |    339 | 		Thread1Semaphore.Signal();
 | 
|  |    340 | 
 | 
|  |    341 | 		if (TheThreadArray[count].iExitType == EExitKill)
 | 
|  |    342 | 			{
 | 
|  |    343 | 			// Wait for it to finish last step
 | 
|  |    344 | 			FinishedOpSemaphore.Wait();
 | 
|  |    345 | 
 | 
|  |    346 | 			User::After(100000);	// let supervisor run
 | 
|  |    347 | 
 | 
|  |    348 | 			// Create NOP thread to call NOP function to check DLL is unloaded
 | 
|  |    349 | 			RThread nopThread;
 | 
|  |    350 | 			r = nopThread.Create(
 | 
|  |    351 | 				KNopThreadName,
 | 
|  |    352 | 				NopThread,
 | 
|  |    353 | 				KDefaultStackSize,
 | 
|  |    354 | 				KHeapSize,
 | 
|  |    355 | 				KHeapSize,
 | 
|  |    356 | 				NULL
 | 
|  |    357 | 			);
 | 
|  |    358 | 			test(r == KErrNone);
 | 
|  |    359 | 
 | 
|  |    360 | 			// Set up notification of thread's death
 | 
|  |    361 | 			TRequestStatus nopStatus;
 | 
|  |    362 | 			nopThread.Logon(nopStatus);
 | 
|  |    363 | 
 | 
|  |    364 | 			// Run thread
 | 
|  |    365 | 			nopThread.Resume();
 | 
|  |    366 | 
 | 
|  |    367 | 			// Wait for it to die
 | 
|  |    368 | 			User::WaitForRequest(nopStatus);
 | 
|  |    369 | 
 | 
|  |    370 | 			// Check the exit info
 | 
|  |    371 | 			test(nopThread.ExitType() == EExitPanic);
 | 
|  |    372 | 			test(nopThread.ExitCategory() == KUnhandledExcCategory);
 | 
|  |    373 | 			test(nopThread.ExitReason() == KUnhandledExcReason);
 | 
|  |    374 | 
 | 
|  |    375 | 			// Close thread handle
 | 
|  |    376 | 			CLOSE_AND_WAIT(nopThread);
 | 
|  |    377 | 
 | 
|  |    378 | 			// Let main thread die now
 | 
|  |    379 | 			Thread1Semaphore.Signal();
 | 
|  |    380 | 			}
 | 
|  |    381 | 
 | 
|  |    382 | 		// Wait for thread to exit
 | 
|  |    383 | 		User::WaitForRequest(status);
 | 
|  |    384 | 
 | 
|  |    385 | 		// Check the exit type & category
 | 
|  |    386 | 		test(thread.ExitType() == TheThreadArray[count].iExitType);
 | 
|  |    387 | 
 | 
|  |    388 | 		// Check category & reason, if appropriate
 | 
|  |    389 | 		if (thread.ExitType() == EExitPanic)
 | 
|  |    390 | 			{
 | 
|  |    391 | 			test(thread.ExitCategory() == KTUnmapPanic);
 | 
|  |    392 | 			test(thread.ExitReason() == EPanickingThread);
 | 
|  |    393 | 			}
 | 
|  |    394 | 
 | 
|  |    395 | 		// Close thread handle
 | 
|  |    396 | 		thread.Close();
 | 
|  |    397 | 		}
 | 
|  |    398 | 
 | 
|  |    399 | 	// Turn off kernel heap checking
 | 
|  |    400 | 	CheckKernelHeap = EFalse;
 | 
|  |    401 | 
 | 
|  |    402 | 	test.Next(_L("Run threads against each other"));
 | 
|  |    403 | 	for (count = 0; count < TheThreadCount ; ++count)
 | 
|  |    404 | 		{
 | 
|  |    405 | 		for (count2 = 0; count2 < TheThreadCount ; ++count2)
 | 
|  |    406 | 			{
 | 
|  |    407 | 			// Can't run the same threads back to back
 | 
|  |    408 | 			if (count == count2)
 | 
|  |    409 | 				{
 | 
|  |    410 | 				continue;
 | 
|  |    411 | 				}
 | 
|  |    412 | 
 | 
|  |    413 | 			// Set up descriptors for threads' names
 | 
|  |    414 | 			_LIT(KFirstThread, " - 1");
 | 
|  |    415 | 			_LIT(KSecondThread, " - 2");
 | 
|  |    416 | 			TBuf16<11> name(KThread);
 | 
|  |    417 | 			TBuf16<11> name2(KThread);
 | 
|  |    418 | 			name.Append('A' + count);
 | 
|  |    419 | 			name.Append(KFirstThread);
 | 
|  |    420 | 			name2.Append('A' + count2);
 | 
|  |    421 | 			name2.Append(KSecondThread);
 | 
|  |    422 | 
 | 
|  |    423 | 			// Create threads
 | 
|  |    424 | 			RThread thread;
 | 
|  |    425 | 			TInt r = thread.Create(
 | 
|  |    426 | 				name,
 | 
|  |    427 | 				TheThreadArray[count].iThreadFn,
 | 
|  |    428 | 				KDefaultStackSize,
 | 
|  |    429 | 				KHeapSize,
 | 
|  |    430 | 				KHeapSize,
 | 
|  |    431 | 				&Thread1Semaphore
 | 
|  |    432 | 			);
 | 
|  |    433 | 			test(r == KErrNone);
 | 
|  |    434 | 
 | 
|  |    435 | 			RThread thread2;
 | 
|  |    436 | 			r = thread2.Create(
 | 
|  |    437 | 				name2,
 | 
|  |    438 | 				TheThreadArray[count2].iThreadFn,
 | 
|  |    439 | 				KDefaultStackSize,
 | 
|  |    440 | 				KHeapSize,
 | 
|  |    441 | 				KHeapSize,
 | 
|  |    442 | 				&Thread2Semaphore
 | 
|  |    443 | 			);
 | 
|  |    444 | 			test(r == KErrNone);
 | 
|  |    445 | 
 | 
|  |    446 | 			// Set up notification of threads' death
 | 
|  |    447 | 			TRequestStatus status, status2;
 | 
|  |    448 | 			thread.Logon(status);
 | 
|  |    449 | 			thread2.Logon(status2);
 | 
|  |    450 | 
 | 
|  |    451 | 			// Run first thread
 | 
|  |    452 | 			thread.Resume();
 | 
|  |    453 | 
 | 
|  |    454 | 			// Wait until just before it's closed the library handle
 | 
|  |    455 | 			FinishedOpSemaphore.Wait();
 | 
|  |    456 | 
 | 
|  |    457 | 			// Run second thread
 | 
|  |    458 | 			thread2.Resume();
 | 
|  |    459 | 
 | 
|  |    460 | 			// Wait until just before it's closed the library handle
 | 
|  |    461 | 			FinishedOpSemaphore.Wait();
 | 
|  |    462 | 
 | 
|  |    463 | 			// Tell first thread we're ready to go
 | 
|  |    464 | 			TInt signals =	TheThreadArray[count].iMappedSignals +
 | 
|  |    465 | 							TheThreadArray[count].iLeaveSignals +
 | 
|  |    466 | 							((TheThreadArray[count].iExitType == EExitPanic) ? 1 : 2);
 | 
|  |    467 | 			Thread1Semaphore.Signal(signals);
 | 
|  |    468 | 
 | 
|  |    469 | 			// Eat up 'FinishedOp' signals
 | 
|  |    470 | 			while(--signals>0)
 | 
|  |    471 | 				FinishedOpSemaphore.Wait();
 | 
|  |    472 | 
 | 
|  |    473 | 			// Wait for it to finish
 | 
|  |    474 | 			User::WaitForRequest(status);
 | 
|  |    475 | 
 | 
|  |    476 | 			// Check the exit type & category of the first thread
 | 
|  |    477 | 			test(thread.ExitType() == TheThreadArray[count].iExitType);
 | 
|  |    478 | 
 | 
|  |    479 | 			// Check category & reason of the first thread, if appropriate
 | 
|  |    480 | 			if (thread.ExitType() == EExitPanic)
 | 
|  |    481 | 				{
 | 
|  |    482 | 				test(thread.ExitCategory() == KTUnmapPanic);
 | 
|  |    483 | 				test(thread.ExitReason() == EPanickingThread);
 | 
|  |    484 | 				}
 | 
|  |    485 | 
 | 
|  |    486 | 			// Tell second thread we're ready to go
 | 
|  |    487 | 			signals = TheThreadArray[count2].iMappedSignals +
 | 
|  |    488 | 					  TheThreadArray[count2].iLeaveSignals +
 | 
|  |    489 | 					  ((TheThreadArray[count2].iExitType == EExitPanic) ? 1 : 2);
 | 
|  |    490 | 			Thread2Semaphore.Signal(signals);
 | 
|  |    491 | 
 | 
|  |    492 | 			// Eat up 'FinishedOp' signals
 | 
|  |    493 | 			while(--signals>0)
 | 
|  |    494 | 				FinishedOpSemaphore.Wait();
 | 
|  |    495 | 
 | 
|  |    496 | 			// Wait for it to finish
 | 
|  |    497 | 			User::WaitForRequest(status2);
 | 
|  |    498 | 
 | 
|  |    499 | 			// Check the exit type & category of the second thread
 | 
|  |    500 | 			test(thread2.ExitType() == TheThreadArray[count2].iExitType);
 | 
|  |    501 | 
 | 
|  |    502 | 			// Check category & reason of the second thread, if appropriate
 | 
|  |    503 | 			if (thread2.ExitType() == EExitPanic)
 | 
|  |    504 | 				{
 | 
|  |    505 | 				test(thread2.ExitCategory() == KTUnmapPanic);
 | 
|  |    506 | 				test(thread2.ExitReason() == EPanickingThread);
 | 
|  |    507 | 				}
 | 
|  |    508 | 
 | 
|  |    509 | 			// Close thread handles
 | 
|  |    510 | 			CLOSE_AND_WAIT(thread);
 | 
|  |    511 | 			CLOSE_AND_WAIT(thread2);
 | 
|  |    512 | 			}
 | 
|  |    513 | 		}
 | 
|  |    514 | 
 | 
|  |    515 | 	// Test two processes at once to deal with race conditions
 | 
|  |    516 | 	test.Printf(_L("Create two processes at once to map the same library\n"));
 | 
|  |    517 | 	RSemaphore procSem1, procSem2;
 | 
|  |    518 | 	test(procSem1.CreateGlobal(KNullDesC, 0)==KErrNone);
 | 
|  |    519 | 	test(procSem2.CreateGlobal(KNullDesC, 0)==KErrNone);
 | 
|  |    520 | 	RProcess proc1, proc2;
 | 
|  |    521 | 	test(proc1.Create(_L("T_UNMAP2"), KNullDesC)==KErrNone);
 | 
|  |    522 | 	test(proc2.Create(_L("T_UNMAP2"), KNullDesC)==KErrNone);
 | 
|  |    523 | 	test(proc1.SetParameter(1, procSem1)==KErrNone);
 | 
|  |    524 | 	test(proc1.SetParameter(2, procSem2)==KErrNone);
 | 
|  |    525 | 	test(proc2.SetParameter(1, procSem2)==KErrNone);
 | 
|  |    526 | 	test(proc2.SetParameter(2, procSem1)==KErrNone);
 | 
|  |    527 | 	TRequestStatus proc1stat, proc2stat;
 | 
|  |    528 | 	proc1.Logon(proc1stat);
 | 
|  |    529 | 	proc2.Logon(proc2stat);
 | 
|  |    530 | 	test.Printf(_L("Start processes\n"));
 | 
|  |    531 | 	proc1.Resume();
 | 
|  |    532 | 	proc2.Resume();
 | 
|  |    533 | 	test.Printf(_L("Wait for them to exit\n"));
 | 
|  |    534 | 	User::WaitForRequest(proc1stat);
 | 
|  |    535 | 	test(proc1.ExitType() == EExitKill);
 | 
|  |    536 | 	test(proc1.ExitReason() == KErrNone);
 | 
|  |    537 | 	User::WaitForRequest(proc2stat);
 | 
|  |    538 | 	test(proc2.ExitType() == EExitKill);
 | 
|  |    539 | 	test(proc2.ExitReason()==KErrNone);
 | 
|  |    540 | 	CLOSE_AND_WAIT(proc1);
 | 
|  |    541 | 	CLOSE_AND_WAIT(proc2);
 | 
|  |    542 | 	procSem1.Close();
 | 
|  |    543 | 	procSem2.Close();
 | 
|  |    544 | 
 | 
|  |    545 | 	// Test load while unload
 | 
|  |    546 | 	TestLoadWhileUnload();
 | 
|  |    547 | 
 | 
|  |    548 | 	// Restore JIT setting
 | 
|  |    549 | 	User::SetJustInTime(jit);
 | 
|  |    550 | 
 | 
|  |    551 | 	// Close synchronisation semaphores
 | 
|  |    552 | 	Thread1Semaphore.Close();
 | 
|  |    553 | 	Thread2Semaphore.Close();
 | 
|  |    554 | 	FinishedOpSemaphore.Close();
 | 
|  |    555 | 
 | 
|  |    556 | 	__KHEAP_MARKEND;
 | 
|  |    557 | 	}
 | 
|  |    558 | 
 | 
|  |    559 | // Test loading a library while another thread is unloading it in an unwind
 | 
|  |    560 | void TestLoadWhileUnload()
 | 
|  |    561 | 	{
 | 
|  |    562 | 	// Set up descriptor for thread's name
 | 
|  |    563 | 	TBuf16<7> name(KThread);
 | 
|  |    564 | 	name.Append('H');
 | 
|  |    565 | 
 | 
|  |    566 | 	// Create thread
 | 
|  |    567 | 	RThread thread;
 | 
|  |    568 | 	TInt r = thread.Create(
 | 
|  |    569 | 		name,
 | 
|  |    570 | 		ThreadH,
 | 
|  |    571 | 		KDefaultStackSize,
 | 
|  |    572 | 		KHeapSize,
 | 
|  |    573 | 		KHeapSize,
 | 
|  |    574 | 		&Thread1Semaphore
 | 
|  |    575 | 	);
 | 
|  |    576 | 	test(r == KErrNone);
 | 
|  |    577 | 
 | 
|  |    578 | 	// Set up notification of thread's death
 | 
|  |    579 | 	TRequestStatus status;
 | 
|  |    580 | 	thread.Logon(status);
 | 
|  |    581 | 
 | 
|  |    582 | 	// Run thread
 | 
|  |    583 | 	thread.Resume();
 | 
|  |    584 | 
 | 
|  |    585 | 	// Wait until it has an open handle to the library
 | 
|  |    586 | 	FinishedOpSemaphore.Wait();
 | 
|  |    587 | 
 | 
|  |    588 | 	// Tell it to go ahead and leave
 | 
|  |    589 | 	Thread1Semaphore.Signal();
 | 
|  |    590 | 
 | 
|  |    591 | 	// Wait for it to start unwinding
 | 
|  |    592 | 	FinishedOpSemaphore.Wait();
 | 
|  |    593 | 
 | 
|  |    594 | 	// Tell it to go ahead and close the library handle
 | 
|  |    595 | 	Thread1Semaphore.Signal();
 | 
|  |    596 | 
 | 
|  |    597 | 	// Wait for it to have closed the library
 | 
|  |    598 | 	FinishedOpSemaphore.Wait();
 | 
|  |    599 | 
 | 
|  |    600 | 	// Load library
 | 
|  |    601 | 	RLibrary library;
 | 
|  |    602 | 	r = library.Load(KLeavingDll);
 | 
|  |    603 | 	test(r == KErrNone);
 | 
|  |    604 | 
 | 
|  |    605 | 	// Remember the address of the NOP function
 | 
|  |    606 | 	NopThreadInfo.iFunc = library.Lookup(2);
 | 
|  |    607 | #ifdef __WINS__
 | 
|  |    608 | 	NopThreadInfo.iOriginalContents = *(TInt32*)NopThreadInfo.iFunc;
 | 
|  |    609 | #endif
 | 
|  |    610 | 
 | 
|  |    611 | 	User::After(100000);	// let supervisor run
 | 
|  |    612 | 
 | 
|  |    613 | 	// Check User::Leave() library unloading behaviour
 | 
|  |    614 | 	for (TInt i = 0; i < 2; ++i)
 | 
|  |    615 | 		{
 | 
|  |    616 | 		// Create NOP thread to call NOP function to check whether DLL is still loaded
 | 
|  |    617 | 		RThread nopThread;
 | 
|  |    618 | 		r = nopThread.Create(
 | 
|  |    619 | 			KNopThreadName,
 | 
|  |    620 | 			NopThread,
 | 
|  |    621 | 			KDefaultStackSize,
 | 
|  |    622 | 			KHeapSize,
 | 
|  |    623 | 			KHeapSize,
 | 
|  |    624 | 			NULL
 | 
|  |    625 | 		);
 | 
|  |    626 | 		test(r == KErrNone);
 | 
|  |    627 | 
 | 
|  |    628 | 		// Set up notification of thread's death
 | 
|  |    629 | 		TRequestStatus nopStatus;
 | 
|  |    630 | 		nopThread.Logon(nopStatus);
 | 
|  |    631 | 
 | 
|  |    632 | 		// Run thread
 | 
|  |    633 | 		nopThread.Resume();
 | 
|  |    634 | 
 | 
|  |    635 | 		// Wait for it to die
 | 
|  |    636 | 		User::WaitForRequest(nopStatus);
 | 
|  |    637 | 
 | 
|  |    638 | 		// Check the exit info
 | 
|  |    639 | 		test(nopThread.ExitType() == EExitKill);
 | 
|  |    640 | 		test(nopThread.ExitReason() == KErrNone);
 | 
|  |    641 | 
 | 
|  |    642 | 		// Close thread handle
 | 
|  |    643 | 		CLOSE_AND_WAIT(nopThread);
 | 
|  |    644 | 
 | 
|  |    645 | 		// Tell it we're ready to go
 | 
|  |    646 | 		Thread1Semaphore.Signal();
 | 
|  |    647 | 
 | 
|  |    648 | 		// Wait for it to finish next step
 | 
|  |    649 | 		if (i==0)
 | 
|  |    650 | 			FinishedOpSemaphore.Wait();
 | 
|  |    651 | 		}
 | 
|  |    652 | 
 | 
|  |    653 | 	// Wait for thread to exit
 | 
|  |    654 | 	User::WaitForRequest(status);
 | 
|  |    655 | 
 | 
|  |    656 | 	// Check the exit type & category
 | 
|  |    657 | 	test(thread.ExitType() == EExitKill);
 | 
|  |    658 | 
 | 
|  |    659 | 	// Close thread handle
 | 
|  |    660 | 	CLOSE_AND_WAIT(thread);
 | 
|  |    661 | 
 | 
|  |    662 | 	// Close our handle to the library
 | 
|  |    663 | 	library.Close();
 | 
|  |    664 | 
 | 
|  |    665 | 	// Create NOP thread to call NOP function to check DLL is unloaded
 | 
|  |    666 | 	RThread nopThread;
 | 
|  |    667 | 	r = nopThread.Create(
 | 
|  |    668 | 		KNopThreadName,
 | 
|  |    669 | 		NopThread,
 | 
|  |    670 | 		KDefaultStackSize,
 | 
|  |    671 | 		KHeapSize,
 | 
|  |    672 | 		KHeapSize,
 | 
|  |    673 | 		NULL
 | 
|  |    674 | 	);
 | 
|  |    675 | 	test(r == KErrNone);
 | 
|  |    676 | 
 | 
|  |    677 | 	// Set up notification of thread's death
 | 
|  |    678 | 	TRequestStatus nopStatus;
 | 
|  |    679 | 	nopThread.Logon(nopStatus);
 | 
|  |    680 | 
 | 
|  |    681 | 	// Run thread
 | 
|  |    682 | 	nopThread.Resume();
 | 
|  |    683 | 
 | 
|  |    684 | 	// Wait for it to die
 | 
|  |    685 | 	User::WaitForRequest(nopStatus);
 | 
|  |    686 | 
 | 
|  |    687 | 	// Check the exit info
 | 
|  |    688 | 	test(nopThread.ExitType() == EExitPanic);
 | 
|  |    689 | 	test(nopThread.ExitCategory() == KUnhandledExcCategory);
 | 
|  |    690 | 	test(nopThread.ExitReason() == KUnhandledExcReason);
 | 
|  |    691 | 
 | 
|  |    692 | 	// Close thread handle
 | 
|  |    693 | 	CLOSE_AND_WAIT(nopThread);
 | 
|  |    694 | 	}	
 | 
|  |    695 | 
 | 
|  |    696 | //
 | 
|  |    697 | // Cleanup operations
 | 
|  |    698 | //
 | 
|  |    699 | 
 | 
|  |    700 | void Checkpoint(TAny* aSemaphore)
 | 
|  |    701 | 	{
 | 
|  |    702 | 	if(aSemaphore)
 | 
|  |    703 | 		{
 | 
|  |    704 | 		FinishedOpSemaphore.Signal();
 | 
|  |    705 | 		static_cast<RSemaphore*>(aSemaphore)->Wait();
 | 
|  |    706 | 		}
 | 
|  |    707 | 	}
 | 
|  |    708 | 
 | 
|  |    709 | void DieDieDie(TAny* aSemaphore)
 | 
|  |    710 | 	{
 | 
|  |    711 | 	// Check-point before panicking
 | 
|  |    712 | 	Checkpoint(aSemaphore);
 | 
|  |    713 | 	
 | 
|  |    714 | 	User::Panic(KTUnmapPanic, EPanickingThread);
 | 
|  |    715 | 	}
 | 
|  |    716 | 
 | 
|  |    717 | void PauseLeaving(TAny* aSemaphore)
 | 
|  |    718 | 	{
 | 
|  |    719 | 	Checkpoint(aSemaphore);
 | 
|  |    720 | 	}
 | 
|  |    721 | 
 | 
|  |    722 | void TrapLeave(TAny* aSemaphore)
 | 
|  |    723 | 	{
 | 
|  |    724 | 	TRAP_IGNORE(
 | 
|  |    725 | 		{
 | 
|  |    726 | 		CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |    727 | 
 | 
|  |    728 | 		Checkpoint(aSemaphore);
 | 
|  |    729 | 
 | 
|  |    730 | 		User::Leave(KErrGeneral);
 | 
|  |    731 | 
 | 
|  |    732 | 		CleanupStack::Pop(); // pause op
 | 
|  |    733 | 		}
 | 
|  |    734 | 	);
 | 
|  |    735 | 
 | 
|  |    736 | 	Checkpoint(aSemaphore);
 | 
|  |    737 | 	}
 | 
|  |    738 | 
 | 
|  |    739 | void TrapLeaveAndDie(TAny* aSemaphore)
 | 
|  |    740 | 	{
 | 
|  |    741 | 	TRAP_IGNORE(
 | 
|  |    742 | 		{
 | 
|  |    743 | 		CleanupStack::PushL(TCleanupItem(&DieDieDie, aSemaphore));
 | 
|  |    744 | 
 | 
|  |    745 | 		Checkpoint(aSemaphore);
 | 
|  |    746 | 
 | 
|  |    747 | 		User::Leave(KErrGeneral);
 | 
|  |    748 | 
 | 
|  |    749 | 		CleanupStack::Pop(); //DieDieDie op
 | 
|  |    750 | 		}
 | 
|  |    751 | 	);
 | 
|  |    752 | 	}
 | 
|  |    753 | 
 | 
|  |    754 | void TrapLeaveAndClose_ThreadE(TAny* aSemaphore)
 | 
|  |    755 | 	{
 | 
|  |    756 | 	CleanupStack::Pop(&ThreadELibraryHandle);
 | 
|  |    757 | 
 | 
|  |    758 | 	TRAP_IGNORE(
 | 
|  |    759 | 		{
 | 
|  |    760 | 		CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |    761 | 
 | 
|  |    762 | 		CleanupClosePushL(ThreadELibraryHandle);
 | 
|  |    763 | 
 | 
|  |    764 | 		CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |    765 | 
 | 
|  |    766 | 		Checkpoint(aSemaphore);
 | 
|  |    767 | 
 | 
|  |    768 | 		User::Leave(KErrGeneral);
 | 
|  |    769 | 
 | 
|  |    770 | 		CleanupStack::Pop(); //pre-close pause op
 | 
|  |    771 | 
 | 
|  |    772 | 		CleanupStack::Pop(&ThreadELibraryHandle);
 | 
|  |    773 | 
 | 
|  |    774 | 		CleanupStack::Pop(); //post-close pause op
 | 
|  |    775 | 		}
 | 
|  |    776 | 	);
 | 
|  |    777 | 
 | 
|  |    778 | 	Checkpoint(aSemaphore);
 | 
|  |    779 | 	}
 | 
|  |    780 | 
 | 
|  |    781 | void TrapLeaveCloseAndDie_ThreadF(TAny* aSemaphore)
 | 
|  |    782 | 	{
 | 
|  |    783 | 	CleanupStack::Pop(&ThreadFLibraryHandle);
 | 
|  |    784 | 
 | 
|  |    785 | 	TRAP_IGNORE(
 | 
|  |    786 | 		{
 | 
|  |    787 | 		CleanupStack::PushL(TCleanupItem(&DieDieDie, aSemaphore));
 | 
|  |    788 | 
 | 
|  |    789 | 		CleanupClosePushL(ThreadFLibraryHandle);
 | 
|  |    790 | 
 | 
|  |    791 | 		CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |    792 | 
 | 
|  |    793 | 		Checkpoint(aSemaphore);
 | 
|  |    794 | 
 | 
|  |    795 | 		User::Leave(KErrGeneral);
 | 
|  |    796 | 
 | 
|  |    797 | 		CleanupStack::Pop(); //pre-close pause op
 | 
|  |    798 | 
 | 
|  |    799 | 		CleanupStack::Pop(&ThreadFLibraryHandle);
 | 
|  |    800 | 
 | 
|  |    801 | 		CleanupStack::Pop(); //DieDieDie op
 | 
|  |    802 | 		}
 | 
|  |    803 | 	);
 | 
|  |    804 | 	}
 | 
|  |    805 | 
 | 
|  |    806 | 
 | 
|  |    807 | /**
 | 
|  |    808 | Here's a list of interesting things that could happen to a thread which
 | 
|  |    809 | has an open handle to library on cleanup stack:
 | 
|  |    810 | 
 | 
|  |    811 | a)	Panicks
 | 
|  |    812 | b)	Closes handle normally
 | 
|  |    813 | c)	Leaves and panicks before closing handle
 | 
|  |    814 | d)	Recursively leaves and panicks before closing handle
 | 
|  |    815 | e)	Recursively leaves and closes handle in recursive leave
 | 
|  |    816 | f)	Recursively leaves and panicks in recursive leave, after closing handle
 | 
|  |    817 | g)	Recursively leaves and returns to first leave without closing handle; first leave closes handle
 | 
|  |    818 | h)	Leaves and closes handle
 | 
|  |    819 | i)	Leaves and closes handle, then recursively leaves
 | 
|  |    820 | j)	Leaves and closes handle, then recursively leaves and panicks in recursive leave
 | 
|  |    821 | k)	Leaves and panicks after closing handle, but before leave completes
 | 
|  |    822 | 
 | 
|  |    823 | Other ideas yet to be done:
 | 
|  |    824 | 
 | 
|  |    825 | l)	TRAPs a leave, then closes handle
 | 
|  |    826 | m)	TRAPs a recusive leave, then closes handle
 | 
|  |    827 | 
 | 
|  |    828 | The thread functions below correspond to these.
 | 
|  |    829 | 
 | 
|  |    830 | These are the ways a library's code seg can be held open by a process:
 | 
|  |    831 | 
 | 
|  |    832 | a)	Open handle to the library
 | 
|  |    833 | b)	Open reference to code seg because the last reference to the library was closed during a leave and
 | 
|  |    834 | 	the process has not gone leave-idle
 | 
|  |    835 | 
 | 
|  |    836 | We then test both these by testing at extra points during the sequences above that
 | 
|  |    837 | the code segments are either mapped or unmapped, as appropriate.
 | 
|  |    838 | */
 | 
|  |    839 | 
 | 
|  |    840 | TInt ThreadA(TAny* aSemaphore)
 | 
|  |    841 | 	{
 | 
|  |    842 | 	__UHEAP_MARK;
 | 
|  |    843 | 	if (CheckKernelHeap)
 | 
|  |    844 | 		{
 | 
|  |    845 | 		__KHEAP_MARK;
 | 
|  |    846 | 		}
 | 
|  |    847 | 
 | 
|  |    848 | 	new (&testThreadA) RTest(KNullDesC);
 | 
|  |    849 | 	testThreadA.Start(KNullDesC);
 | 
|  |    850 |    	//
 | 
|  |    851 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |    852 |    	TInt r = KErrNoMemory;
 | 
|  |    853 |    	if (cleanup)
 | 
|  |    854 |    		{
 | 
|  |    855 | 		TRAP(r, DoThreadAL(aSemaphore));
 | 
|  |    856 | 
 | 
|  |    857 | 		// Check-point after closing the library handle
 | 
|  |    858 | 		Checkpoint(aSemaphore);
 | 
|  |    859 | 
 | 
|  |    860 | 		testThreadA.Printf(_L("A: Returned %d, expected %d\n"), r, KErrNone);
 | 
|  |    861 | 		testThreadA(r == KErrNone);
 | 
|  |    862 |    		}
 | 
|  |    863 | 
 | 
|  |    864 | 	delete cleanup;
 | 
|  |    865 |    	//
 | 
|  |    866 | 	testThreadA.End();
 | 
|  |    867 | 	testThreadA.Close();
 | 
|  |    868 | 
 | 
|  |    869 | 	if (CheckKernelHeap)
 | 
|  |    870 | 		{
 | 
|  |    871 | 		User::After(100000);	// let supervisor run
 | 
|  |    872 | 		__KHEAP_MARKEND;
 | 
|  |    873 | 		}
 | 
|  |    874 |    	__UHEAP_MARKEND;
 | 
|  |    875 | 
 | 
|  |    876 |    	return r;
 | 
|  |    877 | 	}
 | 
|  |    878 | 
 | 
|  |    879 | TInt DoThreadAL(TAny* aSemaphore)
 | 
|  |    880 | 	{
 | 
|  |    881 | 	testThreadA.Printf(_L("A: Loading DLL.\n"));
 | 
|  |    882 | 	User::LeaveIfError(ThreadALibraryHandle.Load(KLeavingDll));
 | 
|  |    883 | 
 | 
|  |    884 |     testThreadA.Printf(_L("A: Pushing cleanup item to kill this thread!\n"));
 | 
|  |    885 | 	CleanupStack::PushL(TCleanupItem(&DieDieDie, aSemaphore));
 | 
|  |    886 | 
 | 
|  |    887 | 	testThreadA.Printf(_L("A: Cleaning up Panic operation.\n"));
 | 
|  |    888 | 	CleanupStack::PopAndDestroy(); // DieDieDie op
 | 
|  |    889 | 
 | 
|  |    890 | 	ThreadALibraryHandle.Close();
 | 
|  |    891 | 
 | 
|  |    892 | 	return KErrNone;
 | 
|  |    893 | 	}
 | 
|  |    894 | 
 | 
|  |    895 | TInt ThreadB(TAny* aSemaphore)
 | 
|  |    896 | 	{
 | 
|  |    897 | 	__UHEAP_MARK;
 | 
|  |    898 | 	if (CheckKernelHeap)
 | 
|  |    899 | 		{
 | 
|  |    900 | 		__KHEAP_MARK;
 | 
|  |    901 | 		}
 | 
|  |    902 | 
 | 
|  |    903 | 	new (&testThreadB) RTest(KNullDesC);
 | 
|  |    904 | 	testThreadB.Start(KNullDesC);
 | 
|  |    905 |    	//
 | 
|  |    906 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |    907 |    	TInt r = KErrNoMemory;
 | 
|  |    908 |    	if (cleanup)
 | 
|  |    909 |    		{
 | 
|  |    910 | 		TRAP(r, DoThreadBL(aSemaphore));
 | 
|  |    911 | 
 | 
|  |    912 | 		// Check-point after closing the library handle
 | 
|  |    913 | 		Checkpoint(aSemaphore);
 | 
|  |    914 | 
 | 
|  |    915 | 		testThreadB.Printf(_L("B: Returned %d, expected %d\n"), r, KErrNone);
 | 
|  |    916 | 		testThreadB(r == KErrNone);
 | 
|  |    917 |    		}
 | 
|  |    918 | 
 | 
|  |    919 | 	delete cleanup;
 | 
|  |    920 |    	//
 | 
|  |    921 | 	testThreadB.End();
 | 
|  |    922 | 	testThreadB.Close();
 | 
|  |    923 | 
 | 
|  |    924 | 	if (CheckKernelHeap)
 | 
|  |    925 | 		{
 | 
|  |    926 | 		User::After(100000);	// let supervisor run
 | 
|  |    927 | 		__KHEAP_MARKEND;
 | 
|  |    928 | 		}
 | 
|  |    929 |    	__UHEAP_MARKEND;
 | 
|  |    930 | 
 | 
|  |    931 | 	return r;
 | 
|  |    932 | 	}
 | 
|  |    933 | 
 | 
|  |    934 | TInt DoThreadBL(TAny* aSemaphore)
 | 
|  |    935 | 	{
 | 
|  |    936 |     testThreadB.Printf(_L("B: Loading DLL.\n"));
 | 
|  |    937 | 	User::LeaveIfError(ThreadBLibraryHandle.Load(KLeavingDll));
 | 
|  |    938 | 
 | 
|  |    939 |     testThreadB.Printf(_L("B: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |    940 | 	CleanupClosePushL(ThreadBLibraryHandle);
 | 
|  |    941 | 
 | 
|  |    942 | 	// Check-point whilst holding the open library handle
 | 
|  |    943 | 	Checkpoint(aSemaphore);
 | 
|  |    944 | 	
 | 
|  |    945 | 	testThreadB.Printf(_L("B: Cleaning up DLL handle.\n"));
 | 
|  |    946 | 	CleanupStack::PopAndDestroy(&ThreadBLibraryHandle);
 | 
|  |    947 | 
 | 
|  |    948 | 	return KErrNone;
 | 
|  |    949 | 	}
 | 
|  |    950 | 
 | 
|  |    951 | TInt ThreadC(TAny* aSemaphore)
 | 
|  |    952 | 	{
 | 
|  |    953 | 	__UHEAP_MARK;
 | 
|  |    954 | 	if (CheckKernelHeap)
 | 
|  |    955 | 		{
 | 
|  |    956 | 		__KHEAP_MARK;
 | 
|  |    957 | 		}
 | 
|  |    958 | 
 | 
|  |    959 | 	new (&testThreadC) RTest(KNullDesC);
 | 
|  |    960 | 	testThreadC.Start(KNullDesC);
 | 
|  |    961 |    	//
 | 
|  |    962 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |    963 |    	TInt r = KErrNoMemory;
 | 
|  |    964 |    	if (cleanup)
 | 
|  |    965 |    		{
 | 
|  |    966 | 		TRAP(r, DoThreadCL(aSemaphore));
 | 
|  |    967 | 
 | 
|  |    968 | 		// Check-point after closing the library handle
 | 
|  |    969 | 		Checkpoint(aSemaphore);
 | 
|  |    970 | 
 | 
|  |    971 | 		testThreadC.Printf(_L("C: Returned %d, expected %d\n"), r, KErrGeneral);
 | 
|  |    972 | 		testThreadC(r == KErrGeneral);
 | 
|  |    973 | 
 | 
|  |    974 | 		r = KErrNone;
 | 
|  |    975 |    		}
 | 
|  |    976 | 
 | 
|  |    977 | 	delete cleanup;
 | 
|  |    978 |    	//
 | 
|  |    979 | 	testThreadC.End();
 | 
|  |    980 | 	testThreadC.Close();
 | 
|  |    981 | 
 | 
|  |    982 | 	if (CheckKernelHeap)
 | 
|  |    983 | 		{
 | 
|  |    984 | 		User::After(100000);	// let supervisor run
 | 
|  |    985 | 		__KHEAP_MARKEND;
 | 
|  |    986 | 		}
 | 
|  |    987 |    	__UHEAP_MARKEND;
 | 
|  |    988 | 
 | 
|  |    989 | 	return r;
 | 
|  |    990 | 	}
 | 
|  |    991 | 
 | 
|  |    992 | TInt DoThreadCL(TAny* aSemaphore)
 | 
|  |    993 | 	{
 | 
|  |    994 |     testThreadC.Printf(_L("C: Loading DLL.\n"));
 | 
|  |    995 | 	User::LeaveIfError(ThreadCLibraryHandle.Load(KLeavingDll));
 | 
|  |    996 | 
 | 
|  |    997 |     testThreadC.Printf(_L("C: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |    998 | 	CleanupClosePushL(ThreadCLibraryHandle);
 | 
|  |    999 | 
 | 
|  |   1000 | 	testThreadC.Printf(_L("C: Pushing cleanup item to kill this thread before closing handle!\n"));
 | 
|  |   1001 | 	CleanupStack::PushL(TCleanupItem(&DieDieDie, aSemaphore));
 | 
|  |   1002 | 
 | 
|  |   1003 |     testThreadC.Printf(_L("C: Looking up leaving function.\n"));
 | 
|  |   1004 | 	TLibraryFunction leaving = ThreadCLibraryHandle.Lookup(1);
 | 
|  |   1005 | 	User::LeaveIfNull((TAny*)leaving);
 | 
|  |   1006 | 
 | 
|  |   1007 | 	// Check-point whilst holding the open library handle
 | 
|  |   1008 | 	Checkpoint(aSemaphore);
 | 
|  |   1009 | 	
 | 
|  |   1010 | 	testThreadC.Printf(_L("C: Calling leaving function.\n"));
 | 
|  |   1011 | 	(*leaving)();
 | 
|  |   1012 | 
 | 
|  |   1013 | 	testThreadC.Printf(_L("C: Cleaning up Panic operation.\n"));
 | 
|  |   1014 | 	CleanupStack::Pop(aSemaphore); // DieDieDie op
 | 
|  |   1015 | 
 | 
|  |   1016 | 	testThreadC.Printf(_L("C: Cleaning up DLL handle.\n"));
 | 
|  |   1017 | 	CleanupStack::PopAndDestroy(&ThreadCLibraryHandle);
 | 
|  |   1018 | 
 | 
|  |   1019 | 	return KErrNone;
 | 
|  |   1020 | 	}
 | 
|  |   1021 | 
 | 
|  |   1022 | TInt ThreadD(TAny* aSemaphore)
 | 
|  |   1023 | 	{
 | 
|  |   1024 | 	__UHEAP_MARK;
 | 
|  |   1025 | 	if (CheckKernelHeap)
 | 
|  |   1026 | 		{
 | 
|  |   1027 | 		__KHEAP_MARK;
 | 
|  |   1028 | 		}
 | 
|  |   1029 | 
 | 
|  |   1030 | 	new (&testThreadD) RTest(KNullDesC);
 | 
|  |   1031 | 	testThreadD.Start(KNullDesC);
 | 
|  |   1032 |    	//
 | 
|  |   1033 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |   1034 |    	TInt r = KErrNoMemory;
 | 
|  |   1035 |    	if (cleanup)
 | 
|  |   1036 |    		{
 | 
|  |   1037 | 		TRAP(r, DoThreadDL(aSemaphore));
 | 
|  |   1038 | 
 | 
|  |   1039 | 		// Check-point after closing the library handle
 | 
|  |   1040 | 		Checkpoint(aSemaphore);
 | 
|  |   1041 | 
 | 
|  |   1042 | 		testThreadD.Printf(_L("D: Returned %d, expected %d\n"), r, KErrGeneral);
 | 
|  |   1043 | 		testThreadD(r == KErrGeneral);
 | 
|  |   1044 | 
 | 
|  |   1045 | 		r = KErrNone;
 | 
|  |   1046 |    		}
 | 
|  |   1047 | 
 | 
|  |   1048 | 	delete cleanup;
 | 
|  |   1049 |    	//
 | 
|  |   1050 | 	testThreadD.End();
 | 
|  |   1051 | 	testThreadD.Close();
 | 
|  |   1052 | 
 | 
|  |   1053 | 	if (CheckKernelHeap)
 | 
|  |   1054 | 		{
 | 
|  |   1055 | 		User::After(100000);	// let supervisor run
 | 
|  |   1056 | 		__KHEAP_MARKEND;
 | 
|  |   1057 | 		}
 | 
|  |   1058 |    	__UHEAP_MARKEND;
 | 
|  |   1059 | 
 | 
|  |   1060 | 	return r;
 | 
|  |   1061 | 	}
 | 
|  |   1062 | 
 | 
|  |   1063 | TInt DoThreadDL(TAny* aSemaphore)
 | 
|  |   1064 | 	{
 | 
|  |   1065 |     testThreadD.Printf(_L("D: Loading DLL.\n"));
 | 
|  |   1066 | 	User::LeaveIfError(ThreadDLibraryHandle.Load(KLeavingDll));
 | 
|  |   1067 | 
 | 
|  |   1068 |     testThreadD.Printf(_L("D: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |   1069 | 	CleanupClosePushL(ThreadDLibraryHandle);
 | 
|  |   1070 | 
 | 
|  |   1071 | 	testThreadD.Printf(_L("D: Pushing cleanup item to recursively leave and then kill this thread before closing handle!\n"));
 | 
|  |   1072 | 	CleanupStack::PushL(TCleanupItem(&TrapLeaveAndDie, aSemaphore));
 | 
|  |   1073 | 
 | 
|  |   1074 |     testThreadD.Printf(_L("D: Looking up leaving function.\n"));
 | 
|  |   1075 | 	TLibraryFunction leaving = ThreadDLibraryHandle.Lookup(1);
 | 
|  |   1076 | 	User::LeaveIfNull((TAny*)leaving);
 | 
|  |   1077 | 
 | 
|  |   1078 | 	// Check-point whilst holding the open library handle
 | 
|  |   1079 | 	Checkpoint(aSemaphore);
 | 
|  |   1080 | 
 | 
|  |   1081 | 	testThreadD.Printf(_L("D: Calling leaving function.\n"));
 | 
|  |   1082 | 	(*leaving)();
 | 
|  |   1083 | 
 | 
|  |   1084 | 	testThreadD.Printf(_L("D: Cleaning up DLL handle.\n"));
 | 
|  |   1085 | 	CleanupStack::PopAndDestroy(&ThreadDLibraryHandle);
 | 
|  |   1086 | 
 | 
|  |   1087 | 	testThreadD.Printf(_L("D: Cleaning up recursive leave operation.\n"));
 | 
|  |   1088 | 	CleanupStack::Pop(aSemaphore); // recursive leave op
 | 
|  |   1089 | 
 | 
|  |   1090 | 	return KErrNone;
 | 
|  |   1091 | 	}
 | 
|  |   1092 | 
 | 
|  |   1093 | TInt ThreadE(TAny* aSemaphore)
 | 
|  |   1094 | 	{
 | 
|  |   1095 | 	__UHEAP_MARK;
 | 
|  |   1096 | 	if (CheckKernelHeap)
 | 
|  |   1097 | 		{
 | 
|  |   1098 | 		__KHEAP_MARK;
 | 
|  |   1099 | 		}
 | 
|  |   1100 | 
 | 
|  |   1101 | 	new (&testThreadE) RTest(KNullDesC);
 | 
|  |   1102 | 	testThreadE.Start(KNullDesC);
 | 
|  |   1103 |    	//
 | 
|  |   1104 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |   1105 |    	TInt r = KErrNoMemory;
 | 
|  |   1106 |    	if (cleanup)
 | 
|  |   1107 |    		{
 | 
|  |   1108 | 		TRAP(r, DoThreadEL(aSemaphore));
 | 
|  |   1109 | 
 | 
|  |   1110 | 		// Check-point after closing the library handle
 | 
|  |   1111 | 		Checkpoint(aSemaphore);
 | 
|  |   1112 | 
 | 
|  |   1113 | 		testThreadE.Printf(_L("E: Returned %d, expected %d\n"), r, KErrGeneral);
 | 
|  |   1114 | 		testThreadE(r == KErrGeneral);
 | 
|  |   1115 | 
 | 
|  |   1116 | 		r = KErrNone;
 | 
|  |   1117 |    		}
 | 
|  |   1118 | 
 | 
|  |   1119 | 	delete cleanup;
 | 
|  |   1120 |    	//
 | 
|  |   1121 | 	testThreadE.End();
 | 
|  |   1122 | 	testThreadE.Close();
 | 
|  |   1123 | 
 | 
|  |   1124 | 	if (CheckKernelHeap)
 | 
|  |   1125 | 		{
 | 
|  |   1126 | 		User::After(100000);	// let supervisor run
 | 
|  |   1127 | 		__KHEAP_MARKEND;
 | 
|  |   1128 | 		}
 | 
|  |   1129 |    	__UHEAP_MARKEND;
 | 
|  |   1130 | 
 | 
|  |   1131 | 	return r;
 | 
|  |   1132 | 	}
 | 
|  |   1133 | 
 | 
|  |   1134 | TInt DoThreadEL(TAny* aSemaphore)
 | 
|  |   1135 | 	{
 | 
|  |   1136 |     testThreadE.Printf(_L("E: Loading DLL.\n"));
 | 
|  |   1137 | 	User::LeaveIfError(ThreadELibraryHandle.Load(KLeavingDll));
 | 
|  |   1138 | 
 | 
|  |   1139 |     testThreadE.Printf(_L("E: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |   1140 | 	CleanupClosePushL(ThreadELibraryHandle);
 | 
|  |   1141 | 
 | 
|  |   1142 | 	testThreadE.Printf(_L("E: Pushing cleanup item to recursively leave and then close the handle in the recursive leave\n"));
 | 
|  |   1143 | 	CleanupStack::PushL(TCleanupItem(&TrapLeaveAndClose_ThreadE, aSemaphore));
 | 
|  |   1144 | 
 | 
|  |   1145 |     testThreadE.Printf(_L("E: Looking up leaving function.\n"));
 | 
|  |   1146 | 	TLibraryFunction leaving = ThreadELibraryHandle.Lookup(1);
 | 
|  |   1147 | 	User::LeaveIfNull((TAny*)leaving);
 | 
|  |   1148 | 
 | 
|  |   1149 | 	// Check-point whilst holding the open library handle
 | 
|  |   1150 | 	Checkpoint(aSemaphore);
 | 
|  |   1151 | 
 | 
|  |   1152 | 	testThreadE.Printf(_L("E: Calling leaving function.\n"));
 | 
|  |   1153 | 	(*leaving)();
 | 
|  |   1154 | 
 | 
|  |   1155 | 	testThreadE.Printf(_L("E: Cleaning up recursive leave operation.\n"));
 | 
|  |   1156 | 	CleanupStack::Pop(aSemaphore); // recursive leave op
 | 
|  |   1157 | 
 | 
|  |   1158 | 	// NB: library handle removed from cleanup stack
 | 
|  |   1159 | 
 | 
|  |   1160 | 	return KErrNone;
 | 
|  |   1161 | 	}
 | 
|  |   1162 | 
 | 
|  |   1163 | TInt ThreadF(TAny* aSemaphore)
 | 
|  |   1164 | 	{
 | 
|  |   1165 | 	__UHEAP_MARK;
 | 
|  |   1166 | 	if (CheckKernelHeap)
 | 
|  |   1167 | 		{
 | 
|  |   1168 | 		__KHEAP_MARK;
 | 
|  |   1169 | 		}
 | 
|  |   1170 | 
 | 
|  |   1171 | 	new (&testThreadF) RTest(KNullDesC);
 | 
|  |   1172 | 	testThreadF.Start(KNullDesC);
 | 
|  |   1173 |    	//
 | 
|  |   1174 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |   1175 |    	TInt r = KErrNoMemory;
 | 
|  |   1176 |    	if (cleanup)
 | 
|  |   1177 |    		{
 | 
|  |   1178 | 		TRAP(r, DoThreadFL(aSemaphore));
 | 
|  |   1179 | 
 | 
|  |   1180 | 		// Check-point after closing the library handle
 | 
|  |   1181 | 		Checkpoint(aSemaphore);
 | 
|  |   1182 | 
 | 
|  |   1183 | 		testThreadF.Printf(_L("F: Returned %d, expected %d\n"), r, KErrGeneral);
 | 
|  |   1184 | 		testThreadF(r == KErrGeneral);
 | 
|  |   1185 | 
 | 
|  |   1186 | 		r = KErrNone;
 | 
|  |   1187 |    		}
 | 
|  |   1188 | 
 | 
|  |   1189 | 	delete cleanup;
 | 
|  |   1190 |    	//
 | 
|  |   1191 | 	testThreadF.End();
 | 
|  |   1192 | 	testThreadF.Close();
 | 
|  |   1193 | 
 | 
|  |   1194 | 	if (CheckKernelHeap)
 | 
|  |   1195 | 		{
 | 
|  |   1196 | 		User::After(100000);	// let supervisor run
 | 
|  |   1197 | 		__KHEAP_MARKEND;
 | 
|  |   1198 | 		}
 | 
|  |   1199 |    	__UHEAP_MARKEND;
 | 
|  |   1200 | 
 | 
|  |   1201 | 	return r;
 | 
|  |   1202 | 	}
 | 
|  |   1203 | 
 | 
|  |   1204 | TInt DoThreadFL(TAny* aSemaphore)
 | 
|  |   1205 | 	{
 | 
|  |   1206 |     testThreadF.Printf(_L("F: Loading DLL.\n"));
 | 
|  |   1207 | 	User::LeaveIfError(ThreadFLibraryHandle.Load(KLeavingDll));
 | 
|  |   1208 | 
 | 
|  |   1209 |     testThreadF.Printf(_L("F: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |   1210 | 	CleanupClosePushL(ThreadFLibraryHandle);
 | 
|  |   1211 | 
 | 
|  |   1212 | 	testThreadF.Printf(_L("F: Pushing cleanup item to recursively leave and then panic in recursive leave after closing the library handle\n"));
 | 
|  |   1213 | 	CleanupStack::PushL(TCleanupItem(&TrapLeaveCloseAndDie_ThreadF, aSemaphore));
 | 
|  |   1214 | 
 | 
|  |   1215 |     testThreadF.Printf(_L("F: Looking up leaving function.\n"));
 | 
|  |   1216 | 	TLibraryFunction leaving = ThreadFLibraryHandle.Lookup(1);
 | 
|  |   1217 | 	User::LeaveIfNull((TAny*)leaving);
 | 
|  |   1218 | 
 | 
|  |   1219 | 	// Check-point whilst holding the open library handle
 | 
|  |   1220 | 	Checkpoint(aSemaphore);
 | 
|  |   1221 | 
 | 
|  |   1222 | 	testThreadF.Printf(_L("F: Calling leaving function.\n"));
 | 
|  |   1223 | 	(*leaving)();
 | 
|  |   1224 | 
 | 
|  |   1225 | 	testThreadF.Printf(_L("F: Cleaning up recursive leave operation.\n"));
 | 
|  |   1226 | 	CleanupStack::Pop(aSemaphore); // recursive leave op
 | 
|  |   1227 | 
 | 
|  |   1228 | 	// NB: library handle removed from cleanup stack
 | 
|  |   1229 | 
 | 
|  |   1230 | 	return KErrNone;
 | 
|  |   1231 | 	}
 | 
|  |   1232 | 
 | 
|  |   1233 | TInt ThreadG(TAny* aSemaphore)
 | 
|  |   1234 | 	{
 | 
|  |   1235 | 	__UHEAP_MARK;
 | 
|  |   1236 | 	if (CheckKernelHeap)
 | 
|  |   1237 | 		{
 | 
|  |   1238 | 		__KHEAP_MARK;
 | 
|  |   1239 | 		}
 | 
|  |   1240 | 
 | 
|  |   1241 | 	new (&testThreadG) RTest(KNullDesC);
 | 
|  |   1242 | 	testThreadG.Start(KNullDesC);
 | 
|  |   1243 |    	//
 | 
|  |   1244 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |   1245 |    	TInt r = KErrNoMemory;
 | 
|  |   1246 |    	if (cleanup)
 | 
|  |   1247 |    		{
 | 
|  |   1248 | 		TRAP(r, DoThreadGL(aSemaphore));
 | 
|  |   1249 | 
 | 
|  |   1250 | 		// Check-point after closing the library handle
 | 
|  |   1251 | 		Checkpoint(aSemaphore);
 | 
|  |   1252 | 
 | 
|  |   1253 | 		testThreadG.Printf(_L("G: Returned %d, expected %d\n"), r, KErrGeneral);
 | 
|  |   1254 | 		testThreadG(r == KErrGeneral);
 | 
|  |   1255 | 
 | 
|  |   1256 | 		r = KErrNone;
 | 
|  |   1257 |    		}
 | 
|  |   1258 | 
 | 
|  |   1259 | 	delete cleanup;
 | 
|  |   1260 |    	//
 | 
|  |   1261 | 	testThreadG.End();
 | 
|  |   1262 | 	testThreadG.Close();
 | 
|  |   1263 | 
 | 
|  |   1264 | 	if (CheckKernelHeap)
 | 
|  |   1265 | 		{
 | 
|  |   1266 | 		User::After(100000);	// let supervisor run
 | 
|  |   1267 | 		__KHEAP_MARKEND;
 | 
|  |   1268 | 		}
 | 
|  |   1269 |    	__UHEAP_MARKEND;
 | 
|  |   1270 | 
 | 
|  |   1271 | 	return r;
 | 
|  |   1272 | 	}
 | 
|  |   1273 | 
 | 
|  |   1274 | TInt DoThreadGL(TAny* aSemaphore)
 | 
|  |   1275 | 	{
 | 
|  |   1276 |     testThreadG.Printf(_L("G: Loading DLL.\n"));
 | 
|  |   1277 | 	User::LeaveIfError(ThreadGLibraryHandle.Load(KLeavingDll));
 | 
|  |   1278 | 
 | 
|  |   1279 |     testThreadG.Printf(_L("G: Pushing cleanup item to synchronise after closing library handle.\n"));
 | 
|  |   1280 | 	CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |   1281 | 
 | 
|  |   1282 |     testThreadG.Printf(_L("G: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |   1283 | 	CleanupClosePushL(ThreadGLibraryHandle);
 | 
|  |   1284 | 
 | 
|  |   1285 | 	testThreadG.Printf(_L("G: Pushing cleanup item to recursively leave, doing nothing, before closing handle\n"));
 | 
|  |   1286 | 	CleanupStack::PushL(TCleanupItem(&TrapLeave, aSemaphore));
 | 
|  |   1287 | 
 | 
|  |   1288 |     testThreadG.Printf(_L("G: Looking up leaving function.\n"));
 | 
|  |   1289 | 	TLibraryFunction leaving = ThreadGLibraryHandle.Lookup(1);
 | 
|  |   1290 | 	User::LeaveIfNull((TAny*)leaving);
 | 
|  |   1291 | 
 | 
|  |   1292 | 	// Check-point whilst holding the open library handle
 | 
|  |   1293 | 	Checkpoint(aSemaphore);
 | 
|  |   1294 | 
 | 
|  |   1295 | 	testThreadG.Printf(_L("G: Calling leaving function.\n"));
 | 
|  |   1296 | 	(*leaving)();
 | 
|  |   1297 | 
 | 
|  |   1298 | 	testThreadG.Printf(_L("G: Cleaning up recursive leave operation.\n"));
 | 
|  |   1299 | 	CleanupStack::Pop(aSemaphore); // trap leave op
 | 
|  |   1300 | 
 | 
|  |   1301 | 	testThreadG.Printf(_L("G: Cleaning up DLL handle.\n"));
 | 
|  |   1302 | 	CleanupStack::PopAndDestroy(&ThreadGLibraryHandle);
 | 
|  |   1303 | 
 | 
|  |   1304 | 	return KErrNone;
 | 
|  |   1305 | 	}
 | 
|  |   1306 | 
 | 
|  |   1307 | TInt ThreadH(TAny* aSemaphore)
 | 
|  |   1308 | 	{
 | 
|  |   1309 | 	__UHEAP_MARK;
 | 
|  |   1310 | 	if (CheckKernelHeap)
 | 
|  |   1311 | 		{
 | 
|  |   1312 | 		__KHEAP_MARK;
 | 
|  |   1313 | 		}
 | 
|  |   1314 | 
 | 
|  |   1315 | 	new (&testThreadH) RTest(KNullDesC);
 | 
|  |   1316 | 	testThreadH.Start(KNullDesC);
 | 
|  |   1317 |    	//
 | 
|  |   1318 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |   1319 |    	TInt r = KErrNoMemory;
 | 
|  |   1320 |    	if (cleanup)
 | 
|  |   1321 |    		{
 | 
|  |   1322 | 		TRAP(r, DoThreadHL(aSemaphore));
 | 
|  |   1323 | 
 | 
|  |   1324 | 		// Check-point after closing the library handle
 | 
|  |   1325 | 		Checkpoint(aSemaphore);
 | 
|  |   1326 | 
 | 
|  |   1327 | 		testThreadH.Printf(_L("H: Returned %d, expected %d\n"), r, KErrGeneral);
 | 
|  |   1328 | 		testThreadH(r == KErrGeneral);
 | 
|  |   1329 | 
 | 
|  |   1330 | 		r = KErrNone;
 | 
|  |   1331 |    		}
 | 
|  |   1332 | 
 | 
|  |   1333 | 	delete cleanup;
 | 
|  |   1334 |    	//
 | 
|  |   1335 | 	testThreadH.End();
 | 
|  |   1336 | 	testThreadH.Close();
 | 
|  |   1337 | 
 | 
|  |   1338 | 	if (CheckKernelHeap)
 | 
|  |   1339 | 		{
 | 
|  |   1340 | 		User::After(100000);	// let supervisor run
 | 
|  |   1341 | 		__KHEAP_MARKEND;
 | 
|  |   1342 | 		}
 | 
|  |   1343 |    	__UHEAP_MARKEND;
 | 
|  |   1344 | 
 | 
|  |   1345 | 	return r;
 | 
|  |   1346 | 	}
 | 
|  |   1347 | 
 | 
|  |   1348 | TInt DoThreadHL(TAny* aSemaphore)
 | 
|  |   1349 | 	{
 | 
|  |   1350 |     testThreadH.Printf(_L("H: Loading DLL.\n"));
 | 
|  |   1351 | 	User::LeaveIfError(ThreadHLibraryHandle.Load(KLeavingDll));
 | 
|  |   1352 | 
 | 
|  |   1353 |     testThreadH.Printf(_L("H: Pushing cleanup item to synchronise after closing library handle.\n"));
 | 
|  |   1354 | 	CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |   1355 | 
 | 
|  |   1356 |     testThreadH.Printf(_L("H: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |   1357 | 	CleanupClosePushL(ThreadHLibraryHandle);
 | 
|  |   1358 | 
 | 
|  |   1359 | 	testThreadH.Printf(_L("H: Pushing cleanup item to synchronise during leave\n"));
 | 
|  |   1360 | 	CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |   1361 | 
 | 
|  |   1362 |     testThreadH.Printf(_L("H: Looking up leaving function.\n"));
 | 
|  |   1363 | 	TLibraryFunction leaving = ThreadHLibraryHandle.Lookup(1);
 | 
|  |   1364 | 	User::LeaveIfNull((TAny*)leaving);
 | 
|  |   1365 | 
 | 
|  |   1366 | 	// Check-point whilst holding the open library handle
 | 
|  |   1367 | 	Checkpoint(aSemaphore);
 | 
|  |   1368 | 
 | 
|  |   1369 | 	testThreadH.Printf(_L("H: Calling leaving function.\n"));
 | 
|  |   1370 | 	(*leaving)();
 | 
|  |   1371 | 
 | 
|  |   1372 | 	testThreadH.Printf(_L("H: Cleaning up leave pausing operation.\n"));
 | 
|  |   1373 | 	CleanupStack::Pop(aSemaphore); // pause leave op
 | 
|  |   1374 | 
 | 
|  |   1375 | 	testThreadH.Printf(_L("H: Cleaning up DLL handle.\n"));
 | 
|  |   1376 | 	CleanupStack::PopAndDestroy(&ThreadHLibraryHandle);
 | 
|  |   1377 | 
 | 
|  |   1378 | 	return KErrNone;
 | 
|  |   1379 | 	}
 | 
|  |   1380 | 
 | 
|  |   1381 | TInt ThreadI(TAny* aSemaphore)
 | 
|  |   1382 | 	{
 | 
|  |   1383 | 	__UHEAP_MARK;
 | 
|  |   1384 | 	if (CheckKernelHeap)
 | 
|  |   1385 | 		{
 | 
|  |   1386 | 		__KHEAP_MARK;
 | 
|  |   1387 | 		}
 | 
|  |   1388 | 
 | 
|  |   1389 | 	new (&testThreadI) RTest(KNullDesC);
 | 
|  |   1390 | 	testThreadI.Start(KNullDesC);
 | 
|  |   1391 |    	//
 | 
|  |   1392 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |   1393 |    	TInt r = KErrNoMemory;
 | 
|  |   1394 |    	if (cleanup)
 | 
|  |   1395 |    		{
 | 
|  |   1396 | 		TRAP(r, DoThreadIL(aSemaphore));
 | 
|  |   1397 | 
 | 
|  |   1398 | 		// Check-point after closing the library handle
 | 
|  |   1399 | 		Checkpoint(aSemaphore);
 | 
|  |   1400 | 
 | 
|  |   1401 | 		testThreadI.Printf(_L("I: Returned %d, expected %d\n"), r, KErrGeneral);
 | 
|  |   1402 | 		testThreadI(r == KErrGeneral);
 | 
|  |   1403 | 
 | 
|  |   1404 | 		r = KErrNone;
 | 
|  |   1405 |    		}
 | 
|  |   1406 | 
 | 
|  |   1407 | 	delete cleanup;
 | 
|  |   1408 |    	//
 | 
|  |   1409 | 	testThreadI.End();
 | 
|  |   1410 | 	testThreadI.Close();
 | 
|  |   1411 | 
 | 
|  |   1412 | 	if (CheckKernelHeap)
 | 
|  |   1413 | 		{
 | 
|  |   1414 | 		User::After(100000);	// let supervisor run
 | 
|  |   1415 | 		__KHEAP_MARKEND;
 | 
|  |   1416 | 		}
 | 
|  |   1417 |    	__UHEAP_MARKEND;
 | 
|  |   1418 | 
 | 
|  |   1419 | 	return r;
 | 
|  |   1420 | 	}
 | 
|  |   1421 | 
 | 
|  |   1422 | TInt DoThreadIL(TAny* aSemaphore)
 | 
|  |   1423 | 	{
 | 
|  |   1424 |     testThreadI.Printf(_L("I: Loading DLL.\n"));
 | 
|  |   1425 | 	User::LeaveIfError(ThreadILibraryHandle.Load(KLeavingDll));
 | 
|  |   1426 | 
 | 
|  |   1427 | 	testThreadI.Printf(_L("I: Pushing cleanup item to recursively leave, doing nothing, after closing handle\n"));
 | 
|  |   1428 | 	CleanupStack::PushL(TCleanupItem(&TrapLeave, aSemaphore));
 | 
|  |   1429 | 
 | 
|  |   1430 |     testThreadI.Printf(_L("I: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |   1431 | 	CleanupClosePushL(ThreadILibraryHandle);
 | 
|  |   1432 | 
 | 
|  |   1433 | 	testThreadI.Printf(_L("I: Pushing cleanup item to synchronise during leave\n"));
 | 
|  |   1434 | 	CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |   1435 | 
 | 
|  |   1436 |     testThreadI.Printf(_L("I: Looking up leaving function.\n"));
 | 
|  |   1437 | 	TLibraryFunction leaving = ThreadILibraryHandle.Lookup(1);
 | 
|  |   1438 | 	User::LeaveIfNull((TAny*)leaving);
 | 
|  |   1439 | 
 | 
|  |   1440 | 	// Check-point whilst holding the open library handle
 | 
|  |   1441 | 	Checkpoint(aSemaphore);
 | 
|  |   1442 | 
 | 
|  |   1443 | 	testThreadI.Printf(_L("I: Calling leaving function.\n"));
 | 
|  |   1444 | 	(*leaving)();
 | 
|  |   1445 | 
 | 
|  |   1446 | 	testThreadI.Printf(_L("I: Cleaning up leave pausing operation.\n"));
 | 
|  |   1447 | 	CleanupStack::Pop(aSemaphore); // pause leave op
 | 
|  |   1448 | 
 | 
|  |   1449 | 	testThreadI.Printf(_L("I: Cleaning up DLL handle.\n"));
 | 
|  |   1450 | 	CleanupStack::PopAndDestroy(&ThreadILibraryHandle);
 | 
|  |   1451 | 
 | 
|  |   1452 | 	testThreadI.Printf(_L("I: Cleaning up recursive leave operation.\n"));
 | 
|  |   1453 | 	CleanupStack::Pop(); // trap leave op
 | 
|  |   1454 | 
 | 
|  |   1455 | 	return KErrNone;
 | 
|  |   1456 | 	}
 | 
|  |   1457 | 
 | 
|  |   1458 | TInt ThreadJ(TAny* aSemaphore)
 | 
|  |   1459 | 	{
 | 
|  |   1460 | 	__UHEAP_MARK;
 | 
|  |   1461 | 	if (CheckKernelHeap)
 | 
|  |   1462 | 		{
 | 
|  |   1463 | 		__KHEAP_MARK;
 | 
|  |   1464 | 		}
 | 
|  |   1465 | 
 | 
|  |   1466 | 	new (&testThreadJ) RTest(KNullDesC);
 | 
|  |   1467 | 	testThreadJ.Start(KNullDesC);
 | 
|  |   1468 |    	//
 | 
|  |   1469 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |   1470 |    	TInt r = KErrNoMemory;
 | 
|  |   1471 |    	if (cleanup)
 | 
|  |   1472 |    		{
 | 
|  |   1473 | 		TRAP(r, DoThreadJL(aSemaphore));
 | 
|  |   1474 | 
 | 
|  |   1475 | 		// Check-point after closing the library handle
 | 
|  |   1476 | 		Checkpoint(aSemaphore);
 | 
|  |   1477 | 
 | 
|  |   1478 | 		testThreadJ.Printf(_L("J: Returned %d, expected %d\n"), r, KErrGeneral);
 | 
|  |   1479 | 		testThreadJ(r == KErrGeneral);
 | 
|  |   1480 | 
 | 
|  |   1481 | 		r = KErrNone;
 | 
|  |   1482 |    		}
 | 
|  |   1483 | 
 | 
|  |   1484 | 	delete cleanup;
 | 
|  |   1485 |    	//
 | 
|  |   1486 | 	testThreadJ.End();
 | 
|  |   1487 | 	testThreadJ.Close();
 | 
|  |   1488 | 
 | 
|  |   1489 | 	if (CheckKernelHeap)
 | 
|  |   1490 | 		{
 | 
|  |   1491 | 		User::After(100000);	// let supervisor run
 | 
|  |   1492 | 		__KHEAP_MARKEND;
 | 
|  |   1493 | 		}
 | 
|  |   1494 |    	__UHEAP_MARKEND;
 | 
|  |   1495 | 
 | 
|  |   1496 | 	return r;
 | 
|  |   1497 | 	}
 | 
|  |   1498 | 
 | 
|  |   1499 | TInt DoThreadJL(TAny* aSemaphore)
 | 
|  |   1500 | 	{
 | 
|  |   1501 |     testThreadJ.Printf(_L("J: Loading DLL.\n"));
 | 
|  |   1502 | 	User::LeaveIfError(ThreadJLibraryHandle.Load(KLeavingDll));
 | 
|  |   1503 | 
 | 
|  |   1504 | 	testThreadJ.Printf(_L("J: Pushing cleanup item to recursively leave and panic, after closing handle\n"));
 | 
|  |   1505 | 	CleanupStack::PushL(TCleanupItem(&TrapLeaveAndDie, aSemaphore));
 | 
|  |   1506 | 
 | 
|  |   1507 |     testThreadJ.Printf(_L("J: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |   1508 | 	CleanupClosePushL(ThreadJLibraryHandle);
 | 
|  |   1509 | 
 | 
|  |   1510 | 	testThreadJ.Printf(_L("J: Pushing cleanup item to synchronise during leave\n"));
 | 
|  |   1511 | 	CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |   1512 | 
 | 
|  |   1513 |     testThreadJ.Printf(_L("J: Looking up leaving function.\n"));
 | 
|  |   1514 | 	TLibraryFunction leaving = ThreadJLibraryHandle.Lookup(1);
 | 
|  |   1515 | 	User::LeaveIfNull((TAny*)leaving);
 | 
|  |   1516 | 
 | 
|  |   1517 | 	// Check-point whilst holding the open library handle
 | 
|  |   1518 | 	Checkpoint(aSemaphore);
 | 
|  |   1519 | 
 | 
|  |   1520 | 	testThreadJ.Printf(_L("J: Calling leaving function.\n"));
 | 
|  |   1521 | 	(*leaving)();
 | 
|  |   1522 | 
 | 
|  |   1523 | 	testThreadJ.Printf(_L("J: Cleaning up leave pausing operation.\n"));
 | 
|  |   1524 | 	CleanupStack::Pop(aSemaphore); // pause leave op
 | 
|  |   1525 | 
 | 
|  |   1526 | 	testThreadJ.Printf(_L("J: Cleaning up DLL handle.\n"));
 | 
|  |   1527 | 	CleanupStack::PopAndDestroy(&ThreadJLibraryHandle);
 | 
|  |   1528 | 
 | 
|  |   1529 | 	testThreadJ.Printf(_L("J: Cleaning up recursive leave operation.\n"));
 | 
|  |   1530 | 	CleanupStack::Pop(); // leave and die op
 | 
|  |   1531 | 
 | 
|  |   1532 | 	return KErrNone;
 | 
|  |   1533 | 	}
 | 
|  |   1534 | 
 | 
|  |   1535 | TInt ThreadK(TAny* aSemaphore)
 | 
|  |   1536 | 	{
 | 
|  |   1537 | 	__UHEAP_MARK;
 | 
|  |   1538 | 	if (CheckKernelHeap)
 | 
|  |   1539 | 		{
 | 
|  |   1540 | 		__KHEAP_MARK;
 | 
|  |   1541 | 		}
 | 
|  |   1542 | 
 | 
|  |   1543 | 	new (&testThreadK) RTest(KNullDesC);
 | 
|  |   1544 | 	testThreadK.Start(KNullDesC);
 | 
|  |   1545 |    	//
 | 
|  |   1546 |    	CTrapCleanup* cleanup = CTrapCleanup::New();
 | 
|  |   1547 |    	TInt r = KErrNoMemory;
 | 
|  |   1548 |    	if (cleanup)
 | 
|  |   1549 |    		{
 | 
|  |   1550 | 		TRAP(r, DoThreadKL(aSemaphore));
 | 
|  |   1551 | 
 | 
|  |   1552 | 		// Check-point after closing the library handle
 | 
|  |   1553 | 		Checkpoint(aSemaphore);
 | 
|  |   1554 | 
 | 
|  |   1555 | 		testThreadK.Printf(_L("K: Returned %d, expected %d\n"), r, KErrGeneral);
 | 
|  |   1556 | 		testThreadK(r == KErrGeneral);
 | 
|  |   1557 | 
 | 
|  |   1558 | 		r = KErrNone;
 | 
|  |   1559 |    		}
 | 
|  |   1560 | 
 | 
|  |   1561 | 	delete cleanup;
 | 
|  |   1562 |    	//
 | 
|  |   1563 | 	testThreadK.End();
 | 
|  |   1564 | 	testThreadK.Close();
 | 
|  |   1565 | 
 | 
|  |   1566 | 	if (CheckKernelHeap)
 | 
|  |   1567 | 		{
 | 
|  |   1568 | 		User::After(100000);	// let supervisor run
 | 
|  |   1569 | 		__KHEAP_MARKEND;
 | 
|  |   1570 | 		}
 | 
|  |   1571 |    	__UHEAP_MARKEND;
 | 
|  |   1572 | 
 | 
|  |   1573 | 	return r;
 | 
|  |   1574 | 	}
 | 
|  |   1575 | 
 | 
|  |   1576 | TInt DoThreadKL(TAny* aSemaphore)
 | 
|  |   1577 | 	{
 | 
|  |   1578 |     testThreadK.Printf(_L("K: Loading DLL.\n"));
 | 
|  |   1579 | 	User::LeaveIfError(ThreadKLibraryHandle.Load(KLeavingDll));
 | 
|  |   1580 | 
 | 
|  |   1581 | 	testThreadK.Printf(_L("K: Pushing cleanup item to panic, after closing handle\n"));
 | 
|  |   1582 | 	CleanupStack::PushL(TCleanupItem(&DieDieDie, aSemaphore));
 | 
|  |   1583 | 
 | 
|  |   1584 |     testThreadK.Printf(_L("K: Pushing handle to dynamically loaded DLL onto the cleanup stack.\n"));
 | 
|  |   1585 | 	CleanupClosePushL(ThreadKLibraryHandle);
 | 
|  |   1586 | 
 | 
|  |   1587 | 	testThreadK.Printf(_L("K: Pushing cleanup item to synchronise during leave\n"));
 | 
|  |   1588 | 	CleanupStack::PushL(TCleanupItem(&PauseLeaving, aSemaphore));
 | 
|  |   1589 | 
 | 
|  |   1590 |     testThreadK.Printf(_L("K: Looking up leaving function.\n"));
 | 
|  |   1591 | 	TLibraryFunction leaving = ThreadKLibraryHandle.Lookup(1);
 | 
|  |   1592 | 	User::LeaveIfNull((TAny*)leaving);
 | 
|  |   1593 | 
 | 
|  |   1594 | 	// Check-point whilst holding the open library handle
 | 
|  |   1595 | 	Checkpoint(aSemaphore);
 | 
|  |   1596 | 
 | 
|  |   1597 | 	testThreadK.Printf(_L("K: Calling leaving function.\n"));
 | 
|  |   1598 | 	(*leaving)();
 | 
|  |   1599 | 
 | 
|  |   1600 | 	testThreadK.Printf(_L("K: Cleaning up leave pausing operation.\n"));
 | 
|  |   1601 | 	CleanupStack::Pop(aSemaphore); // pause leave op
 | 
|  |   1602 | 
 | 
|  |   1603 | 	testThreadK.Printf(_L("K: Cleaning up DLL handle.\n"));
 | 
|  |   1604 | 	CleanupStack::PopAndDestroy(&ThreadKLibraryHandle);
 | 
|  |   1605 | 
 | 
|  |   1606 | 	testThreadK.Printf(_L("K: Cleaning up panic operation.\n"));
 | 
|  |   1607 | 	CleanupStack::Pop(); // die op
 | 
|  |   1608 | 
 | 
|  |   1609 | 	return KErrNone;
 | 
|  |   1610 | 	}
 |