|  | /*****************************************************************************\ | 
|  | *  print.c - squeue print job functions | 
|  | ***************************************************************************** | 
|  | *  Copyright (C) 2002-2007 The Regents of the University of California. | 
|  | *  Copyright (C) 2008-2010 Lawrence Livermore National Security. | 
|  | *  Copyright (C) SchedMD LLC. | 
|  | *  Produced at Lawrence Livermore National Laboratory (cf, DISCLAIMER). | 
|  | *  Written by Joey Ekstrom <ekstrom1@llnl.gov>, et. al. | 
|  | *  CODE-OCEC-09-009. All rights reserved. | 
|  | * | 
|  | *  This file is part of Slurm, a resource management program. | 
|  | *  For details, see <https://slurm.schedmd.com/>. | 
|  | *  Please also read the included file: DISCLAIMER. | 
|  | * | 
|  | *  Slurm is free software; you can redistribute it and/or modify it under | 
|  | *  the terms of the GNU General Public License as published by the Free | 
|  | *  Software Foundation; either version 2 of the License, or (at your option) | 
|  | *  any later version. | 
|  | * | 
|  | *  In addition, as a special exception, the copyright holders give permission | 
|  | *  to link the code of portions of this program with the OpenSSL library under | 
|  | *  certain conditions as described in each individual source file, and | 
|  | *  distribute linked combinations including the two. You must obey the GNU | 
|  | *  General Public License in all respects for all of the code used other than | 
|  | *  OpenSSL. If you modify file(s) with this exception, you may extend this | 
|  | *  exception to your version of the file(s), but you are not obligated to do | 
|  | *  so. If you do not wish to do so, delete this exception statement from your | 
|  | *  version.  If you delete this exception statement from all source files in | 
|  | *  the program, then also delete it here. | 
|  | * | 
|  | *  Slurm is distributed in the hope that it will be useful, but WITHOUT ANY | 
|  | *  WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | 
|  | *  FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more | 
|  | *  details. | 
|  | * | 
|  | *  You should have received a copy of the GNU General Public License along | 
|  | *  with Slurm; if not, write to the Free Software Foundation, Inc., | 
|  | *  51 Franklin Street, Fifth Floor, Boston, MA 02110-1301  USA. | 
|  | \*****************************************************************************/ | 
|  |  | 
|  | #include <grp.h> | 
|  | #include <pwd.h> | 
|  | #include <stdio.h> | 
|  | #include <string.h> | 
|  | #include <time.h> | 
|  | #include <sys/types.h> | 
|  |  | 
|  | #include "src/common/cpu_frequency.h" | 
|  | #include "src/common/hostlist.h" | 
|  | #include "src/common/list.h" | 
|  | #include "src/common/macros.h" | 
|  | #include "src/common/parse_time.h" | 
|  | #include "src/interfaces/select.h" | 
|  | #include "src/interfaces/acct_gather_profile.h" | 
|  | #include "src/common/uid.h" | 
|  | #include "src/common/xmalloc.h" | 
|  | #include "src/common/xstring.h" | 
|  |  | 
|  | #include "src/squeue/print.h" | 
|  | #include "src/common/print_fields.h" | 
|  | #include "src/squeue/squeue.h" | 
|  |  | 
|  | static void _combine_pending_array_tasks(list_t *l); | 
|  | static bool _filter_job(job_info_t *job); | 
|  | static int	_filter_job_part(char *part_name); | 
|  | static int	_filter_step(job_step_info_t * step); | 
|  | static void	_job_list_del(void *x); | 
|  | static uint32_t	_part_get_prio_tier(char *part_name); | 
|  | static void	_part_state_free(void); | 
|  | static void	_part_state_load(void); | 
|  | static int	_print_str(char *str, int width, bool right, bool cut_output); | 
|  |  | 
|  | static int _print_job_from_format(void *x, void *arg); | 
|  | static int _print_step_from_format(void *x, void *arg); | 
|  |  | 
|  | static partition_info_msg_t *part_info_msg = NULL; | 
|  |  | 
|  | /***************************************************************************** | 
|  | * Global Print Functions | 
|  | *****************************************************************************/ | 
|  | typedef struct { | 
|  | int *count; | 
|  | list_t *req_list; | 
|  | job_info_t *job_ptr; | 
|  | } foreach_prio_job_req_arg_t; | 
|  |  | 
|  | static int _foreach_create_prio_job_req(void *x, void *arg) | 
|  | { | 
|  | char *part_name = x; | 
|  | foreach_prio_job_req_arg_t *req_arg = arg; | 
|  | job_info_t *job_ptr = req_arg->job_ptr; | 
|  | squeue_job_rec_t *job_rec_ptr; | 
|  |  | 
|  | (*req_arg->count)++; | 
|  |  | 
|  | if (_filter_job_part(part_name)) | 
|  | return SLURM_SUCCESS; | 
|  |  | 
|  | job_rec_ptr = xmalloc(sizeof(squeue_job_rec_t)); | 
|  | job_rec_ptr->job_ptr = req_arg->job_ptr; | 
|  | job_rec_ptr->part_name = xstrdup(part_name); | 
|  | job_rec_ptr->part_prio = _part_get_prio_tier(part_name); | 
|  |  | 
|  | if (IS_JOB_PENDING(job_ptr) && job_ptr->priority_array) { | 
|  | job_rec_ptr->job_prio = | 
|  | job_ptr->priority_array[(*req_arg->count) - 1]; | 
|  | } else { | 
|  | job_rec_ptr->job_prio = job_ptr->priority; | 
|  | } | 
|  | list_append(req_arg->req_list, job_rec_ptr); | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | static void _create_priority_list(list_t *l, | 
|  | job_info_t *job_ptr) | 
|  | { | 
|  | char *tmp; | 
|  | int j = 0; | 
|  | foreach_prio_job_req_arg_t arg = {0}; | 
|  | list_t *part_names = list_create(xfree_ptr); | 
|  |  | 
|  | /* | 
|  | * If the job requests multiple partitions then priority_array_names | 
|  | * will exist. When a job is running the controller only sends the | 
|  | * partition that the job is running in in job_ptr->partition so we can | 
|  | * just use job_ptr->partition and job_ptr->priority. | 
|  | */ | 
|  | if (IS_JOB_PENDING(job_ptr) && job_ptr->priority_array_names) | 
|  | tmp = job_ptr->priority_array_names; | 
|  | else | 
|  | tmp = job_ptr->partition; | 
|  | slurm_addto_char_list(part_names, tmp); | 
|  |  | 
|  | arg.count = &j; | 
|  | arg.job_ptr = job_ptr; | 
|  | arg.req_list = l; | 
|  | list_for_each(part_names, _foreach_create_prio_job_req, &arg); | 
|  | } | 
|  |  | 
|  | extern void print_jobs_array(job_info_t *jobs, int size, list_t *format) | 
|  | { | 
|  | squeue_job_rec_t *job_rec_ptr; | 
|  | int i; | 
|  | list_t *l; | 
|  |  | 
|  | l = list_create(_job_list_del); | 
|  | if (!params.no_header) | 
|  | _print_job_from_format(NULL, format); | 
|  | if (!params.only_state) | 
|  | _part_state_load(); | 
|  |  | 
|  | /* Filter out the jobs of interest */ | 
|  | for (i = 0; i < size; i++) { | 
|  | if (_filter_job(&jobs[i])) | 
|  | continue; | 
|  | if (params.priority_flag) { | 
|  | _create_priority_list(l, &jobs[i]); | 
|  | } else { | 
|  | if (_filter_job_part(jobs[i].partition)) | 
|  | continue; | 
|  | job_rec_ptr = xmalloc(sizeof(squeue_job_rec_t)); | 
|  | job_rec_ptr->job_ptr = jobs + i; | 
|  | /* | 
|  | * Note: for multi-partition definitions the following | 
|  | * is ill-defined, and you should really use the -P | 
|  | * option for a consistent view. | 
|  | * This will sort such jobs alphabetically, but that | 
|  | * may be confusing as to why "foo,bar" and "bar,foo" | 
|  | * submissions end up in different places. | 
|  | */ | 
|  | job_rec_ptr->part_name = xstrdup(jobs[i].partition); | 
|  | list_append(l, (void *) job_rec_ptr); | 
|  | } | 
|  | } | 
|  |  | 
|  | _combine_pending_array_tasks(l); | 
|  | _part_state_free(); | 
|  | sort_jobs_by_start_time (l); | 
|  | sort_job_list (l); | 
|  |  | 
|  | /* Print the jobs of interest */ | 
|  | list_for_each(l, _print_job_from_format, format); | 
|  | FREE_NULL_LIST(l); | 
|  | } | 
|  |  | 
|  | extern void print_steps_array(job_step_info_t *steps, int size, list_t *format) | 
|  | { | 
|  | if (!params.no_header) | 
|  | _print_step_from_format(NULL, format); | 
|  |  | 
|  | if (size > 0) { | 
|  | int i; | 
|  | list_t *step_list; | 
|  |  | 
|  | step_list = list_create(NULL); | 
|  |  | 
|  | /* Filter out the jobs of interest */ | 
|  | for (i = 0; i < size; i++) { | 
|  | if (_filter_step(&steps[i])) | 
|  | continue; | 
|  | list_append(step_list, (void *) &steps[i]); | 
|  | } | 
|  |  | 
|  | sort_step_list(step_list); | 
|  |  | 
|  | /* Print the steps of interest */ | 
|  | list_for_each(step_list, _print_step_from_format, format); | 
|  | FREE_NULL_LIST(step_list); | 
|  | } | 
|  | } | 
|  |  | 
|  | extern void squeue_filter_jobs_for_json(job_info_msg_t *job_info) | 
|  | { | 
|  | int new_array_size = 0; | 
|  | job_info_t *tmp_jobs = xcalloc(job_info->record_count, | 
|  | sizeof(job_info_t)); | 
|  |  | 
|  | for (int i = 0; i < job_info->record_count; i++) { | 
|  | if (!(_filter_job(&job_info->job_array[i])) && | 
|  | !(_filter_job_part(job_info->job_array[i].partition))) { | 
|  | tmp_jobs[new_array_size] = job_info->job_array[i]; | 
|  | new_array_size++; | 
|  | } else { | 
|  | slurm_free_job_info_members(&job_info->job_array[i]); | 
|  | } | 
|  | } | 
|  |  | 
|  | xrecalloc(tmp_jobs, new_array_size, sizeof(job_info_t)); | 
|  | xfree(job_info->job_array); | 
|  | job_info->job_array = tmp_jobs; | 
|  | job_info->record_count = new_array_size; | 
|  | } | 
|  |  | 
|  | /* Combine a job array's task "reason" into the master job array record | 
|  | * reason as needed */ | 
|  | static void _merge_job_reason(job_info_t *job_ptr, job_info_t *task_ptr) | 
|  | { | 
|  | const char *task_desc; | 
|  |  | 
|  | if (job_ptr->state_reason == task_ptr->state_reason) | 
|  | return; | 
|  |  | 
|  | if (!job_ptr->state_desc) { | 
|  | job_ptr->state_desc = | 
|  | xstrdup(job_state_reason_string(job_ptr->state_reason)); | 
|  | } | 
|  | task_desc = job_state_reason_string(task_ptr->state_reason); | 
|  | if (strstr(job_ptr->state_desc, task_desc)) | 
|  | return; | 
|  | xstrfmtcat(job_ptr->state_desc, ",%s", task_desc); | 
|  | } | 
|  |  | 
|  | /* Combine pending tasks of a job array into a single record. | 
|  | * The tasks may have been split into separate job records because they were | 
|  | * modified or started, but the records can be re-combined if pending. */ | 
|  | static void _combine_pending_array_tasks(list_t *job_list) | 
|  | { | 
|  | squeue_job_rec_t *job_rec_ptr, *task_rec_ptr; | 
|  | list_itr_t *job_iterator, *task_iterator; | 
|  | bitstr_t *task_bitmap; | 
|  | int bitmap_size, update_cnt; | 
|  |  | 
|  | if (params.array_flag)	/* Want to see each task separately */ | 
|  | return; | 
|  |  | 
|  | job_iterator = list_iterator_create(job_list); | 
|  | while ((job_rec_ptr = list_next(job_iterator))) { | 
|  | if (!IS_JOB_PENDING(job_rec_ptr->job_ptr) || | 
|  | !job_rec_ptr->job_ptr->array_task_str || | 
|  | !job_rec_ptr->job_ptr->array_bitmap) | 
|  | continue; | 
|  | update_cnt = 0; | 
|  | task_bitmap = job_rec_ptr->job_ptr->array_bitmap; | 
|  | bitmap_size = bit_size(task_bitmap); | 
|  | task_iterator = list_iterator_create(job_list); | 
|  | while ((task_rec_ptr = list_next(task_iterator))) { | 
|  | if (!IS_JOB_PENDING(task_rec_ptr->job_ptr)) | 
|  | continue;	/* Not pending */ | 
|  | if ((task_rec_ptr == job_rec_ptr) || | 
|  | (task_rec_ptr->job_ptr->array_job_id != | 
|  | job_rec_ptr->job_ptr->array_job_id) || | 
|  | (task_rec_ptr->job_ptr->array_task_id >= | 
|  | bitmap_size)) | 
|  | continue;	/* Different job array ID */ | 
|  | if (xstrcmp(task_rec_ptr->job_ptr->name, | 
|  | job_rec_ptr->job_ptr->name)) | 
|  | continue;	/* Different name */ | 
|  | if (xstrcmp(task_rec_ptr->job_ptr->partition, | 
|  | job_rec_ptr->job_ptr->partition)) | 
|  | continue;	/* Different partition */ | 
|  | /* Combine this task into master job array record */ | 
|  | update_cnt++; | 
|  | _merge_job_reason(job_rec_ptr->job_ptr, | 
|  | task_rec_ptr->job_ptr); | 
|  | bit_set(task_bitmap, | 
|  | task_rec_ptr->job_ptr->array_task_id); | 
|  | list_delete_item(task_iterator); | 
|  | } | 
|  | list_iterator_destroy(task_iterator); | 
|  | if (update_cnt) { | 
|  | int bitstr_len = -1; | 
|  | char *bitstr_len_str = getenv("SLURM_BITSTR_LEN"); | 
|  | if (bitstr_len_str) | 
|  | bitstr_len = atoi(bitstr_len_str); | 
|  | if (bitstr_len < 0) | 
|  | bitstr_len = 64; | 
|  | xfree(job_rec_ptr->job_ptr->array_task_str); | 
|  | job_rec_ptr->job_ptr->array_task_str = | 
|  | xmalloc(bitstr_len); | 
|  | if (bitstr_len > 0) | 
|  | bit_fmt(job_rec_ptr->job_ptr->array_task_str, | 
|  | bitstr_len, task_bitmap); | 
|  | else { | 
|  | /* Print the full bitmap's string | 
|  | * representation.  For huge bitmaps this can | 
|  | * take roughly one minute, so let the client do | 
|  | * the work */ | 
|  | job_rec_ptr->job_ptr->array_task_str = | 
|  | bit_fmt_full(task_bitmap); | 
|  | } | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(job_iterator); | 
|  | } | 
|  |  | 
|  | static void _job_list_del(void *x) | 
|  | { | 
|  | squeue_job_rec_t *job_rec_ptr = (squeue_job_rec_t *) x; | 
|  | xfree(job_rec_ptr->part_name); | 
|  | xfree(job_rec_ptr); | 
|  | } | 
|  |  | 
|  | static uint32_t _part_get_prio_tier(char *part_name) | 
|  | { | 
|  | partition_info_t *part_ptr; | 
|  | uint32_t part_prio = 1;	/* Default partition priority */ | 
|  | int i; | 
|  |  | 
|  | for (i = 0, part_ptr = part_info_msg->partition_array; | 
|  | i < part_info_msg->record_count; i++, part_ptr++) { | 
|  | if (!xstrcmp(part_ptr->name, part_name)) { | 
|  | part_prio = part_ptr->priority_tier; | 
|  | break; | 
|  | } | 
|  | } | 
|  | return part_prio; | 
|  | } | 
|  |  | 
|  | static void _part_state_free(void) | 
|  | { | 
|  | if (part_info_msg) { | 
|  | slurm_free_partition_info_msg(part_info_msg); | 
|  | part_info_msg = NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  | static void _part_state_load(void) | 
|  | { | 
|  | int rc; | 
|  |  | 
|  | rc = slurm_load_partitions(0, &part_info_msg, SHOW_ALL); | 
|  | if (rc != SLURM_SUCCESS) | 
|  | slurm_perror ("slurm_load_partitions"); | 
|  | } | 
|  |  | 
|  | static int _print_str(char *str, int width, bool right, bool cut_output) | 
|  | { | 
|  | char format[64]; | 
|  | int printed = 0; | 
|  |  | 
|  | if (right == true && width > 0) | 
|  | snprintf(format, 64, "%%%ds", width); | 
|  | else if (width > 0) | 
|  | snprintf(format, 64, "%%.%ds", width); | 
|  | else if (width < 0) { | 
|  | format[0] = '%'; | 
|  | format[1] = 's'; | 
|  | format[2] = ' '; | 
|  | format[3] = '\0'; | 
|  | } else if (width == 0) { | 
|  | format[0] = '%'; | 
|  | format[1] = 's'; | 
|  | format[2] = '\0'; | 
|  | } | 
|  |  | 
|  | if ((width <= 0) || (cut_output == false) ) { | 
|  | if ((printed = printf(format, str)) < 0) | 
|  | return printed; | 
|  | } else { | 
|  | char temp[width + 1]; | 
|  | snprintf(temp, width + 1, format, str); | 
|  | if ((printed = printf("%s",temp)) < 0) | 
|  | return printed; | 
|  | } | 
|  |  | 
|  | while (printed++ < width) | 
|  | printf(" "); | 
|  |  | 
|  | return printed; | 
|  | } | 
|  |  | 
|  | int _print_nodes(char *nodes, int width, bool right, bool cut) | 
|  | { | 
|  | char *buf = NULL; | 
|  | int retval; | 
|  | buf = slurm_sort_node_list_str(nodes); | 
|  | retval = _print_str(buf, width, right, false); | 
|  | xfree(buf); | 
|  | return retval; | 
|  | } | 
|  |  | 
|  |  | 
|  | int _print_int(int number, int width, bool right, bool cut_output) | 
|  | { | 
|  | char buf[32]; | 
|  |  | 
|  | snprintf(buf, 32, "%d", number); | 
|  | return _print_str(buf, width, right, cut_output); | 
|  | } | 
|  |  | 
|  |  | 
|  | int _print_secs(long time, int width, bool right, bool cut_output) | 
|  | { | 
|  | char str[FORMAT_STRING_SIZE]; | 
|  | long days, hours, minutes, seconds; | 
|  |  | 
|  | seconds =  time % 60; | 
|  | minutes = (time / 60)   % 60; | 
|  | hours   = (time / 3600) % 24; | 
|  | days    =  time / 86400; | 
|  |  | 
|  | if ((time < 0) || (time > YEAR_SECONDS)) | 
|  | snprintf(str, FORMAT_STRING_SIZE, "INVALID"); | 
|  | else if (days) | 
|  | snprintf(str, FORMAT_STRING_SIZE, | 
|  | "%ld-%2.2ld:%2.2ld:%2.2ld", | 
|  | days, hours, minutes, seconds); | 
|  | else if (hours) | 
|  | snprintf(str, FORMAT_STRING_SIZE, | 
|  | "%ld:%2.2ld:%2.2ld", | 
|  | hours, minutes, seconds); | 
|  | else | 
|  | snprintf(str, FORMAT_STRING_SIZE, | 
|  | "%ld:%2.2ld", | 
|  | minutes, seconds); | 
|  |  | 
|  | _print_str(str, width, right, cut_output); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_time(time_t t, int level, int width, bool right) | 
|  | { | 
|  | if (t) { | 
|  | char time_str[256]; | 
|  | slurm_make_time_str(&t, time_str, sizeof(time_str)); | 
|  | _print_str(time_str, width, right, true); | 
|  | } else | 
|  | _print_str("N/A", width, right, true); | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | /***************************************************************************** | 
|  | * Job Print Functions | 
|  | *****************************************************************************/ | 
|  | static int _print_one_job_from_format(job_info_t *job, list_t *list) | 
|  | { | 
|  | list_itr_t *iter = list_iterator_create(list); | 
|  | job_format_t *current; | 
|  |  | 
|  | while ((current = list_next(iter))) { | 
|  | if (current-> | 
|  | function(job, current->width, current->right_justify, | 
|  | current->suffix) | 
|  | != SLURM_SUCCESS) | 
|  | return SLURM_ERROR; | 
|  | } | 
|  | list_iterator_destroy(iter); | 
|  |  | 
|  | printf("\n"); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | static int _print_job_from_format(void *x, void *arg) | 
|  | { | 
|  | int i, i_first, i_last; | 
|  | bitstr_t *bitmap; | 
|  | squeue_job_rec_t *job_rec_ptr = (squeue_job_rec_t *) x; | 
|  | list_t *list = arg; | 
|  |  | 
|  | if (!job_rec_ptr) { | 
|  | _print_one_job_from_format(NULL, list); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | if (job_rec_ptr->part_name) { | 
|  | xfree(job_rec_ptr->job_ptr->partition); | 
|  | job_rec_ptr->job_ptr->partition = xstrdup(job_rec_ptr-> | 
|  | part_name); | 
|  | } | 
|  |  | 
|  | if (job_rec_ptr->job_prio) | 
|  | job_rec_ptr->job_ptr->priority = job_rec_ptr->job_prio; | 
|  |  | 
|  | if (job_rec_ptr->job_ptr->array_task_str && params.array_flag) { | 
|  | char *p; | 
|  |  | 
|  | if ((p = strchr(job_rec_ptr->job_ptr->array_task_str, '%'))) | 
|  | *p = 0; | 
|  | bitmap = bit_alloc(slurm_conf.max_array_sz); | 
|  | bit_unfmt(bitmap, job_rec_ptr->job_ptr->array_task_str); | 
|  | xfree(job_rec_ptr->job_ptr->array_task_str); | 
|  | i_first = bit_ffs(bitmap); | 
|  | if (i_first == -1) | 
|  | i_last = -2; | 
|  | else | 
|  | i_last = bit_fls(bitmap); | 
|  | for (i = i_first; i <= i_last; i++) { | 
|  | if (!bit_test(bitmap, i)) | 
|  | continue; | 
|  | job_rec_ptr->job_ptr->array_task_id = i; | 
|  | _print_one_job_from_format(job_rec_ptr->job_ptr, list); | 
|  | } | 
|  | FREE_NULL_BITMAP(bitmap); | 
|  | } else { | 
|  | _print_one_job_from_format(job_rec_ptr->job_ptr, list); | 
|  | } | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int job_format_add_function(list_t *list, int width, bool right, char *suffix, | 
|  | int (*function)(job_info_t *, int, bool, char *)) | 
|  | { | 
|  | job_format_t *tmp = (job_format_t *) xmalloc(sizeof(job_format_t)); | 
|  | tmp->function = function; | 
|  | tmp->width = width; | 
|  | tmp->right_justify = right; | 
|  | tmp->suffix = suffix; | 
|  | list_append(list, tmp); | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_array_job_id(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | char id[FORMAT_STRING_SIZE]; | 
|  |  | 
|  | if (job == NULL) {	/* Print the Header instead */ | 
|  | _print_str("ARRAY_JOB_ID", width, right, true); | 
|  | } else if (job->array_task_str || | 
|  | (job->array_task_id != NO_VAL)) { | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u", job->array_job_id); | 
|  | _print_str(id, width, right, true); | 
|  | } else { | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u", job->job_id); | 
|  | _print_str(id, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_array_task_id(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) {	/* Print the Header instead */ | 
|  | _print_str("ARRAY_TASK_ID", width, right, true); | 
|  | } else if (job->array_task_str) { | 
|  | _print_str(job->array_task_str, width, right, true); | 
|  | } else if (job->array_task_id != NO_VAL) { | 
|  | char id[FORMAT_STRING_SIZE]; | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u", job->array_task_id); | 
|  | _print_str(id, width, right, true); | 
|  | } else { | 
|  | _print_str("N/A", width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_batch_host(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("EXEC_HOST", width, right, true); | 
|  | else { | 
|  | char *eh = job->batch_flag ? job->batch_host : job->alloc_node; | 
|  | _print_str(eh ? eh : "n/a", width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_burst_buffer(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("BURST_BUFFER", width, right, true); | 
|  | else { | 
|  | _print_str(job->burst_buffer, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_burst_buffer_state(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("BURST_BUFFER_STATE", width, right, true); | 
|  | else { | 
|  | _print_str(job->burst_buffer_state, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_cluster_name(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("CLUSTER", width, right, true); | 
|  | else | 
|  | _print_str(job->cluster, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_container(job_info_t *job, int width, bool right, char *suffix) | 
|  | { | 
|  | if (!job)		/* Print the Header instead */ | 
|  | _print_str("CONTAINER", width, right, true); | 
|  | else | 
|  | _print_str(job->container, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_container_id(job_info_t *job, int width, bool right, | 
|  | char *suffix) | 
|  | { | 
|  | if (!job) /* Print the Header instead */ | 
|  | _print_str("CONTAINERID", width, right, true); | 
|  | else | 
|  | _print_str(job->container_id, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_core_spec(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | char spec[FORMAT_STRING_SIZE]; | 
|  |  | 
|  | if (job == NULL) {	/* Print the Header instead */ | 
|  | _print_str("CORE_SPEC", width, right, true); | 
|  | } else if (job->core_spec == NO_VAL16) { | 
|  | _print_str("N/A", width, right, true); | 
|  | } else if (job->core_spec & CORE_SPEC_THREAD) { | 
|  | snprintf(spec, FORMAT_STRING_SIZE, "%d Threads", | 
|  | (job->core_spec & (~CORE_SPEC_THREAD))); | 
|  | _print_str(spec, width, right, true); | 
|  | } else { | 
|  | _print_int(job->core_spec, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_delay_boot(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("DELAY_BOOT", width, right, true); | 
|  | else | 
|  | _print_secs((long)job->delay_boot, width, right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_job_id(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | char id[FORMAT_STRING_SIZE]; | 
|  | int len; | 
|  | char *buf; | 
|  |  | 
|  | if (job == NULL) {	/* Print the Header instead */ | 
|  | _print_str("JOBID", width, right, true); | 
|  | } else if (job->array_task_str) { | 
|  | if (getenv("SLURM_BITSTR_LEN")) { | 
|  | len = strlen(job->array_task_str) + 64; | 
|  | buf = xmalloc(len); | 
|  | sprintf(buf, "%u_[%s]", job->array_job_id, | 
|  | job->array_task_str); | 
|  | _print_str(buf, width, right, false); | 
|  | xfree(buf); | 
|  | } else { | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u_[%s]", | 
|  | job->array_job_id, job->array_task_str); | 
|  | _print_str(id, width, right, true); | 
|  | } | 
|  | } else if (job->array_task_id != NO_VAL) { | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u_%u", | 
|  | job->array_job_id, job->array_task_id); | 
|  | _print_str(id, width, right, true); | 
|  | } else if (job->het_job_id) { | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u+%u", | 
|  | job->het_job_id, job->het_job_offset); | 
|  | _print_str(id, width, right, true); | 
|  | } else { | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u", job->job_id); | 
|  | _print_str(id, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_job_id2(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL) {	/* Print the Header instead */ | 
|  | _print_str("JOBID", width, right, true); | 
|  | } else { | 
|  | char id[FORMAT_STRING_SIZE]; | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u", job->job_id); | 
|  | _print_str(id, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_partition(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("PARTITION", width, right, true); | 
|  | else { | 
|  | _print_str(job->partition, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_prefix(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_reason(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)        /* Print the Header instead */ | 
|  | _print_str("REASON", width, right, true); | 
|  | else { | 
|  | const char *reason; | 
|  | if (job->state_desc) | 
|  | reason = job->state_desc; | 
|  | else | 
|  | reason = job_state_reason_string(job->state_reason); | 
|  | _print_str((char *)reason, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_name(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("NAME", width, right, true); | 
|  | else | 
|  | _print_str(job->name, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_licenses(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("LICENSES", width, right, true); | 
|  | else | 
|  | _print_str(job->licenses, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_licenses_alloc(job_info_t *job, int width, bool right, | 
|  | char *suffix) | 
|  | { | 
|  | if (job == NULL) /* Print the Header instead */ | 
|  | _print_str("LICENSES_ALLOC", width, right, true); | 
|  | else | 
|  | _print_str(job->licenses_allocated, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_wckey(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("WCKEY", width, right, true); | 
|  | else | 
|  | _print_str(job->wckey, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_user_id(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("UID", width, right, true); | 
|  | else | 
|  | _print_int(job->user_id, width, right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_user_name(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("USER", width, right, true); | 
|  | else { | 
|  | char *uname = uid_to_string_cached((uid_t) job->user_id); | 
|  | _print_str(uname, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_group_id(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("GROUP", width, right, true); | 
|  | else | 
|  | _print_int(job->group_id, width, right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_group_name(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("GROUP", width, right, true); | 
|  | else { | 
|  | char *group = gid_to_string(job->group_id); | 
|  | _print_str(group, width, right, true); | 
|  | xfree(group); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_job_state(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("STATE", width, right, true); | 
|  | else | 
|  | _print_str(job_state_string(job->job_state), width, right, | 
|  | true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_last_sched_eval(job_info_t *job, int width, bool right, | 
|  | char *suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("LAST_SCHED_EVAL", width, right, true); | 
|  | else | 
|  | _print_time(job->last_sched_eval, 0, width, right); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_job_state_compact(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("ST", width, right, true); | 
|  | else | 
|  | _print_str(job_state_string_compact(job->job_state), width, | 
|  | right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_time_left(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("TIME_LEFT", width, right, true); | 
|  | else if (job->time_limit == INFINITE) | 
|  | _print_str("UNLIMITED", width, right, true); | 
|  | else if (job->time_limit == NO_VAL) | 
|  | _print_str("NOT_SET", width, right, true); | 
|  | else { | 
|  | time_t time_left = job->time_limit * 60 - job_time_used(job); | 
|  | _print_secs(time_left, width, right, false); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_time_limit(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("TIME_LIMIT", width, right, true); | 
|  | else if (job->time_limit == INFINITE) | 
|  | _print_str("UNLIMITED", width, right, true); | 
|  | else if (job->time_limit == NO_VAL) | 
|  | _print_str("NOT_SET", width, right, true); | 
|  | else | 
|  | _print_secs((job->time_limit*60), width, right, false); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  | int _print_job_het_job_offset(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | char id[FORMAT_STRING_SIZE]; | 
|  |  | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("HET_JOB_OFFSET", width, right, true); | 
|  | else if (job->het_job_id == 0) | 
|  | _print_str("N/A", width, right, true); | 
|  | else { | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u", job->het_job_offset); | 
|  | _print_str(id, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_het_job_id(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | char id[FORMAT_STRING_SIZE]; | 
|  |  | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("HET_JOB_ID", width, right, true); | 
|  | else if (job->het_job_id == 0) | 
|  | _print_str("N/A", width, right, true); | 
|  | else { | 
|  | snprintf(id, FORMAT_STRING_SIZE, "%u", job->het_job_id); | 
|  | _print_str(id, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_het_job_id_set(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("HET_JOB_ID_SET", width, right, true); | 
|  | else if (job->het_job_id == 0) | 
|  | _print_str("N/A", width, right, true); | 
|  | else | 
|  | _print_str(job->het_job_id_set, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  | int _print_job_time_used(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("TIME", width, right, true); | 
|  | else | 
|  | _print_secs(job_time_used(job), width, right, false); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | long job_time_used(job_info_t * job_ptr) | 
|  | { | 
|  | time_t end_time; | 
|  |  | 
|  | if ((job_ptr->start_time == 0) || IS_JOB_PENDING(job_ptr)) | 
|  | return 0L; | 
|  |  | 
|  | if (IS_JOB_SUSPENDED(job_ptr)) | 
|  | return (long) job_ptr->pre_sus_time; | 
|  |  | 
|  | if (IS_JOB_RUNNING(job_ptr) || (job_ptr->end_time == 0)) | 
|  | end_time = time(NULL); | 
|  | else | 
|  | end_time = job_ptr->end_time; | 
|  |  | 
|  | if (job_ptr->suspend_time) | 
|  | return (long) (difftime(end_time, job_ptr->suspend_time) | 
|  | + job_ptr->pre_sus_time); | 
|  | return (long) (difftime(end_time, job_ptr->start_time)); | 
|  | } | 
|  |  | 
|  | int _print_job_time_submit(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)        /* Print the Header instead */ | 
|  | _print_str("SUBMIT_TIME", width, right, true); | 
|  | else | 
|  | _print_time(job->submit_time, 0, width, right); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_time_pending(job_info_t *job, int width, bool right, | 
|  | char *suffix) | 
|  | { | 
|  | time_t now = time(NULL); | 
|  |  | 
|  | /* | 
|  | * If the job has started, defined as (start - submit). | 
|  | * Else, defined as (now - submit). | 
|  | */ | 
|  |  | 
|  | if (!job)	/* Print the Header instead */ | 
|  | _print_str("PENDING_TIME", width, right, true); | 
|  | else if (job->start_time && (job->start_time < now)) | 
|  | _print_int((job->start_time - job->submit_time), width, right, | 
|  | true); | 
|  | else | 
|  | _print_int((now - job->submit_time), width, right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_time_start(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("START_TIME", width, right, true); | 
|  | else | 
|  | _print_time(job->start_time, 0, width, right); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  | int _print_job_deadline(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)        /* Print the Header instead */ | 
|  | _print_str("DEADLINE", width, right, true); | 
|  | else | 
|  | _print_time(job->deadline, 0, width, right); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  | int _print_job_time_end(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("END_TIME", width, right, true); | 
|  | else if ((job->time_limit == INFINITE) && | 
|  | (job->end_time > time(NULL))) | 
|  | _print_str("NONE", width, right, true); | 
|  | else | 
|  | _print_time(job->end_time, 0, width, right); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_priority(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | char temp[FORMAT_STRING_SIZE]; | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("PRIORITY", width, right, true); | 
|  | else { | 
|  | double prio = (double) job->priority / | 
|  | (double) ((uint32_t) 0xffffffff); | 
|  | sprintf(temp, "%16.14f", prio); | 
|  | _print_str(temp, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_priority_long(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | char temp[FORMAT_STRING_SIZE]; | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("PRIORITY", width, right, true); | 
|  | else { | 
|  | sprintf(temp, "%u", job->priority); | 
|  | _print_str(temp, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_nodes(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("NODELIST", width, right, false); | 
|  | else | 
|  | _print_nodes(job->nodes, width, right, false); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_schednodes(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("SCHEDNODES", width, right, false); | 
|  | else | 
|  | _print_str(job->sched_nodes, width, right, false); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_reason_list(job_info_t * job, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) {	/* Print the Header instead */ | 
|  | _print_str("NODELIST(REASON)", width, right, false); | 
|  | } else if (!IS_JOB_COMPLETING(job) | 
|  | && (IS_JOB_PENDING(job) | 
|  | || IS_JOB_STAGE_OUT(job) | 
|  | || IS_JOB_TIMEOUT(job) | 
|  | || IS_JOB_OOM(job) | 
|  | || IS_JOB_DEADLINE(job) | 
|  | || IS_JOB_FAILED(job))) { | 
|  | char *reason_fmt = NULL; | 
|  | const char *reason = NULL; | 
|  | if (job->state_desc) | 
|  | reason = job->state_desc; | 
|  | else | 
|  | reason = job_state_reason_string(job->state_reason); | 
|  | xstrfmtcat(reason_fmt, "(%s)", reason); | 
|  | _print_str(reason_fmt, width, right, true); | 
|  | xfree(reason_fmt); | 
|  | } else | 
|  | _print_nodes(job->nodes, width, right, false); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_node_inx(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("NODE_BY_INDEX", width, right, true); | 
|  | else { | 
|  | int *current = job->node_inx; | 
|  | int curr_width = 0; | 
|  | while (*current != -1 && curr_width < width) { | 
|  | if (curr_width) | 
|  | printf(","); | 
|  | curr_width += _print_int(*current, width, right, true); | 
|  | current++; | 
|  | } | 
|  | while (curr_width < width) | 
|  | curr_width += printf(" "); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_num_cpus(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | char tmp_char[18]; | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("CPUS", width, right, true); | 
|  | else { | 
|  | snprintf(tmp_char, sizeof(tmp_char), "%u", job->num_cpus); | 
|  | _print_str(tmp_char, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_num_nodes(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | char tmp_char[8]; | 
|  |  | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("NODES", width, right_justify, true); | 
|  | else { | 
|  | snprintf(tmp_char, sizeof(tmp_char), "%d", job->num_nodes); | 
|  | _print_str(tmp_char, width, right_justify, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_num_sct(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | char sockets[10]; | 
|  | char cores[10]; | 
|  | char threads[10]; | 
|  | char *sct = NULL; | 
|  | if (job) { | 
|  | if (job->sockets_per_node == NO_VAL16) | 
|  | strcpy(sockets, "*"); | 
|  | else | 
|  | convert_num_unit((float)job->sockets_per_node, sockets, | 
|  | sizeof(sockets), UNIT_NONE, NO_VAL, | 
|  | params.convert_flags); | 
|  | if (job->cores_per_socket == NO_VAL16) | 
|  | strcpy(cores, "*"); | 
|  | else | 
|  | convert_num_unit((float)job->cores_per_socket, cores, | 
|  | sizeof(cores), UNIT_NONE, NO_VAL, | 
|  | params.convert_flags); | 
|  | if (job->threads_per_core == NO_VAL16) | 
|  | strcpy(threads, "*"); | 
|  | else | 
|  | convert_num_unit((float)job->threads_per_core, threads, | 
|  | sizeof(threads), UNIT_NONE, NO_VAL, | 
|  | params.convert_flags); | 
|  | xstrfmtcat(sct, "%s:%s:%s", sockets, cores, threads); | 
|  | _print_str(sct, width, right_justify, true); | 
|  | xfree(sct); | 
|  | } else { | 
|  | _print_str("S:C:T", width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_num_tasks(job_info_t * job, int width, bool right, char* suffix) | 
|  | { | 
|  | char tmp_char[18]; | 
|  | if (job == NULL) {	/* Print the Header instead */ | 
|  | _print_str("TASKS", width, right, true); | 
|  | } else { | 
|  | snprintf(tmp_char, sizeof(tmp_char), "%u", job->num_tasks); | 
|  | _print_str(tmp_char, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_over_subscribe(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) {	/* Print the Header instead */ | 
|  | _print_str("OVER_SUBSCRIBE", width, right_justify, true); | 
|  | } else { | 
|  | _print_str(job_share_string(job->shared), | 
|  | width, right_justify, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_contiguous(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("CONTIGUOUS", width, right_justify, true); | 
|  | else { | 
|  | _print_int(job->contiguous, width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_pn_min_cpus(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | char tmp_char[8]; | 
|  |  | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("MIN_CPUS", width, right_justify, true); | 
|  | else { | 
|  | convert_num_unit((float)job->pn_min_cpus, tmp_char, | 
|  | sizeof(tmp_char), UNIT_NONE, NO_VAL, | 
|  | params.convert_flags); | 
|  | _print_str(tmp_char, width, right_justify, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_sockets(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | char tmp_char[8]; | 
|  |  | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("SOCKETS_PER_NODE", width, right_justify, true); | 
|  | else { | 
|  | if (job->sockets_per_node == NO_VAL16) | 
|  | strcpy(tmp_char, "*"); | 
|  | else | 
|  | convert_num_unit((float)job->sockets_per_node, tmp_char, | 
|  | sizeof(tmp_char), UNIT_NONE, NO_VAL, | 
|  | params.convert_flags); | 
|  | _print_str(tmp_char, width, right_justify, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_cores(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | char tmp_char[8]; | 
|  |  | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("CORES_PER_SOCKET", width, right_justify, true); | 
|  | else { | 
|  | if (job->cores_per_socket == NO_VAL16) | 
|  | strcpy(tmp_char, "*"); | 
|  | else | 
|  | convert_num_unit((float)job->cores_per_socket, tmp_char, | 
|  | sizeof(tmp_char), UNIT_NONE, NO_VAL, | 
|  | params.convert_flags); | 
|  | _print_str(tmp_char, width, right_justify, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_threads(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | char tmp_char[8]; | 
|  |  | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("THREADS_PER_CORE", width, right_justify, true); | 
|  | else { | 
|  | if (job->threads_per_core == NO_VAL16) | 
|  | strcpy(tmp_char, "*"); | 
|  | else | 
|  | convert_num_unit((float)job->threads_per_core, tmp_char, | 
|  | sizeof(tmp_char), UNIT_NONE, NO_VAL, | 
|  | params.convert_flags); | 
|  | _print_str(tmp_char, width, right_justify, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_pn_min_memory(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | char min_mem[10]; | 
|  |  | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("MIN_MEMORY", width, right_justify, true); | 
|  | else { | 
|  | job->pn_min_memory &= (~MEM_PER_CPU); | 
|  | convert_num_unit((float)job->pn_min_memory, min_mem, | 
|  | sizeof(min_mem), UNIT_MEGA, NO_VAL, | 
|  | params.convert_flags); | 
|  | _print_str(min_mem, width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int | 
|  | _print_pn_min_tmp_disk(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | char tmp_char[10]; | 
|  |  | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("MIN_TMP_DISK", width, right_justify, true); | 
|  | else { | 
|  | convert_num_unit((float)job->pn_min_tmp_disk, tmp_char, | 
|  | sizeof(tmp_char), UNIT_MEGA, NO_VAL, | 
|  | params.convert_flags); | 
|  | _print_str(tmp_char, width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_req_nodes(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("REQ_NODES", width, right_justify, true); | 
|  | else | 
|  | _print_nodes(job->req_nodes, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_exc_nodes(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("EXC_NODES", width, right_justify, true); | 
|  | else | 
|  | _print_nodes(job->exc_nodes, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int | 
|  | _print_job_req_node_inx(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("REQ_NODES_BY_INX", width, right_justify, true); | 
|  | else { | 
|  | int *current = job->req_node_inx; | 
|  | int curr_width = 0; | 
|  | while (*current != -1 && curr_width < width) { | 
|  | curr_width += | 
|  | _print_int(*current, width, right_justify, | 
|  | true); | 
|  | printf(","); | 
|  | } | 
|  | while (curr_width < width) | 
|  | curr_width += printf(" "); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int | 
|  | _print_job_exc_node_inx(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("EXC_NODES_BY_INX", width, right_justify, true); | 
|  | else { | 
|  | int *current = job->exc_node_inx; | 
|  | int curr_width = 0; | 
|  | while (*current != -1 && curr_width < width) { | 
|  | curr_width += | 
|  | _print_int(*current, width, right_justify, | 
|  | true); | 
|  | printf(","); | 
|  | } | 
|  | while (curr_width < width) | 
|  | curr_width += printf(" "); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_features(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("FEATURES", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->features, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_cluster_features(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("CLUSTER_FEATURES", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->cluster_features, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_prefer(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("PREFER", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->prefer, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_account(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	 /* Print the Header instead */ | 
|  | _print_str("ACCOUNT", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->account, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_admin_comment(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	 /* Print the Header instead */ | 
|  | _print_str("ADMIN_COMMENT", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->admin_comment, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_system_comment(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	 /* Print the Header instead */ | 
|  | _print_str("SYSTEM_COMMENT", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->system_comment, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_comment(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	 /* Print the Header instead */ | 
|  | _print_str("COMMENT", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->comment, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_dependency(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	/* Print the Header instead */ | 
|  | _print_str("DEPENDENCY", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->dependency, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_qos(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	 /* Print the Header instead */ | 
|  | _print_str("QOS", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->qos, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_reservation(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL)	 /* Print the Header instead */ | 
|  | _print_str("RESERVATION", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->resv_name, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_command(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("COMMAND", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->command, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_work_dir(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("WORK_DIR", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->work_dir, width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_nice(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("NICE", width, right_justify, true); | 
|  | else { | 
|  | int nice = (int) job->nice; | 
|  | nice -= NICE_OFFSET; | 
|  | _print_int(nice, width, right_justify, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_accrue_time(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("ACCRUE_TIME", width, right_justify, true); | 
|  | else { | 
|  | _print_time(job->accrue_time, 0, width, right_justify); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_alloc_nodes(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("ALLOC_NODES", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->alloc_node, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_alloc_sid(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("ALLOC_SID", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->alloc_sid, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_assoc_id(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("ASSOC_ID", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->assoc_id, width, right_justify,true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  |  | 
|  | } | 
|  |  | 
|  | int _print_job_batch_flag(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("BATCH_FLAG", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->batch_flag, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_boards_per_node(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("BOARDS_PER_NODE", width, right_justify, true); | 
|  | else if (job->boards_per_node == NO_VAL16) | 
|  | _print_str("N/A", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->boards_per_node, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_cpus_per_task(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("CPUS_PER_TASK", width, right_justify, true); | 
|  | else if (job->cpus_per_task == NO_VAL16) | 
|  | _print_str("N/A", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->cpus_per_task, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_derived_ec(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | uint16_t exit_status = 0, term_sig = 0; | 
|  | char *out = NULL; | 
|  |  | 
|  | if (!job) | 
|  | _print_str("DERIVED_EC", width, right_justify, true); | 
|  | else if (job->derived_ec != NO_VAL) { | 
|  | if (WIFSIGNALED(job->derived_ec)) | 
|  | term_sig = WTERMSIG(job->derived_ec); | 
|  | else if (WIFEXITED(job->derived_ec)) | 
|  | exit_status = WEXITSTATUS(job->derived_ec); | 
|  |  | 
|  | xstrfmtcat(out, "%u:%u", exit_status, term_sig); | 
|  | _print_str(out, width, right_justify, true); | 
|  | xfree(out); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  |  | 
|  | } | 
|  |  | 
|  | int _print_job_eligible_time(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("ELIGIBLE_TIME", width, right_justify, true); | 
|  | else { | 
|  | _print_time(job->eligible_time, 0, width, right_justify); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_exit_code(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | uint16_t exit_status = 0, term_sig = 0; | 
|  | char *out = NULL; | 
|  |  | 
|  | if (!job) | 
|  | _print_str("EXIT_CODE", width, right_justify, true); | 
|  | else if (job->exit_code != NO_VAL) { | 
|  | if (WIFSIGNALED(job->exit_code)) | 
|  | term_sig = WTERMSIG(job->exit_code); | 
|  | else if (WIFEXITED(job->exit_code)) | 
|  | exit_status = WEXITSTATUS(job->exit_code); | 
|  |  | 
|  | xstrfmtcat(out, "%u:%u", exit_status, term_sig); | 
|  | _print_str(out, width, right_justify, true); | 
|  | xfree(out); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_fed_origin(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("ORIGIN", width, right_justify, true); | 
|  | else { | 
|  | if (job->fed_origin_str) | 
|  | _print_str(job->fed_origin_str, width, right_justify, | 
|  | true); | 
|  | else | 
|  | _print_str("NA", width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_fed_origin_raw(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("ORIGIN_RAW", width, right_justify, true); | 
|  | else { | 
|  | int id = job->job_id >> 26; | 
|  | if (id) | 
|  | _print_int(id, width, right_justify, true); | 
|  | else | 
|  | _print_str("NA", width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_fed_siblings_active(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("ACTIVE_SIBLINGS", width, right_justify, true); | 
|  | else { | 
|  | if (job->fed_siblings_active_str) | 
|  | _print_str(job->fed_siblings_active_str, width, right_justify, | 
|  | true); | 
|  | else | 
|  | _print_str("NA", width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_fed_siblings_active_raw(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("ACTIVE_SIBLINGS_RAW", width, right_justify, true); | 
|  | else { | 
|  | int bit = 1; | 
|  | char *ids = NULL; | 
|  | uint64_t tmp_sibs = job->fed_siblings_active; | 
|  | while (tmp_sibs) { | 
|  | if (tmp_sibs & 1) | 
|  | xstrfmtcat(ids, "%s%d", (ids) ? "," : "", bit); | 
|  |  | 
|  | tmp_sibs >>= 1; | 
|  | bit++; | 
|  | } | 
|  | if (ids) | 
|  | _print_str(ids, width, right_justify, true); | 
|  | else | 
|  | _print_str("NA", width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_fed_siblings_viable(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("VIABLE_SIBLINGS", width, right_justify, true); | 
|  | else { | 
|  | if (job->fed_siblings_viable_str) | 
|  | _print_str(job->fed_siblings_viable_str, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("NA", width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_fed_siblings_viable_raw(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("VIALBLE_SIBLINGS_RAW", width, right_justify, true); | 
|  | else { | 
|  | int bit = 1; | 
|  | char *ids = NULL; | 
|  | uint64_t tmp_sibs = job->fed_siblings_viable; | 
|  | while (tmp_sibs) { | 
|  | if (tmp_sibs & 1) | 
|  | xstrfmtcat(ids, "%s%d", (ids) ? "," : "", bit); | 
|  |  | 
|  | tmp_sibs >>= 1; | 
|  | bit++; | 
|  | } | 
|  | if (ids) | 
|  | _print_str(ids, width, right_justify, true); | 
|  | else | 
|  | _print_str("NA", width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_max_cpus(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("MAX_CPUS", width, right_justify, true); | 
|  | else if (job->max_cpus != 0) | 
|  | _print_int(job->max_cpus, width, right_justify, true); | 
|  | else | 
|  | _print_int(job->num_cpus, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_max_nodes(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("MAX_NODES", width, right_justify, true); | 
|  | else if (job->max_nodes != 0) | 
|  | _print_int(job->max_nodes, width, right_justify, true); | 
|  | else | 
|  | _print_int(job->num_nodes, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_network(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("NETWORK", width, right_justify, true); | 
|  | else | 
|  | _print_str(job->network, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_ntasks_per_core(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("NTASKS_PER_CORE", width, right_justify, true); | 
|  | else if ((job->ntasks_per_core == NO_VAL16) || | 
|  | (job->ntasks_per_core == INFINITE16)) | 
|  | _print_str("N/A", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->ntasks_per_core, width, right_justify, true); | 
|  |  | 
|  | if(suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_ntasks_per_node(job_info_t * job, int width, bool right_justify, | 
|  | char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("NTASKS_PER_NODE", width, right_justify, true); | 
|  | else if ((job->ntasks_per_node == NO_VAL16) || | 
|  | (job->ntasks_per_node == INFINITE16)) | 
|  | _print_str("N/A", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->ntasks_per_node, width, right_justify, true); | 
|  |  | 
|  | if(suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_ntasks_per_socket(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("NTASKS_PER_SOCKET", width, right_justify, true); | 
|  | else if ((job->ntasks_per_socket == NO_VAL16) || | 
|  | (job->ntasks_per_socket == INFINITE16)) | 
|  | _print_str("N/A", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->ntasks_per_socket, width, right_justify, true); | 
|  |  | 
|  | if(suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_ntasks_per_board(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("NTASKS_PER_BOARD", width, right_justify, true); | 
|  | else if ((job->ntasks_per_board == NO_VAL16) || | 
|  | (job->ntasks_per_board == INFINITE16)) | 
|  | _print_str("N/A", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->ntasks_per_board, width, right_justify, true); | 
|  |  | 
|  | if(suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_preempt_time(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("PREEMPT_TIME", width, right_justify, true); | 
|  | else if (job->preempt_time == INFINITE) | 
|  | _print_str("UNLIMITED", width, right_justify, true); | 
|  | else if (job->preempt_time == NO_VAL) | 
|  | _print_str("NOT_SET", width, right_justify, true); | 
|  | else | 
|  | _print_time(job->preempt_time, 0, width, right_justify); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  |  | 
|  | } | 
|  |  | 
|  | int _print_job_profile(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("PROFILE", width, right_justify, true); | 
|  | else | 
|  | _print_str(acct_gather_profile_to_string(job->profile), | 
|  | width, right_justify, true); | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_reboot(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("REBOOT", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->reboot, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_req_switch(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("REQ_SWITCH", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->req_switch, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_requeue(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("REQUEUE", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->requeue, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s",suffix); | 
|  | return SLURM_SUCCESS; | 
|  |  | 
|  | } | 
|  |  | 
|  | int _print_job_resize_time(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("RESIZE_TIME", width, right_justify, true); | 
|  | else if (job->resize_time) | 
|  | _print_secs((job->resize_time*60), width, right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, right_justify, false); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  |  | 
|  | } | 
|  |  | 
|  | int _print_job_restart_cnt(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("RESTART_COUNT", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->restart_cnt, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_segment_size(job_info_t *job, int width, bool right_justify, | 
|  | char *suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("SEGMENT_SIZE", width, right_justify, true); | 
|  | else if (job->segment_size) | 
|  | _print_int(job->segment_size, width, right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_sockets_per_board(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("SOCKETS_PER_BOARD", width, right_justify, true); | 
|  | else | 
|  | _print_int(job->sockets_per_board, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  |  | 
|  | } | 
|  |  | 
|  | int _print_job_std_err(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("STDERR", width, right_justify, true); | 
|  | else if (params.expand_patterns) { | 
|  | char *tmp_str = slurm_expand_job_stdio_fields(job->std_err, | 
|  | job); | 
|  | _print_str(tmp_str, width, right_justify, true); | 
|  | xfree(tmp_str); | 
|  | } else | 
|  | _print_str(job->std_err, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_std_in(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("STDIN", width, right_justify, true); | 
|  | else if (params.expand_patterns) { | 
|  | char *tmp_str = slurm_expand_job_stdio_fields(job->std_in, job); | 
|  | _print_str(tmp_str, width, right_justify, true); | 
|  | xfree(tmp_str); | 
|  | } else | 
|  | _print_str(job->std_in, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  |  | 
|  | } | 
|  |  | 
|  | int _print_job_std_out(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | /* | 
|  | * Populate the default patterns in std_out for batch jobs. | 
|  | */ | 
|  | if (job && !job->std_out && job->batch_flag) { | 
|  | if (job->array_job_id) | 
|  | xstrfmtcat(job->std_out, "%s/slurm-%%A_%%a.out", | 
|  | job->work_dir); | 
|  | else | 
|  | xstrfmtcat(job->std_out, "%s/slurm-%%j.out", | 
|  | job->work_dir); | 
|  | } | 
|  |  | 
|  | if (job == NULL) | 
|  | _print_str("STDOUT", width, right_justify, true); | 
|  | else if (params.expand_patterns) { | 
|  | char *tmp_str = slurm_expand_job_stdio_fields(job->std_out, | 
|  | job); | 
|  | _print_str(tmp_str, width, right_justify, true); | 
|  | xfree(tmp_str); | 
|  | } else | 
|  | _print_str(job->std_out, width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_min_time(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("TIME_MIN", width, right_justify, true); | 
|  | else | 
|  | _print_secs((job->time_min*60), width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_wait4switch(job_info_t * job, int width, | 
|  | bool right_justify, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("WAIT4SWITCH", width, right_justify, true); | 
|  | else | 
|  | _print_secs(job->wait4switch, | 
|  | width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_cpus_per_tres(job_info_t *job, int width, | 
|  | bool right_justify, char *suffix) | 
|  | { | 
|  | if (job == NULL) { | 
|  | _print_str("CPUS_PER_TRES", width, right_justify, true); | 
|  | } else { | 
|  | if (job->cpus_per_tres) | 
|  | _print_str(job->cpus_per_tres, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, | 
|  | right_justify, true); | 
|  |  | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_mem_per_tres(job_info_t *job, int width, | 
|  | bool right_justify, char *suffix) | 
|  | { | 
|  | if (job == NULL) { | 
|  | _print_str("MEM_PER_TRES", width, right_justify, true); | 
|  | } else { | 
|  | if (job->mem_per_tres) | 
|  | _print_str(job->mem_per_tres, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, | 
|  | right_justify, true); | 
|  |  | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_tres_alloc(job_info_t *job, int width, | 
|  | bool right_justify, char *suffix) | 
|  | { | 
|  | if (job == NULL) { | 
|  | _print_str("TRES_ALLOC", width, right_justify, true); | 
|  | } else { | 
|  | if (job->tres_alloc_str) | 
|  | _print_str(job->tres_alloc_str, width, | 
|  | right_justify, true); | 
|  | else if (job->tres_req_str) | 
|  | _print_str(job->tres_req_str, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, | 
|  | right_justify, true); | 
|  |  | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_tres_bind(job_info_t *job, int width, | 
|  | bool right_justify, char *suffix) | 
|  | { | 
|  | if (job == NULL) { | 
|  | _print_str("TRES_BIND", width, right_justify, true); | 
|  | } else { | 
|  | if (job->tres_bind) | 
|  | _print_str(job->tres_bind, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, | 
|  | right_justify, true); | 
|  |  | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_tres_freq(job_info_t *job, int width, | 
|  | bool right_justify, char *suffix) | 
|  | { | 
|  | if (job == NULL) { | 
|  | _print_str("TRES_FREQ", width, right_justify, true); | 
|  | } else { | 
|  | if (job->tres_freq) | 
|  | _print_str(job->tres_freq, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, | 
|  | right_justify, true); | 
|  |  | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_tres_per_job(job_info_t *job, int width, | 
|  | bool right_justify, char *suffix) | 
|  | { | 
|  | if (job == NULL) { | 
|  | _print_str("TRES_PER_JOB", width, right_justify, true); | 
|  | } else { | 
|  | if (job->tres_per_job) | 
|  | _print_str(job->tres_per_job, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, | 
|  | right_justify, true); | 
|  |  | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_tres_per_node(job_info_t *job, int width, | 
|  | bool right_justify, char *suffix) | 
|  | { | 
|  | if (job == NULL) { | 
|  | _print_str("TRES_PER_NODE", width, right_justify, true); | 
|  | } else { | 
|  | if (job->tres_per_node) | 
|  | _print_str(job->tres_per_node, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, | 
|  | right_justify, true); | 
|  |  | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_tres_per_socket(job_info_t *job, int width, | 
|  | bool right_justify, char *suffix) | 
|  | { | 
|  | if (job == NULL) { | 
|  | _print_str("TRES_PER_SOCKET", width, right_justify, true); | 
|  | } else { | 
|  | if (job->tres_per_socket) | 
|  | _print_str(job->tres_per_socket, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, | 
|  | right_justify, true); | 
|  |  | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_tres_per_task(job_info_t *job, int width, | 
|  | bool right_justify, char *suffix) | 
|  | { | 
|  | if (job == NULL) { | 
|  | _print_str("TRES_PER_TASK", width, right_justify, true); | 
|  | } else { | 
|  | if (job->tres_per_task) | 
|  | _print_str(job->tres_per_task, width, | 
|  | right_justify, true); | 
|  | else | 
|  | _print_str("N/A", width, | 
|  | right_justify, true); | 
|  |  | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_mcs_label(job_info_t * job, int width, | 
|  | bool right, char* suffix) | 
|  | { | 
|  | if (job == NULL) | 
|  | _print_str("MCSLABEL", width, right, true); | 
|  | else | 
|  | _print_str(job->mcs_label, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | /***************************************************************************** | 
|  | * Job Step Print Functions | 
|  | *****************************************************************************/ | 
|  | static int _print_step_from_format(void *x, void *arg) | 
|  | { | 
|  | job_step_info_t *job_step = (job_step_info_t *) x; | 
|  | list_t *list = arg; | 
|  | list_itr_t *i = list_iterator_create(list); | 
|  | step_format_t *current; | 
|  |  | 
|  | while ((current = list_next(i))) { | 
|  | if (current-> | 
|  | function(job_step, current->width, | 
|  | current->right_justify, current->suffix) | 
|  | != SLURM_SUCCESS) | 
|  | return SLURM_ERROR; | 
|  | } | 
|  | list_iterator_destroy(i); | 
|  | printf("\n"); | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int step_format_add_function( | 
|  | list_t *list, int width, bool right_justify, char *suffix, | 
|  | int (*function) (job_step_info_t *, int, bool, char *)) | 
|  | { | 
|  | step_format_t *tmp = | 
|  | (step_format_t *) xmalloc(sizeof(step_format_t)); | 
|  | tmp->function = function; | 
|  | tmp->width = width; | 
|  | tmp->right_justify = right_justify; | 
|  | tmp->suffix = suffix; | 
|  | list_append(list, tmp); | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_cluster_name(job_step_info_t *step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("CLUSTER", width, right, true); | 
|  | else | 
|  | _print_str(step->cluster, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_container(job_step_info_t *step, int width, bool right, | 
|  | char *suffix) | 
|  | { | 
|  | if (!step)		/* Print the Header instead */ | 
|  | _print_str("CONTAINER", width, right, true); | 
|  | else | 
|  | _print_str(step->container, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_container_id(job_step_info_t *step, int width, bool right, | 
|  | char *suffix) | 
|  | { | 
|  | if (!step) /* Print the Header instead */ | 
|  | _print_str("CONTAINERID", width, right, true); | 
|  | else | 
|  | _print_str(step->container_id, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_id(job_step_info_t * step, int width, bool right, char* suffix) | 
|  | { | 
|  | char id[FORMAT_STRING_SIZE]; | 
|  |  | 
|  | if (step == NULL) {	/* Print the Header instead */ | 
|  | _print_str("STEPID", width, right, true); | 
|  | } else { | 
|  | uint16_t flags = STEP_ID_FLAG_NO_PREFIX; | 
|  | int len = FORMAT_STRING_SIZE; | 
|  | int pos = 0; | 
|  | if (step->array_job_id) { | 
|  | pos = snprintf(id, len, "%u_%u.", | 
|  | step->array_job_id, step->array_task_id); | 
|  | flags |= STEP_ID_FLAG_NO_JOB; | 
|  | len -= pos; | 
|  | } | 
|  |  | 
|  | log_build_step_id_str(&step->step_id, | 
|  | id+pos, len, flags); | 
|  |  | 
|  | _print_str(id, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_partition(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("PARTITION", width, right, true); | 
|  | else { | 
|  | _print_str(step->partition, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_prefix(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_std_err(job_step_info_t *step, int width, bool right_justify, | 
|  | char *suffix) | 
|  | { | 
|  | if (step == NULL) { | 
|  | _print_str("STDERR", width, right_justify, true); | 
|  | } else if (!step->std_err) { | 
|  | _print_str("", width, right_justify, true); | 
|  | } else if (params.expand_patterns) { | 
|  | char *tmp_str = slurm_expand_step_stdio_fields(step->std_err, | 
|  | step); | 
|  | _print_str(tmp_str, width, right_justify, true); | 
|  | xfree(tmp_str); | 
|  | } else { | 
|  | _print_str(step->std_err, width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_std_in(job_step_info_t *step, int width, bool right_justify, | 
|  | char *suffix) | 
|  | { | 
|  | if (step == NULL) { | 
|  | _print_str("STDIN", width, right_justify, true); | 
|  | } else if (!step->std_in) { | 
|  | _print_str("", width, right_justify, true); | 
|  | } else if (params.expand_patterns) { | 
|  | char *tmp_str = slurm_expand_step_stdio_fields(step->std_in, | 
|  | step); | 
|  | _print_str(tmp_str, width, right_justify, true); | 
|  | xfree(tmp_str); | 
|  | } else { | 
|  | _print_str(step->std_in, width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_std_out(job_step_info_t *step, int width, bool right_justify, | 
|  | char *suffix) | 
|  | { | 
|  | if (step == NULL) { | 
|  | _print_str("STDOUT", width, right_justify, true); | 
|  | } else if (!step->std_out) { | 
|  | _print_str("", width, right_justify, true); | 
|  | } else if (params.expand_patterns) { | 
|  | char *tmp_str = slurm_expand_step_stdio_fields(step->std_out, | 
|  | step); | 
|  | _print_str(tmp_str, width, right_justify, true); | 
|  | xfree(tmp_str); | 
|  | } else { | 
|  | _print_str(step->std_out, width, right_justify, true); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  |  | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_user_id(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("UID", width, right, true); | 
|  | else | 
|  | _print_int(step->user_id, width, right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_user_name(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("USER", width, right, true); | 
|  | else { | 
|  | char *uname = uid_to_string_cached((uid_t) step->user_id); | 
|  | _print_str(uname, width, right, true); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_time_limit(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("TIME_LIMIT", width, right, true); | 
|  | else if (step->time_limit == INFINITE) | 
|  | _print_str("UNLIMITED", width, right, true); | 
|  | else | 
|  | _print_secs(step->time_limit * 60, width, right, false); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_time_start(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("START_TIME", width, false, true); | 
|  | else | 
|  | _print_time(step->start_time, 0, width, right); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_time_used(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("TIME", width, right, true); | 
|  | else { | 
|  | long delta_t = step->run_time; | 
|  | _print_secs(delta_t, width, right, false); | 
|  | } | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_name(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("NAME", width, right, true); | 
|  | else | 
|  | _print_str(step->name, width, right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_nodes(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("NODELIST", width, right, false); | 
|  | else | 
|  | _print_nodes(step->nodes, width, right, false); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_num_tasks(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL)	/* Print the Header instead */ | 
|  | _print_str("TASKS", width, right, true); | 
|  | else | 
|  | _print_int(step->num_tasks, width, right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_array_job_id(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("ARRAY_JOB_ID", width, right, true); | 
|  | else if (step->array_job_id != NO_VAL) | 
|  | _print_int(step->array_job_id, width, right, true); | 
|  | else | 
|  | _print_int(step->step_id.job_id, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_array_task_id(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("ARRAY_TASK_ID", width, right, true); | 
|  | else if (step->array_task_id != NO_VAL) | 
|  | _print_int(step->array_task_id, width, right, true); | 
|  | else | 
|  | _print_str("N/A", width, right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  |  | 
|  | } | 
|  |  | 
|  | int _print_step_job_id(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("JOB_ID", width, right, true); | 
|  | else | 
|  | _print_int(step->step_id.job_id, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_network(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("NETWORK", width, right, true); | 
|  | else | 
|  | _print_str(step->network, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_node_inx(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("NODE_INDEX", width, right, true); | 
|  | else { | 
|  | int *current = step->node_inx; | 
|  | int curr_width = 0; | 
|  | while (*current != -1 && curr_width < width) { | 
|  | if (curr_width) | 
|  | printf(","); | 
|  | curr_width += _print_int(*current, width, right, true); | 
|  | current++; | 
|  | } | 
|  | while (curr_width < width) | 
|  | curr_width += printf(" "); | 
|  | } | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_num_cpus(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("NUM_CPUS", width, right, true); | 
|  | else | 
|  | _print_int(step->num_cpus, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_cpu_freq(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | char bfm[16], bfx[16], bfg[16], bfall[48]; | 
|  |  | 
|  | if (step == NULL) { | 
|  | _print_str("CPU_FREQ", width, right, true); | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  | cpu_freq_to_string(bfm, sizeof(bfm), step->cpu_freq_min); | 
|  | cpu_freq_to_string(bfx, sizeof(bfx), step->cpu_freq_max); | 
|  | cpu_freq_to_string(bfg, sizeof(bfg), step->cpu_freq_gov); | 
|  | snprintf(bfall, sizeof(bfall), "%s-%s:%s", bfm, bfx, bfg); | 
|  | _print_str(bfall, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_resv_ports(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("RESERVED_PORTS", width, right, true); | 
|  | else | 
|  | _print_str(step->resv_ports, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_state(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("STATE", width, right, true); | 
|  | else | 
|  | _print_str(job_state_string(step->state), width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_cpus_per_tres(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("CPUS_PER_TRES", width, right, true); | 
|  | else | 
|  | _print_str(step->cpus_per_tres, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_mem_per_tres(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("MEM_PER_TRES", width, right, true); | 
|  | else | 
|  | _print_str(step->mem_per_tres, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_tres_bind(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("TRES_BIND", width, right, true); | 
|  | else | 
|  | _print_str(step->tres_bind, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_tres_freq(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("TRES_FREQ", width, right, true); | 
|  | else | 
|  | _print_str(step->tres_freq, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_tres_per_step(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("TRES_PER_STEP", width, right, true); | 
|  | else | 
|  | _print_str(step->tres_per_step, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_tres_per_node(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("TRES_PER_JOB", width, right, true); | 
|  | else | 
|  | _print_str(step->tres_per_node, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_tres_per_socket(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("TRES_PER_SOCKET", width, right, true); | 
|  | else | 
|  | _print_str(step->tres_per_socket, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_step_tres_per_task(job_step_info_t * step, int width, bool right, | 
|  | char* suffix) | 
|  | { | 
|  | if (step == NULL) | 
|  | _print_str("TRES_PER_TASK", width, right, true); | 
|  | else | 
|  | _print_str(step->tres_per_task, width, right, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | static int _find_any_resv(void *x, void *arg) | 
|  | { | 
|  | return list_find_first(arg, slurm_find_char_exact_in_list, x) ? 1 : 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Filter job records per input specifications. | 
|  | * Returns true if the job should be filtered out (not printed). | 
|  | */ | 
|  | static bool _filter_job(job_info_t *job) | 
|  | { | 
|  | int i; | 
|  | list_itr_t *iterator; | 
|  | uint32_t *user; | 
|  | uint32_t *state_id; | 
|  | char *account, *license, *qos, *name; | 
|  | squeue_job_step_t *job_step_id; | 
|  | bool partial_array = false; | 
|  |  | 
|  | if (job->job_id == 0) | 
|  | return true; | 
|  |  | 
|  | if (params.job_list) { | 
|  | bool filter = true; | 
|  | iterator = list_iterator_create(params.job_list); | 
|  | while ((job_step_id = list_next(iterator))) { | 
|  | if (((job_step_id->array_id == NO_VAL)             && | 
|  | ((job_step_id->step_id.job_id == | 
|  | job->array_job_id) || | 
|  | (job_step_id->step_id.job_id == | 
|  | job->job_id))) || | 
|  | ((job_step_id->array_id == job->array_task_id) && | 
|  | (job_step_id->step_id.job_id == | 
|  | job->array_job_id))) { | 
|  | filter = false; | 
|  | break; | 
|  | } | 
|  | if ((job_step_id->array_id != NO_VAL)             && | 
|  | (job_step_id->step_id.job_id == | 
|  | job->array_job_id) && | 
|  | (job->array_bitmap && | 
|  | bit_test(job->array_bitmap, | 
|  | job_step_id->array_id))) { | 
|  | filter = false; | 
|  | partial_array = true; | 
|  | break; | 
|  | } | 
|  | if (job_step_id->step_id.job_id == job->het_job_id) { | 
|  | filter = false; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (params.licenses_list) { | 
|  | char *token = NULL, *last = NULL, *tmp_name = NULL; | 
|  | char *tmp_token; | 
|  | bool filter = true; | 
|  |  | 
|  | if (job->licenses) { | 
|  | tmp_name = xstrdup(job->licenses); | 
|  | token = strtok_r(tmp_name, ",", &last); | 
|  | } | 
|  | while (token && filter) { | 
|  | /* Consider license name only, ignore ":" lic count */ | 
|  | tmp_token = token; | 
|  | while (*tmp_token) { | 
|  | if (*tmp_token == ':') { | 
|  | *tmp_token = '\0'; | 
|  | break; | 
|  | } | 
|  | tmp_token++; | 
|  | } | 
|  | iterator = list_iterator_create(params.licenses_list); | 
|  | while ((license = list_next(iterator))) { | 
|  | if (xstrcmp(token, license) == 0) { | 
|  | filter = false; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | token = strtok_r(NULL, ",", &last); | 
|  | } | 
|  | xfree(tmp_name); | 
|  | if (filter) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (params.account_list) { | 
|  | bool filter = true; | 
|  | iterator = list_iterator_create(params.account_list); | 
|  | while ((account = list_next(iterator))) { | 
|  | if ((job->account != NULL) && | 
|  | (xstrcasecmp(account, job->account) == 0)) { | 
|  | filter = false; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (params.qos_list) { | 
|  | bool filter = true; | 
|  | iterator = list_iterator_create(params.qos_list); | 
|  | while ((qos = list_next(iterator))) { | 
|  | if ((job->qos != NULL) && | 
|  | (xstrcasecmp(qos, job->qos) == 0)) { | 
|  | filter = false; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (params.all_states) { | 
|  | } else if (params.state_list) { | 
|  | bool filter = true; | 
|  | iterator = list_iterator_create(params.state_list); | 
|  | while ((state_id = list_next(iterator))) { | 
|  | bool match = false; | 
|  | if (*state_id &  JOB_STATE_FLAGS) { | 
|  | if (*state_id &  job->job_state) | 
|  | match = true; | 
|  | } else if (*state_id == job->job_state) | 
|  | match = true; | 
|  | if (match) { | 
|  | filter = false; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter) | 
|  | return true; | 
|  | } else { | 
|  | if (!IS_JOB_PENDING(job) && | 
|  | !IS_JOB_RUNNING(job) && | 
|  | !IS_JOB_STAGE_OUT(job) && | 
|  | !IS_JOB_SUSPENDED(job) && | 
|  | !IS_JOB_COMPLETING(job)) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if ((params.nodes) | 
|  | && ((job->nodes == NULL) | 
|  | || (!hostset_intersects(params.nodes, job->nodes)))) | 
|  | return true; | 
|  |  | 
|  | if (params.notme_flag && (getuid() == job->user_id)) | 
|  | return true; | 
|  |  | 
|  | if (params.user_list) { | 
|  | bool filter = true; | 
|  | iterator = list_iterator_create(params.user_list); | 
|  | while ((user = list_next(iterator))) { | 
|  | if (*user == job->user_id) { | 
|  | filter = false; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (params.reservation) { | 
|  | bool filter = false; | 
|  | list_t *spec_resv_list = NULL, *job_resv_list = NULL; | 
|  |  | 
|  | if (!job->resv_name) | 
|  | return true; | 
|  |  | 
|  | spec_resv_list = list_create(xfree_ptr); | 
|  | slurm_addto_char_list_with_case(spec_resv_list, | 
|  | params.reservation, false); | 
|  |  | 
|  | job_resv_list = list_create(xfree_ptr); | 
|  | slurm_addto_char_list_with_case(job_resv_list, job->resv_name, | 
|  | false); | 
|  |  | 
|  | if (!list_find_first(spec_resv_list, _find_any_resv, | 
|  | job_resv_list)) | 
|  | filter = true; | 
|  |  | 
|  | FREE_NULL_LIST(spec_resv_list); | 
|  | FREE_NULL_LIST(job_resv_list); | 
|  |  | 
|  | if (filter) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (params.name_list) { | 
|  | bool filter = true; | 
|  | iterator = list_iterator_create(params.name_list); | 
|  | while ((name = list_next(iterator))) { | 
|  | if ((job->name != NULL) && | 
|  | (xstrcasecmp(name, job->name) == 0)) { | 
|  | filter = false; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (partial_array) { | 
|  | /* Print this record, but perhaps only some job array records */ | 
|  | bitstr_t *new_array_bitmap; | 
|  | int array_len = bit_size(job->array_bitmap); | 
|  | new_array_bitmap = bit_alloc(array_len); | 
|  | iterator = list_iterator_create(params.job_list); | 
|  | while ((job_step_id = list_next(iterator))) { | 
|  | if ((job_step_id->step_id.job_id == | 
|  | job->array_job_id) && | 
|  | (job_step_id->array_id < array_len)) { | 
|  | bit_set(new_array_bitmap,job_step_id->array_id); | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | bit_and(job->array_bitmap, new_array_bitmap); | 
|  | FREE_NULL_BITMAP(new_array_bitmap); | 
|  | xfree(job->array_task_str); | 
|  | i = bit_set_count(job->array_bitmap); | 
|  | if (i == 1) { | 
|  | job->array_task_id = bit_ffs(job->array_bitmap); | 
|  | FREE_NULL_BITMAP(job->array_bitmap); | 
|  | } else { | 
|  | i = i * 16 + 10; | 
|  | job->array_task_str = xmalloc(i); | 
|  | (void) bit_fmt(job->array_task_str, i, | 
|  | job->array_bitmap); | 
|  | } | 
|  | } | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /* Return 0 if supplied partition name is to be printed, otherwise return 2 */ | 
|  | static int _filter_job_part(char *part_name) | 
|  | { | 
|  | char *token = NULL, *last = NULL, *tmp_name = NULL, *part; | 
|  | list_itr_t *iterator; | 
|  | int rc = 2; | 
|  |  | 
|  | if (!params.part_list) | 
|  | return 0; | 
|  |  | 
|  | if (part_name) { | 
|  | tmp_name = xstrdup(part_name); | 
|  | token = strtok_r(tmp_name, ",", &last); | 
|  | } | 
|  | while (token && (rc != 0)) { | 
|  | iterator = list_iterator_create(params.part_list); | 
|  | while ((part = list_next(iterator))) { | 
|  | if (xstrcmp(part, token) == 0) { | 
|  | rc = 0; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | token = strtok_r(NULL, ",", &last); | 
|  | } | 
|  | xfree(tmp_name); | 
|  |  | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | /* filter step records per input specifications, | 
|  | * returns 1 if step should be filter out (not printed) */ | 
|  | static int _filter_step(job_step_info_t * step) | 
|  | { | 
|  | int filter; | 
|  | list_itr_t *iterator; | 
|  | uint32_t *user; | 
|  | char *part; | 
|  | squeue_job_step_t *job_step_id; | 
|  |  | 
|  | if (step->state == JOB_PENDING) | 
|  | return 1; | 
|  |  | 
|  | if (params.job_list) { | 
|  | filter = 1; | 
|  | iterator = list_iterator_create(params.job_list); | 
|  | while ((job_step_id = list_next(iterator))) { | 
|  | if (((job_step_id->array_id == NO_VAL)   && | 
|  | ((job_step_id->step_id.job_id == | 
|  | step->array_job_id) || | 
|  | (job_step_id->step_id.job_id == | 
|  | step->step_id.job_id))) || | 
|  | ((job_step_id->array_id == step->array_task_id) && | 
|  | (job_step_id->step_id.job_id == | 
|  | step->array_job_id))) { | 
|  | filter = 0; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter == 1) | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | if (params.part_list) { | 
|  | filter = 1; | 
|  | iterator = list_iterator_create(params.part_list); | 
|  | while ((part = list_next(iterator))) { | 
|  | if (xstrcmp(part, step->partition) == 0) { | 
|  | filter = 0; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter == 1) | 
|  | return 2; | 
|  | } | 
|  |  | 
|  | if (params.step_list) { | 
|  | filter = 1; | 
|  | iterator = list_iterator_create(params.step_list); | 
|  | while ((job_step_id = list_next(iterator))) { | 
|  | if (job_step_id->step_id.step_id != | 
|  | step->step_id.step_id) | 
|  | continue; | 
|  | if (((job_step_id->array_id == NO_VAL) && | 
|  | ((job_step_id->step_id.job_id == | 
|  | step->array_job_id) || | 
|  | (job_step_id->step_id.job_id == | 
|  | step->step_id.job_id))) || | 
|  | ((job_step_id->array_id == step->array_task_id) && | 
|  | (job_step_id->step_id.job_id == | 
|  | step->array_job_id))) { | 
|  | filter = 0; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter == 1) | 
|  | return 3; | 
|  | } | 
|  |  | 
|  | if ((params.nodes) | 
|  | && ((step->nodes == NULL) | 
|  | || (!hostset_intersects(params.nodes, step->nodes)))) | 
|  | return 5; | 
|  |  | 
|  | if (params.user_list) { | 
|  | filter = 1; | 
|  | iterator = list_iterator_create(params.user_list); | 
|  | while ((user = list_next(iterator))) { | 
|  | if (*user == step->user_id) { | 
|  | filter = 0; | 
|  | break; | 
|  | } | 
|  | } | 
|  | list_iterator_destroy(iterator); | 
|  | if (filter == 1) | 
|  | return 6; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int _print_com_invalid(void * p, int width, bool right, char* suffix) | 
|  | { | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } | 
|  |  | 
|  | int _print_job_cron_flag(job_info_t *job, int width, bool right_justify, | 
|  | char *suffix) | 
|  | { | 
|  | if (!job) | 
|  | _print_str("CRON_JOB", width, right_justify, true); | 
|  | else if (job->bitflags & CRON_JOB) | 
|  | _print_str("Yes", width, right_justify, true); | 
|  | else | 
|  | _print_str("No", width, right_justify, true); | 
|  |  | 
|  | if (suffix) | 
|  | printf("%s", suffix); | 
|  | return SLURM_SUCCESS; | 
|  | } |