00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034 #ifndef _WIN32
00035 #include "config.h"
00036 #else
00037 #include "winconfig.h"
00038 #endif
00039
00040 #include "include/ams.h"
00041 #include "include/agent.h"
00042 #include "include/mc_platform.h"
00043
00044 int
00045 ams_Destroy(ams_p ams)
00046 {
00047 MUTEX_DESTROY(ams->runflag_lock);
00048 free(ams->runflag_lock);
00049 COND_DESTROY(ams->runflag_cond);
00050 free(ams->runflag_cond);
00051 MUTEX_DESTROY(ams->waiting_lock);
00052 free(ams->waiting_lock);
00053 COND_DESTROY(ams->waiting_cond);
00054 free(ams->waiting_cond);
00055 free(ams);
00056 return MC_SUCCESS;
00057 }
00058
00059 ams_p
00060 ams_Initialize(mc_platform_p mc_platform)
00061 {
00062 ams_p ams;
00063 ams = (ams_p)malloc(sizeof(ams_t));
00064 CHECK_NULL(ams, exit(0););
00065 ams->mc_platform = mc_platform;
00066
00067 ams->runflag_lock = (MUTEX_T*)malloc(sizeof(MUTEX_T));
00068 CHECK_NULL(ams->runflag_lock, exit(0););
00069 MUTEX_INIT(ams->runflag_lock);
00070
00071 ams->runflag_cond = (COND_T*)malloc(sizeof(COND_T));
00072 CHECK_NULL(ams->runflag_cond, exit(0););
00073 COND_INIT(ams->runflag_cond);
00074
00075 ams->run = 0;
00076
00077 ams->waiting = 0;
00078 ams->waiting_lock = (MUTEX_T*)malloc(sizeof(MUTEX_T));
00079 MUTEX_INIT(ams->waiting_lock);
00080 ams->waiting_cond = (COND_T*)malloc(sizeof(COND_T));
00081 COND_INIT(ams->waiting_cond);
00082
00083 return ams;
00084 }
00085
00086 void
00087 ams_Print(ams_p ams)
00088 {
00089 list_t* alist;
00090
00091 alist = ams->mc_platform->agent_queue;
00092
00093 ListRDLock(alist);
00094 if(ListGetSize(alist) == 0)
00095 {
00096 return;
00097 }
00098
00099
00100 printf("%d total agents on board.\n", alist->size);
00101 ListForEachCB(alist, (ListElemGenericFunc_t)agent_Print);
00102 ListRDUnlock(alist);
00103
00104 return;
00105 }
00106
00107 extern int
00108 ams_ManageAgentList(ams_p ams)
00109 {
00110
00111 MCAgent_t current_agent;
00112 int index = 0;
00113 list_t* alist;
00114 mc_platform_p global;
00115 message_p message;
00116
00117 alist = ams->mc_platform->agent_queue;
00118 global = ams->mc_platform;
00119
00120
00121
00122 ListRDLock(alist);
00123 for(index=0; index<alist->size; index++)
00124 {
00125 if((current_agent = (MCAgent_t)ListSearch(alist, index)))
00126 {
00127 if(current_agent->binary) {continue;}
00128 MUTEX_LOCK(current_agent->lock);
00129 current_agent->orphan = 0;
00130 MUTEX_LOCK(global->quit_lock);
00131 MUTEX_LOCK(current_agent->agent_status_lock);
00132 if(global->quit && current_agent->agent_status != MC_WAIT_MESSGSEND) {
00133 MUTEX_UNLOCK(current_agent->agent_status_lock);
00134 MUTEX_UNLOCK(global->quit_lock);
00135 MUTEX_UNLOCK(current_agent->lock);
00136 MC_TerminateAgent(current_agent);
00137
00138 MUTEX_LOCK(current_agent->run_lock);
00139 MUTEX_UNLOCK(current_agent->run_lock);
00140 continue;
00141 } else {
00142 MUTEX_UNLOCK(current_agent->agent_status_lock);
00143 }
00144 MUTEX_UNLOCK(global->quit_lock);
00145 MUTEX_LOCK(current_agent->agent_status_lock);
00146 switch(current_agent->agent_status)
00147 {
00148 case MC_WAIT_CH :
00149 MUTEX_UNLOCK(current_agent->lock);
00150 MUTEX_UNLOCK(current_agent->agent_status_lock);
00151 agent_RunChScript(current_agent, global);
00152 break;
00153 case MC_AGENT_ACTIVE :
00154 MUTEX_UNLOCK(current_agent->lock);
00155 MUTEX_UNLOCK(current_agent->agent_status_lock);
00156
00157 break;
00158 case MC_WAIT_MESSGSEND :
00159 current_agent->agent_status = MC_WAIT_FINISHED;
00160 COND_BROADCAST(current_agent->agent_status_cond);
00161 MUTEX_UNLOCK(current_agent->agent_status_lock);
00162 MUTEX_LOCK(ams->runflag_lock);
00163 ams->run = 1;
00164 MUTEX_UNLOCK(ams->runflag_lock);
00165 MUTEX_UNLOCK(current_agent->lock);
00166 message = message_New();
00167 if (
00168 message_InitializeFromAgent
00169 (
00170 ams->mc_platform,
00171 message,
00172 current_agent
00173 )
00174 )
00175 {
00176 fprintf(stderr, "Error initializing message from agent. %s:%d\n", __FILE__, __LINE__);
00177 message_Destroy(message);
00178 message = NULL;
00179 } else {
00180
00181
00182 current_agent->name = (char*)realloc(
00183 current_agent->name,
00184 sizeof(char) * (strlen(current_agent->name) + 10)
00185 );
00186 strcat(current_agent->name, "_SENDING");
00187 ListWRLock(ams->mc_platform->message_queue);
00188 ListAdd(
00189 ams->mc_platform->message_queue,
00190 message
00191 );
00192 ListWRUnlock(ams->mc_platform->message_queue);
00193 }
00194 break;
00195 case MC_AGENT_NEUTRAL :
00196 MUTEX_UNLOCK(current_agent->agent_status_lock);
00197 MUTEX_UNLOCK(current_agent->lock);
00198 break;
00199 case MC_WAIT_FINISHED :
00200 MUTEX_UNLOCK(current_agent->agent_status_lock);
00201 MUTEX_UNLOCK(current_agent->lock);
00202 ListRDtoWR(alist);
00203 ListDelete(alist, index);
00204 ListWRtoRD(alist);
00205
00206
00207
00208 index--;
00209 break;
00210 default :
00211 printf("ERROR IN AGENT FORMAT");
00212 printf("Agent Format %d not recognized.",
00213 current_agent->agent_status);
00214
00215 current_agent->agent_status = MC_WAIT_FINISHED;
00216 COND_BROADCAST(current_agent->agent_status_cond);
00217 MUTEX_UNLOCK(current_agent->agent_status_lock);
00218 MUTEX_UNLOCK(current_agent->lock);
00219 }
00220 }
00221 }
00222 ListRDUnlock(alist);
00223 return 0 ;
00224 }
00225
00226 void
00227 ams_Start(mc_platform_p mc_platform)
00228 {
00229 ams_p ams = mc_platform->ams;
00230 #ifndef _WIN32
00231 pthread_attr_t attr;
00232 pthread_attr_init(&attr);
00233 if(mc_platform->stack_size[MC_THREAD_AMS] != -1) {
00234 pthread_attr_setstacksize
00235 (
00236 &attr,
00237 mc_platform->stack_size[MC_THREAD_AMS]
00238 );
00239 }
00240 #else
00241 int stack_size;
00242 if (mc_platform->stack_size[MC_THREAD_AMS] < 1) {
00243
00244 stack_size = mc_platform->stack_size[MC_THREAD_AMS]+1;
00245 } else {
00246 stack_size = mc_platform->stack_size[MC_THREAD_AMS];
00247 }
00248 #endif
00249 THREAD_CREATE
00250 (
00251 &ams->thread,
00252 ams_Thread,
00253 mc_platform
00254 );
00255 }
00256 #ifndef _WIN32
00257 void*
00258 ams_Thread(void* arg)
00259 #else
00260 DWORD WINAPI
00261 ams_Thread( LPVOID arg )
00262 #endif
00263 {
00264 mc_platform_p mc_platform = (mc_platform_p)arg;
00265 ams_p ams = mc_platform->ams;
00266 while(1) {
00267 MUTEX_LOCK(ams->runflag_lock);
00268 MUTEX_LOCK(mc_platform->quit_lock);
00269 while(ams->run == 0 && !mc_platform->quit) {
00270 MUTEX_UNLOCK(mc_platform->quit_lock);
00271
00272 MUTEX_LOCK(ams->waiting_lock);
00273 ams->waiting = 1;
00274 COND_BROADCAST(ams->waiting_cond);
00275 MUTEX_UNLOCK(ams->waiting_lock);
00276
00277 COND_WAIT
00278 (
00279 ams->runflag_cond,
00280 ams->runflag_lock
00281 );
00282 MUTEX_LOCK(mc_platform->quit_lock);
00283 }
00284
00285 MUTEX_LOCK(ams->waiting_lock);
00286 ams->waiting = 0;
00287 COND_BROADCAST(ams->waiting_cond);
00288 MUTEX_UNLOCK(ams->waiting_lock);
00289 if (ams->run == 0 && mc_platform->quit) {
00290 MUTEX_UNLOCK(mc_platform->quit_lock);
00291 MUTEX_UNLOCK(ams->runflag_lock);
00292 ams_ManageAgentList(ams);
00293 THREAD_EXIT();
00294 }
00295 ams->run = 0;
00296 MUTEX_UNLOCK(mc_platform->quit_lock);
00297 MUTEX_UNLOCK(ams->runflag_lock);
00298 ams_ManageAgentList(ams);
00299 }
00300 THREAD_EXIT();
00301 }