00001 /* 00002 Copyright (c) 2000-2010 Nokia Corporation and/or its subsidiary(-ies). All rights reserved. 00003 00004 Redistribution and use in source and binary forms, with or without 00005 modification, are permitted provided that the following conditions are met: 00006 00007 * Redistributions of source code must retain the above copyright notice, this 00008 list of conditions and the following disclaimer. 00009 * Redistributions in binary form must reproduce the above copyright notice, 00010 this list of conditions and the following disclaimer in the documentation 00011 and/or other materials provided with the distribution. 00012 * Neither the name of Nokia Corporation nor the names of its contributors 00013 may be used to endorse or promote products derived from this software 00014 without specific prior written permission. 00015 00016 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 00017 AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 00018 IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 00019 DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE 00020 FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 00021 DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 00022 SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 00023 CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 00024 OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 00025 OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 00026 00027 Description: 00028 Example shows use of the cleanup stack fucntions PushL() 00029 and PopAndDestroy(). 00030 NOTE: the structure of this example is different to standard E32 examples 00031 */ 00032 00033 00034 00035 00036 #include <e32cons.h> 00037 00038 // All messages written to this 00039 LOCAL_D CConsoleBase* console; 00040 00041 // Flag which determines whether the doSomething() member function 00042 // of the CExample class should leave when called. 00043 LOCAL_D TBool leaveFlag = ETrue; 00044 00045 // Parameter for __UHEAP_SETFAIL 00046 // Allocation guaranteed to fail at this number of allocation attempts; 00047 // i.e. if set to n, allocation fails on the nth attempt. 00048 // NB only used in debug mode 00049 #ifdef _DEBUG 00050 LOCAL_D TInt allocFailNumber = 1; 00051 #endif 00052 00053 // Function prototypes 00054 LOCAL_C void doExampleL(); 00055 LOCAL_C void callExampleL(); 00056 00057 00058 00060 // 00061 // -----> CExample (definition) 00062 // 00063 // The class is used by the example code 00064 // 00066 class CExample : public CBase 00067 { 00068 public : 00069 void DoSomethingL(); 00070 public : 00071 TInt iInt; 00072 }; 00073 00074 00076 // 00077 // -----> CExample (implementation) 00078 // 00080 void CExample::DoSomethingL() 00081 { 00082 // Leave if the global flag is set 00083 if (leaveFlag) 00084 { 00085 _LIT(KMsgLeaving,"DoSomethingL leaving.\n"); 00086 console->Printf(KMsgLeaving); 00087 User::Leave(KErrGeneral); 00088 } 00089 _LIT(KFormat1,"Value of iInt is %d.\n"); 00090 console->Printf(KFormat1,iInt); 00091 } 00092 00094 // 00095 // Main function called by E32 00096 // 00098 GLDEF_C TInt E32Main() 00099 { 00100 // Get cleanup stack 00101 CTrapCleanup* cleanup=CTrapCleanup::New(); 00102 00103 // Some more initialization, then do the example 00104 TRAPD(error,callExampleL()); 00105 00106 // callExampleL() should never leave. 00107 _LIT(KMsgPanicEpoc32ex,"EPOC32EX"); 00108 __ASSERT_ALWAYS(!error,User::Panic(KMsgPanicEpoc32ex,error)); 00109 00110 // destroy the cleanup stack 00111 delete cleanup; 00112 00113 // return 00114 return 0; 00115 } 00116 00117 00119 // 00120 // 00121 // 00123 LOCAL_C void callExampleL() 00124 { 00125 // Initialize and call the example code under cleanup stack. 00126 _LIT(KMsgExampleCode,"Symbian platform Example Code"); 00127 console = Console::NewL(KMsgExampleCode,TSize(KConsFullScreen,KConsFullScreen)); 00128 // Put console onto the cleanup stack. 00129 CleanupStack::PushL(console); 00130 00131 // Perform the example function under the protection of a 00132 // TRAP harness. 00133 TRAPD(error,doExampleL()); 00134 _LIT(KMsgOK,"ok"); 00135 _LIT(KFormat2,"failed: leave code = %d"); 00136 if (error) 00137 console->Printf(KFormat2,error); 00138 else 00139 console->Printf(KMsgOK); 00140 00141 // Continue 00142 _LIT(KMsgPressAnyKey," [press any key]"); 00143 console->Printf(KMsgPressAnyKey); 00144 console->Getch(); 00145 00146 // Remove the console object from the cleanupstack 00147 // and destroy it. 00148 CleanupStack::PopAndDestroy(); 00149 } 00150 00151 00153 // 00154 // Do the example 00155 // 00156 // Surround something that might leave with a PushL() and a PopAndDestroy() 00157 // Compare this with the example: EUCLNA5 00158 // 00160 void doExampleL() 00161 { 00162 // Memory alloc fails on the 'allocFailNumber' attempt. 00163 __UHEAP_SETFAIL(RHeap::EDeterministic,allocFailNumber); 00164 // Allocate - leave if allocation fails. 00165 CExample* myExample = new (ELeave) CExample; 00166 // Do something that cannot leave (no protection needed) 00167 myExample->iInt = 5; 00168 // Do something that can leave. 00169 // Use the cleanup stack - put the pointer to the CExample 00170 // object on the cleanup stack 00171 CleanupStack::PushL(myExample); 00172 // This is something that might leave 00173 myExample->DoSomethingL(); 00174 // It didn't leave, so pop the pointer off the stack and 00175 // delete the CExample object in one operation 00176 // (Equivalent to: CleanupStack::Pop(); 00177 // delete myExample; 00178 // in example UsingPushLAndPop) 00179 CleanupStack::PopAndDestroy(); 00180 } 00181 00182 00183 00184 00185 00186 00187 00188