THR_CREATE(3C) Standard C Library Functions THR_CREATE(3C)

NAME


thr_create - create a thread

SYNOPSIS


cc -mt [ flag... ] file...[ library... ]
#include <thread.h>

int thr_create(void *stack_base, size_t stack_size,
void *(*start_func) (void*), void *arg, long flags,
thread_t *new_thread_ID);


DESCRIPTION


Thread creation adds a new thread of control to the current process.
The procedure main() is a single thread of control. Each thread
executes concurrently with all other threads within the calling
process and with other threads from other active processes.


Although a newly created thread shares all of the calling process's
global data with the other threads in the process, it has its own set
of attributes and private execution stack. The new thread inherits
the calling thread's signal mask and scheduling priority. Pending
signals for a new thread are not inherited and will be empty.


The call to create a thread takes the address of a user-defined
function, specified by start_func, as one of its arguments. This
function is the complete execution routine for the new thread.


The lifetime of a thread begins with the successful return from
thr_create(), which calls start_func() and ends with one of the
following:

o the normal completion of start_func(),

o an explicit call to thr_exit(3C), or

o the conclusion of the calling process (see exit(2)).


The new thread performs by calling the function defined by start_func
with only one argument, arg. If more than one argument needs to be
passed to start_func, the arguments can be packed into a structure,
the address of which can be passed to arg.


If start_func returns, the thread terminates with the exit status set
to the start_func return value (see thr_exit(3C)).


When the thread from which main() originated returns, the effect is
the same as if an implicit call to exit() were made using the return
value of main() as the exit status. This behavior differs from a
start_func return. If main() calls thr_exit(3C), only the main
thread exits, not the entire process.


If the thread creation fails, a new thread is not created and the
contents of the location referenced by the pointer to the new thread
are undefined.


The flags argument specifies which attributes are modifiable for the
created thread. The value in flags is determined by the bitwise
inclusive-OR of the following:

THR_BOUND
This flag is obsolete and is maintained for
compatibility.


THR_DETACHED
This flag affects the detachstate attribute of the
thread. The new thread is created detached. The
exit status of a detached thread is not accessible
to other threads. Its thread ID and other resources
may be re-used as soon as the thread terminates.
thr_join(3C) will not wait for a detached thread.


THR_NEW_LWP
This flag is obsolete and is maintained for
compatibility.


THR_SUSPENDED
This flag affects the suspended attribute of the
thread. The new thread is created suspended and
will not execute start_func until it is started by
thr_continue().


THR_DAEMON
This flag affects the daemon attribute of the
thread. In addition to being created detached
(THR_DAEMON implies THR_DETACHED), the thread is
marked as a daemon. Daemon threads do not interfere
with the exit conditions for a process. A process
will terminate when the last non-daemon thread
exits or the process calls exit(2). Also, a thread
that is waiting in thr_join(3C) for any thread to
terminate will return EDEADLK when all remaining
threads in the process are either daemon threads or
other threads waiting in thr_join(). Daemon threads
are most useful in libraries that want to use
threads.


Default thread creation:

thread_t tid;
void *start_func(void *), *arg;
thr_create(NULL, 0, start_func, arg, 0, &tid);


Create a detached thread whose thread ID we do not care about:

thr_create(NULL, 0, start_func, arg, THR_DETACHED, NULL);


If stack_base is not NULL, the new thread uses the stack beginning at
the address specified by stack_base and continuing for stack_size
bytes, where stack_size must be greater than or equal to
THR_MIN_STACK. If stack_base is NULL, thr_create() allocates a stack
for the new thread with at least stack_size bytes. If stack_size is
0, a default size is used. If stack_size is not 0, it must be greater
than or equal to THR_MIN_STACK. See NOTES.


When new_thread_ID is not NULL, it points to a location where the ID
of the new thread is stored if thr_create() is successful. The ID is
only valid within the calling process.

RETURN VALUES


If successful, the thr_create() function returns 0. Otherwise, an
error value is returned to indicate the error.

ERRORS


EAGAIN
A resource control limit on the total number of threads in
a process, task, project, or zone has been exceeded or
some system resource has been exceeded.


EINVAL
The stack_base argument is not NULL and stack_size is less
than THR_MIN_STACK, or the stack_base argument is NULL and
stack_size is not 0 and is less than THR_MIN_STACK.


ENOMEM
The system cannot allocate stack for the thread.


The thr_create() function may use mmap() to allocate thread stacks
from MAP_PRIVATE, MAP_NORESERVE, and MAP_ANON memory mappings if
stack_base is NULL, and consequently may return upon failure the
relevant error values returned by mmap(). See the mmap(2) manual page
for these error values.

EXAMPLES


The following is an example of concurrency with multithreading. Since
POSIX threads and Solaris threads are fully compatible even within
the same process, this example uses pthread_create() if you execute
a.out 0, or thr_create() if you execute a.out 1.


Five threads are created that simultaneously perform a time-consuming
function, sleep(10). If the execution of this process is timed, the
results will show that all five individual calls to sleep for ten-
seconds completed in about ten seconds, even on a uniprocessor. If a
single-threaded process calls sleep(10) five times, the execution
time will be about 50-seconds.


The command-line to time this process is:


/usr/bin/time a.out 0 (for POSIX threading)


or


/usr/bin/time a.out 1 (for Solaris threading)

Example 1: An example of concurrency with multithreading.



#define _REENTRANT /* basic 3-lines for threads */
#include <pthread.h>
#include <thread.h>
#define NUM_THREADS 5
#define SLEEP_TIME 10

void *sleeping(void *); /* thread routine */
int i;
thread_t tid[NUM_THREADS]; /* array of thread IDs */

int
main(int argc, char *argv[])
{
if (argc == 1) {
printf("use 0 as arg1 to use pthread_create()\n");
printf("or use 1 as arg1 to use thr_create()\n");
return (1);
}

switch (*argv[1]) {
case '0': /* POSIX */
for ( i = 0; i < NUM_THREADS; i++)
pthread_create(&tid[i], NULL, sleeping,
(void *)SLEEP_TIME);
for ( i = 0; i < NUM_THREADS; i++)
pthread_join(tid[i], NULL);
break;

case '1': /* Solaris */
for ( i = 0; i < NUM_THREADS; i++)
thr_create(NULL, 0, sleeping, (void *)SLEEP_TIME, 0,
&tid[i]);
while (thr_join(0, NULL, NULL) == 0)
continue;
break;
} /* switch */
printf("main() reporting that all %d threads have
terminated\n", i);
return (0);
} /* main */

void *
sleeping(void *arg)
{
int sleep_time = (int)arg;
printf("thread %d sleeping %d seconds ...\n", thr_self(),
sleep_time);
sleep(sleep_time);
printf("\nthread %d awakening\n", thr_self());
return (NULL);
}


Had main() not waited for the completion of the other threads (using
pthread_join(3C) or thr_join(3C)), it would have continued to process
concurrently until it reached the end of its routine and the entire
process would have exited prematurely (see exit(2)).


Example 2: Creating a default thread with a new signal mask.




The following example demonstrates how to create a default thread
with a new signal mask. The new_mask argument is assumed to have a
value different from the creator's signal mask (orig_mask). The
new_mask argument is set to block all signals except for SIGINT. The
creator's signal mask is changed so that the new thread inherits a
different mask, and is restored to its original value after
thr_create() returns.


This example assumes that SIGINT is also unmasked in the creator.
If it is masked by the creator, then unmasking the signal opens the
creator to this signal. The other alternative is to have the new
thread set its own signal mask in its start routine.


thread_t tid;
sigset_t new_mask, orig_mask;
int error;

(void)sigfillset(&new_mask);
(void)sigdelset(&new_mask, SIGINT);
(void)thr_sigsetmask(SIG_SETMASK, &new_mask, &orig_mask);
error = thr_create(NULL, 0, do_func, NULL, 0, &tid);
(void)thr_sigsetmask(SIG_SETMASK, &orig_mask, NULL);


ATTRIBUTES


See attributes(7) for descriptions of the following attributes:


+---------------+-----------------+
|ATTRIBUTE TYPE | ATTRIBUTE VALUE |
+---------------+-----------------+
|MT-Level | MT-Safe |
+---------------+-----------------+

SEE ALSO


exit(2), getrlimit(2), mmap(2), exit(3C), sleep(3C), thr_exit(3C),
thr_join(3C), thr_min_stack(3C), thr_setconcurrency(3C),
thr_suspend(3C), attributes(7), standards(7), threads(7)

NOTES


Since multithreaded-application threads execute independently of each
other, their relative behavior is unpredictable. It is therefore
possible for the thread executing main() to finish before all other
user-application threads.


Using thr_join(3C) in the following syntax,

while (thr_join(0, NULL, NULL) == 0)
continue;


will cause the invoking thread (which may be main()) to wait for the
termination of all non-daemon threads, excluding threads that are
themselves waiting in thr_join(); however, the second and third
arguments to thr_join() need not necessarily be NULL.


A thread has not terminated until thr_exit() has finished. The only
way to determine this is by thr_join(). When thr_join() returns a
departed thread, it means that this thread has terminated and its
resources are reclaimable. For instance, if a user specified a stack
to thr_create(), this stack can only be reclaimed after thr_join()
has reported this thread as a departed thread. It is not possible
to determine when a detached thread has terminated. A detached
thread disappears without leaving a trace.


Typically, thread stacks allocated by thr_create() begin on page
boundaries and any specified (a red-zone) size is rounded up to the
next page boundary. A page with no access permission is appended to
the top of the stack so that most stack overflows will result in a
SIGSEGV signal being sent to the offending thread. Thread stacks
allocated by the caller are used as is.


Using a default stack size for the new thread, instead of passing a
user-specified stack size, results in much better thr_create()
performance. The default stack size for a user-thread is 1 megabyte
in a 32-bit process and 2 megabyte in a 64-bit process.


A user-specified stack size must be greater than or equal to
THR_MIN_STACK. A minimum stack size may not accommodate the stack
frame for the user thread function start_func. If a stack size is
specified, it must accommodate start_func requirements and the
functions that it may call in turn, in addition to the minimum
requirement.


It is usually very difficult to determine the runtime stack
requirements for a thread. THR_MIN_STACK specifies how much stack
storage is required to execute a trivial start_func. The total
runtime requirements for stack storage are dependent on the storage
required to do runtime linking, the amount of storage required by
library runtimes (like printf()) that your thread calls. Since these
storage parameters are not known before the program runs, it is best
to use default stacks. If you know your runtime requirements or
decide to use stacks that are larger than the default, then it makes
sense to specify your own stacks.

March 16, 2009 THR_CREATE(3C)

tribblix@gmail.com :: GitHub :: Privacy