Title: COMP309/509 - Lecture 8 class: middle, center, inverse
#include <pthread.h>
int pthread_create(pthread_t *thread,
pthread_attr_t * attr,
void * (*start_routine)(void *),
void * arg);
Continued from last slide:
#include <pthread.h>
int pthread_join(pthread_t thread, void *value);
waitpid()
.pthread_join(t,r)
suspends the calling thread until the thread identified by t terminates.r
is not NULL
, then return value of t is stored in the location pointed to by r
.pthread_exit
, orPTHREAD_CANCELED
if t was cancelled.#include <pthread.h>
void pthread_exit(void *value);
pthread_join
and specifies the terminating thread.pthread_exit
!atexit()
routines that may exist.#include "process.h"
#include "headers.h"
int main(void){
pthread_t thread_1, thread_2;
char *msg_1 = "Hello", *msg_2 = "World";
/* error checking left out for legibility */
pthread_create(&thread_1, NULL, pr_msg_fn, (void *)msg_1);
pthread_create(&thread_2, NULL, pr_msg_fn, (void *)msg_2);
return 0;
}
A few points…
If we run this program, we may actually not see any output. Why?
exit
is a process primitive.sleep
.sleep
is process primitive!!man 3 sleep
on a Linux machine, and then look at sleep in the standard.pthread_join
to sort this out:#include "process.h"
#include "headers.h"
int main(void){
pthread_t thread_1, thread_2;
char *msg_1 = "Hello", *msg_2 = "World";
/* error checking left out for legibility */
pthread_create(&thread_1, NULL, pr_msg_fn, (void *)msg_1);
pthread_create(&thread_2, NULL, pr_msg_fn, (void *)msg_2);
pthread_join(thread_1,NULL);
pthread_join(thread_2,NULL);
fprintf(stderr, "\n");
return 0;
}
#include "process.h"
#include "headers.h"
int main(void){
int i;
pthread_t threads[2];
char *msgs[] = {"Hello", "World"};
/* error checking left out for legibility */
for(i = 0; i < 2 ; i++)
pthread_create(&threads[i], NULL, pr_msg_fn, (void *)msgs[i]);
for(i = 0; i < 2 ; i++)
pthread_join(threads[i],NULL);
fprintf(stderr, "\n");
return 0;
}
struct
could be huge)pthread_join
’s second argument.struct
s to store things:#include "headers.h"
/* error checking left out for legibility */
/* stuff to pass into thread */
typedef struct thread_arg {
int* id;
char* str;
float cost; } arg_t;
/* stuff thread returns */
typedef struct thread_val {
int id;
float cost; } val_t;
void *entry_point(void *arg){
arg_t input;
val_t *output;
input = *((arg_t *)arg);
output = (val_t *)malloc(sizeof(val_t));
fprintf(stderr, input.str, *input.id, input.cost);
(*input.id)++;
output->id = *input.id;
output->cost = input.cost * (*input.id);
pthread_exit(output);
}
int main(int argc, char *argv[]){
pthread_t thread_id;
arg_t thread_arg;
val_t *thread_val;
int local = 7;
char str[] = "Me be threaded: %d %f!!\n";
fprintf(stderr, "local is now = %d\n", local);
thread_arg.id = &local;
thread_arg.str = str;
thread_arg.cost = 3.14;
pthread_create(&thread_id,NULL,entry_point, &thread_arg);
pthread_join(thread_id,(void **)&thread_val);
fprintf(stderr,
"the thread produced: %d %f\n",
thread_val->id,
thread_val->cost);
fprintf(stderr, "local is now = %d\n", local);
exit(EXIT_SUCCESS);
}
entry_point
#include "headers.h"
/* error checking left out for legibility */
/* size of thread bevy */
#define TH_NO 10
/* stuff to pass into thread */
typedef struct thread_arg {
int id;
float cost; } arg_t;
/* stuff thread returns */
typedef struct thread_val {
int id;
float cost; } val_t;
void *entry_point(void *arg){
arg_t input;
val_t *output;
announce();
input = *((arg_t *)arg);
output = (val_t *)malloc(sizeof(val_t));
output->id = input.id;
output->cost = input.cost * input.id;
pthread_exit(output); }
void announce(void){
fprintf(stderr,
"Thread %lu of Process %ld with Parent %ld\n",
(unsigned long int)pthread_self(), (long)getpid(), (long)getppid());
}
int main(int argc, char *argv[]){
int i;
pthread_t thread_ids[TH_NO];
arg_t thread_args[TH_NO];
val_t *thread_vals[TH_NO];
announce();
for(i = 0; i < TH_NO; i++){
thread_args[i].id = i;
thread_args[i].cost = 3.14;
}
...
...
for(i = 0; i < TH_NO; i++){
pthread_create(&thread_ids[i], NULL, entry_point, &thread_args[i]);
}
for(i = 0; i < TH_NO; i++)
pthread_join(thread_ids[i],(void **)&thread_vals[i]);
for(i = 0; i < TH_NO; i++)
fprintf(stderr,
"Thread %d produced: %d %f\n",
i,
thread_vals[i]->id,
thread_vals[i]->cost);
exit(EXIT_SUCCESS);
}
for(i = 0; i < TH_NO; i++)
pthread_create(&thread_ids[i],NULL,entry_point,(void *)&i);
i
is changing in the loop)There are three recognised models for implementing at the processor level:
1:1 - kernel-level threading: In this apprach, each thread to a single shedulable entity in the kernel (one thread per kernel-level process)
N:1 - user-level threading: In this approach, all threads map to a single entity in the kernel.
M:N - hybrid threading : In this appraoch, M threads map to N processes.
The basic distiction is The basic distinction is who schedules the thread, and who does it compete with for other resources.
PTHREAD_THREADS_MAX
in the Linux Threads/glibc sources and recompile.PTHREAD_THREADS_MAX
is not defined - it’s not a POSIX requirement.class: middle, center, inverse