Introduction to Open C |
The previous section gave an overview of the amount of work that, needs to be done when programming on Symbian and UNIX environments. This section deals with the more general scenario that will be used frequently in a typical operating system.
Assuming that the user is new to the UNIX environment, writing a simple program that displays a string on a console does not take more than a day’s effort. Its simply 4-5 lines of code as shown below:
#include <stdio.h> int main(int argc, char** argv) { printf(“Hello world\n”); return 0; }
The code contains a main() , one printf statement, and one #include< >. By convention UNIX applications will map descriptor 0 with standard input (STDIN), descriptor 1 with standard output (STDOUT), and descriptor 2 with standard error (STDERROR) of the process. So, there is no need to create/open standard input/output/error for a process before using them. Notice also that, main( ) takes two arguments. These arguments will be read from the command prompt before invoking main( ) and will be passed to main. These arguments are not mandatory; they can be ignored. No additional work is needed for having these arguments.
If the user attempts to display a string on a console using Symbian OS, a lot of effort goes into collecting the materials, and for coding. The first time the developer may find the code a little complex to understand.
#include <e32std.h> #include <e32cons.h> int HelloWorldL() { CConsoleBase* console = Console::NewL(_L("Hello World"), TSize( KConsFullScreen, KConsFullScreen)); console->Printf(_L(“Hello World\n”)); delete console; return 0; } TInt E32Main() { __UHEAP_MARK; CTrapCleanup* cleanup = CTrapCleanup::New(); TInt retVal = KErrNone; if (cleanup) { TRAP(retVal, HelloWorldL()); __ASSERT_ALWAYS(!retVal, User::Panic(_L("Hello-World PANIC"), retVal)); //Destroy cleanup stack delete cleanup; } __UHEAP_MARKEND; return retVal; }
The code above has some header inclusions, two functions, and some complicated codes. In Symbian OS, the entry function for an EXE is E32Main (not main as in UNIX). The following actions are done to print a message on a console:
In Symbian OS, a console is not created by default; it has to be created explicitly. Symbian OS does not give a rich set of APIs needed to perform I/O operations on a console. The I/O operations that can be performed on a console in Symbian OS are limited to :
In case the user is interested in command line arguments, it takes some more additional work. The user must read explicitly from the command prompt by using the APIs provided by Symbian OS.
This section compares the ease of programming in the Open C environment against programming on Symbian OS.
/* File I/O in UNIX environment */ FILE* fp = fopen(“file.txt”, w); if (fp) { fprintf(fp, “Write some data”); fclose(); }
//File I/O in Symbian OS RFs fSession; User::LeaveIfError(fSession.Connect()); RFile file; ret = file.Open(fSession, _L(“file.txt”), EFileWrite); if (ret) { ret = file.Create(fSession, _L(“file.txt”), EFileWrite); } if (!ret) { file.Write(_L8(“Write some data”)); } file.Close(); fSession.Close();
The example above shows the complexity of the code in Symbian OS for doing simple file I/O. In addition to coding complexity, the user must include and link with :
This section deals with thread creation in the UNIX and Symbian OS environments.
/* Thread creation in UNIX environment*/ void* ThreadEntryPoint( void* aParam ); … int exitReason = 0; int retVal = 0; pthread_t threadID = 0; pthread_attr_t threadAttr; pthread_attr_init( &threadAttr ); retVal = pthread_create( &threadID, &threadAttr, ThreadEntryPoint, (void*)NULL ); if(retVal == 0) { retVal = pthread_join(threadID1, (void**)&exitReason ); }
// Thread creation in Symbian OS TInt ThreadEntryPoint( TAny* aData ); … RThread thread; thread.Create(_L("MY-THREAD"), ThreadEntryPoint, 4096, NULL, NULL); TRequestStatus stat; thread.Logon(stat); //Start executing the thread. thread.Resume(); //Wait for the thread to die. User::WaitForRequest(stat); exitReason = thread.ExitReason(); thread.Close();
In addition the difference in the code for creating threads using pthread_create and RThread::Create, a thread created using RThread::Create also expects the ThreadEntryPoint function to create its own vcleanup stack and a top-level TRAP, if required. Else, the thread may crash or panic. But this does not have to be done for a thread created using pthread_create. pthread_create does that for the developer.