int status_job( job *pjob, /* ptr to job to status */ struct batch_request *preq, svrattrl *pal, /* specific attributes to status */ tlist_head *pstathd, /* RETURN: head of list to append status to */ int *bad) /* RETURN: index of first bad pbs_attribute */ { struct brp_status *pstat; int IsOwner = 0; long query_others = 0; /* see if the client is authorized to status this job */ if (svr_authorize_jobreq(preq, pjob) == 0) IsOwner = 1; get_svr_attr_l(SRV_ATR_query_others, &query_others); if (!query_others) { if (IsOwner == 0) { return(PBSE_PERM); } } /* allocate reply structure and fill in header portion */ if ((pstat = calloc(1, sizeof(struct brp_status))) == NULL) { return(PBSE_SYSTEM); } CLEAR_LINK(pstat->brp_stlink); pstat->brp_objtype = MGR_OBJ_JOB; strcpy(pstat->brp_objname, pjob->ji_qs.ji_jobid); CLEAR_HEAD(pstat->brp_attr); append_link(pstathd, &pstat->brp_stlink, pstat); /* add attributes to the status reply */ *bad = 0; if (status_attrib( pal, job_attr_def, pjob->ji_wattr, JOB_ATR_LAST, preq->rq_perm, &pstat->brp_attr, bad, IsOwner)) { return(PBSE_NOATTR); } return (0); } /* END status_job() */
int status_job( job *pjob, /* ptr to job to status */ batch_request *preq, svrattrl *pal, /* specific attributes to status */ tlist_head *pstathd, /* RETURN: head of list to append status to */ bool condensed, int *bad) /* RETURN: index of first bad pbs_attribute */ { struct brp_status *pstat; int IsOwner = 0; long query_others = 0; long condensed_timeout = JOB_CONDENSED_TIMEOUT; /* Make sure procct is removed from the job resource attributes */ remove_procct(pjob); /* see if the client is authorized to status this job */ if (svr_authorize_jobreq(preq, pjob) == 0) IsOwner = 1; get_svr_attr_l(SRV_ATR_query_others, &query_others); if (!query_others) { if (IsOwner == 0) { return(PBSE_PERM); } } get_svr_attr_l(SRV_ATR_job_full_report_time, &condensed_timeout); // if the job has been modified within the timeout, send the full output if ((condensed == true) && (time(NULL) < pjob->ji_mod_time + condensed_timeout)) condensed = false; /* allocate reply structure and fill in header portion */ if ((pstat = (struct brp_status *)calloc(1, sizeof(struct brp_status))) == NULL) { return(PBSE_SYSTEM); } CLEAR_LINK(pstat->brp_stlink); pstat->brp_objtype = MGR_OBJ_JOB; strcpy(pstat->brp_objname, pjob->ji_qs.ji_jobid); CLEAR_HEAD(pstat->brp_attr); append_link(pstathd, &pstat->brp_stlink, pstat); /* add attributes to the status reply */ *bad = 0; if (status_attrib( pal, job_attr_def, pjob->ji_wattr, JOB_ATR_LAST, preq->rq_perm, &pstat->brp_attr, condensed, bad, IsOwner)) { return(PBSE_NOATTR); } return (0); } /* END status_job() */
int req_releasearray( batch_request *preq) /* I */ { job *pjob; job_array *pa; char *range; int rc; int index; pa = get_array(preq->rq_ind.rq_release.rq_objname); if (pa == NULL) { req_reject(PBSE_IVALREQ,0,preq,NULL,"Cannot find array"); return(PBSE_NONE); } mutex_mgr pa_mutex = mutex_mgr(pa->ai_mutex, true); while (TRUE) { if (((index = first_job_index(pa)) == -1) || (pa->job_ids[index] == NULL)) { return(PBSE_NONE); } if ((pjob = svr_find_job(pa->job_ids[index], FALSE)) == NULL) { free(pa->job_ids[index]); pa->job_ids[index] = NULL; } else break; } mutex_mgr pjob_mutex = mutex_mgr(pjob->ji_mutex, true); if (svr_authorize_jobreq(preq, pjob) == -1) { req_reject(PBSE_PERM,0,preq,NULL,NULL); return(PBSE_NONE); } pjob_mutex.unlock(); range = preq->rq_extend; if ((range != NULL) && (strstr(range,ARRAY_RANGE) != NULL)) { /* parse the array range */ /* ai_mutex is locked going into release_array_range and returns locked as well */ if ((rc = release_array_range(pa,preq,range)) != 0) { req_reject(rc,0,preq,NULL,NULL); return(PBSE_NONE); } } /* pa->ai_mutex remains locked in and out of release_whole_array */ else if ((rc = release_whole_array(pa,preq)) != 0) { req_reject(rc,0,preq,NULL,NULL); return(PBSE_NONE); } reply_ack(preq); return(PBSE_NONE); } /* END req_releasearray() */
void chk_job_req_permissions( job **pjob_ptr, /* M */ struct batch_request *preq) /* I */ { job *pjob = *pjob_ptr; char tmpLine[MAXLINE]; char log_buf[LOCAL_LOG_BUF_SIZE]; if (svr_authorize_jobreq(preq, pjob) == -1) { sprintf(log_buf, msg_permlog, preq->rq_type, "Job", pjob->ji_qs.ji_jobid, preq->rq_user, preq->rq_host); log_event(PBSEVENT_SECURITY,PBS_EVENTCLASS_JOB,pjob->ji_qs.ji_jobid,log_buf); req_reject(PBSE_PERM, 0, preq, NULL, "operation not permitted"); unlock_ji_mutex(pjob, __func__, "1", LOGLEVEL); *pjob_ptr = NULL; } else if (pjob->ji_qs.ji_state >= JOB_STATE_EXITING) { /* job has completed */ switch (preq->rq_type) { case PBS_BATCH_Rerun: /* allow re-run to be executed for completed jobs */ /* NO-OP */ break; default: sprintf(log_buf, "%s %s", pbse_to_txt(PBSE_BADSTATE), PJobState[pjob->ji_qs.ji_state]); log_event(PBSEVENT_DEBUG,PBS_EVENTCLASS_JOB,pjob->ji_qs.ji_jobid,log_buf); snprintf(tmpLine, sizeof(tmpLine), "invalid state for job - %s", PJobState[pjob->ji_qs.ji_state]); req_reject(PBSE_BADSTATE, 0, preq, NULL, tmpLine); unlock_ji_mutex(pjob, __func__, "2", LOGLEVEL); *pjob_ptr = NULL; break; } /* END switch (preq->rq_type) */ } /* END if (pjob->ji_qs.ji_state >= JOB_STATE_EXITING) */ /* SUCCESS - request is valid */ } /* END chk_job_req_permissions() */
/** * @brief * status_subjob - status a single subjob (of an Array Job) * Works by statusing the parrent unless subjob is actually running. * * @param[in,out] pjob - ptr to parent Array * @param[in] preq - request structure * @param[in] pal - specific attributes to status * @param[in] subj - if not = -1 then include subjob [n] * @param[in,out] pstathd - RETURN: head of list to append status to * @param[out] bad - RETURN: index of first bad attribute * * @return int * @retval 0 : success * @retval PBSE_PERM : client is not authorized to status the job * @retval PBSE_SYSTEM : memory allocation error * @retval PBSE_IVALREQ : something wrong with the flags */ int status_subjob(job *pjob, struct batch_request *preq, svrattrl *pal, int subj, pbs_list_head *pstathd, int *bad) { int limit = (int)JOB_ATR_LAST; struct brp_status *pstat; job *psubjob; /* ptr to job to status */ char realstate; int rc = 0; int oldeligflags = 0; int oldatypflags = 0; int subjob_state = -1; char *old_subjob_comment = NULL; /* see if the client is authorized to status this job */ if (! server.sv_attr[(int)SRV_ATR_query_others].at_val.at_long) if (svr_authorize_jobreq(preq, pjob)) return (PBSE_PERM); if ((pjob->ji_qs.ji_svrflags & JOB_SVFLG_ArrayJob) == 0) return PBSE_IVALREQ; /* if subjob is running, use real job structure */ if (get_subjob_state(pjob, subj) == JOB_STATE_RUNNING) { psubjob = find_job(mk_subjob_id(pjob, subj)); if (psubjob) status_job(psubjob, preq, pal, pstathd, bad); return 0; } /* otherwise we fake it with info from the parent */ /* allocate reply structure and fill in header portion */ /* for the general case, we don't want to include the parent's */ /* array related attrbutes as they belong only to the Array */ if (pal == NULL) limit = JOB_ATR_array; pstat = (struct brp_status *)malloc(sizeof(struct brp_status)); if (pstat == (struct brp_status *)0) return (PBSE_SYSTEM); CLEAR_LINK(pstat->brp_stlink); pstat->brp_objtype = MGR_OBJ_JOB; (void)strcpy(pstat->brp_objname, mk_subjob_id(pjob, subj)); CLEAR_HEAD(pstat->brp_attr); append_link(pstathd, &pstat->brp_stlink, pstat); /* add attributes to the status reply */ *bad = 0; /* * fake the job state and comment by setting the parent job's state * and comment to that of the subjob */ subjob_state = get_subjob_state(pjob, subj); realstate = pjob->ji_wattr[(int)JOB_ATR_state].at_val.at_char; pjob->ji_wattr[(int)JOB_ATR_state].at_val.at_char = statechars[subjob_state]; pjob->ji_wattr[(int)JOB_ATR_state].at_flags |= ATR_VFLAG_MODCACHE; if (subjob_state == JOB_STATE_EXPIRED || subjob_state == JOB_STATE_FINISHED) { if (pjob->ji_ajtrk->tkm_tbl[subj].trk_substate == JOB_SUBSTATE_FINISHED) { if (pjob->ji_wattr[(int)JOB_ATR_Comment].at_flags & ATR_VFLAG_SET) { old_subjob_comment = strdup(pjob->ji_wattr[(int)JOB_ATR_Comment].at_val.at_str); if (old_subjob_comment == (char *)0) return (PBSE_SYSTEM); } if (job_attr_def[(int)JOB_ATR_Comment].at_decode(&pjob->ji_wattr[(int)JOB_ATR_Comment], (char *)0, (char *)0, "Subjob finished") == PBSE_SYSTEM) { free(old_subjob_comment); return (PBSE_SYSTEM); } } else if (pjob->ji_ajtrk->tkm_tbl[subj].trk_substate == JOB_SUBSTATE_FAILED) { if (pjob->ji_wattr[(int)JOB_ATR_Comment].at_flags & ATR_VFLAG_SET) { old_subjob_comment = strdup(pjob->ji_wattr[(int)JOB_ATR_Comment].at_val.at_str); if (old_subjob_comment == (char *)0) return (PBSE_SYSTEM); } if (job_attr_def[(int)JOB_ATR_Comment].at_decode(&pjob->ji_wattr[(int)JOB_ATR_Comment], (char *)0, (char *)0, "Subjob failed") == PBSE_SYSTEM) { free(old_subjob_comment); return (PBSE_SYSTEM); } } else if (pjob->ji_ajtrk->tkm_tbl[subj].trk_substate == JOB_SUBSTATE_TERMINATED) { if (pjob->ji_wattr[(int)JOB_ATR_Comment].at_flags & ATR_VFLAG_SET) { old_subjob_comment = strdup(pjob->ji_wattr[(int)JOB_ATR_Comment].at_val.at_str); if (old_subjob_comment == (char *)0) return (PBSE_SYSTEM); } if (job_attr_def[(int)JOB_ATR_Comment].at_decode(&pjob->ji_wattr[(int)JOB_ATR_Comment], (char *)0, (char *)0, "Subjob terminated") == PBSE_SYSTEM) { free(old_subjob_comment); return (PBSE_SYSTEM); } } } /* when eligible_time_enable is off, */ /* clear the set flag so that eligible_time and accrue_type dont show */ if (server.sv_attr[(int)SRV_ATR_EligibleTimeEnable].at_val.at_long == 0) { oldeligflags = pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags; pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags &= ~ATR_VFLAG_SET; pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags |= ATR_VFLAG_MODCACHE; oldatypflags = pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_flags; pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_flags &= ~ATR_VFLAG_SET; pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_flags |= ATR_VFLAG_MODCACHE; /* Note: ATR_VFLAG_MODCACHE must be set because of svr_cached() does */ /* not correctly check ATR_VFLAG_SET */ } if (status_attrib(pal, job_attr_def, pjob->ji_wattr, limit, preq->rq_perm, &pstat->brp_attr, bad)) rc = PBSE_NOATTR; /* Set the parent state back to what it really is */ pjob->ji_wattr[(int)JOB_ATR_state].at_val.at_char = realstate; pjob->ji_wattr[(int)JOB_ATR_state].at_flags |= ATR_VFLAG_MODCACHE; /* Set the parent comment back to what it really is */ if (old_subjob_comment != NULL) { if (job_attr_def[(int)JOB_ATR_Comment].at_decode(&pjob->ji_wattr[(int)JOB_ATR_Comment], (char *)0, (char *)0, old_subjob_comment) == PBSE_SYSTEM) { free(old_subjob_comment); return (PBSE_SYSTEM); } free(old_subjob_comment); } /* reset the flags */ if (server.sv_attr[(int)SRV_ATR_EligibleTimeEnable].at_val.at_long == 0) { pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags = oldeligflags; pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_flags = oldatypflags; } return (rc); }
int status_job(job *pjob, struct batch_request *preq, svrattrl *pal, pbs_list_head *pstathd, int *bad) { struct brp_status *pstat; time_t tm; long oldtime = 0; int old_elig_flags = 0; int old_atyp_flags = 0; /* see if the client is authorized to status this job */ if (! server.sv_attr[(int)SRV_ATR_query_others].at_val.at_long) if (svr_authorize_jobreq(preq, pjob)) return (PBSE_PERM); if (pjob->ji_qs.ji_svrflags & JOB_SVFLG_ArrayJob) { /* for Array Job, if array_indices_remaining is modified */ /* then need to recalculate the string value */ char *pnewstr; attribute *premain; premain = &pjob->ji_wattr[(int)JOB_ATR_array_indices_remaining]; if (premain->at_flags & ATR_VFLAG_MODCACHE) { pnewstr = cvt_range(pjob->ji_ajtrk, JOB_STATE_QUEUED); if (pnewstr == NULL) pnewstr = "-"; job_attr_def[JOB_ATR_array_indices_remaining].at_free(premain); job_attr_def[JOB_ATR_array_indices_remaining].at_decode(premain, 0, 0, pnewstr); /* also update value of attribute "array_state_count" */ update_subjob_state_ct(pjob); } } /* calc eligible time on the fly and return, don't save. */ if (server.sv_attr[(int)SRV_ATR_EligibleTimeEnable].at_val.at_long != 0) { if (pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_val.at_long == JOB_ELIGIBLE) { time(&tm); oldtime = pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_val.at_long; pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_val.at_long += ((long)tm - pjob->ji_wattr[(int)JOB_ATR_sample_starttime].at_val.at_long); pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags |= ATR_VFLAG_MODCACHE; /* Note: ATR_VFLAG_MODCACHE must be set because of svr_cached() does */ /* not correctly check ATR_VFLAG_SET */ } } else { /* eligible_time_enable is off so, */ /* clear set flag so that eligible_time and accrue type dont show */ old_elig_flags = pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags; pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags &= ~ATR_VFLAG_SET; pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags |= (ATR_VFLAG_MODCACHE); old_atyp_flags = pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_flags; pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_flags &= ~ATR_VFLAG_SET; pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_flags |= (ATR_VFLAG_MODCACHE); /* Note: ATR_VFLAG_MODCACHE must be set because of svr_cached() does */ /* not correctly check ATR_VFLAG_SET */ } /* allocate reply structure and fill in header portion */ pstat = (struct brp_status *)malloc(sizeof(struct brp_status)); if (pstat == (struct brp_status *)0) return (PBSE_SYSTEM); CLEAR_LINK(pstat->brp_stlink); pstat->brp_objtype = MGR_OBJ_JOB; (void)strcpy(pstat->brp_objname, pjob->ji_qs.ji_jobid); CLEAR_HEAD(pstat->brp_attr); append_link(pstathd, &pstat->brp_stlink, pstat); /* add attributes to the status reply */ *bad = 0; if (status_attrib(pal, job_attr_def, pjob->ji_wattr, JOB_ATR_LAST, preq->rq_perm, &pstat->brp_attr, bad)) return (PBSE_NOATTR); /* reset eligible time, it was calctd on the fly, real calctn only when accrue_type changes */ if (server.sv_attr[(int)SRV_ATR_EligibleTimeEnable].at_val.at_long != 0) { if (pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_val.at_long == JOB_ELIGIBLE) { pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_val.at_long = oldtime; pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags |= ATR_VFLAG_MODCACHE; /* Note: ATR_VFLAG_MODCACHE must be set because of svr_cached() does */ /* not correctly check ATR_VFLAG_SET */ } } else { /* reset the set flags */ pjob->ji_wattr[(int)JOB_ATR_eligible_time].at_flags = old_elig_flags; pjob->ji_wattr[(int)JOB_ATR_accrue_type].at_flags = old_atyp_flags; } return (0); }
job * chk_job_request(char *jobid, struct batch_request *preq, int *rc) { int t; int histerr = 0; job *pjob; int deletehist = 0; char *p1; char *p2; if (preq->rq_extend && strstr(preq->rq_extend, DELETEHISTORY)) deletehist = 1; t = is_job_array(jobid); if ((t == IS_ARRAY_NO) || (t == IS_ARRAY_ArrayJob)) pjob = find_job(jobid); /* regular or ArrayJob itself */ else pjob = find_arrayparent(jobid); /* subjob(s) */ *rc = t; if (pjob == NULL) { log_event(PBSEVENT_DEBUG, PBS_EVENTCLASS_JOB, LOG_INFO, jobid, msg_unkjobid); req_reject(PBSE_UNKJOBID, 0, preq); return NULL; } else { histerr = svr_chk_histjob(pjob); if (histerr && deletehist == 0) { req_reject(histerr, 0, preq); return NULL; } if (deletehist ==1&& pjob->ji_qs.ji_state == JOB_STATE_MOVED && pjob->ji_qs.ji_substate != JOB_SUBSTATE_FINISHED) { job_purge(pjob); req_reject(PBSE_UNKJOBID, 0, preq); return NULL; } } /* * The job was found using the job ID in the request, but it may not * match exactly (i.e. FQDN vs. unqualified hostname). Overwrite the * host portion of the job ID in the request with the host portion of * the one from the server job structure. Do not modify anything * before the first dot in the job ID because it may be an array job. * This will allow find_job() to look for an exact match when the * request is serviced by MoM. */ p1 = strchr(pjob->ji_qs.ji_jobid, '.'); if (p1) { p2 = strchr(jobid, '.'); if (p2) *p2 = '\0'; strncat(jobid, p1, PBS_MAXSVRJOBID-1); } if (svr_authorize_jobreq(preq, pjob) == -1) { (void)sprintf(log_buffer, msg_permlog, preq->rq_type, "Job", pjob->ji_qs.ji_jobid, preq->rq_user, preq->rq_host); log_event(PBSEVENT_SECURITY, PBS_EVENTCLASS_JOB, LOG_INFO, pjob->ji_qs.ji_jobid, log_buffer); req_reject(PBSE_PERM, 0, preq); return NULL; } if ((t == IS_ARRAY_NO) && (pjob->ji_qs.ji_state == JOB_STATE_EXITING)) { /* special case Deletejob with "force" */ if ((preq->rq_type == PBS_BATCH_DeleteJob) && (preq->rq_extend != NULL) && (strcmp(preq->rq_extend, "force") == 0)) { return (pjob); } (void)sprintf(log_buffer, "%s, state=%d", msg_badstate, pjob->ji_qs.ji_state); log_event(PBSEVENT_DEBUG, PBS_EVENTCLASS_JOB, LOG_INFO, pjob->ji_qs.ji_jobid, log_buffer); req_reject(PBSE_BADSTATE, 0, preq); return NULL; } return (pjob); }
int req_releasearray( void *vp) /* I */ { job *pjob; job_array *pa; char *range; int rc; int index; struct batch_request *preq = (struct batch_request *)vp; pa = get_array(preq->rq_ind.rq_release.rq_objname); if (pa == NULL) { req_reject(PBSE_IVALREQ,0,preq,NULL,"Cannot find array"); return(PBSE_NONE); } while (TRUE) { if (((index = first_job_index(pa)) == -1) || (pa->job_ids[index] == NULL)) { unlock_ai_mutex(pa, __func__, (char *)"1", LOGLEVEL); return(PBSE_NONE); } if ((pjob = svr_find_job(pa->job_ids[index], FALSE)) == NULL) { free(pa->job_ids[index]); pa->job_ids[index] = NULL; } else break; } if (svr_authorize_jobreq(preq, pjob) == -1) { req_reject(PBSE_PERM,0,preq,NULL,NULL); unlock_ai_mutex(pa, __func__, (char *)"2", LOGLEVEL); unlock_ji_mutex(pjob, __func__, (char *)"1", LOGLEVEL); return(PBSE_NONE); } unlock_ji_mutex(pjob, __func__, (char *)"2", LOGLEVEL); range = preq->rq_extend; if ((range != NULL) && (strstr(range,ARRAY_RANGE) != NULL)) { /* parse the array range */ if ((rc = release_array_range(pa,preq,range)) != 0) { unlock_ai_mutex(pa, __func__, (char *)"3", LOGLEVEL); req_reject(rc,0,preq,NULL,NULL); return(PBSE_NONE); } } else if ((rc = release_whole_array(pa,preq)) != 0) { unlock_ai_mutex(pa, __func__, (char *)"4", LOGLEVEL); req_reject(rc,0,preq,NULL,NULL); return(PBSE_NONE); } unlock_ai_mutex(pa, __func__, (char *)"5", LOGLEVEL); reply_ack(preq); return(PBSE_NONE); } /* END req_releasearray() */
int req_releasearray( void *vp) /* I */ { char log_buf[LOCAL_LOG_BUF_SIZE]; job *pjob; job_array *pa; char *range; int rc; int index; struct batch_request *preq = (struct batch_request *)vp; pa = get_array(preq->rq_ind.rq_release.rq_objname); if (pa == NULL) { req_reject(PBSE_IVALREQ,0,preq,NULL,"Cannot find array"); return(PBSE_NONE); } while (TRUE) { if (((index = first_job_index(pa)) == -1) || (pa->job_ids[index] == NULL)) { pthread_mutex_unlock(pa->ai_mutex); return(PBSE_NONE); } if ((pjob = svr_find_job(pa->job_ids[index], FALSE)) == NULL) { free(pa->job_ids[index]); pa->job_ids[index] = NULL; } else break; } if (svr_authorize_jobreq(preq, pjob) == -1) { req_reject(PBSE_PERM,0,preq,NULL,NULL); pthread_mutex_unlock(pa->ai_mutex); unlock_ji_mutex(pjob, __func__, "1", LOGLEVEL); return(PBSE_NONE); } unlock_ji_mutex(pjob, __func__, "2", LOGLEVEL); range = preq->rq_extend; if ((range != NULL) && (strstr(range,ARRAY_RANGE) != NULL)) { /* parse the array range */ if ((rc = release_array_range(pa,preq,range)) != 0) { if (LOGLEVEL >= 7) { sprintf(log_buf, "%s: unlocking ai_mutex", __func__); log_event(PBSEVENT_JOB, PBS_EVENTCLASS_JOB, pa->ai_qs.parent_id, log_buf); } pthread_mutex_unlock(pa->ai_mutex); req_reject(rc,0,preq,NULL,NULL); return(PBSE_NONE); } } else if ((rc = release_whole_array(pa,preq)) != 0) { if (LOGLEVEL >= 7) { sprintf(log_buf, "%s: unlocking ai_mutex", __func__); log_event(PBSEVENT_JOB, PBS_EVENTCLASS_JOB, pjob->ji_qs.ji_jobid, log_buf); } pthread_mutex_unlock(pa->ai_mutex); req_reject(rc,0,preq,NULL,NULL); return(PBSE_NONE); } if (LOGLEVEL >= 7) { sprintf(log_buf, "%s: unlocking ai_mutex", __func__); log_event(PBSEVENT_JOB, PBS_EVENTCLASS_JOB, pjob->ji_qs.ji_jobid, log_buf); } pthread_mutex_unlock(pa->ai_mutex); reply_ack(preq); return(PBSE_NONE); } /* END req_releasearray() */