1 /////////////////////////////////////////////////////////////////////////////
2 // Name: samples/console/console.cpp
3 // Purpose: a sample console (as opposed to GUI) progam using wxWindows
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
22 #include <wx/string.h>
26 // ----------------------------------------------------------------------------
27 // conditional compilation
28 // ----------------------------------------------------------------------------
34 //#define TEST_STRINGS
37 //#define TEST_LONGLONG
39 // ============================================================================
41 // ============================================================================
43 // ----------------------------------------------------------------------------
45 // ----------------------------------------------------------------------------
49 #include <wx/longlong.h>
52 static void TestSpeed()
54 static const long max
= 100000000;
61 for ( n
= 0; n
< max
; n
++ )
66 printf("Summing longs took %ld milliseconds.\n", sw
.Time());
73 for ( n
= 0; n
< max
; n
++ )
78 printf("Summing __int64s took %ld milliseconds.\n", sw
.Time());
85 for ( n
= 0; n
< max
; n
++ )
90 printf("Summing wxLongLongs took %ld milliseconds.\n", sw
.Time());
94 static void TestDivision()
96 #define MAKE_LL(x1, x2, x3, x4) wxLongLong((x1 << 16) | x2, (x3 << 16) | x3)
98 // seed pseudo random generator
99 //srand((unsigned)time(NULL));
102 for ( size_t n
= 0; n
< 10000; n
++ )
104 // get a random wxLongLong (shifting by 12 the MSB ensures that the
105 // multiplication will not overflow)
106 wxLongLong ll
= MAKE_LL((rand() >> 12), rand(), rand(), rand());
108 wxASSERT( (ll
* 1000l)/1000l == ll
);
113 printf("\n*** Tested %u divisions/multiplications: ok\n", nTested
);
118 #endif // TEST_LONGLONG
120 // ----------------------------------------------------------------------------
122 // ----------------------------------------------------------------------------
126 #include <wx/datetime.h>
128 // this test miscellaneous static wxDateTime functions
129 static void TestTimeStatic()
131 puts("\n*** wxDateTime static methods test ***");
133 // some info about the current date
134 int year
= wxDateTime::GetCurrentYear();
135 printf("Current year %d is %sa leap one and has %d days.\n",
137 wxDateTime::IsLeapYear(year
) ? "" : "not ",
138 wxDateTime::GetNumberOfDays(year
));
140 wxDateTime::Month month
= wxDateTime::GetCurrentMonth();
141 printf("Current month is '%s' ('%s') and it has %d days\n",
142 wxDateTime::GetMonthName(month
, TRUE
).c_str(),
143 wxDateTime::GetMonthName(month
).c_str(),
144 wxDateTime::GetNumberOfDays(month
));
147 static const size_t nYears
= 5;
148 static const size_t years
[2][nYears
] =
150 // first line: the years to test
151 { 1990, 1976, 2000, 2030, 1984, },
153 // second line: TRUE if leap, FALSE otherwise
154 { FALSE
, TRUE
, TRUE
, FALSE
, TRUE
}
157 for ( size_t n
= 0; n
< nYears
; n
++ )
159 int year
= years
[0][n
];
160 bool should
= years
[1][n
] != 0;
162 printf("Year %d is %sa leap year (should be: %s)\n",
164 wxDateTime::IsLeapYear(year
) ? "" : "not ",
165 should
? "yes" : "no");
167 wxASSERT( should
== wxDateTime::IsLeapYear(year
) );
171 // test constructing wxDateTime objects
172 static void TestTimeSet()
174 puts("\n*** wxDateTime construction test ***");
176 printf("Current time:\t%s\n", wxDateTime::Now().Format().c_str());
177 printf("Unix epoch:\t%s\n", wxDateTime((time_t)0).Format().c_str());
178 printf("Today noon:\t%s\n", wxDateTime(12, 0).Format().c_str());
179 printf("May 29, 1976:\t%s\n", wxDateTime(29, wxDateTime::May
, 1976).Format().c_str());
182 // test time zones stuff
183 static void TestTimeZones()
185 puts("\n*** wxDateTime timezone test ***");
187 wxDateTime now
= wxDateTime::Now();
189 printf("Current GMT time:\t%s\n", now
.ToGMT().Format().c_str());
190 //TODO printf("Unix epoch (GMT):\t%s\n", wxDateTime((time_t)0).MakeGMT().Format().c_str());
191 printf("Current time in Paris:\t%s\n", now
.ToTimezone(wxDateTime::CET
).Format().c_str());
192 printf(" Moscow:\t%s\n", now
.ToTimezone(wxDateTime::MSK
).Format().c_str());
193 printf(" New York:\t%s\n", now
.ToTimezone(wxDateTime::EST
).Format().c_str());
196 // test some minimal support for the dates outside the standard range
197 static void TestTimeRange()
199 puts("\n*** wxDateTime out-of-standard-range dates test ***");
201 printf("JDN 0: \t%s\n",
202 wxDateTime(0.0).Format().c_str());
203 printf("Jan 1, 1AD:\t%s\n",
204 wxDateTime(1, wxDateTime::Jan
, 1).Format().c_str());
205 printf("May 29, 2099:\t%s\n",
206 wxDateTime(29, wxDateTime::May
, 2099).Format().c_str());
209 // test conversions to JDN &c
210 static void TestTimeJulian()
212 puts("\n*** wxDateTime to JDN test ***");
214 printf("JDN of current time:\t%f\n", wxDateTime::Now().GetJulianDayNumber());
215 printf("JDN of Jan 1, 1900: \t%f\n",
216 wxDateTime(1, wxDateTime::Jan
, 1900).GetJulianDayNumber());
217 printf("JDN of Jan 1, 1BC: \t%f\n",
218 wxDateTime(1, wxDateTime::Jan
, 0).GetJulianDayNumber());
219 printf("JDN 0: \t%f\n",
220 wxDateTime(24, wxDateTime::Nov
, -4713, 12, 0, 0).GetJulianDayNumber());
225 // ----------------------------------------------------------------------------
227 // ----------------------------------------------------------------------------
231 #include <wx/thread.h>
233 static size_t gs_counter
= (size_t)-1;
234 static wxCriticalSection gs_critsect
;
235 static wxCondition gs_cond
;
237 class MyJoinableThread
: public wxThread
240 MyJoinableThread(size_t n
) : wxThread(wxTHREAD_JOINABLE
)
241 { m_n
= n
; Create(); }
243 // thread execution starts here
244 virtual ExitCode
Entry();
250 wxThread::ExitCode
MyJoinableThread::Entry()
252 unsigned long res
= 1;
253 for ( size_t n
= 1; n
< m_n
; n
++ )
257 // it's a loooong calculation :-)
261 return (ExitCode
)res
;
264 class MyDetachedThread
: public wxThread
267 MyDetachedThread(size_t n
, char ch
)
276 // thread execution starts here
277 virtual ExitCode
Entry();
280 virtual void OnExit();
283 size_t m_n
; // number of characters to write
284 char m_ch
; // character to write
286 bool m_cancelled
; // FALSE if we exit normally
289 wxThread::ExitCode
MyDetachedThread::Entry()
292 wxCriticalSectionLocker
lock(gs_critsect
);
293 if ( gs_counter
== (size_t)-1 )
299 for ( size_t n
= 0; n
< m_n
; n
++ )
311 wxThread::Sleep(100);
317 void MyDetachedThread::OnExit()
319 wxLogTrace("thread", "Thread %ld is in OnExit", GetId());
321 wxCriticalSectionLocker
lock(gs_critsect
);
322 if ( !--gs_counter
&& !m_cancelled
)
326 void TestDetachedThreads()
328 puts("\n*** Testing detached threads ***");
330 static const size_t nThreads
= 3;
331 MyDetachedThread
*threads
[nThreads
];
333 for ( n
= 0; n
< nThreads
; n
++ )
335 threads
[n
] = new MyDetachedThread(10, 'A' + n
);
338 threads
[0]->SetPriority(WXTHREAD_MIN_PRIORITY
);
339 threads
[1]->SetPriority(WXTHREAD_MAX_PRIORITY
);
341 for ( n
= 0; n
< nThreads
; n
++ )
346 // wait until all threads terminate
352 void TestJoinableThreads()
354 puts("\n*** Testing a joinable thread (a loooong calculation...) ***");
356 // calc 10! in the background
357 MyJoinableThread
thread(10);
360 printf("\nThread terminated with exit code %lu.\n",
361 (unsigned long)thread
.Wait());
364 void TestThreadSuspend()
366 puts("\n*** Testing thread suspend/resume functions ***");
368 MyDetachedThread
*thread
= new MyDetachedThread(15, 'X');
372 // this is for this demo only, in a real life program we'd use another
373 // condition variable which would be signaled from wxThread::Entry() to
374 // tell us that the thread really started running - but here just wait a
375 // bit and hope that it will be enough (the problem is, of course, that
376 // the thread might still not run when we call Pause() which will result
378 wxThread::Sleep(300);
380 for ( size_t n
= 0; n
< 3; n
++ )
384 puts("\nThread suspended");
387 // don't sleep but resume immediately the first time
388 wxThread::Sleep(300);
390 puts("Going to resume the thread");
395 puts("Waiting until it terminates now");
397 // wait until the thread terminates
403 void TestThreadDelete()
405 // As above, using Sleep() is only for testing here - we must use some
406 // synchronisation object instead to ensure that the thread is still
407 // running when we delete it - deleting a detached thread which already
408 // terminated will lead to a crash!
410 puts("\n*** Testing thread delete function ***");
412 MyDetachedThread
*thread0
= new MyDetachedThread(30, 'W');
416 puts("\nDeleted a thread which didn't start to run yet.");
418 MyDetachedThread
*thread1
= new MyDetachedThread(30, 'Y');
422 wxThread::Sleep(300);
426 puts("\nDeleted a running thread.");
428 MyDetachedThread
*thread2
= new MyDetachedThread(30, 'Z');
432 wxThread::Sleep(300);
438 puts("\nDeleted a sleeping thread.");
440 MyJoinableThread
thread3(20);
445 puts("\nDeleted a joinable thread.");
447 MyJoinableThread
thread4(2);
450 wxThread::Sleep(300);
454 puts("\nDeleted a joinable thread which already terminated.");
459 #endif // TEST_THREADS
461 // ----------------------------------------------------------------------------
463 // ----------------------------------------------------------------------------
467 void PrintArray(const char* name
, const wxArrayString
& array
)
469 printf("Dump of the array '%s'\n", name
);
471 size_t nCount
= array
.GetCount();
472 for ( size_t n
= 0; n
< nCount
; n
++ )
474 printf("\t%s[%u] = '%s'\n", name
, n
, array
[n
].c_str());
478 #endif // TEST_ARRAYS
480 // ----------------------------------------------------------------------------
482 // ----------------------------------------------------------------------------
486 #include "wx/timer.h"
498 for (int i
= 0; i
< 1000000; ++i
)
502 c
= "! How'ya doin'?";
505 c
= "Hello world! What's up?";
510 printf ("TestString elapsed time: %ld\n", sw
.Time());
521 for (int i
= 0; i
< 1000000; ++i
)
524 strcpy (b
, " world");
525 strcpy (c
, "! How'ya doin'?");
528 strcpy (c
, "Hello world! What's up?");
529 if (strcmp (c
, a
) == 0)
533 printf ("TestPChar elapsed time: %ld\n", sw
.Time());
536 #endif // TEST_STRINGS
538 // ----------------------------------------------------------------------------
540 // ----------------------------------------------------------------------------
542 int main(int argc
, char **argv
)
544 if ( !wxInitialize() )
546 fprintf(stderr
, "Failed to initialize the wxWindows library, aborting.");
552 #endif // TEST_STRINGS
563 puts("*** Initially:");
565 PrintArray("a1", a1
);
567 wxArrayString
a2(a1
);
568 PrintArray("a2", a2
);
570 wxSortedArrayString
a3(a1
);
571 PrintArray("a3", a3
);
573 puts("*** After deleting a string from a1");
576 PrintArray("a1", a1
);
577 PrintArray("a2", a2
);
578 PrintArray("a3", a3
);
580 puts("*** After reassigning a1 to a2 and a3");
582 PrintArray("a2", a2
);
583 PrintArray("a3", a3
);
584 #endif // TEST_ARRAYS
588 for ( size_t n
= 0; n
< 8000; n
++ )
590 s
<< (char)('A' + (n
% 26));
594 msg
.Printf("A very very long message: '%s', the end!\n", s
.c_str());
596 // this one shouldn't be truncated
599 // but this one will because log functions use fixed size buffer
600 // (note that it doesn't need '\n' at the end neither - will be added
602 wxLogMessage("A very very long message 2: '%s', the end!", s
.c_str());
606 if ( argc
> 1 && argv
[1][0] == 't' )
607 wxLog::AddTraceMask("thread");
610 TestDetachedThreads();
612 TestJoinableThreads();
618 #endif // TEST_THREADS
625 #endif // TEST_LONGLONG