Difference between revisions of "User:Kr/A Thread's Life/Example Code"

From Apache OpenOffice Wiki
Jump to: navigation, search
m (Now creates randomly activities and daemons ...)
m (Worked in some code review comments, improved structure.)
Line 16: Line 16:
 
     Daemon * next_;
 
     Daemon * next_;
  
 +
    void (*func_)(void);
 
     pthread_t hd_;
 
     pthread_t hd_;
 
};
 
};
Line 23: Line 24:
 
static pthread_mutex_t daemons_mutex_ = PTHREAD_MUTEX_INITIALIZER;
 
static pthread_mutex_t daemons_mutex_ = PTHREAD_MUTEX_INITIALIZER;
  
static Daemon * new_daemon(void) {
+
/* Unregister the given "Daemon" struct
    Daemon * daemon = malloc(sizeof(Daemon));
+
  and terminate this thread.            */
   
+
static void daemon_terminate(Daemon * daemon) {
    pthread_mutex_lock(&daemons_mutex_);
+
 
+
    daemon->previous_ = NULL;
+
    daemon->next_ = firstDaemon_;
+
 
+
    if (firstDaemon_)
+
        firstDaemon_->previous_ = daemon;
+
 
+
    firstDaemon_ = daemon;
+
 
+
    ++ daemons_;
+
 
+
    pthread_mutex_unlock(&daemons_mutex_);
+
 
+
    return daemon;
+
}
+
 
+
static void end_of_daemon(void * context) {
+
    Daemon * daemon = (Daemon *)context;
+
 
+
 
     pthread_mutex_lock(&daemons_mutex_);
 
     pthread_mutex_lock(&daemons_mutex_);
 
     -- daemons_;
 
     -- daemons_;
Line 62: Line 43:
 
     free(daemon);
 
     free(daemon);
  
     // This thread is not in the list of daemons anymore,  
+
     /* This daemon is not in the list of daemons anymore,  
    // and therefor can't be joined.
+
      and therefor can't be joined. */
 
     pthread_detach(pthread_self());
 
     pthread_detach(pthread_self());
 
}
 
}
  
static void create_a_daemon(void);
+
static void * daemon_func(Daemon * daemon) {
 +
    pthread_cleanup_push((void (*)(void *))daemon_terminate, daemon);
  
static void * daemon_func(void * para) {
+
     daemon->func_();
     pthread_cleanup_push (end_of_daemon, para);
+
 
+
    int n = 10;
+
    while (n) {
+
        sleep(1);
+
       
+
        if (rand() < (RAND_MAX / (daemons_ + 1)))
+
            create_a_daemon();
+
 
+
        -- n;
+
    }
+
  
 
     pthread_cleanup_pop(1);
 
     pthread_cleanup_pop(1);
Line 87: Line 58:
 
}
 
}
  
static void join_daemons(void) {
+
/* Create a "Daemon" struct and register it.
     fprintf(stderr, "%s ... ", __PRETTY_FUNCTION__, pthread_self());
+
  Start the "Daemon". */
 +
static void daemon_create_(void (*func)(void)) {
 +
    Daemon * daemon = malloc(sizeof(Daemon));
 +
    daemon->func_ = func;
 +
 
 +
    pthread_mutex_lock(&daemons_mutex_);
 +
 
 +
    daemon->previous_ = NULL;
 +
    daemon->next_ = firstDaemon_;
 +
 
 +
    if (firstDaemon_)
 +
        firstDaemon_->previous_ = daemon;
 +
 
 +
    firstDaemon_ = daemon;
 +
    ++ daemons_;
 +
 
 +
    pthread_mutex_unlock(&daemons_mutex_);
 +
 
 +
 
 +
    pthread_create(&daemon->hd_, NULL, (void *(*)(void *))daemon_func, daemon);
 +
}
 +
 
 +
/* Ask all registered daemons for termination and join them. */
 +
static void daemons_join_(void) {
 +
     fprintf(stderr, "Joining daemons ... ");
  
 
     pthread_mutex_lock(&daemons_mutex_);
 
     pthread_mutex_lock(&daemons_mutex_);
 
     while (firstDaemon_) {
 
     while (firstDaemon_) {
 
         pthread_t hd;
 
         pthread_t hd;
 
 
         memcpy(&hd, &firstDaemon_->hd_, sizeof(pthread_t));
 
         memcpy(&hd, &firstDaemon_->hd_, sizeof(pthread_t));
  
Line 104: Line 98:
  
 
         pthread_join(hd, NULL);
 
         pthread_join(hd, NULL);
 +
        fprintf(stderr, "# ");
  
 
         pthread_mutex_lock(&daemons_mutex_);
 
         pthread_mutex_lock(&daemons_mutex_);
Line 110: Line 105:
  
 
     fprintf(stderr, "done\n");
 
     fprintf(stderr, "done\n");
}
 
 
static void create_a_daemon(void) {
 
    Daemon * daemon = new_daemon();
 
    pthread_create(&daemon->hd_, NULL, daemon_func, daemon);
 
 
}
 
}
  
Line 121: Line 111:
 
static pthread_mutex_t activity_mutex_ = PTHREAD_MUTEX_INITIALIZER;
 
static pthread_mutex_t activity_mutex_ = PTHREAD_MUTEX_INITIALIZER;
  
static void new_activity(void) {
+
static void activity_terminate_(void) {
    pthread_mutex_lock(&activity_mutex_);
+
    ++ activities_;
+
    pthread_mutex_unlock(&activity_mutex_);
+
}
+
 
+
static void end_of_activity(void) {
+
 
     int use_exit = 0;
 
     int use_exit = 0;
 
     pthread_mutex_lock(&activity_mutex_);
 
     pthread_mutex_lock(&activity_mutex_);
Line 142: Line 126:
 
}
 
}
  
static void create_an_activity(void);
+
static void * activity_func_(void (*func)(void)) {
 +
    func();
  
static void * activity(void * gr) {
+
     activity_terminate_();
     int n = 3;
+
    while (n) {
+
        sleep(1);
+
       
+
        if (rand() < (RAND_MAX / (activities_ + 1)))
+
            create_an_activity();
+
  
        if (rand() < (RAND_MAX / (daemons_ + 1)))
+
     return NULL;
            create_a_daemon();
+
 
+
        -- n;
+
     }
+
 
+
    end_of_activity();
+
 
}
 
}
  
static void create_an_activity(void) {
+
static void activity_create_(void (*func)(void)) {
 
     pthread_attr_t attr;
 
     pthread_attr_t attr;
 
      
 
      
 +
    /* Activities need to be detached. */
 
     pthread_attr_init(&attr);
 
     pthread_attr_init(&attr);
 
     pthread_attr_setdetachstate(&attr, 1);
 
     pthread_attr_setdetachstate(&attr, 1);
 
      
 
      
 
     pthread_t bt;
 
     pthread_t bt;
     pthread_create(&bt, &attr, activity, NULL);
+
     pthread_create(&bt, &attr, (void *(*)(void *))activity_func_, func);
 
     pthread_attr_destroy(&attr);
 
     pthread_attr_destroy(&attr);
 
      
 
      
     new_activity();
+
     pthread_mutex_lock(&activity_mutex_);
 +
    ++ activities_;
 +
    pthread_mutex_unlock(&activity_mutex_);
 
}
 
}
  
  
static void * print_daemon(void * gr) {
+
/* Some daemon functions. */
 +
static void log_threads(void) {
 +
    fprintf(stderr, "activities: %u  daemons: %u\n", activities_, daemons_);
 +
}
 +
 
 +
static void log_daemon_(void) {
 
     while (1) {
 
     while (1) {
         fprintf(stderr, "activities: %u  daemons: %u\n", activities_, daemons_);
+
         log_threads();
  
         sleep(1);
+
         sleep(1); /* This is a cancellation point. */
 
     }
 
     }
  
Line 185: Line 166:
 
}
 
}
  
 +
static void some_daemon_(void) {
 +
    int n = 10;
 +
    while (n) {
 +
        sleep(1);
 +
       
 +
        if (rand() < (RAND_MAX / (daemons_ + 1)))
 +
            daemon_create_(some_daemon_);
  
static pthread_t print_daemon_hd;
+
        -- n;
 +
    }
 +
}
  
static void join_print_daemon(void) {
+
/* Some activity functions. */
     fprintf(stderr, "%s - tid: %x\n", __PRETTY_FUNCTION__, pthread_self());
+
static void some_activity_(void) {
 +
     int n = 3;
 +
    while (n) {
 +
        sleep(1);
 +
       
 +
        if (rand() < (RAND_MAX / (activities_ + 1)))
 +
            activity_create_(some_activity_);
  
    // You may want to cancel a thread actively during termination:
+
        if (rand() < (RAND_MAX / (daemons_ + 1)))
    pthread_cancel(print_daemon_hd);
+
            daemon_create_(some_daemon_);
    // respectively
+
    //pthread_kill();
+
  
    fprintf(stderr, "%s - joining print daemon ...\n", __PRETTY_FUNCTION__);
+
        -- n;
     pthread_join(print_daemon_hd, NULL);
+
     }
 
}
 
}
  
static void create_print_daemon(void) {
+
static void main_activity_(void) {
     pthread_create(&print_daemon_hd, NULL, print_daemon, NULL);
+
     srand (time (0));
    atexit(join_print_daemon);
+
}
+
  
 +
    daemon_create_(log_daemon_);
 +
    daemon_create_(some_daemon_);
 +
    activity_create_(some_activity_);
  
static void print_something(void) {
+
    sleep(3);
     fprintf(stderr, "activities: %u  daemons: %u\n", activities_, daemons_);
+
    daemon_create_(some_daemon_);
 +
 
 +
     sleep(5);
 
}
 
}
  
 
int main(void) {
 
int main(void) {
     srand (time (0));
+
     atexit(log_threads);
 +
    atexit(daemons_join_);
  
     atexit(print_something);
+
     main_activity_();
    atexit(join_daemons);
+
  
     create_print_daemon();
+
     activity_terminate_();
    create_a_daemon();
+
    create_an_activity();
+
 
+
    sleep(3);
+
    create_a_daemon();
+
 
+
    sleep(5);
+
    end_of_activity();
+
  
 
     return 0;
 
     return 0;
 
}
 
}
 
</code>
 
</code>

Revision as of 09:45, 6 September 2007

[cpp] // // gcc -g main.c -o main.bin -lpthread //

  1. include <stdlib.h>
  2. include <stdio.h>
  3. include <pthread.h>
  4. include <string.h>


typedef struct Daemon Daemon;

struct Daemon {

   Daemon * previous_;
   Daemon * next_;
   void (*func_)(void);
   pthread_t hd_;

};

static Daemon * firstDaemon_ = NULL; static unsigned int daemons_ = 0; static pthread_mutex_t daemons_mutex_ = PTHREAD_MUTEX_INITIALIZER;

/* Unregister the given "Daemon" struct

  and terminate this thread.            */

static void daemon_terminate(Daemon * daemon) {

   pthread_mutex_lock(&daemons_mutex_);
   -- daemons_;
   if (daemon->next_)
       daemon->next_->previous_ = daemon->previous_;
   if (daemon->previous_)
       daemon->previous_->next_ = daemon->next_;
   if (firstDaemon_ == daemon)
       firstDaemon_ = daemon->next_;
   pthread_mutex_unlock(&daemons_mutex_);
   free(daemon);
   /* This daemon is not in the list of daemons anymore, 
      and therefor can't be joined. */
   pthread_detach(pthread_self());

}

static void * daemon_func(Daemon * daemon) {

   pthread_cleanup_push((void (*)(void *))daemon_terminate, daemon);
   daemon->func_();
   pthread_cleanup_pop(1);
   return NULL;

}

/* Create a "Daemon" struct and register it.

  Start the "Daemon". */

static void daemon_create_(void (*func)(void)) {

   Daemon * daemon = malloc(sizeof(Daemon));
   daemon->func_ = func;
   pthread_mutex_lock(&daemons_mutex_);
   daemon->previous_ = NULL;
   daemon->next_ = firstDaemon_;
   if (firstDaemon_) 
       firstDaemon_->previous_ = daemon;
   firstDaemon_ = daemon;
   ++ daemons_;
   pthread_mutex_unlock(&daemons_mutex_);


   pthread_create(&daemon->hd_, NULL, (void *(*)(void *))daemon_func, daemon);

}

/* Ask all registered daemons for termination and join them. */ static void daemons_join_(void) {

   fprintf(stderr, "Joining daemons ... ");
   pthread_mutex_lock(&daemons_mutex_);
   while (firstDaemon_) {
       pthread_t hd;
       memcpy(&hd, &firstDaemon_->hd_, sizeof(pthread_t));
       pthread_mutex_unlock(&daemons_mutex_);
       // We may want to cancel a thread actively during termination:
       pthread_cancel(hd);
       // respectively
       //pthread_kill();
       pthread_join(hd, NULL);
       fprintf(stderr, "# ");
       pthread_mutex_lock(&daemons_mutex_);
   }
   pthread_mutex_unlock(&daemons_mutex_);
   fprintf(stderr, "done\n");

}


static unsigned int activities_ = 1; // The "main" thread is the first "activity". static pthread_mutex_t activity_mutex_ = PTHREAD_MUTEX_INITIALIZER;

static void activity_terminate_(void) {

   int use_exit = 0;
   pthread_mutex_lock(&activity_mutex_);
   -- activities_;
   use_exit = activities_ == 0;
   pthread_mutex_unlock(&activity_mutex_);
  // Terminate the process, if this was the last "activity"!
   if (use_exit) 
       exit(0);
   else
       pthread_exit(NULL);

}

static void * activity_func_(void (*func)(void)) {

   func();
   activity_terminate_();
   return NULL;

}

static void activity_create_(void (*func)(void)) {

   pthread_attr_t attr;
   
   /* Activities need to be detached. */
   pthread_attr_init(&attr);
   pthread_attr_setdetachstate(&attr, 1);
   
   pthread_t bt;
   pthread_create(&bt, &attr, (void *(*)(void *))activity_func_, func);
   pthread_attr_destroy(&attr);
   
   pthread_mutex_lock(&activity_mutex_);
   ++ activities_;
   pthread_mutex_unlock(&activity_mutex_);

}


/* Some daemon functions. */ static void log_threads(void) {

   fprintf(stderr, "activities: %u  daemons: %u\n", activities_, daemons_);

}

static void log_daemon_(void) {

   while (1) {
       log_threads();
       sleep(1); /* This is a cancellation point. */
   }
   pthread_exit(NULL);

}

static void some_daemon_(void) {

   int n = 10;
   while (n) {
       sleep(1);
       
       if (rand() < (RAND_MAX / (daemons_ + 1)))
           daemon_create_(some_daemon_);
       -- n;
   }

}

/* Some activity functions. */ static void some_activity_(void) {

   int n = 3;
   while (n) {
       sleep(1);
       
       if (rand() < (RAND_MAX / (activities_ + 1)))
           activity_create_(some_activity_);
       if (rand() < (RAND_MAX / (daemons_ + 1)))
           daemon_create_(some_daemon_);
       -- n;
   }

}

static void main_activity_(void) {

   srand (time (0));
   daemon_create_(log_daemon_);
   daemon_create_(some_daemon_);
   activity_create_(some_activity_);
   sleep(3);
   daemon_create_(some_daemon_);
   sleep(5);

}

int main(void) {

   atexit(log_threads);
   atexit(daemons_join_);
   main_activity_();
   activity_terminate_();
   return 0;

}

Personal tools