Tizen Native API

Facilities to run heavy tasks in different threads to avoid blocking the main loop.

The EFL is, for the most part, not thread safe. This means that if you have some task running in another thread and you have, for example, an Evas object to show the status progress of this task, you cannot update the object from within the thread. This can only be done from the main thread, the one running the main loop. This problem can be solved by running a thread that sends messages to the main one using an Ecore_Pipe, but when you need to handle other things like cancelling the thread, your code grows in complexity and gets much harder to maintain.

Ecore Thread is here to solve that problem. It is not a simple wrapper around standard POSIX threads (or an equivalent in other systems) and it's not meant to be used to run parallel tasks throughout the entire duration of the program, especially when these tasks are performance critical, as Ecore manages these tasks using a pool of threads based on system configuration.

What Ecore Thread does is it makes it a lot easier to dispatch a worker function to perform some heavy tasks and then get the result once it completes, without blocking the application's UI. In addition, cancelling and rescheduling comes practically for free and the developer need not worry about how many threads are launched, since Ecore schedules them according to the number of processors the system has and the maximum amount of concurrent threads set for the application.

At the system level, Ecore starts a new thread on an as-needed basis until the maximum set is reached. When no more threads can be launched, new worker functions are queued in a waiting list until a thread becomes available. This way, system threads are shared throughout different worker functions, but running only one at a time. At the same time, a worker function that is rescheduled may be run on a different thread the next time.

The Ecore_Thread handler has two meanings, depending on what context it is on. The one returned when starting a worker with any of the functions ecore_thread_run() or ecore_thread_feedback_run() is an identifier of that specific instance of the function and can be used from the main loop with the ecore_thread_cancel() and ecore_thread_check() functions. This handler must not be shared with the worker function running in the thread. This same handler is the one received on the end, cancel, and feedback callbacks.

The worker function, that's the one running in the thread, also receives an Ecore_Thread handler that can be used with ecore_thread_cancel() and ecore_thread_check(), sharing the flag with the main loop. But this handler is also associated with the thread where the function is running. This has strong implications when working with thread local data.

There are two kinds of worker threads that Ecore handles: simple or short, workers, and feedback workers.

The first kind is for simple functions that perform a usually small but time consuming task. Ecore runs this function in a thread as soon as one becomes available and notifies the calling user of its completion once the task is done.

The following image shows the flow of a program running four tasks on a pool of two threads.

ecore_thread.png

For larger tasks that may require continuous communication with the main program, the feedback workers provide the same functionality plus a way for the function running in the thread to send messages to the main thread.

The next diagram omits some details shown in the previous one regarding how threads are spawned and tasks are queued, but illustrates how feedback jobs communicate with the main loop and the special case of threads running out of the pool.

ecore_thread_feedback.png

Functions

Ecore_Threadecore_thread_run (Ecore_Thread_Cb func_blocking, Ecore_Thread_Cb func_end, Ecore_Thread_Cb func_cancel, const void *data)
 Schedules a task to run in a parallel thread to avoid locking the main loop.
Ecore_Threadecore_thread_feedback_run (Ecore_Thread_Cb func_heavy, Ecore_Thread_Notify_Cb func_notify, Ecore_Thread_Cb func_end, Ecore_Thread_Cb func_cancel, const void *data, Eina_Bool try_no_queue)
 Launches a thread to run a task that can talk back to the main thread.
Eina_Bool ecore_thread_cancel (Ecore_Thread *thread)
 Cancels a running thread.
Eina_Bool ecore_thread_check (Ecore_Thread *thread)
 Checks whether a thread is in pending cancellation.
Eina_Bool ecore_thread_feedback (Ecore_Thread *thread, const void *msg_data)
 Sends data from the worker thread to the main loop.
Eina_Bool ecore_thread_reschedule (Ecore_Thread *thread)
 Asks for the function in the thread to be called again at a later period.
int ecore_thread_active_get (void)
 Gets the number of active threads running jobs.
int ecore_thread_pending_get (void)
 Gets the number of short jobs waiting for a thread to run.
int ecore_thread_pending_feedback_get (void)
 Gets the number of feedback jobs waiting for a thread to run.
int ecore_thread_pending_total_get (void)
 Gets the total number of pending jobs.
int ecore_thread_max_get (void)
 Gets the maximum number of threads that can run simultaneously.
void ecore_thread_max_set (int num)
 Sets the maximum number of threads allowed to run simultaneously.
void ecore_thread_max_reset (void)
 Resets the maximum number of concurrently running threads to the default.
int ecore_thread_available_get (void)
 Gets the number of threads available for running tasks.
Eina_Bool ecore_thread_local_data_add (Ecore_Thread *thread, const char *key, void *value, Eina_Free_Cb cb, Eina_Bool direct)
 Adds some data present in the hash local to the thread.
void * ecore_thread_local_data_set (Ecore_Thread *thread, const char *key, void *value, Eina_Free_Cb cb)
 Sets some data present in the hash local to the given thread.
void * ecore_thread_local_data_find (Ecore_Thread *thread, const char *key)
 Gets data stored in the hash local to the given thread.
Eina_Bool ecore_thread_local_data_del (Ecore_Thread *thread, const char *key)
 Deletes the data corresponding to the given key from the thread's hash.
Eina_Bool ecore_thread_global_data_add (const char *key, void *value, Eina_Free_Cb cb, Eina_Bool direct)
 Adds some data to a hash shared by all threads.
void * ecore_thread_global_data_set (const char *key, void *value, Eina_Free_Cb cb)
 Sets some data in the hash shared by all threads.
void * ecore_thread_global_data_find (const char *key)
 Gets data stored in the hash shared by all threads.
Eina_Bool ecore_thread_global_data_del (const char *key)
 Deletes the data corresponding to the given key from the shared hash.
void * ecore_thread_global_data_wait (const char *key, double seconds)
 Gets data stored in the shared hash or waits for it if it doesn't exist.

Typedefs

typedef struct _Ecore_Thread Ecore_Thread
 A handle for threaded jobs.
typedef void(* Ecore_Thread_Cb )(void *data, Ecore_Thread *thread)
 Called to be used by Ecore_Thread helper.
typedef void(* Ecore_Thread_Notify_Cb )(void *data, Ecore_Thread *thread, void *msg_data)
 Called to be used by the main loop to receive data sent by an Ecore Thread.

Function Documentation

int ecore_thread_active_get ( void  )

Gets the number of active threads running jobs.

This returns the number of threads currently running jobs of any type through the Ecore_Thread API.

Since :
2.3.1
Remarks:
Jobs started through the ecore_thread_feedback_run() function with the try_no_queue parameter set to EINA_TRUE are not accounted for in the return of this function unless the thread creation fails and it falls back to using one from the pool.
Returns:
The number of active threads running jobs

Gets the number of threads available for running tasks.

Since :
2.3.1
Remarks:
This is same as doing ecore_thread_max_get() - ecore_thread_active_get().
This function may return a negative number only in the case when the user changes the maximum number of running threads while other tasks are running.
Returns:
The number of available threads

Cancels a running thread.

This function cancels a running thread. If thread can be immediately cancelled (its still pending execution after creation or rescheduling), then the cancel callback is called, thread is freed and the function returns EINA_TRUE.

Since :
2.3.1
Remarks:
If the thread is already running, then this function returns EINA_FALSE after marking the thread as pending cancellation. For the thread to actually be terminated, it needs to return from the user function back into Ecore control. This can happen in several ways:
  • The function ends and returns normally. If it hadn't been cancelled, func_end would be called here, but instead func_cancel happens.
  • The function returns after requesting to be rescheduled with ecore_thread_reschedule().
  • The function is prepared to leave early by checking if ecore_thread_check() returns EINA_TRUE.
The user function can cancel itself by calling ecore_thread_cancel(), but it should always use the Ecore_Thread handle passed to it and never share it with the main loop thread by means of shared user data or in any other way.
thread is freed and should not be used again if this function returns EINA_TRUE or after the func_cancel callback returns.
This function can be called both in the main loop and in the running thread.
Parameters:
[in]threadThe thread to cancel
Returns:
EINA_TRUE if the thread has been cancelled, otherwise EINA_FALSE if it is pending
See also:
ecore_thread_check()

Checks whether a thread is in pending cancellation.

This function can be called both in the main loop and in the running thread.

Since :
2.3.1
Remarks:
When ecore_thread_cancel() is called on an already running task, the thread is marked as pending cancellation. This function returns EINA_TRUE if this mark is set for the given thread and can be used from the main loop thread to check if a still active thread has been cancelled, or from the user function running in the thread to check if it should stop doing what it's doing and return early, effectively cancelling the task.
Parameters:
[in]threadThe thread to test
Returns:
EINA_TRUE if the thread is in pending cancellation, otherwise EINA_FALSE if it is not
See also:
ecore_thread_cancel()
Eina_Bool ecore_thread_feedback ( Ecore_Thread thread,
const void *  msg_data 
)

Sends data from the worker thread to the main loop.

Since :
2.3.1
Remarks:
You should use this function only in the func_heavy call.
Only the address to msg_data is sent and once this function returns EINA_TRUE, the job running in the thread should never touch the contents of it again. The data sent should be malloc()'ed or something similar, as long as it's not the memory that is local to the thread that risks being overwritten or deleted once it goes out of scope or the thread finishes.
Care must be taken that msg_data is properly freed in the func_notify callback set when creating the thread.
Parameters:
[in]threadThe current Ecore_Thread context to send data from
[in]msg_dataThe data to be transmitted to the main loop
Returns:
EINA_TRUE if msg_data is successfully sent to the main loop, otherwise EINA_FALSE if anything goes wrong
See also:
ecore_thread_feedback_run()
Ecore_Thread* ecore_thread_feedback_run ( Ecore_Thread_Cb  func_heavy,
Ecore_Thread_Notify_Cb  func_notify,
Ecore_Thread_Cb  func_end,
Ecore_Thread_Cb  func_cancel,
const void *  data,
Eina_Bool  try_no_queue 
)

Launches a thread to run a task that can talk back to the main thread.

Since :
2.3.1
Remarks:
The difference in the above is that ecore_thread_run() is meant for tasks that don't need to communicate anything until they finish, while this function is provided with a new callback, func_notify, that is called from the main thread for every message sent from func_heavy with ecore_thread_feedback().
Like with ecore_thread_run(), a new thread is launched to run func_heavy unless the maximum number of simultaneous threads has been reached, in which case the function is scheduled to run whenever a running task ends and a thread becomes free. But if try_no_queue is set, Ecore first tries to launch a thread outside of the pool to run the task. If it fails, it reverts to the normal behaviour of using a thread from the pool as if try_no_queue had not been set.
Keep in mind that Ecore handles the thread pool based on the number of CPUs available, but running a thread outside of the pool doesn't count for this, so having too many of them may have drastic effects over the program's performance.
See ecore_thread_run() for a general description of this function.
Parameters:
[in]func_heavyThe function that should run in another thread
[in]func_notifythe function that receives the data sent from the thread
[in]func_endThe function to call from the main loop when func_heavy completes its task successfully
[in]func_cancelThe function to call from the main loop if the thread running func_heavy is cancelled or fails to start
[in]dataThe user context data to pass to all callbacks
[in]try_no_queueThe boolean value that indicates whether to run outside the thread pool
Returns:
A new thread handler, otherwise NULL on failure
See also:
ecore_thread_feedback()
ecore_thread_run()
ecore_thread_cancel()
ecore_thread_reschedule()
ecore_thread_max_set()
Eina_Bool ecore_thread_global_data_add ( const char *  key,
void *  value,
Eina_Free_Cb  cb,
Eina_Bool  direct 
)

Adds some data to a hash shared by all threads.

Since :
2.3.1
Remarks:
Ecore Thread keeps a hash that can be used to share data across several threads, including the main loop thread, without having to manually handle mutexes to do it safely.
This function adds the data value to this hash under the given key. No other value in the hash may have the same key. If you need to change the value under a key, or you don't know if one exists already, you can use ecore_thread_global_data_set().

Neither key nor value may be NULL and key gets copied in the hash, unless direct is set, in which case the string used should not be freed until the data is removed from the hash.

Remarks:
The cb function is called when the data in the hash needs to be freed, be it because it got deleted with ecore_thread_global_data_del() or because Ecore Thread got shut down and the hash got destroyed. This parameter may be NULL, in which case value needs to be manually freed after removing it from the hash with either by ecore_thread_global_data_del() or ecore_thread_global_data_set().

Manually freeing any data that is added to the hash with the cb function is likely to produce a segmentation fault, or any other strange happening at a later stage in the program.

Parameters:
[in]keyThe name under which the data is stored
[in]valueThe data to add
[in]cbThe function to free the data when removed from the hash
[in]directIf true, this does not copy the key string (like eina_hash_direct_add()), otherwise false
Returns:
EINA_TRUE on success, otherwise EINA_FALSE on failure
See also:
ecore_thread_global_data_del()
ecore_thread_global_data_set()
ecore_thread_global_data_find()
Eina_Bool ecore_thread_global_data_del ( const char *  key)

Deletes the data corresponding to the given key from the shared hash.

Since :
2.3.1
Remarks:
If there's any data associated with key that is stored in the global hash, this function removes it from the hash and returns EINA_TRUE. If no data exists or an error occurs, it returns EINA_FALSE.
If the data is added to the hash with a free function, then it is also freed after removing it from the hash, otherwise it requires to be manually freed by the user, which means that if no other reference to it exists before calling this function, it results in a memory leak.

Note, also, that freeing data that other threads may be using results in a crash, so appropriate care must be taken by the application when that possibility exists.

Parameters:
[in]keyThe name under which the data is stored
Returns:
EINA_TRUE on success, otherwise EINA_FALSE on failure
See also:
ecore_thread_global_data_add()
void* ecore_thread_global_data_find ( const char *  key)

Gets data stored in the hash shared by all threads.

Since :
2.3.1

This finds and returns the data stored in the shared hash under the key key.

Remarks:
Keep in mind that the data returned may be used by more than one thread at the same time and no reference counting is done on it by Ecore. Freeing the data or modifying its contents may require additional precautions to be considered, depending on the application's design.
Parameters:
[in]keyThe name under which the data is stored
Returns:
The value under the given key, otherwise NULL on an error
See also:
ecore_thread_global_data_add()
ecore_thread_global_data_wait()
void* ecore_thread_global_data_set ( const char *  key,
void *  value,
Eina_Free_Cb  cb 
)

Sets some data in the hash shared by all threads.

Since :
2.3.1
Remarks:
If no data exists in the hash under the key, this function adds value in the hash under the given key and returns NULL. The key itself is copied.

If the hash already contains something under key, the data is replaced by value and the old value is returned.

NULL is also returned if either key or value is NULL, or if an error occurs.

Parameters:
[in]keyThe name under which the data is stored
[in]valueThe data to add
[in]cbThe function to free the data when removed from the hash
See also:
ecore_thread_global_data_add()
ecore_thread_global_data_del()
ecore_thread_global_data_find()
void* ecore_thread_global_data_wait ( const char *  key,
double  seconds 
)

Gets data stored in the shared hash or waits for it if it doesn't exist.

Since :
2.3.1
Remarks:
This finds and returns the data stored in the shared hash under the key key.

If there's nothing in the hash under the given key, the function blocks and waits for seconds seconds for some other thread to add it with either ecore_thread_global_data_add() or ecore_thread_global_data_set(). If after waiting there's still no data to obtain, NULL is returned.

If seconds is 0, then no waiting happens and this function works like ecore_thread_global_data_find(). If seconds is less than 0, then the function waits indefinitely.

Remarks:
Keep in mind that the data returned may be used by more than one thread at the same time and no reference counting is done on it by Ecore. Freeing the data or modifying its contents may require additional precautions to be considered, depending on the application's design.
Parameters:
[in]keyThe name under which the data is stored
[in]secondsThe amount of time in seconds to wait for the data
Returns:
The value under the given key, otherwise NULL on an error
See also:
ecore_thread_global_data_add()
ecore_thread_global_data_find()
Eina_Bool ecore_thread_local_data_add ( Ecore_Thread thread,
const char *  key,
void *  value,
Eina_Free_Cb  cb,
Eina_Bool  direct 
)

Adds some data present in the hash local to the thread.

Since :
2.3.1
Remarks:
Ecore Thread has a mechanism to share data across several worker functions that run on the same system thread. That is, the data is stored per thread and for a worker function to have access to it, it must be run by the same thread that stored the data.
When there are no more workers pending, the thread is destroyed along with the internal hash and any data left in it is freed with the given cb function.

@ This set of functions is useful to share things around several instances of a function when that thing is costly to create and can be reused, but may only be used by one function at a time.

For example, if you have a program doing requisitions to a database, these requisitions can be done in threads so that waiting for the database to respond doesn't block the UI. Each of these threads run a function, and each function is dependent on a connection to the database, which may not be able to handle more than one request at a time so for each running function you need one connection handle.

The options then are:

  • Each function opens a connection when it's called, does the work and closes the connection when it finishes. This may be costly, wasting a lot of time on resolving hostnames, negotiating permissions, and allocating memory.
  • Open the connections in the main loop and pass it to the threads using the data pointer. Even worse, it's just as costly as before and now it may even be kept with connections open doing nothing until a thread becomes available to run the function.
  • Have a way to share connection handles, so that each instance of the function can check if an available connection exists, and if it doesn't, create one and add it to the pool. When no more connections are needed, they are all closed.

The last option is the most efficient, but it requires a lot of work to be implemented properly. Using thread local data helps to achieve the same result while avoiding all the tracking work on your code. The way to use it would be at the worker function, to ask for the connection using ecore_thread_local_data_find() and if it doesn't exist, then open a new one and save it with ecore_thread_local_data_add(). Complete the work and forget about the connection handle, when everything is done the function just ends. The next worker to run on that thread checks if a connection exists and finds that it does, so the process of opening a new one has been spared. When no more workers exist, the thread is destroyed and the callback used when saving the connection is called to close it.

Remarks:
This function adds the data value to the thread data under the given key. No other value in the hash may have the same key. If you need to change the value under a key, or you don't know if one exists already, you can use ecore_thread_local_data_set().

Neither key nor value may be NULL and key gets copied in the hash, unless direct is set, in which case the string used should not be freed until the data is removed from the hash.

Remarks:
The cb function is called when the data in the hash needs to be freed, be it because it got deleted by ecore_thread_local_data_del() or because thread got terminated and the hash got destroyed. This parameter may be NULL, in which case value needs to be manually freed after removing it from the hash with either ecore_thread_local_data_del() or ecore_thread_local_data_set(), but it's very unlikely that this is what you want.

This function, and all of the others in the ecore_thread_local_data family of functions, can only be called within the worker function running in the thread. Do not call them from the main loop or from a thread other than the one represented by thread.

Parameters:
[in]threadThe thread context the data belongs to
[in]keyThe name under which the data is stored
[in]valueThe data to add
[in]cbThe function to free the data when removed from the hash
[in]directIf true, this does not copy the key string (like eina_hash_direct_add()), otherwise false
Returns:
EINA_TRUE on success, otherwise EINA_FALSE on failure
See also:
ecore_thread_local_data_set()
ecore_thread_local_data_find()
ecore_thread_local_data_del()
Eina_Bool ecore_thread_local_data_del ( Ecore_Thread thread,
const char *  key 
)

Deletes the data corresponding to the given key from the thread's hash.

Since :
2.3.1
Remarks:
If there's any data associated with key that is stored in the global hash, this function removes it from the hash and returns EINA_TRUE. If no data exists or an error occurs, it returns EINA_FALSE.
If the data is added to the hash with a free function, then it is also freed after removing it from the hash, otherwise it requires to be manually freed by the user, which means that if no other reference to it exists before calling this function, it results in a memory leak.
This function, and all the others in the ecore_thread_local_data family of functions, can only be called within the worker function running in the thread. Do not call them from the main loop or from a thread other than the one represented by thread.
Parameters:
[in]threadThe thread context the data belongs to
[in]keyThe name under which the data is stored
Returns:
EINA_TRUE on success, otherwise EINA_FALSE on failure
See also:
ecore_thread_local_data_add()
void* ecore_thread_local_data_find ( Ecore_Thread thread,
const char *  key 
)

Gets data stored in the hash local to the given thread.

Since :
2.3.1

This finds and returns the data stored in the shared hash under the key key.

Remarks:
This function, and all the others in the ecore_thread_local_data family of functions, can only be called within the worker function running in the thread. Do not call them from the main loop or from a thread other than the one represented by thread.
Parameters:
[in]threadThe thread context the data belongs to
[in]keyThe name under which the data is stored
Returns:
The value under the given key, otherwise NULL on an error
See also:
ecore_thread_local_data_add()
ecore_thread_local_data_wait()
void* ecore_thread_local_data_set ( Ecore_Thread thread,
const char *  key,
void *  value,
Eina_Free_Cb  cb 
)

Sets some data present in the hash local to the given thread.

Since :
2.3.1
Remarks:
If no data exists in the hash under the key, this function adds value in the hash under the given key and returns NULL. The key itself is copied.

If the hash already contains something under key, the data is replaced by value and the old value is returned.

NULL is also returned if either key or value are NULL, or if an error occurs.

Remarks:
This function, and all of the others in the ecore_thread_local_data family of functions, can only be called within the worker function running in the thread. Do not call them from the main loop or from a thread other than the one represented by thread.
Parameters:
[in]threadThe thread context the data belongs to
[in]keyThe name under which the data is stored
[in]valueThe data to add
[in]cbThe function to free the data when removed from the hash
See also:
ecore_thread_local_data_add()
ecore_thread_local_data_del()
ecore_thread_local_data_find()
int ecore_thread_max_get ( void  )

Gets the maximum number of threads that can run simultaneously.

This returns the maximum number of Ecore_Thread's that may be running at the same time. If this number is reached, new jobs started by either ecore_thread_run() or ecore_thread_feedback_run() are added to the respective pending queues until one of the running threads finishes its task and becomes available to run a new one.

Since :
2.3.1
Remarks:
By default, this is the number of available CPUs for the running program, or 1 if this value could not be fetched.
Returns:
The maximum possible number of Ecore_Thread's running concurrently
See also:
ecore_thread_max_set()
ecore_thread_max_reset()
void ecore_thread_max_reset ( void  )

Resets the maximum number of concurrently running threads to the default.

This resets the value returned by ecore_thread_max_get() back to its default.

Since :
2.3.1
See also:
ecore_thread_max_get()
ecore_thread_max_set()
void ecore_thread_max_set ( int  num)

Sets the maximum number of threads allowed to run simultaneously.

This sets a new value for the maximum number of concurrently running Ecore_Thread's. It must be an integer between 1 and (16 * x), where x is the number for CPUs available.

Since :
2.3.1
Parameters:
[in]numThe new maximum
See also:
ecore_thread_max_get()
ecore_thread_max_reset()

Gets the number of feedback jobs waiting for a thread to run.

This returns the number of tasks started with ecore_thread_feedback_run() that are pending and waiting for a thread to become available to run them.

Since :
2.3.1
Returns:
The number of pending threads running "feedback" jobs
int ecore_thread_pending_get ( void  )

Gets the number of short jobs waiting for a thread to run.

This returns the number of tasks started with ecore_thread_run() that are pending and waiting for a thread to become available to run them.

Since :
2.3.1
Returns:
The number of pending threads running "short" jobs

Gets the total number of pending jobs.

Since :
2.3.1
Remarks:
This is same as the sum of ecore_thread_pending_get() and ecore_thread_pending_feedback_get().
Returns:
The number of pending threads running jobs

Asks for the function in the thread to be called again at a later period.

Since :
2.3.1
Remarks:
This function should be called only from the function represented by thread.

Calling this function marks the thread for a reschedule, so as soon as it returns, it is added to the end of the list of pending tasks. If no other tasks are waiting or there are sufficient threads available, the rescheduled task is launched again immediately.

This should never return EINA_FALSE, unless it is called from the wrong thread or with the wrong arguments.

Remarks:
The func_end callback set when the thread is created is not called until the function in the thread returns without being rescheduled. Similarly, if the thread is cancelled, the reschedule does not take effect.
Parameters:
[in]threadThe current Ecore_Thread context to reschedule
Returns:
EINA_TRUE if the task is successfully rescheduled, otherwise EINA_FALSE if anything goes wrong
Ecore_Thread* ecore_thread_run ( Ecore_Thread_Cb  func_blocking,
Ecore_Thread_Cb  func_end,
Ecore_Thread_Cb  func_cancel,
const void *  data 
)

Schedules a task to run in a parallel thread to avoid locking the main loop.

This function tries to create a new thread to run func_blocking in, or if the maximum number of concurrent threads has been reached it adds it to the pending list, where it waits until a thread becomes available. The return value is an Ecore_Thread handle that can be used to cancel the thread before its completion.

Since :
2.3.1
Remarks:
This function should always return immediately, but in the rare case that Ecore is built with no thread support, func_blocking is be called here, actually blocking the main loop.
Once a thread becomes available, func_blocking is run in it until it finishes, then func_end is called from the thread containing the main loop to inform the user of its completion. While in func_blocking, no functions from the EFL can be used, except for those from Eina that are marked to be thread-safe. Even for the latter, caution needs to be taken if the data is shared across several threads.
func_end is called from the main thread when func_blocking ends, so here it's safe to use anything from the EFL freely.
The thread can also be cancelled before its completion by calling ecore_thread_cancel(), either from the main thread or func_blocking. In this case, func_cancel is called, also from the main thread to inform of this happening. If the thread could not be created, this function is called and its thread parameter is NULL. It's also safe to call any EFL function here, as it is running in the main thread.
Inside func_blocking, it's possible to call ecore_thread_reschedule() to tell Ecore that this function should be called again.
Be aware that no assumptions can be made about the order in which the func_end callbacks for each task are called. Once the function is running in a different thread, it's the OS that handles its running schedule, and different functions may take longer to finish than others. Also remember that just starting several tasks together doesn't mean they are going to run at the same time. Ecore schedules them based on the number of threads available for the particular system it's running in, so some of the jobs started may be waiting until another one finishes before it can execute its own func_blocking.
Parameters:
[in]func_blockingThe function that should run in another thread
[in]func_endThe function to call from the main loop when func_blocking completes its task successfully (may be NULL)
[in]func_cancelThe function to call from the main loop if the thread running func_blocking is cancelled or fails to start (may be NULL)
[in]dataThe user context data to pass to all callbacks
Returns:
A new thread handler, otherwise NULL on failure
See also:
ecore_thread_feedback_run()
ecore_thread_cancel()
ecore_thread_reschedule()
ecore_thread_max_set()