-
Notifications
You must be signed in to change notification settings - Fork 227
THREAD Module
A thread of execution is the smallest sequence of programmed instructions that can be managed independently by a scheduler, which is a part of the operating system. Multiple threads can exist, executing concurrently and sharing resources such as memory.
Lua RTOS can execute Lua functions into FreeRTOS native threads. Unlike Lua coroutines, threads are scheluded by the operating system, allowing to run Lua programs in real time.
The functions of this module are organized in the following categories:
List all threads.
Arguments:
-
table (optional): if true, thread list is returned in a Lua table, if false thread list is printed on the console.
-
monitor(optional): if true, the thread list is refreshed every 0.5 seconds. Only allowed if table = false.
-
all (optional): if true show all threads. If false (default) shows only the lua threads. There are 3 types of threads:
- task: is a FreeRTOS task managed directly by FreeRTOS. This types of tasks corresponds to system tasks.
- thread: is a FreeRTOS task managed by the pthread API.
- lua: is a FreeRTOS task managed by the Lua thread module.
Returns:
-
if table is false: nothing or an exception.
-
if table is true: a Lua table with the thread list result, or an exception. This table is an array of tables. Each entry corresponds to a thread. Each thread gives the following fields:
- thid: thread id.
- type: thread type, can be either task, thread or lua.
- name: thread name.
- status: for lua threads can be either run (running) or susp (suspended), and for other thread types is blank.
- core: CPU in which thread is pinned.
- stack_size: assigned stack size (in bytes) to the thread.
- free_stack: number of free bytes in the stack.
- used_stack: number of bytes used in the stack.
/ > thread.list()
-----------------------------------------------------------------------------------------
| | | | | | STACK
THID | TYPE | NAME | STATUS | CORE | PRIO | SIZE FREE USED
-----------------------------------------------------------------------------------------
1 lua lua run 0 20 10240 7632 2608 ( 25%)
0 task IDLE 1 0 1024 600 424 ( 41%)
0 task IDLE 0 0 1024 596 428 ( 41%)
0 task Tmr Svc 0 1 2048 1648 400 ( 19%)
0 task ipc0 0 24 1024 576 448 ( 43%)
0 task esp_timer 0 22 4096 3652 444 ( 10%)
0 task signal 0 21 768 312 456 ( 59%)
0 task ipc1 1 24 1024 500 524 ( 51%)
Start a new thread.
Arguments:
- function: function to execute in the thread.
- stack (optional): the number of bytes to allocate for use as the thread's stack. Must be >= 2048.
- priority (optional): the priority at which the thread will execute. Must be >= 3 and <= 25, where 3 is the minimal priority.
- cpu (optional): the CPU at which the thread will execute.
- name (optional): name to assign to the thread, for debug purposes.
Returns: the thread identifier, or an exception. You must store this identifier into a variable for further operations with it.
-- Function to execute
f1 = function()
local led_pin = pio.GPIO4
pio.pin.setdir(pio.OUTPUT, led_pin)
while true do
pio.pin.sethigh(led_pin)
tmr.delayms(250)
pio.pin.setlow(led_pin)
tmr.delayms(250)
end
end
-- Start thread
th1 = thread.start(f1)
Get the thread identifier of the calling thread.
Arguments: none.
Returns: the thread identifier.
thread.start(function()
print("I'm: "..thread.self())
end)
Suspend all threads or a specific thread. This function suspends the execution of the underline operating system thread remaining in memory until it's resumed or stopped.
Arguments:
- thread id: this argument it's optional. If not present function suspend all threads, and if present suspend the specified thread.
Returns: nothing.
-- Function to execute
f1 = function()
local led_pin = pio.GPIO4
pio.pin.setdir(pio.OUTPUT, led_pin)
while true do
pio.pin.sethigh(led_pin)
tmr.delayms(250)
pio.pin.setlow(led_pin)
tmr.delayms(250)
end
end
-- Start thread
th1 = thread.start(f1)
....
-- Suspend thread
thread.suspend(th1)
Resume all threads or a specific thread. This function resume the execution of the underline operating system thread.
Arguments:
- thread id: this argument it's optional. If not present function resume all threads, and if present resume the specified thread.
Returns: nothing.
-- Function to execute
f1 = function()
local led_pin = pio.GPIO4
pio.pin.setdir(pio.OUTPUT, led_pin)
while true do
pio.pin.sethigh(led_pin)
tmr.delayms(250)
pio.pin.setlow(led_pin)
tmr.delayms(250)
end
end
-- Start thread
th1 = thread.start(f1)
....
-- Suspend thread
thread.suspend(th1)
....
-- Resume thread
thread.resume(th1)
Stop all threads or a specific thread.
Arguments:
- thread id: this argument it's optional. If not present function stop all threads, and if present stop the specified thread. This function stop the execution of the underline operating system thread and removes it from memory.
Returns: nothing.
-- Function to execute
f1 = function()
local led_pin = pio.GPIO4
pio.pin.setdir(pio.OUTPUT, led_pin)
while true do
pio.pin.sethigh(led_pin)
tmr.delayms(250)
pio.pin.setlow(led_pin)
tmr.delayms(250)
end
end
-- Start thread
th1 = thread.start(f1)
....
-- Stop thread
thread.stop(th1)
Get the status of a specific thread.
Arguments:
- thread id: thread identifier.
Returns: thread status (running, suspended or nil).
-- Function to execute
f1 = function()
local led_pin = pio.GPIO4
pio.pin.setdir(pio.OUTPUT, led_pin)
while true do
pio.pin.sethigh(led_pin)
tmr.delayms(250)
pio.pin.setlow(led_pin)
tmr.delayms(250)
end
end
-- Start thread
th1 = thread.start(f1)
....
-- Suspend thread
thread.suspend(th1)
-- Get thread status
thread.status(th1)
> suspended
Sleep current thread a number of seconds.
Arguments:
- seconds: seconds to sleep.
Returns: nothing.
-- Sleep for 4 seconds
thread.sleep(4)
Sleep current thread a number of milliseconds.
Arguments:
- milliseconds: milliseconds to sleep.
Returns: nothing.
-- Sleep for 4 milliseconds
thread.sleepms(4)
Sleep current thread a number of micro seconds.
Arguments:
- useconds: micro seconds to sleep.
Returns: nothing.
-- Sleep for 500000 micro seconds (0,5 seconds)
thread.sleepus(500000)
Mutual exclusion (mutex) is a property of concurrency control, which is instituted for the purpose of preventing race conditions; it is the requirement that one thread of execution never enter to a critical section at the same time that another concurrent thread of execution enters in the critical section.
Creates a mutex instance.
Arguments:
- type (optional): type of mutex, can be either thread.Lock or thread.RecursiveLock. The default mutex type is thread.RecursiveLock.
Returns: a mutex instance, or an exception. You must store this instance into a variable for further operations with it.
Locks the mutex instance. If the mutex is locked by other thread, the calling thread is blocked until the mutex is unlocked.
Arguments: nothing.
Returns: nothing.
Unlocks the mutex instance.
Arguments: nothing.
Returns: nothing.
Try to lock the mutex. If the mutex is locked by other thread this functions returns false, and the calling thread is not blocked. If the mutex is not locked by other thread this functions locks the mutex and returns true.
Arguments: nothing.
Returns: true if the mutex has been locked by the calling process, or false if the mutex is locked by other threads.
The following example create 4 threads that increment the global variable timer 20000 times each. When all treads end, the counter value must be 80000.
counter = 0
mtx = thread.createmutex()
thread.start(function()
local i = 0
while (i < 20000) do
mtx:lock()
counter = counter + 1
mtx:unlock()
i = i + 1
end
end)
thread.start(function()
local i = 0
while (i < 20000) do
mtx:lock()
counter = counter + 1
mtx:unlock()
i = i + 1
end
end)
thread.start(function()
local i = 0
while (i < 20000) do
mtx:lock()
counter = counter + 1
mtx:unlock()
i = i + 1
end
end)
thread.start(function()
local i = 0
while (i < 20000) do
mtx:lock()
counter = counter + 1
mtx:unlock()
i = i + 1
end
end)