Update contrib.
1 // Copyright (c) 2003-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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // e32\personality\example\personality.cpp
15 // Example RTOS personality.
19 #include "personality_int.h"
21 /******************************************************************************
22 * Memory pool management
23 ******************************************************************************/
25 // Create a single memory pool consisting of a specified number of equal sized blocks.
26 TInt PMemPool::Create(const poolinfo* aInfo)
28 iBlockSize = aInfo->block_size;
29 TUint bsize = iBlockSize + sizeof(PMemPool*);
30 TInt n = (TInt)aInfo->block_count;
31 __KTRACE_OPT(KBOOT, Kern::Printf("PMemPool::Create %08x iBlockSize=%04x bsize=%04x n=%04x", this, iBlockSize, bsize, n));
32 if (bsize < sizeof(SMemBlock) || (bsize & 3))
34 TInt total_size = n * bsize;
35 iFirstFree = (SMemBlock*)Kern::Alloc(total_size);
36 __KTRACE_OPT(KBOOT, Kern::Printf("PMemPool::Create %08x iFirstFree=%08x", this, iFirstFree));
42 SMemBlock* p = (SMemBlock*)(TLinAddr(iFirstFree) + i*bsize);
43 SMemBlock* q = (i<n-1) ? (SMemBlock*)(TLinAddr(p) + bsize) : NULL;
47 __KTRACE_OPT(KBOOT, Kern::Printf("PMemPool::Create OK"));
51 // Call with interrupts disabled
52 void* PMemPool::Alloc()
54 SMemBlock* p = iFirstFree;
57 iFirstFree = p->iNext;
58 __KTRACE_OPT(KBOOT, Kern::Printf("AL:%08x->%08x", this, &p->iNext));
61 __KTRACE_OPT(KBOOT, Kern::Printf("AL:%08x->0", this));
65 // Call with interrupts disabled
66 void PMemPool::Free(void* aBlock)
68 __KTRACE_OPT(KBOOT, Kern::Printf("FR:%08x<-%08x", this, aBlock));
69 SMemBlock* b = (SMemBlock*)aBlock;
70 __NK_ASSERT_DEBUG(b->iPool==this);
71 b->iNext = iFirstFree;
75 PMemMgr* PMemMgr::TheMgr;
77 // Create a 'size bucket' memory manager consisting of a number of memory pools
78 // each containing blocks of the same size. The block size increases from one
80 void PMemMgr::Create(const poolinfo* aInfo)
83 for (n=0; aInfo[n].block_size; ++n) {}
84 PMemMgr* m = (PMemMgr*)Kern::Alloc(sizeof(PMemMgr) + (n-1)*sizeof(PMemPool));
85 __KTRACE_OPT(KBOOT, Kern::Printf("PMemMgr::Create %08x NumPools=%d", m, n));
86 __NK_ASSERT_ALWAYS(m!=NULL);
92 __NK_ASSERT_ALWAYS(aInfo[i].block_size > prev_sz);
93 prev_sz = aInfo[i].block_size;
94 TInt r = m->iPools[i].Create(aInfo+i);
95 __NK_ASSERT_ALWAYS(r==KErrNone);
100 // Allocate a memory block of the requested size (or the next larger size if necessary).
101 void* PMemMgr::Alloc(size_t aSize)
103 __KTRACE_OPT(KBOOT, Kern::Printf("MA:%04x", aSize));
105 PMemPool* p = &TheMgr->iPools[0];
106 PMemPool* q = p + TheMgr->iPoolCount;
107 for (; p<q && p->iBlockSize < aSize; ++p) {}
110 TInt irq = NKern::DisableAllInterrupts();
112 NKern::RestoreInterrupts(irq);
117 // Free a memory block
118 void PMemMgr::Free(void* aPtr)
120 __KTRACE_OPT(KBOOT, Kern::Printf("MF:%08x", aPtr));
121 SMemBlock* b = _LOFF(aPtr, SMemBlock, iNext);
122 TInt irq = NKern::DisableAllInterrupts();
124 NKern::RestoreInterrupts(irq);
128 /* Memory management APIs */
130 void* alloc_mem_block(size_t size)
132 return PMemMgr::Alloc(size);
135 void free_mem_block(void* block)
137 PMemMgr::Free(block);
141 /******************************************************************************
143 ******************************************************************************/
145 TInt PThread::NumTasks;
146 TInt PThread::MaxTaskId;
147 PThread** PThread::TaskTable;
149 // RTOS priority to nanokernel priority mapping
150 const TUint8 PThread::NThreadPriorityTable[MAX_TASK_PRIORITY+1] =
152 0x00, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x02, 0x02, 0x02, 0x02, 0x03, 0x03, 0x03, 0x03,
153 0x04, 0x04, 0x04, 0x04, 0x05, 0x05, 0x05, 0x05, 0x06, 0x06, 0x06, 0x06, 0x07, 0x07, 0x07, 0x07,
154 0x08, 0x08, 0x08, 0x08, 0x09, 0x09, 0x09, 0x09, 0x0a, 0x0a, 0x0a, 0x0a, 0x0b, 0x0b, 0x0b, 0x0b,
155 0x0c, 0x0c, 0x0c, 0x0c, 0x0d, 0x0d, 0x0d, 0x0d, 0x0e, 0x0e, 0x0e, 0x0e, 0x0f, 0x0f, 0x0f, 0x0f,
156 0x10, 0x10, 0x10, 0x10, 0x11, 0x11, 0x11, 0x11, 0x12, 0x12, 0x12, 0x12, 0x13, 0x13, 0x13, 0x13,
157 0x14, 0x14, 0x14, 0x14, 0x15, 0x15, 0x15, 0x15, 0x16, 0x16, 0x16, 0x16, 0x17, 0x17, 0x17, 0x17,
158 0x18, 0x18, 0x18, 0x18, 0x19, 0x19, 0x19, 0x19, 0x1a, 0x1a, 0x1a, 0x1a, 0x1b, 0x1b, 0x1b, 0x1b,
159 0x1c, 0x1c, 0x1c, 0x1c, 0x1d, 0x1d, 0x1d, 0x1d, 0x1e, 0x1e, 0x1e, 0x1e, 0x1f, 0x1f, 0x1f, 0x1f,
160 0x20, 0x20, 0x20, 0x20, 0x21, 0x21, 0x21, 0x21, 0x22, 0x22, 0x22, 0x22, 0x23, 0x23, 0x23, 0x23,
161 0x24, 0x24, 0x24, 0x24, 0x25, 0x25, 0x25, 0x25, 0x26, 0x26, 0x26, 0x26, 0x27, 0x27, 0x27, 0x27,
162 0x28, 0x28, 0x28, 0x28, 0x29, 0x29, 0x29, 0x29, 0x2a, 0x2a, 0x2a, 0x2a, 0x2b, 0x2b, 0x2b, 0x2b,
163 0x2c, 0x2c, 0x2c, 0x2c, 0x2d, 0x2d, 0x2d, 0x2d, 0x2e, 0x2e, 0x2e, 0x2e, 0x2f, 0x2f, 0x2f, 0x2f,
164 0x30, 0x30, 0x30, 0x30, 0x31, 0x31, 0x31, 0x31, 0x32, 0x32, 0x32, 0x32, 0x33, 0x33, 0x33, 0x33,
165 0x34, 0x34, 0x34, 0x34, 0x35, 0x35, 0x35, 0x35, 0x36, 0x36, 0x36, 0x36, 0x37, 0x37, 0x37, 0x37,
166 0x38, 0x38, 0x38, 0x38, 0x39, 0x39, 0x39, 0x39, 0x3a, 0x3a, 0x3a, 0x3a, 0x3b, 0x3b, 0x3b, 0x3b,
167 0x3c, 0x3c, 0x3c, 0x3c, 0x3d, 0x3d, 0x3d, 0x3d, 0x3e, 0x3e, 0x3e, 0x3e, 0x3f, 0x3f, 0x3f, 0x3f
170 // Handlers for personality layer threads
171 const SNThreadHandlers PThread::Handlers =
173 NULL, // no exit handler
176 NULL // no timeout handler
179 // Create a personality layer thread
180 TInt PThread::Create(PThread*& aThread, const taskinfo* a)
183 return BAD_ENTRY_POINT;
184 if (a->priority < MIN_TASK_PRIORITY || a->priority > MAX_TASK_PRIORITY)
186 if (a->stack_size & 3 || a->stack_size < MIN_STACK_SIZE)
187 return BAD_STACK_SIZE;
190 TInt memsize = sizeof(PThread) + a->stack_size;
191 PThread* t = (PThread*)Kern::Alloc(memsize);
193 return OUT_OF_MEMORY;
194 t->iTaskId = a->task_id;
195 t->iSetPriority = a->priority;
198 t->iISRFirstMsg = NULL;
199 t->iISRLastMsg = NULL;
200 new (&t->iMsgQIDfc) TDfc(&MsgQIDfcFn, t);
202 memset(stack, 0xbb, a->stack_size);
203 SNThreadCreateInfo info;
204 info.iFunction = (NThreadFunction)a->entry_pt;
205 info.iStackBase = stack;
206 info.iStackSize = a->stack_size;
207 info.iPriority = NThreadPriorityTable[a->priority];
208 info.iTimeslice = -1; // no timeslicing
209 info.iAttributes = 0;
210 info.iHandlers = &Handlers;
211 info.iFastExecTable = NULL;
212 info.iSlowExecTable = NULL;
213 info.iParameterBlock = NULL;
214 info.iParameterBlockSize = 0;
215 TInt r = NKern::ThreadCreate(t, info);
216 __NK_ASSERT_ALWAYS(r==KErrNone);
221 // Create all required personality layer threads
222 void PThread::CreateAll(const taskinfo* a)
226 for (; a[n].entry_pt; ++n)
228 if (a[n].task_id > maxid)
229 maxid = a[n].task_id;
233 TaskTable = (PThread**)Kern::AllocZ((maxid+1) * sizeof(PThread*));
234 __NK_ASSERT_ALWAYS(TaskTable != NULL);
236 for (i=0; i<NumTasks; ++i)
238 TInt r = Create(TaskTable[a[i].task_id], a+i);
239 __NK_ASSERT_ALWAYS(r == KErrNone);
242 for (i=0; i<NumTasks; ++i)
245 NKern::ThreadResume(TaskTable[i]);
250 void PThread::StateHandler(NThread* aThread, TInt aOp, TInt aParam)
252 PThread* t = (PThread*)aThread;
255 case NThreadBase::ESuspend:
258 case NThreadBase::EResume:
259 case NThreadBase::EForceResume:
262 case NThreadBase::ERelease:
263 t->HandleRelease(aParam);
265 case NThreadBase::EChangePriority:
266 t->HandlePriorityChange(aParam);
268 case NThreadBase::ETimeout:
271 case NThreadBase::ELeaveCS:
273 __NK_ASSERT_ALWAYS(0);
277 // Exception handler - just fault
278 void PThread::ExceptionHandler(TAny* aContext, NThread* aThread)
281 Exc::Fault(aContext);
284 // Post a message to this thread from an ISR
285 void PThread::ISRPost(msghdr* aM)
288 aM->sending_task_id = TASK_ID_ISR;
289 msghdr* prev = (msghdr*)__e32_atomic_swp_ord_ptr(&iISRLastMsg, aM);
299 // IDFC used to post message from ISR
300 void PThread::MsgQIDfcFn(TAny* aPtr)
302 PThread* t = (PThread*)aPtr;
303 TInt irq = NKern::DisableAllInterrupts();
304 msghdr* m = t->iISRFirstMsg;
305 msghdr* l = t->iISRLastMsg;
306 t->iISRFirstMsg = NULL;
307 t->iISRLastMsg = NULL;
308 NKern::RestoreInterrupts(irq);
312 // Post a chain of messages to this thread from an IDFC or thread
313 // Enter and return with preemption disabled
314 void PThread::Post(msghdr* aFirst, msghdr* aLast)
316 msghdr* l = iLastMsg;
321 return; // queue was not empty so thread can't be waiting
324 if (iNState == EWaitMsgQ)
328 // Dequeue and return the first message if there is one
329 // Return NULL if no messages waiting
330 // Enter and return with preemption disabled
331 msghdr* PThread::GetMsg()
333 msghdr* m = iFirstMsg;
343 void PThread::HandleSuspend()
350 ((PSemaphore*)iWaitObj)->SuspendWaitingThread(this);
353 __NK_ASSERT_ALWAYS(0);
357 void PThread::HandleResume()
364 ((PSemaphore*)iWaitObj)->ResumeWaitingThread(this);
367 __NK_ASSERT_ALWAYS(0);
371 void PThread::HandleRelease(TInt aReturnCode)
377 CheckSuspendThenReady();
381 ((PSemaphore*)iWaitObj)->WaitCancel(this);
383 CheckSuspendThenReady();
386 __NK_ASSERT_ALWAYS(0);
390 void PThread::HandlePriorityChange(TInt aNewPriority)
396 iPriority = (TUint8)aNewPriority;
399 ((PSemaphore*)iWaitObj)->ChangeWaitingThreadPriority(this, aNewPriority);
402 __NK_ASSERT_ALWAYS(0);
406 void PThread::HandleTimeout()
411 CheckSuspendThenReady();
414 ((PSemaphore*)iWaitObj)->WaitCancel(this);
417 __NK_ASSERT_ALWAYS(0);
424 int suspend_task(int id)
426 if (TUint(id) > TUint(PThread::MaxTaskId))
428 PThread* t = PThread::TaskTable[id];
431 NKern::ThreadSuspend(t, 1);
435 int resume_task(int id)
437 if (TUint(id) > TUint(PThread::MaxTaskId))
439 PThread* t = PThread::TaskTable[id];
442 NKern::ThreadResume(t);
446 int get_task_priority(int id)
448 if (TUint(id) > TUint(PThread::MaxTaskId))
450 PThread* t = PThread::TaskTable[id];
453 return t->iSetPriority;
456 int set_task_priority(int id, int priority)
458 if (TUint(id) > TUint(PThread::MaxTaskId))
460 PThread* t = PThread::TaskTable[id];
463 if (priority < MIN_TASK_PRIORITY || priority > MAX_TASK_PRIORITY)
466 t->iSetPriority = priority;
467 t->SetPriority(PThread::NThreadPriorityTable[priority]);
472 int current_task_id(void)
474 TInt c = NKern::CurrentContext();
475 if (c == NKern::EInterrupt)
477 PThread* t = (PThread*)NKern::CurrentThread();
478 if (t->iHandlers == &PThread::Handlers)
480 return TASK_ID_UNKNOWN;
483 void disable_preemption(void)
488 void enable_preemption(void)
493 int disable_interrupts(void)
495 return NKern::DisableAllInterrupts();
498 void restore_interrupts(int level)
500 NKern::RestoreInterrupts(level);
505 int send_msg(int task_id, msghdr* msg)
507 if (TUint(task_id) > TUint(PThread::MaxTaskId))
509 PThread* t = PThread::TaskTable[task_id];
512 TInt c = NKern::CurrentContext();
513 if (c == NKern::EInterrupt)
519 PThread* st = (PThread*)NKern::CurrentThread();
520 msg->sending_task_id = (st->iHandlers == &PThread::Handlers) ? st->iTaskId : TASK_ID_UNKNOWN;
527 int recv_msg(msghdr** msgptr, int time_ticks)
529 if (time_ticks < WAIT_FOREVER)
530 return BAD_TIME_INTERVAL;
531 PThread* t = (PThread*)NKern::CurrentThread();
533 msghdr* m = t->GetMsg();
534 if (!m && time_ticks != NO_WAIT)
536 NKern::NanoBlock(time_ticks>0 ? time_ticks : 0, PThread::EWaitMsgQ, NULL);
537 NKern::PreemptionPoint();
542 return m ? OK : TIMED_OUT;
546 /******************************************************************************
548 ******************************************************************************/
550 TInt PTimer::NumTimers;
551 PTimer* PTimer::TimerTable;
553 // Create all required timers
554 void PTimer::CreateAll()
556 NumTimers = timer_count;
557 TimerTable = new PTimer[timer_count];
558 __NK_ASSERT_ALWAYS(TimerTable != NULL);
562 : NTimer(NTimerExpired, this),
570 void PTimer::NTimerExpired(TAny* aPtr)
572 timer_msg* m = (timer_msg*)alloc_mem_block(sizeof(timer_msg));
574 m->header.msg_id = MSG_ID_TIMEOUT;
575 PTimer* p = (PTimer*)aPtr;
576 TInt irq = NKern::DisableAllInterrupts();
577 PThread* t = p->iThread;
578 m->count = ++p->iExpiryCount;
579 m->cookie = p->iCookie;
581 p->Again(p->iPeriod);
582 NKern::RestoreInterrupts(irq);
583 t->ISRPost(&m->header);
588 unsigned tick_count(void)
590 return NKern::TickCount();
593 void delay(int time_interval)
595 __NK_ASSERT_ALWAYS(time_interval > 0);
596 NKern::Sleep(time_interval);
599 int start_one_shot_timer(int timer_id, int task_id, int time_ticks, void* cookie)
602 return BAD_TIME_INTERVAL;
603 if (TUint(timer_id) >= TUint(PTimer::NumTimers))
605 PTimer* tmr = PTimer::TimerTable + timer_id;
606 if (TUint(task_id) > TUint(PThread::MaxTaskId))
608 PThread* t = PThread::TaskTable[task_id];
612 TInt irq = NKern::DisableAllInterrupts();
618 tmr->iCookie = cookie;
620 tmr->iExpiryCount = 0;
621 tmr->OneShot(time_ticks, EFalse);
623 NKern::RestoreInterrupts(irq);
627 int start_periodic_timer(int timer_id, int task_id, int initial_time_ticks, int period_ticks, void* cookie)
629 if (initial_time_ticks <= 0 || period_ticks <= 0)
630 return BAD_TIME_INTERVAL;
631 if (TUint(timer_id) >= TUint(PTimer::NumTimers))
633 PTimer* tmr = PTimer::TimerTable + timer_id;
634 if (TUint(task_id) > TUint(PThread::MaxTaskId))
636 PThread* t = PThread::TaskTable[task_id];
640 TInt irq = NKern::DisableAllInterrupts();
645 tmr->iPeriod = period_ticks;
646 tmr->iCookie = cookie;
648 tmr->iExpiryCount = 0;
649 tmr->OneShot(initial_time_ticks, EFalse);
651 NKern::RestoreInterrupts(irq);
655 int stop_timer(int timer_id)
657 if (TUint(timer_id) >= TUint(PTimer::NumTimers))
659 PTimer* tmr = PTimer::TimerTable + timer_id;
660 TInt irq = NKern::DisableAllInterrupts();
663 NKern::RestoreInterrupts(irq);
669 /******************************************************************************
670 * Semaphore management
671 ******************************************************************************/
673 TInt PSemaphore::NumSemaphores;
674 PSemaphore* PSemaphore::SemaphoreTable;
676 void PSemaphore::CreateAll()
678 NumSemaphores = semaphore_count;
679 SemaphoreTable = new PSemaphore[semaphore_count];
680 __NK_ASSERT_ALWAYS(SemaphoreTable != NULL);
683 PSemaphore::PSemaphore()
690 void PSemaphore::WaitCancel(PThread* aThread)
692 if (aThread->iSuspendCount == 0)
694 iWaitQ.Remove(aThread);
699 aThread->CheckSuspendThenReady();
702 void PSemaphore::SuspendWaitingThread(PThread* aThread)
704 // do nothing if already suspended
705 if (aThread->iSuspendCount == 0)
707 iWaitQ.Remove(aThread);
709 iSuspendedQ.Add(aThread);
713 void PSemaphore::ResumeWaitingThread(PThread* aThread)
722 aThread->iWaitObj=NULL;
727 void PSemaphore::ChangeWaitingThreadPriority(PThread* aThread, TInt aNewPriority)
729 if (aThread->iSuspendCount == 0)
730 iWaitQ.ChangePriority(aThread, aNewPriority);
732 aThread->iPriority = (TUint8)aNewPriority;
735 void PSemaphore::Signal()
739 // must wake up next thread
740 PThread* t = iWaitQ.First();
742 t->Release(KErrNone);
746 void PSemaphore::ISRSignal()
748 if (__e32_atomic_add_ord32(&iISRCount, 1) == 0)
752 void PSemaphore::IDfcFn(TAny* aPtr)
754 PSemaphore* s = (PSemaphore*)aPtr;
755 TInt count = (TInt)__e32_atomic_swp_ord32(&s->iISRCount, 0);
762 int semaphore_wait(int sem_id, int time_ticks)
764 if (time_ticks < WAIT_FOREVER)
765 return BAD_TIME_INTERVAL;
766 if (TUint(sem_id) >= TUint(PSemaphore::NumSemaphores))
768 PSemaphore* s = PSemaphore::SemaphoreTable + sem_id;
769 PThread* t = (PThread*)NKern::CurrentThread();
772 if (time_ticks == NO_WAIT)
783 NKern::NanoBlock(time_ticks>0 ? time_ticks : 0, PThread::EWaitSemaphore, s);
785 NKern::PreemptionPoint();
786 if (t->iReturnValue == KErrTimedOut)
793 int semaphore_signal(int sem_id)
795 if (TUint(sem_id) >= TUint(PSemaphore::NumSemaphores))
797 PSemaphore* s = PSemaphore::SemaphoreTable + sem_id;
798 TInt c = NKern::CurrentContext();
799 if (c == NKern::EInterrupt)
810 void init_personality(void)
812 __KTRACE_OPT(KBOOT,Kern::Printf("Starting example personality"));
814 PMemMgr::Create(pool_list);
816 PSemaphore::CreateAll();
817 PThread::CreateAll(task_list);
821 /******************************************************************************
822 * Communication with EPOC
823 ******************************************************************************/
824 TPMsgQ* TPMsgQ::ThePMsgQ;
826 TPMsgQ::TPMsgQ(TDfcFn aFunction, TAny* aPtr, TDfcQue* aDfcQ, TInt aPriority)
827 : TDfc(aFunction, aPtr, aDfcQ, aPriority),
834 extern "C" void send_to_epoc(msghdr* m)
836 TPMsgQ* q = TPMsgQ::ThePMsgQ;
838 m->sending_task_id = current_task_id();
840 msghdr* l = q->iLastMsg;
846 return; // queue was not empty so thread can't be waiting
857 void TPMsgQ::Receive()
867 msghdr* TPMsgQ::Get()
870 msghdr* m = iFirstMsg;
881 void TPMsgQ::CancelReceive()