Main Page | Alphabetical List | Data Structures | Directories | File List | Data Fields | Globals

heartbeat.c

Go to the documentation of this file.
00001 /* ==================================================================== 
00002  * The Kannel Software License, Version 1.0 
00003  * 
00004  * Copyright (c) 2001-2008 Kannel Group  
00005  * Copyright (c) 1998-2001 WapIT Ltd.   
00006  * All rights reserved. 
00007  * 
00008  * Redistribution and use in source and binary forms, with or without 
00009  * modification, are permitted provided that the following conditions 
00010  * are met: 
00011  * 
00012  * 1. Redistributions of source code must retain the above copyright 
00013  *    notice, this list of conditions and the following disclaimer. 
00014  * 
00015  * 2. Redistributions in binary form must reproduce the above copyright 
00016  *    notice, this list of conditions and the following disclaimer in 
00017  *    the documentation and/or other materials provided with the 
00018  *    distribution. 
00019  * 
00020  * 3. The end-user documentation included with the redistribution, 
00021  *    if any, must include the following acknowledgment: 
00022  *       "This product includes software developed by the 
00023  *        Kannel Group (http://www.kannel.org/)." 
00024  *    Alternately, this acknowledgment may appear in the software itself, 
00025  *    if and wherever such third-party acknowledgments normally appear. 
00026  * 
00027  * 4. The names "Kannel" and "Kannel Group" must not be used to 
00028  *    endorse or promote products derived from this software without 
00029  *    prior written permission. For written permission, please  
00030  *    contact org@kannel.org. 
00031  * 
00032  * 5. Products derived from this software may not be called "Kannel", 
00033  *    nor may "Kannel" appear in their name, without prior written 
00034  *    permission of the Kannel Group. 
00035  * 
00036  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED 
00037  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 
00038  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 
00039  * DISCLAIMED.  IN NO EVENT SHALL THE KANNEL GROUP OR ITS CONTRIBUTORS 
00040  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,  
00041  * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT  
00042  * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR  
00043  * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,  
00044  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE  
00045  * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,  
00046  * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
00047  * ==================================================================== 
00048  * 
00049  * This software consists of voluntary contributions made by many 
00050  * individuals on behalf of the Kannel Group.  For more information on  
00051  * the Kannel Group, please see <http://www.kannel.org/>. 
00052  * 
00053  * Portions of this software are based upon software originally written at  
00054  * WapIT Ltd., Helsinki, Finland for the Kannel project.  
00055  */ 
00056 
00057 /*
00058  * heartbeat.c - thread for sending heartbeat Msgs to bearerbox
00059  */
00060 
00061 #include <signal.h>
00062 
00063 #include "gwlib/gwlib.h"
00064 #include "msg.h"
00065 #include "heartbeat.h"
00066 
00067 /*
00068  * Each running heartbeat gets one of these.  They are collected in
00069  * the heartbeats List.
00070  */
00071 struct hb_info {
00072     hb_send_func_t *send_func;
00073     double freq;
00074     hb_load_func_t *load_func;
00075     long thread;
00076     volatile sig_atomic_t running;
00077 };
00078 
00079 /* List of struct hb_info. */
00080 static List *heartbeats = NULL;
00081 
00082 /*
00083  * Look for a hb_info in a list, by thread number.
00084  */
00085 static int find_hb(void *item, void *pattern)
00086 {
00087     long *threadnrp;
00088     struct hb_info *info;
00089 
00090     info = item;
00091     threadnrp = pattern;
00092 
00093     return info->thread == *threadnrp;
00094 }
00095 
00096 static void heartbeat_thread(void *arg)
00097 {
00098     struct hb_info *info;
00099     time_t last_hb;
00100 
00101     info = arg;
00102     last_hb = 0;
00103 
00104     while (info->running) {
00105         Msg *msg;
00106 
00107         gwthread_sleep(info->freq);
00108 
00109         /*
00110          * Because the sleep can be interrupted, we might end up sending
00111          * heartbeats faster than the configured heartbeat frequency.
00112          * This is not bad unless we send them way too fast.  Make sure
00113          * our frequency is not more than twice the configured one.
00114          */
00115         if (difftime(time(NULL), last_hb) < info->freq / 2)
00116             continue;
00117 
00118         msg = msg_create(heartbeat);
00119         if (NULL != info->load_func)
00120             msg->heartbeat.load = info->load_func();
00121         info->send_func(msg);
00122         last_hb = time(NULL);
00123     }
00124 }
00125 
00126 long heartbeat_start(hb_send_func_t *send_func, double freq,
00127                      hb_load_func_t *load_func)
00128 {
00129     struct hb_info *info;
00130 
00131     /* can't start with send_funct NULL */
00132     if (send_func == NULL)
00133         return -1;
00134 
00135     info = gw_malloc(sizeof(*info));
00136     info->send_func = send_func;
00137     info->freq = (freq <= 0 ? DEFAULT_HEARTBEAT : freq);
00138     info->load_func = load_func;
00139     info->running = 1;
00140     info->thread = gwthread_create(heartbeat_thread, info);
00141     if (info->thread >= 0) {
00142     if (heartbeats == NULL)
00143         heartbeats = gwlist_create();
00144     gwlist_append(heartbeats, info);
00145         return info->thread;
00146     } else {
00147         gw_free(info);
00148         return -1;
00149     }
00150 }
00151 
00152 /*
00153  * function : heartbeat_stop
00154  * arguments: long hb_thread, the thread number of the heartbeat
00155  *            that is wished to be stopped.
00156  *            if hb_thread == ALL_HEARTBEATS then all heartbeats
00157  *            are stopped.
00158  * returns  : -
00159  */
00160 void heartbeat_stop(long hb_thread)
00161 {
00162     List *matching_info;
00163     struct hb_info *info;
00164 
00165     /*
00166      * First, check if there are heartbeats to stop.
00167      * If not, do not continue, otherwise this function will crash
00168      */
00169     if (heartbeats == NULL)
00170         return;
00171 
00172     if (hb_thread == ALL_HEARTBEATS) {
00173         while (NULL != (info = gwlist_extract_first(heartbeats))) {
00174             gw_assert(info);
00175             info->running = 0;
00176             gwthread_wakeup(info->thread);
00177             gwthread_join(info->thread);
00178             gw_free(info);
00179         }
00180     } else {
00181         matching_info = gwlist_extract_matching(heartbeats, &hb_thread, find_hb);
00182         if (matching_info == NULL) {
00183             warning(0, "Could not stop heartbeat %ld: not found.", hb_thread);
00184         return;
00185         }
00186         gw_assert(gwlist_len(matching_info) == 1);
00187         info = gwlist_extract_first(matching_info);
00188         gwlist_destroy(matching_info, NULL);
00189      
00190         info->running = 0;
00191         gwthread_wakeup(hb_thread);
00192         gwthread_join(hb_thread);
00193         gw_free(info);
00194     }
00195     if (gwlist_len(heartbeats) == 0) {
00196         gwlist_destroy(heartbeats, NULL);
00197         heartbeats = NULL;
00198     }
00199 }
See file LICENSE for details about the license agreement for using, modifying, copying or deriving work from this software.