This thread has been locked.
If you have a related question, please click the "Ask a related question" button in the top right corner. The newly created question will be automatically linked to this question.
Hi.
I really appreciate comments to my following “tricky issue” (we use TI’s TMS320F28069, no RTOS) where we have:
A high-priority PWM ISR executing every 5 usec.
This runs OK in “the background”!
NOTE: This 5 usec interrupt must be handled immediately (other parts of the code must NEVER disable these interrupts etc).
Two other “tasks”:
a) Task-A is running continuously (maintenance stuff etc).
b) Task-B is executed every 650 usec, AND MUST NEVER USE > 150 USEC (otherwise HW will be damaged).
Task-B must be given CPU (at least) every 650 usec.
The scheduler must also pre-empt task-B after 150 usec and give the CPU back to task-A (THIS IS MOST IMPORTANT!).
Q-1: Does the TI-RTOS satisfy these (tough timing) requirements?
Q-2: If not, does anyone know if such a “simple but fast” scheduler is available on the net (for TMSxxxxx) or elsewere (another RTOS maybe)?
Best Regards
Terje Bøhler
Hi Terje,
These are tight timing requirements, indeed. I am in the process of looking at the numbers and will respond back tomorrow.
Steve
Terje,
I'm waiting to speak to one of my colleagues on your timing requirements and will get back to you on that once I do.
I have a couple of questions in the meantime:
Terje Bohler1 said:a) Task-A is running continuously (maintenance stuff etc).
How often do you expect Task A to run? How much time will it need, approximately, to complete the work it is doing?
Terje Bohler1 said:b) Task-B is executed every 650 usec, AND MUST NEVER USE > 150 USEC (otherwise HW will be damaged).
Also for this - How much time will it need, approximately, to complete the work it is doing?
Steve
Terje Bohler1 said:A high-priority PWM ISR executing every 5 usec.
This runs OK in “the background”!
NOTE: This 5 usec interrupt must be handled immediately (other parts of the code must NEVER disable these interrupts etc)
Also, how long will it take to complete the work done in the ISR for this?
Hi Terje,
I discussed your requirements with my colleague and we are confident that SYS/BIOS can handle your tight timing requirements, with one small caveat which I'll talk about below.
1. Your requirement for an interrupt that will fire every 5us can be handled by using the SYS/BIOS Hwi module and the C28x's "zero latency interrupts" feature. The zero latency interrupts will minimize the overhead of a Task swap and Hwi dispatcher in order to run your ISR.
Here's where the caveat comes in. It is very possible that another, (non zero latency) interrupt fires *immediately* before your 5us interrupt does. In this case, the C28x core itself will disable interrupts for a short period of time (there is nothing that SYS/BIOS can do about this; it is done by the core itself). So, in this case, interrupts will be disabled and some registers would be stacked by the hardware.
Once that's done, the SYS/BIOS Hwi dispatcher's prolog would run, pushing some other registers that it needs to save before re-enabling interrupts again (while this code has been minimized, it still does take a little time). So, there is a "dead zone" that could occur in this scenario.
Only once the end of this "dead zone" is reached (when BIOS re-enables interrupts) could your 5us interrupt run, in this scenario. So, you would need to be sure that your ISR code can still complete in time, given the possibility of this extra overhead.
2. Your Task B, which must run every 650us, and NOT run for more than 150us can be handled by a SYS/BIOS Task and a Timer.
This could be solved in more than one way, but here is one possible solution:
The Task would be used to run the actual code that does the work that cannot run for more than 150us.
A Timer object would be used to enable the Task every 650 us, and disable it after 150us.
For example, the Timer object can be created with a period set to 50 us, and configured with a function "myTimer()". The Timer will expire every 50us, at which point it will call "myTimer()". With this 50us period, every 13 ticks will equal 650us, and 3 ticks will be 150us.
.
In the code of myTimer(), you could do something like the following (pseudo code).
myTimer() { static int previousTime; if (ticks % 13 == 0) { /* (re)enable Task B (context switch to Task B) after 650us */ Task_setPri(TaskB, <some high pri that allows TaskB to run>); previousTime = Timer_getCount(); } else if (Timer_getCount() = previousTime + 3) { /* disable Task B: (priority of -1 causes Task to be disabled) after 150us */ Task_setPri(TaskB, -1);
previousTime = 0; } }
I'm not sure if these are the exact times you'd need, this is just some rough pseudo code, but I think you get the idea.
3. For your Task A, which just runs continuously in the background to do maintenance, this could also use a SYS/BIOS Task. It could be set to a priority that is lower than Task B, to allow it to run during the times that the 5us ISR and Task B are not running
Lastly:
Terje Bohler1 said:The 5 usec ISR takes about 2 usec (40% of the total CPU) - assumably.
Since your ISR will be firing during Task B (has only 150ms to complete), 40% leaves only 60us for Task B to complete. Can it complete during that time?
Steve
Thank you very much Steve.
I really appreciate your support.
I'll dive into this matter in a short while, and give you feedback on our results after testing/prototyping.
Best Regards
Terje Bøhler
Terje,
The "dead zone" consists of 2 parts:
1. The C28x hardware itself, will push some registers and do some other work, automatically. During this time, interrupts are disabled. You can see this in the following diagram from
SPRU430F:
2. The SYS/BIOS dispatcher, as mentioned before. The instructions that are run before interrupts are re-enabled are minimal. You can find the code in the file "Hwi_disp.s28" in SYS/BIOS. In particular, here is the worst case run through for a non-zero latency interrupt:
- both the move and short branch instructions will run for the correpsonding interrupt vector in the dispatchZeroTable:
; ; ======== dispatchZero ======== ; This is the Hwi dispatcher when supporting zero latency interrupts ; .sect ".text:_ti_sysbios_family_c28_Hwi_dispatchZero" .clink _ti_sysbios_family_c28_Hwi_dispatchZeroTable: .asmfunc mov @al, #1 sb _ti_sysbios_family_c28_Hwi_dispatchZero, UNC mov @al, #2 ; dispIsr2 sb _ti_sysbios_family_c28_Hwi_dispatchZero, UNC mov @al, #3 ; dispIsr3 sb _ti_sysbios_family_c28_Hwi_dispatchZero, UNC
- The dispatchZero function will run approximately 12 instructions until it re-enables interrupts. You can see this here:
_ti_sysbios_family_c28_Hwi_dispatchZero: ; zero latency dispatcher .asmfunc c28addr pop p ; Load product reg with return addr ; NOTE: p is automatically saved push p ; Put return addr back on stack push xar4 ; Save xar4 early so it can be used here ; Popped at beginning of dispatchRet movl xar4, #_ti_sysbios_family_c28_Hwi_Module__state__V.globalEnable mov al, #0 mov *xar4, al ; Set Module_state globalEnable to 0 movl xar4, #_ti_sysbios_family_c28_Hwi_zeroLatencyIERMask__C mov al, IER and al, *xar4 ; And the IERMask with current value of IER mov IER, al ; Write the IER with the value clrc intm, dbgm ; Re-enable global interrupts
So, you can see that the dead zone is extremely minimal.
Steve