2005-03-14 20:57:21 +00:00
/* -*- C -*-
*
2006-02-16 20:40:23 +00:00
* Copyright ( c ) 2004 - 2006 The Trustees of Indiana University and Indiana
2005-11-05 19:57:48 +00:00
* University Research and Technology
* Corporation . All rights reserved .
* Copyright ( c ) 2004 - 2005 The University of Tennessee and The University
* of Tennessee Research Foundation . All rights
* reserved .
2005-09-20 17:09:11 +00:00
* Copyright ( c ) 2004 - 2005 High Performance Computing Center Stuttgart ,
2005-03-14 20:57:21 +00:00
* University of Stuttgart . All rights reserved .
2005-03-24 12:43:37 +00:00
* Copyright ( c ) 2004 - 2005 The Regents of the University of California .
* All rights reserved .
2006-03-29 00:53:11 +00:00
* Copyright ( c ) 2006 Cisco Systems , Inc . All rights reserved .
2005-03-14 20:57:21 +00:00
* $ COPYRIGHT $
2005-09-20 17:09:11 +00:00
*
2005-03-14 20:57:21 +00:00
* Additional copyrights may follow
2005-09-20 17:09:11 +00:00
*
2005-03-14 20:57:21 +00:00
* $ HEADER $
*/
# include "orte_config.h"
# include <stdio.h>
# ifdef HAVE_UNISTD_H
# include <unistd.h>
# endif
# ifdef HAVE_SYS_PARAM_H
# include <sys/param.h>
# endif
# include <errno.h>
# include <signal.h>
# include <ctype.h>
2005-12-17 22:05:10 +00:00
# ifdef HAVE_SYS_TYPES_H
2005-04-01 00:30:37 +00:00
# include <sys/types.h>
2005-12-17 22:05:10 +00:00
# endif /* HAVE_SYS_TYPES_H */
# ifdef HAVE_SYS_WAIT_H
2005-04-01 00:30:37 +00:00
# include <sys/wait.h>
2005-12-17 22:05:10 +00:00
# endif /* HAVE_SYS_WAIT_H */
2006-03-23 16:53:11 +00:00
# ifdef HAVE_LIBGEN_H
2006-02-28 11:52:12 +00:00
# include <libgen.h>
# endif
2005-03-14 20:57:21 +00:00
2005-07-03 23:09:55 +00:00
# include "opal/event/event.h"
2005-09-19 17:20:01 +00:00
# include "opal/mca/base/base.h"
# include "opal/threads/condition.h"
2005-07-04 00:13:44 +00:00
# include "opal/util/argv.h"
2005-09-19 17:20:01 +00:00
# include "opal/util/basename.h"
2005-07-04 00:13:44 +00:00
# include "opal/util/cmd_line.h"
2005-09-19 17:20:01 +00:00
# include "opal/util/opal_environ.h"
2005-07-03 23:31:27 +00:00
# include "opal/util/output.h"
2005-07-04 02:38:44 +00:00
# include "opal/util/show_help.h"
2005-09-19 17:20:01 +00:00
# include "opal/util/trace.h"
2006-06-09 17:21:23 +00:00
# include "opal/version.h"
2005-09-19 17:20:01 +00:00
2006-02-12 01:33:29 +00:00
# include "orte/orte_constants.h"
2005-09-19 17:20:01 +00:00
# include "orte/class/orte_pointer_array.h"
# include "orte/util/proc_info.h"
# include "orte/util/sys_info.h"
# include "orte/util/universe_setup_file_io.h"
2005-03-14 20:57:21 +00:00
2005-09-19 17:20:01 +00:00
# include "orte/mca/ns/ns.h"
# include "orte/mca/gpr/gpr.h"
# include "orte/mca/rmgr/rmgr.h"
# include "orte/mca/schema/schema.h"
# include "orte/mca/errmgr/errmgr.h"
2005-03-14 20:57:21 +00:00
2005-09-19 17:20:01 +00:00
# include "orte/runtime/runtime.h"
# include "orte/runtime/orte_wait.h"
2005-03-14 20:57:21 +00:00
2005-08-31 16:15:59 +00:00
# include "orterun.h"
2005-08-31 00:47:52 +00:00
# include "totalview.h"
2005-03-14 20:57:21 +00:00
/*
* Globals
*/
2005-07-03 23:09:55 +00:00
static struct opal_event term_handler ;
static struct opal_event int_handler ;
2006-07-11 05:24:08 +00:00
# ifndef __WINDOWS__
2006-06-08 18:27:17 +00:00
static struct opal_event sigusr1_handler ;
static struct opal_event sigusr2_handler ;
2006-07-11 05:24:08 +00:00
# endif /* __WINDOWS__ */
2005-03-14 20:57:21 +00:00
static orte_jobid_t jobid = ORTE_JOBID_MAX ;
2005-07-03 04:02:01 +00:00
static orte_pointer_array_t * apps_pa ;
2005-03-14 20:57:21 +00:00
static bool wait_for_job_completion = true ;
2005-04-12 16:01:30 +00:00
static char * orterun_basename = NULL ;
2005-04-15 21:52:58 +00:00
static int max_display_aborted = 1 ;
static int num_aborted = 0 ;
static int num_killed = 0 ;
2005-08-08 16:42:28 +00:00
static char * * global_mca_env = NULL ;
2006-07-10 21:25:33 +00:00
static bool have_zero_np = false ;
2006-08-15 19:54:10 +00:00
static orte_std_cntr_t total_num_apps = 0 ;
2005-03-14 20:57:21 +00:00
/*
* setup globals for catching orterun command line options
*/
struct globals_t {
bool help ;
2006-06-09 17:21:23 +00:00
bool version ;
2005-03-14 20:57:21 +00:00
bool verbose ;
2006-06-26 18:21:45 +00:00
bool quiet ;
2005-03-14 20:57:21 +00:00
bool exit ;
bool no_wait_for_job_completion ;
(copied from a mail that has a lengthy description of this commit)
I spoke with Tim about this the other day -- he gave me the green
light to go ahead with this, but it turned into a bigger job than I
thought it would be. I revamped how the default RAS scheduling and
round_robin RMAPS mapping occurs. The previous algorithms were pretty
brain dead, and ignored the "slots" and "max_slots" tokens in
hostfiles. I considered this a big enough problem to fix it for the
beta (because there is currently no way to control where processes are
launched on SMPs).
There's still some more bells and whistles that I'd like to implement,
but there's no hurry, and they can go on the trunk at any time. My
patches below are for what I considered "essential", and do the
following:
- honor the "slots" and "max-slots" tokens in the hostfile (and all
their synonyms), meaning that we allocate/map until we fill slots,
and if there are still more processes to allocate/map, we keep going
until we fill max-slots (i.e., only oversubscribe a node if we have
to).
- offer two different algorithms, currently supported by two new
options to orterun. Remember that there are two parts here -- slot
allocation and process mapping. Slot allocation controls how many
processes we'll be running on a node. After that decision has been
made, process mapping effectively controls where the ranks of
MPI_COMM_WORLD (MCW) are placed. Some of the examples given below
don't make sense unless you remember that there is a difference
between the two (which makes total sense, but you have to think
about it in terms of both things):
1. "-bynode": allocates/maps one process per node in a round-robin
fashion until all slots on the node are taken. If we still have more
processes after all slots are taken, then keep going until all
max-slots are taken. Examples:
- The hostfile:
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -bynode -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 2
vogon: MCW ranks 1, 3, 4, 5
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4
vogon: MCW ranks 1, 3, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until each
node's max_slots is hit, of course)
- orterun -bynode -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4, 6
vogon: MCW ranks 1, 3, 5, 7, 8, 9, 10, 11
2. "-byslot" (this is the default if you don't specify -bynode):
greedily takes all available slots on a node for a job before moving
on to the next node. If we still have processes to allocate/schedule,
then oversubscribe all nodes equally (i.e., go round robin on all
nodes until each node's max_slots is hit). Examples:
- The hostfile
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 1
vogon: MCW ranks 2, 3, 4, 5
- orterun -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2
vogon: MCW ranks 3, 4, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until max_slots
is hit)
- orterun -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2, 3
vogon: MCW ranks 4, 5, 6, 7, 8, 9, 10, 11
The above examples are fairly contrived, and it's not clear from them
that you can get different allocation answers in all cases (the
mapping differences are obvious). Consider the following allocation
example:
- The hostfile
eddie count=4
vogon count=4
earth count=4
deep-thought count=4
- orterun -np 8 -hostfile hostfile a.out
eddie: 4 slots will be allocated
vogon: 4 slots will be allocated
earth: no slots allocated
deep-thought: no slots allocated
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: 2 slots will be allocated
vogon: 2 slots will be allocated
earth: 2 slots will be allocated
deep-thought: 2 slots will be allocated
This commit was SVN r5894.
2005-05-31 16:36:53 +00:00
bool by_node ;
bool by_slot ;
2006-07-10 21:25:33 +00:00
bool no_oversubscribe ;
2005-11-20 16:06:53 +00:00
bool debugger ;
2006-07-04 20:12:35 +00:00
bool no_local_schedule ;
2006-08-15 19:54:10 +00:00
orte_std_cntr_t num_procs ;
2005-04-29 00:36:07 +00:00
int exit_status ;
2005-03-14 20:57:21 +00:00
char * hostfile ;
char * env_val ;
char * appfile ;
char * wdir ;
char * path ;
2005-07-03 22:45:48 +00:00
opal_mutex_t lock ;
opal_condition_t cond ;
2005-03-14 20:57:21 +00:00
} orterun_globals ;
2005-03-18 23:58:36 +00:00
static bool globals_init = false ;
2005-03-14 20:57:21 +00:00
2005-04-15 21:52:58 +00:00
struct proc_info_t {
bool reported ;
2005-04-29 00:36:07 +00:00
int32_t exit_status ;
2005-04-15 21:52:58 +00:00
} ;
struct proc_info_t * proc_infos = NULL ;
2005-03-14 20:57:21 +00:00
2005-07-04 00:13:44 +00:00
opal_cmd_line_init_t cmd_line_init [ ] = {
2005-03-14 20:57:21 +00:00
/* Various "obvious" options */
2005-09-04 20:54:19 +00:00
{ NULL , NULL , NULL , ' h ' , NULL , " help " , 0 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . help , OPAL_CMD_LINE_TYPE_BOOL ,
2005-03-14 20:57:21 +00:00
" This help message " } ,
2006-06-09 17:21:23 +00:00
{ NULL , NULL , NULL , ' V ' , NULL , " version " , 0 ,
& orterun_globals . version , OPAL_CMD_LINE_TYPE_BOOL ,
" Print version and exit " } ,
2005-03-14 20:57:21 +00:00
{ NULL , NULL , NULL , ' v ' , NULL , " verbose " , 0 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . verbose , OPAL_CMD_LINE_TYPE_BOOL ,
2005-03-14 20:57:21 +00:00
" Be verbose " } ,
2006-06-26 18:21:45 +00:00
{ NULL , NULL , NULL , ' q ' , NULL , " quiet " , 0 ,
& orterun_globals . quiet , OPAL_CMD_LINE_TYPE_BOOL ,
" Suppress helpful messages " } ,
2005-03-14 20:57:21 +00:00
/* Use an appfile */
{ NULL , NULL , NULL , ' \0 ' , NULL , " app " , 1 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . appfile , OPAL_CMD_LINE_TYPE_STRING ,
2005-03-14 20:57:21 +00:00
" Provide an appfile; ignore all other command line options " } ,
/* Number of processes; -c, -n, --n, -np, and --np are all
synonyms */
{ NULL , NULL , NULL , ' c ' , " np " , " np " , 1 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . num_procs , OPAL_CMD_LINE_TYPE_SIZE_T ,
2005-03-14 20:57:21 +00:00
" Number of processes to run " } ,
{ NULL , NULL , NULL , ' \0 ' , " n " , " n " , 1 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . num_procs , OPAL_CMD_LINE_TYPE_SIZE_T ,
2005-03-14 20:57:21 +00:00
" Number of processes to run " } ,
2006-07-10 21:25:33 +00:00
2005-03-14 20:57:21 +00:00
/* Set a hostfile */
2005-03-18 23:40:08 +00:00
{ " rds " , " hostfile " , " path " , ' \0 ' , " hostfile " , " hostfile " , 1 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_STRING ,
2005-03-18 23:40:08 +00:00
" Provide a hostfile " } ,
{ " rds " , " hostfile " , " path " , ' \0 ' , " machinefile " , " machinefile " , 1 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_STRING ,
2005-03-14 20:57:21 +00:00
" Provide a hostfile " } ,
/* Don't wait for the process to finish before exiting */
{ NULL , NULL , NULL , ' \0 ' , " nw " , " nw " , 0 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . no_wait_for_job_completion , OPAL_CMD_LINE_TYPE_BOOL ,
2005-03-14 20:57:21 +00:00
" Launch the processes and do not wait for their completion (i.e., let orterun complete as soon a successful launch occurs) " } ,
2005-04-15 21:52:58 +00:00
/* Set the max number of aborted processes to show */
{ NULL , NULL , NULL , ' \0 ' , " aborted " , " aborted " , 1 ,
2005-07-04 00:13:44 +00:00
& max_display_aborted , OPAL_CMD_LINE_TYPE_INT ,
2005-04-15 21:52:58 +00:00
" The maximum number of aborted processes to display " } ,
2005-03-14 20:57:21 +00:00
/* Export environment variables; potentially used multiple times,
so it does not make sense to set into a variable */
{ NULL , NULL , NULL , ' x ' , NULL , NULL , 1 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_NULL ,
2005-03-14 20:57:21 +00:00
" Export an environment variable, optionally specifying a value (e.g., \" -x foo \" exports the environment variable foo and takes its value from the current environment; \" -x foo=bar \" exports the environment variable name foo and sets its value to \" bar \" in the started processes) " } ,
/* Specific mapping (C, cX, N, nX) */
2005-10-01 15:51:20 +00:00
#if 0
/* JJH --map is not currently implemented so don't advertise it until it is */
2005-03-14 20:57:21 +00:00
{ NULL , NULL , NULL , ' \0 ' , NULL , " map " , 1 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_STRING ,
2005-03-14 20:57:21 +00:00
" Mapping of processes to nodes / CPUs " } ,
2005-10-01 15:51:20 +00:00
# endif
(copied from a mail that has a lengthy description of this commit)
I spoke with Tim about this the other day -- he gave me the green
light to go ahead with this, but it turned into a bigger job than I
thought it would be. I revamped how the default RAS scheduling and
round_robin RMAPS mapping occurs. The previous algorithms were pretty
brain dead, and ignored the "slots" and "max_slots" tokens in
hostfiles. I considered this a big enough problem to fix it for the
beta (because there is currently no way to control where processes are
launched on SMPs).
There's still some more bells and whistles that I'd like to implement,
but there's no hurry, and they can go on the trunk at any time. My
patches below are for what I considered "essential", and do the
following:
- honor the "slots" and "max-slots" tokens in the hostfile (and all
their synonyms), meaning that we allocate/map until we fill slots,
and if there are still more processes to allocate/map, we keep going
until we fill max-slots (i.e., only oversubscribe a node if we have
to).
- offer two different algorithms, currently supported by two new
options to orterun. Remember that there are two parts here -- slot
allocation and process mapping. Slot allocation controls how many
processes we'll be running on a node. After that decision has been
made, process mapping effectively controls where the ranks of
MPI_COMM_WORLD (MCW) are placed. Some of the examples given below
don't make sense unless you remember that there is a difference
between the two (which makes total sense, but you have to think
about it in terms of both things):
1. "-bynode": allocates/maps one process per node in a round-robin
fashion until all slots on the node are taken. If we still have more
processes after all slots are taken, then keep going until all
max-slots are taken. Examples:
- The hostfile:
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -bynode -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 2
vogon: MCW ranks 1, 3, 4, 5
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4
vogon: MCW ranks 1, 3, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until each
node's max_slots is hit, of course)
- orterun -bynode -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4, 6
vogon: MCW ranks 1, 3, 5, 7, 8, 9, 10, 11
2. "-byslot" (this is the default if you don't specify -bynode):
greedily takes all available slots on a node for a job before moving
on to the next node. If we still have processes to allocate/schedule,
then oversubscribe all nodes equally (i.e., go round robin on all
nodes until each node's max_slots is hit). Examples:
- The hostfile
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 1
vogon: MCW ranks 2, 3, 4, 5
- orterun -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2
vogon: MCW ranks 3, 4, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until max_slots
is hit)
- orterun -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2, 3
vogon: MCW ranks 4, 5, 6, 7, 8, 9, 10, 11
The above examples are fairly contrived, and it's not clear from them
that you can get different allocation answers in all cases (the
mapping differences are obvious). Consider the following allocation
example:
- The hostfile
eddie count=4
vogon count=4
earth count=4
deep-thought count=4
- orterun -np 8 -hostfile hostfile a.out
eddie: 4 slots will be allocated
vogon: 4 slots will be allocated
earth: no slots allocated
deep-thought: no slots allocated
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: 2 slots will be allocated
vogon: 2 slots will be allocated
earth: 2 slots will be allocated
deep-thought: 2 slots will be allocated
This commit was SVN r5894.
2005-05-31 16:36:53 +00:00
{ NULL , NULL , NULL , ' \0 ' , " bynode " , " bynode " , 0 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . by_node , OPAL_CMD_LINE_TYPE_BOOL ,
(copied from a mail that has a lengthy description of this commit)
I spoke with Tim about this the other day -- he gave me the green
light to go ahead with this, but it turned into a bigger job than I
thought it would be. I revamped how the default RAS scheduling and
round_robin RMAPS mapping occurs. The previous algorithms were pretty
brain dead, and ignored the "slots" and "max_slots" tokens in
hostfiles. I considered this a big enough problem to fix it for the
beta (because there is currently no way to control where processes are
launched on SMPs).
There's still some more bells and whistles that I'd like to implement,
but there's no hurry, and they can go on the trunk at any time. My
patches below are for what I considered "essential", and do the
following:
- honor the "slots" and "max-slots" tokens in the hostfile (and all
their synonyms), meaning that we allocate/map until we fill slots,
and if there are still more processes to allocate/map, we keep going
until we fill max-slots (i.e., only oversubscribe a node if we have
to).
- offer two different algorithms, currently supported by two new
options to orterun. Remember that there are two parts here -- slot
allocation and process mapping. Slot allocation controls how many
processes we'll be running on a node. After that decision has been
made, process mapping effectively controls where the ranks of
MPI_COMM_WORLD (MCW) are placed. Some of the examples given below
don't make sense unless you remember that there is a difference
between the two (which makes total sense, but you have to think
about it in terms of both things):
1. "-bynode": allocates/maps one process per node in a round-robin
fashion until all slots on the node are taken. If we still have more
processes after all slots are taken, then keep going until all
max-slots are taken. Examples:
- The hostfile:
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -bynode -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 2
vogon: MCW ranks 1, 3, 4, 5
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4
vogon: MCW ranks 1, 3, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until each
node's max_slots is hit, of course)
- orterun -bynode -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4, 6
vogon: MCW ranks 1, 3, 5, 7, 8, 9, 10, 11
2. "-byslot" (this is the default if you don't specify -bynode):
greedily takes all available slots on a node for a job before moving
on to the next node. If we still have processes to allocate/schedule,
then oversubscribe all nodes equally (i.e., go round robin on all
nodes until each node's max_slots is hit). Examples:
- The hostfile
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 1
vogon: MCW ranks 2, 3, 4, 5
- orterun -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2
vogon: MCW ranks 3, 4, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until max_slots
is hit)
- orterun -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2, 3
vogon: MCW ranks 4, 5, 6, 7, 8, 9, 10, 11
The above examples are fairly contrived, and it's not clear from them
that you can get different allocation answers in all cases (the
mapping differences are obvious). Consider the following allocation
example:
- The hostfile
eddie count=4
vogon count=4
earth count=4
deep-thought count=4
- orterun -np 8 -hostfile hostfile a.out
eddie: 4 slots will be allocated
vogon: 4 slots will be allocated
earth: no slots allocated
deep-thought: no slots allocated
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: 2 slots will be allocated
vogon: 2 slots will be allocated
earth: 2 slots will be allocated
deep-thought: 2 slots will be allocated
This commit was SVN r5894.
2005-05-31 16:36:53 +00:00
" Whether to allocate/map processes round-robin by node " } ,
{ NULL , NULL , NULL , ' \0 ' , " byslot " , " byslot " , 0 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . by_slot , OPAL_CMD_LINE_TYPE_BOOL ,
(copied from a mail that has a lengthy description of this commit)
I spoke with Tim about this the other day -- he gave me the green
light to go ahead with this, but it turned into a bigger job than I
thought it would be. I revamped how the default RAS scheduling and
round_robin RMAPS mapping occurs. The previous algorithms were pretty
brain dead, and ignored the "slots" and "max_slots" tokens in
hostfiles. I considered this a big enough problem to fix it for the
beta (because there is currently no way to control where processes are
launched on SMPs).
There's still some more bells and whistles that I'd like to implement,
but there's no hurry, and they can go on the trunk at any time. My
patches below are for what I considered "essential", and do the
following:
- honor the "slots" and "max-slots" tokens in the hostfile (and all
their synonyms), meaning that we allocate/map until we fill slots,
and if there are still more processes to allocate/map, we keep going
until we fill max-slots (i.e., only oversubscribe a node if we have
to).
- offer two different algorithms, currently supported by two new
options to orterun. Remember that there are two parts here -- slot
allocation and process mapping. Slot allocation controls how many
processes we'll be running on a node. After that decision has been
made, process mapping effectively controls where the ranks of
MPI_COMM_WORLD (MCW) are placed. Some of the examples given below
don't make sense unless you remember that there is a difference
between the two (which makes total sense, but you have to think
about it in terms of both things):
1. "-bynode": allocates/maps one process per node in a round-robin
fashion until all slots on the node are taken. If we still have more
processes after all slots are taken, then keep going until all
max-slots are taken. Examples:
- The hostfile:
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -bynode -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 2
vogon: MCW ranks 1, 3, 4, 5
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4
vogon: MCW ranks 1, 3, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until each
node's max_slots is hit, of course)
- orterun -bynode -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4, 6
vogon: MCW ranks 1, 3, 5, 7, 8, 9, 10, 11
2. "-byslot" (this is the default if you don't specify -bynode):
greedily takes all available slots on a node for a job before moving
on to the next node. If we still have processes to allocate/schedule,
then oversubscribe all nodes equally (i.e., go round robin on all
nodes until each node's max_slots is hit). Examples:
- The hostfile
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 1
vogon: MCW ranks 2, 3, 4, 5
- orterun -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2
vogon: MCW ranks 3, 4, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until max_slots
is hit)
- orterun -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2, 3
vogon: MCW ranks 4, 5, 6, 7, 8, 9, 10, 11
The above examples are fairly contrived, and it's not clear from them
that you can get different allocation answers in all cases (the
mapping differences are obvious). Consider the following allocation
example:
- The hostfile
eddie count=4
vogon count=4
earth count=4
deep-thought count=4
- orterun -np 8 -hostfile hostfile a.out
eddie: 4 slots will be allocated
vogon: 4 slots will be allocated
earth: no slots allocated
deep-thought: no slots allocated
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: 2 slots will be allocated
vogon: 2 slots will be allocated
earth: 2 slots will be allocated
deep-thought: 2 slots will be allocated
This commit was SVN r5894.
2005-05-31 16:36:53 +00:00
" Whether to allocate/map processes round-robin by slot (the default) " } ,
2006-07-14 14:15:03 +00:00
{ NULL , NULL , NULL , ' \0 ' , " nooversubscribe " , " nooversubscribe " , 0 ,
2006-07-10 21:25:33 +00:00
& orterun_globals . no_oversubscribe , OPAL_CMD_LINE_TYPE_BOOL ,
" Nodes are not to be oversubscribed, even if the system supports such operation " } ,
2005-03-14 20:57:21 +00:00
/* mpiexec-like arguments */
{ NULL , NULL , NULL , ' \0 ' , " wdir " , " wdir " , 1 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . wdir , OPAL_CMD_LINE_TYPE_STRING ,
2005-03-14 20:57:21 +00:00
" Set the working directory of the started processes " } ,
{ NULL , NULL , NULL , ' \0 ' , " path " , " path " , 1 ,
2005-07-04 00:13:44 +00:00
& orterun_globals . path , OPAL_CMD_LINE_TYPE_STRING ,
2005-03-14 20:57:21 +00:00
" PATH to be used to look for executables to start processes " } ,
/* These arguments can be specified multiple times */
2005-09-20 08:56:02 +00:00
#if 0
/* JMS: Removed because it's not really implemented */
2005-03-14 20:57:21 +00:00
{ NULL , NULL , NULL , ' \0 ' , " arch " , " arch " , 1 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_STRING ,
2005-03-14 20:57:21 +00:00
" Architecture to start processes on " } ,
2005-09-20 08:56:02 +00:00
# endif
2005-03-14 20:57:21 +00:00
{ NULL , NULL , NULL , ' H ' , " host " , " host " , 1 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_STRING ,
2005-03-14 20:57:21 +00:00
" List of hosts to invoke processes on " } ,
2005-11-20 16:06:53 +00:00
2006-07-04 20:12:35 +00:00
/* OSC mpiexec-like arguments */
{ NULL , NULL , NULL , ' \0 ' , " nolocal " , " nolocal " , 0 ,
& orterun_globals . no_local_schedule , OPAL_CMD_LINE_TYPE_BOOL ,
" Do not run any MPI applications on the local node " } ,
2005-11-20 16:06:53 +00:00
/* User-level debugger arguments */
{ NULL , NULL , NULL , ' \0 ' , " tv " , " tv " , 0 ,
& orterun_globals . debugger , OPAL_CMD_LINE_TYPE_BOOL ,
" Deprecated backwards compatibility flag; synonym for \" --debug \" " } ,
{ NULL , NULL , NULL , ' \0 ' , " debug " , " debug " , 0 ,
& orterun_globals . debugger , OPAL_CMD_LINE_TYPE_BOOL ,
" Invoke the user-level debugger indicated by the orte_base_user_debugger MCA parameter " } ,
{ " orte " , " base " , " user_debugger " , ' \0 ' , " debugger " , " debugger " , 1 ,
NULL , OPAL_CMD_LINE_TYPE_STRING ,
" Sequence of debuggers to search for when \" --debug \" is used " } ,
2005-05-12 21:44:23 +00:00
/* OpenRTE arguments */
2005-11-20 16:06:53 +00:00
{ " orte " , " debug " , NULL , ' d ' , NULL , " debug-devel " , 0 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_BOOL ,
2005-05-12 21:44:23 +00:00
" Enable debugging of OpenRTE " } ,
{ " orte " , " debug " , " daemons " , ' \0 ' , NULL , " debug-daemons " , 0 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_INT ,
2005-05-12 21:44:23 +00:00
" Enable debugging of any OpenRTE daemons used by this application " } ,
{ " orte " , " debug " , " daemons_file " , ' \0 ' , NULL , " debug-daemons-file " , 0 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_BOOL ,
2005-05-12 21:44:23 +00:00
" Enable debugging of any OpenRTE daemons used by this application, storing output in files " } ,
2005-05-24 15:02:50 +00:00
{ " orte " , " no_daemonize " , NULL , ' \0 ' , NULL , " no-daemonize " , 0 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_BOOL ,
2005-05-24 15:02:50 +00:00
" Do not detach OpenRTE daemons used by this application " } ,
2005-05-12 21:44:23 +00:00
{ " universe " , NULL , NULL , ' \0 ' , NULL , " universe " , 1 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_STRING ,
2005-05-12 21:44:23 +00:00
" Set the universe name as username@hostname:universe_name for this application " } ,
{ NULL , NULL , NULL , ' \0 ' , NULL , " tmpdir " , 1 ,
2005-07-04 00:13:44 +00:00
& orte_process_info . tmpdir_base , OPAL_CMD_LINE_TYPE_STRING ,
2005-05-12 21:44:23 +00:00
" Set the root for the session directory tree for orterun ONLY " } ,
2006-02-28 11:52:12 +00:00
{ NULL , NULL , NULL , ' \0 ' , NULL , " prefix " , 1 ,
NULL , OPAL_CMD_LINE_TYPE_STRING ,
" Prefix where Open MPI is installed on remote nodes " } ,
2006-03-23 16:53:11 +00:00
2005-03-14 20:57:21 +00:00
/* End of list */
{ NULL , NULL , NULL , ' \0 ' , NULL , NULL , 0 ,
2005-07-04 00:13:44 +00:00
NULL , OPAL_CMD_LINE_TYPE_NULL , NULL }
2005-03-14 20:57:21 +00:00
} ;
2006-08-23 02:35:00 +00:00
# if !defined(__WINDOWS__)
extern char * * environ ;
# endif /* !defined(__WINDOWS__) */
2005-03-14 20:57:21 +00:00
/*
* Local functions
*/
static void exit_callback ( int fd , short event , void * arg ) ;
2006-06-26 15:12:52 +00:00
static void abort_signal_callback ( int fd , short event , void * arg ) ;
static void signal_forward_callback ( int fd , short event , void * arg ) ;
2005-03-14 20:57:21 +00:00
static int create_app ( int argc , char * argv [ ] , orte_app_context_t * * app ,
2005-08-08 16:42:28 +00:00
bool * made_app , char * * * app_env ) ;
2005-03-14 20:57:21 +00:00
static int init_globals ( void ) ;
static int parse_globals ( int argc , char * argv [ ] ) ;
static int parse_locals ( int argc , char * argv [ ] ) ;
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
static int parse_appfile ( char * filename , char * * * env ) ;
2005-03-14 20:57:21 +00:00
static void job_state_callback ( orte_jobid_t jobid , orte_proc_state_t state ) ;
2005-08-31 16:15:59 +00:00
int orterun ( int argc , char * argv [ ] )
2005-03-14 20:57:21 +00:00
{
orte_app_context_t * * apps ;
2005-07-08 18:48:25 +00:00
int rc , i , num_apps , array_size , j ;
2005-05-24 15:02:50 +00:00
int id , iparam ;
2006-02-08 17:40:11 +00:00
orte_proc_state_t cb_states ;
2005-03-14 20:57:21 +00:00
2005-11-20 16:06:53 +00:00
/* Setup MCA params */
mca_base_param_init ( ) ;
orte_register_params ( false ) ;
2006-06-26 15:12:52 +00:00
/* find our basename (the name of the executable) so that we can
use it in pretty - print error messages */
2005-07-04 00:13:44 +00:00
orterun_basename = opal_basename ( argv [ 0 ] ) ;
2006-06-08 18:27:17 +00:00
2005-03-14 20:57:21 +00:00
/* Check for some "global" command line params */
parse_globals ( argc , argv ) ;
/* If we're still here, parse each app */
parse_locals ( argc , argv ) ;
/* Convert the list of apps to an array of orte_app_context_t
pointers */
2005-07-08 18:48:25 +00:00
array_size = orte_pointer_array_get_size ( apps_pa ) ;
2006-08-23 02:35:00 +00:00
apps = ( orte_app_context_t * * ) malloc ( sizeof ( orte_app_context_t * ) * array_size ) ;
2005-03-14 20:57:21 +00:00
if ( NULL = = apps ) {
2006-02-16 20:40:23 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:call-failed " ,
true , orterun_basename , " system " , " malloc returned NULL " , errno ) ;
2005-03-14 20:57:21 +00:00
exit ( 1 ) ;
}
2005-07-08 18:48:25 +00:00
num_apps = 0 ;
for ( j = i = 0 ; i < array_size ; + + i ) {
2005-09-04 20:54:19 +00:00
apps [ num_apps ] = ( orte_app_context_t * )
2005-07-03 04:02:01 +00:00
orte_pointer_array_get_item ( apps_pa , i ) ;
2005-08-08 16:42:28 +00:00
if ( NULL ! = apps [ num_apps ] ) {
2005-07-08 18:48:25 +00:00
j + = apps [ num_apps ] - > num_procs ;
num_apps + + ;
}
}
if ( 0 = = num_apps ) {
/* This should never happen -- this case should be caught in
create_app ( ) , but let ' s just double check . . . */
2005-09-04 20:54:19 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:nothing-to-do " ,
2005-07-08 18:48:25 +00:00
true , orterun_basename ) ;
exit ( 1 ) ;
2005-04-15 21:52:58 +00:00
}
2006-08-23 02:35:00 +00:00
proc_infos = ( struct proc_info_t * ) malloc ( sizeof ( struct proc_info_t ) * j ) ;
2005-04-15 21:52:58 +00:00
if ( NULL = = proc_infos ) {
2006-02-16 20:40:23 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:call-failed " ,
true , orterun_basename , " system " , " malloc returned NULL " , errno ) ;
2005-04-15 21:52:58 +00:00
exit ( 1 ) ;
}
for ( i = 0 ; i < j ; + + i ) {
proc_infos [ i ] . reported = false ;
2005-04-29 00:36:07 +00:00
proc_infos [ i ] . exit_status = 0 ;
2005-03-14 20:57:21 +00:00
}
/* Intialize our Open RTE environment */
2005-08-26 20:13:35 +00:00
/* Set the flag telling orte_init that I am NOT a
2005-06-24 16:59:37 +00:00
* singleton , but am " infrastructure " - prevents setting
* up incorrect infrastructure that only a singleton would
* require
*/
2005-08-26 20:13:35 +00:00
if ( ORTE_SUCCESS ! = ( rc = orte_init ( true ) ) ) {
2005-07-04 02:38:44 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:init-failure " , true ,
2005-03-14 20:57:21 +00:00
" orte_init() " , rc ) ;
return rc ;
}
2005-05-24 15:02:50 +00:00
/* check for daemon flags and push them into the environment
* since this isn ' t being automatically done
*/
2005-08-04 18:15:47 +00:00
id = mca_base_param_reg_int_name ( " orte_debug " , " daemons " ,
" Whether to debug the ORTE daemons or not " ,
2005-08-07 04:17:10 +00:00
false , false , ( int ) false , & iparam ) ;
2005-05-24 15:02:50 +00:00
if ( iparam ) {
2005-08-04 18:15:47 +00:00
char * tmp = mca_base_param_environ_variable ( " orte " , " debug " , " daemons " ) ;
2006-08-23 02:35:00 +00:00
if ( ORTE_SUCCESS ! = ( rc = opal_setenv ( tmp , " 1 " , true , NULL ) ) ) {
2005-07-28 21:17:48 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:environ " , false ,
2005-08-04 18:15:47 +00:00
orterun_basename , tmp , " 1 " , rc ) ;
free ( tmp ) ;
2005-05-24 15:02:50 +00:00
return rc ;
}
2005-08-04 18:15:47 +00:00
free ( tmp ) ;
2005-05-24 15:02:50 +00:00
}
2005-08-04 18:15:47 +00:00
id = mca_base_param_reg_int_name ( " orte " , " debug " ,
" Top-level ORTE debug switch " ,
false , false , 0 , & iparam ) ;
2005-05-24 15:02:50 +00:00
if ( iparam ) {
2005-08-04 18:15:47 +00:00
char * tmp = mca_base_param_environ_variable ( " orte " , NULL , " debug " ) ;
2006-08-23 02:35:00 +00:00
if ( ORTE_SUCCESS ! = ( rc = opal_setenv ( tmp , " 1 " , true , NULL ) ) ) {
2005-07-28 21:17:48 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:environ " , false ,
2005-08-04 18:15:47 +00:00
orterun_basename , tmp , " 1 " , rc ) ;
free ( tmp ) ;
2005-05-24 15:02:50 +00:00
return rc ;
}
2005-08-04 18:15:47 +00:00
free ( tmp ) ;
2005-05-24 15:02:50 +00:00
}
2005-08-04 18:15:47 +00:00
id = mca_base_param_reg_int_name ( " orte_debug " , " daemons_file " ,
" Whether want stdout/stderr of daemons to go to a file or not " ,
false , false , 0 , & iparam ) ;
2005-05-24 15:02:50 +00:00
if ( iparam ) {
2005-09-04 20:54:19 +00:00
char * tmp = mca_base_param_environ_variable ( " orte " , " debug " ,
2005-08-04 18:15:47 +00:00
" daemons_file " ) ;
2006-08-23 02:35:00 +00:00
if ( ORTE_SUCCESS ! = ( rc = opal_setenv ( tmp , " 1 " , true , NULL ) ) ) {
2005-07-28 21:17:48 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:environ " , false ,
2005-08-04 18:15:47 +00:00
orterun_basename , tmp , " 1 " , rc ) ;
free ( tmp ) ;
2005-05-24 15:02:50 +00:00
return rc ;
}
2005-08-04 18:15:47 +00:00
free ( tmp ) ;
2005-05-24 15:02:50 +00:00
}
2005-08-04 18:15:47 +00:00
id = mca_base_param_reg_int_name ( " orte " , " no_daemonize " ,
" Whether to properly daemonize the ORTE daemons or not " ,
false , false , 0 , & iparam ) ;
2005-05-24 15:02:50 +00:00
if ( iparam ) {
2005-09-04 20:54:19 +00:00
char * tmp = mca_base_param_environ_variable ( " orte " , NULL ,
2005-08-04 18:15:47 +00:00
" no_daemonize " ) ;
2006-08-23 02:35:00 +00:00
if ( ORTE_SUCCESS ! = ( rc = opal_setenv ( tmp , " 1 " , true , NULL ) ) ) {
2005-07-28 21:17:48 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:environ " , false ,
2005-08-04 18:15:47 +00:00
orterun_basename , tmp , " 1 " , rc ) ;
free ( tmp ) ;
2005-05-24 15:02:50 +00:00
return rc ;
}
2005-08-04 18:15:47 +00:00
free ( tmp ) ;
2005-05-24 15:02:50 +00:00
}
2005-09-04 20:54:19 +00:00
/* Prep to start the application */
2005-03-14 20:57:21 +00:00
2006-06-08 18:27:17 +00:00
/** setup callbacks for abort signals */
2005-09-20 17:09:11 +00:00
opal_signal_set ( & term_handler , SIGTERM ,
2006-06-26 15:12:52 +00:00
abort_signal_callback , & term_handler ) ;
2005-09-10 23:22:37 +00:00
opal_signal_add ( & term_handler , NULL ) ;
2005-09-20 17:09:11 +00:00
opal_signal_set ( & int_handler , SIGINT ,
2006-06-26 15:12:52 +00:00
abort_signal_callback , & int_handler ) ;
2005-09-10 23:22:37 +00:00
opal_signal_add ( & int_handler , NULL ) ;
2005-03-14 20:57:21 +00:00
2006-07-11 05:24:08 +00:00
# ifndef __WINDOWS__
2006-06-26 15:12:52 +00:00
/** setup callbacks for signals we should foward */
2006-06-08 18:27:17 +00:00
opal_signal_set ( & sigusr1_handler , SIGUSR1 ,
2006-06-26 15:12:52 +00:00
signal_forward_callback , & sigusr1_handler ) ;
2006-06-08 18:27:17 +00:00
opal_signal_add ( & sigusr1_handler , NULL ) ;
opal_signal_set ( & sigusr2_handler , SIGUSR2 ,
2006-06-26 15:12:52 +00:00
signal_forward_callback , & sigusr2_handler ) ;
2006-06-08 18:27:17 +00:00
opal_signal_add ( & sigusr2_handler , NULL ) ;
2006-07-11 05:24:08 +00:00
# endif /* __WINDOWS__ */
2005-08-30 17:29:43 +00:00
orte_totalview_init_before_spawn ( ) ;
2005-03-14 20:57:21 +00:00
/* Spawn the job */
2005-09-04 20:54:19 +00:00
2006-02-08 17:40:11 +00:00
cb_states = ORTE_PROC_STATE_ABORTED | ORTE_PROC_STATE_TERMINATED | ORTE_PROC_STATE_AT_STG1 ;
rc = orte_rmgr . spawn ( apps , num_apps , & jobid , job_state_callback , cb_states ) ;
2005-03-14 20:57:21 +00:00
if ( ORTE_SUCCESS ! = rc ) {
/* JMS show_help */
2005-07-03 23:31:27 +00:00
opal_output ( 0 , " %s: spawn failed with errno=%d \n " , orterun_basename , rc ) ;
2005-03-14 20:57:21 +00:00
} else {
2005-08-30 17:29:43 +00:00
2005-03-14 20:57:21 +00:00
/* Wait for the app to complete */
if ( wait_for_job_completion ) {
2005-07-03 22:45:48 +00:00
OPAL_THREAD_LOCK ( & orterun_globals . lock ) ;
2005-03-14 20:57:21 +00:00
while ( ! orterun_globals . exit ) {
2005-09-04 20:54:19 +00:00
opal_condition_wait ( & orterun_globals . cond ,
2005-03-14 20:57:21 +00:00
& orterun_globals . lock ) ;
}
2005-04-07 15:57:34 +00:00
/* Make sure we propagate the exit code */
2005-04-29 00:36:07 +00:00
if ( WIFEXITED ( orterun_globals . exit_status ) ) {
rc = WEXITSTATUS ( orterun_globals . exit_status ) ;
} else {
2006-02-15 22:41:29 +00:00
/* If a process was killed by a signal, then make the
* exit code of orterun be " signo + 128 " so that " prog "
* and " orterun prog " will both set the same status
* value for the shell */
rc = WTERMSIG ( orterun_globals . exit_status ) + 128 ;
2005-04-29 00:36:07 +00:00
}
2005-07-03 22:45:48 +00:00
OPAL_THREAD_UNLOCK ( & orterun_globals . lock ) ;
2005-04-15 21:52:58 +00:00
/* If we showed more abort messages than were allowed,
show a followup message here */
if ( num_aborted > max_display_aborted ) {
i = num_aborted - max_display_aborted ;
printf ( " %d additional process%s aborted (not shown) \n " ,
i , ( ( i > 1 ) ? " es " : " " ) ) ;
}
if ( num_killed > 0 ) {
printf ( " %d process%s killed (possibly by Open MPI) \n " ,
num_killed , ( ( num_killed > 1 ) ? " es " : " " ) ) ;
}
2005-03-14 20:57:21 +00:00
}
}
/* All done */
for ( i = 0 ; i < num_apps ; + + i ) {
OBJ_RELEASE ( apps [ i ] ) ;
}
free ( apps ) ;
2005-07-03 04:02:01 +00:00
OBJ_RELEASE ( apps_pa ) ;
2005-03-14 20:57:21 +00:00
orte_finalize ( ) ;
2005-04-13 15:26:33 +00:00
free ( orterun_basename ) ;
2005-09-03 01:22:11 +00:00
free ( proc_infos ) ;
2005-03-14 20:57:21 +00:00
return rc ;
}
2005-03-31 19:39:02 +00:00
/*
2005-09-04 20:54:19 +00:00
* On abnormal termination - dump the
2005-03-31 19:39:02 +00:00
* exit status of the aborted procs .
*/
static void dump_aborted_procs ( orte_jobid_t jobid )
{
char * segment ;
orte_gpr_value_t * * values = NULL ;
2006-08-15 19:54:10 +00:00
orte_std_cntr_t i , k , num_values = 0 ;
2005-03-31 19:39:02 +00:00
int rc ;
2005-04-29 00:36:07 +00:00
int32_t exit_status = 0 ;
bool exit_status_set ;
2005-03-31 19:39:02 +00:00
char * keys [ ] = {
ORTE_PROC_NAME_KEY ,
ORTE_PROC_PID_KEY ,
ORTE_PROC_RANK_KEY ,
ORTE_PROC_EXIT_CODE_KEY ,
ORTE_NODE_NAME_KEY ,
NULL
} ;
2005-09-20 17:09:11 +00:00
OPAL_TRACE_ARG1 ( 1 , jobid ) ;
2005-03-31 19:39:02 +00:00
/* query the job segment on the registry */
if ( ORTE_SUCCESS ! = ( rc = orte_schema . get_job_segment_name ( & segment , jobid ) ) ) {
ORTE_ERROR_LOG ( rc ) ;
return ;
}
rc = orte_gpr . get (
2005-03-31 22:30:22 +00:00
ORTE_GPR_KEYS_OR | ORTE_GPR_TOKENS_OR ,
2005-03-31 19:39:02 +00:00
segment ,
NULL ,
keys ,
& num_values ,
& values
) ;
if ( rc ! = ORTE_SUCCESS ) {
ORTE_ERROR_LOG ( rc ) ;
free ( segment ) ;
return ;
}
2005-04-28 13:18:52 +00:00
for ( i = 0 ; i < num_values ; i + + ) {
2005-03-31 19:39:02 +00:00
orte_gpr_value_t * value = values [ i ] ;
2006-02-07 03:32:36 +00:00
orte_process_name_t name , * nptr ;
pid_t pid = 0 , * pidptr ;
2006-08-15 19:54:10 +00:00
orte_std_cntr_t rank = 0 , * sptr ;
2005-05-01 00:47:35 +00:00
bool rank_found = false ;
2005-03-31 19:39:02 +00:00
char * node_name = NULL ;
2006-02-07 03:32:36 +00:00
orte_exit_code_t * ecptr ;
2005-03-31 19:39:02 +00:00
2005-04-29 00:36:07 +00:00
exit_status = 0 ;
exit_status_set = false ;
2005-03-31 19:39:02 +00:00
for ( k = 0 ; k < value - > cnt ; k + + ) {
orte_gpr_keyval_t * keyval = value - > keyvals [ k ] ;
if ( strcmp ( keyval - > key , ORTE_PROC_NAME_KEY ) = = 0 ) {
2006-02-07 03:32:36 +00:00
if ( ORTE_SUCCESS ! = ( rc = orte_dss . get ( ( void * * ) & nptr , keyval - > value , ORTE_NAME ) ) ) {
ORTE_ERROR_LOG ( rc ) ;
continue ;
}
name = * nptr ;
2005-03-31 19:39:02 +00:00
continue ;
}
if ( strcmp ( keyval - > key , ORTE_PROC_PID_KEY ) = = 0 ) {
2006-02-07 03:32:36 +00:00
if ( ORTE_SUCCESS ! = ( rc = orte_dss . get ( ( void * * ) & pidptr , keyval - > value , ORTE_PID ) ) ) {
ORTE_ERROR_LOG ( rc ) ;
continue ;
}
pid = * pidptr ;
2005-03-31 19:39:02 +00:00
continue ;
}
if ( strcmp ( keyval - > key , ORTE_PROC_RANK_KEY ) = = 0 ) {
2006-08-16 16:35:09 +00:00
if ( ORTE_SUCCESS ! = ( rc = orte_dss . get ( ( void * * ) & sptr , keyval - > value , ORTE_STD_CNTR ) ) ) {
2006-02-07 03:32:36 +00:00
ORTE_ERROR_LOG ( rc ) ;
continue ;
}
2005-05-01 00:47:35 +00:00
rank_found = true ;
2006-02-07 03:32:36 +00:00
rank = * sptr ;
2005-03-31 19:39:02 +00:00
continue ;
}
if ( strcmp ( keyval - > key , ORTE_PROC_EXIT_CODE_KEY ) = = 0 ) {
2006-02-07 03:32:36 +00:00
if ( ORTE_SUCCESS ! = ( rc = orte_dss . get ( ( void * * ) & ecptr , keyval - > value , ORTE_EXIT_CODE ) ) ) {
ORTE_ERROR_LOG ( rc ) ;
continue ;
}
exit_status = * ecptr ;
2005-04-29 00:36:07 +00:00
exit_status_set = true ;
2005-03-31 19:39:02 +00:00
continue ;
}
if ( strcmp ( keyval - > key , ORTE_NODE_NAME_KEY ) = = 0 ) {
2006-02-07 03:32:36 +00:00
node_name = ( char * ) ( keyval - > value - > data ) ;
2005-03-31 19:39:02 +00:00
continue ;
}
}
2005-05-01 00:47:35 +00:00
if ( rank_found & & exit_status_set ) {
2005-04-29 00:36:07 +00:00
proc_infos [ rank ] . exit_status = exit_status ;
2005-04-15 21:52:58 +00:00
}
2005-05-31 19:25:42 +00:00
2006-02-16 20:40:23 +00:00
if ( rank_found & & ! proc_infos [ rank ] . reported ) {
2005-04-15 21:52:58 +00:00
proc_infos [ rank ] . reported = true ;
2006-02-16 20:40:23 +00:00
if ( WIFSIGNALED ( exit_status ) ) {
if ( 9 = = WTERMSIG ( exit_status ) ) {
+ + num_killed ;
} else {
if ( num_aborted < max_display_aborted ) {
opal_show_help ( " help-orterun.txt " , " orterun:proc-aborted " , false ,
orterun_basename , ( unsigned long ) rank , ( unsigned long ) pid ,
node_name , WTERMSIG ( exit_status ) ) ;
}
+ + num_aborted ;
2005-04-15 21:52:58 +00:00
}
}
2005-04-28 13:18:52 +00:00
}
2005-04-07 15:57:34 +00:00
2005-05-01 09:53:30 +00:00
/* If we haven't done so already, hold the exit_status so we
can return it when exiting . Specifically , keep the first
2005-04-28 13:18:52 +00:00
non - zero entry . If they all return zero , we ' ll return
2005-05-01 09:53:30 +00:00
zero . We already have the globals . lock ( from
job_state_callback ) , so don ' t try to get it again . */
2005-04-28 13:18:52 +00:00
2005-04-29 00:36:07 +00:00
if ( 0 = = orterun_globals . exit_status & & exit_status_set ) {
orterun_globals . exit_status = exit_status ;
2005-03-31 19:39:02 +00:00
}
2005-04-28 13:18:52 +00:00
2005-03-31 19:39:02 +00:00
OBJ_RELEASE ( value ) ;
}
2005-04-28 13:18:52 +00:00
if ( NULL ! = values ) {
2005-03-31 19:39:02 +00:00
free ( values ) ;
}
2005-09-03 01:22:11 +00:00
free ( segment ) ;
2005-03-31 19:39:02 +00:00
}
2005-03-14 20:57:21 +00:00
/*
* signal main thread when application completes
*/
static void job_state_callback ( orte_jobid_t jobid , orte_proc_state_t state )
{
2005-09-20 17:09:11 +00:00
OPAL_TRACE_ARG2 ( 1 , jobid , state ) ;
2005-07-03 22:45:48 +00:00
OPAL_THREAD_LOCK ( & orterun_globals . lock ) ;
2005-04-15 21:52:58 +00:00
2006-02-16 20:40:23 +00:00
/* Note that there's only three states that we're interested in
2005-04-15 21:52:58 +00:00
here :
ABORTED : which means that one or more processes have aborted
( terminated abnormally ) . In which case , we probably
want to print out some information .
TERMINATED : which means that all the processes in the job have
completed ( normally and / or abnormally ) .
2006-02-16 20:40:23 +00:00
AT_STG1 : which means that everyone has hit stage gate 1 , so we
can do the parallel debugger startup stuff .
2005-04-15 21:52:58 +00:00
Remember that the rmgr itself will also be called for the
ABORTED state and call the pls . terminate_job , which will result
in killing all the other processes . */
2005-09-04 20:54:19 +00:00
2005-08-30 17:29:43 +00:00
if ( orte_debug_flag ) {
opal_output ( 0 , " spawn: in job_state_callback(jobid = %d, state = 0x%x) \n " ,
jobid , state ) ;
}
2005-03-14 20:57:21 +00:00
switch ( state ) {
case ORTE_PROC_STATE_ABORTED :
2005-03-31 19:39:02 +00:00
dump_aborted_procs ( jobid ) ;
2005-05-13 17:52:50 +00:00
orte_rmgr . terminate_job ( jobid ) ;
2005-04-15 21:52:58 +00:00
break ;
2005-03-31 19:39:02 +00:00
case ORTE_PROC_STATE_TERMINATED :
2006-08-16 16:35:09 +00:00
orterun_globals . exit_status = 0 ; /* set the exit status to indicate normal termination */
2005-03-14 20:57:21 +00:00
orterun_globals . exit = true ;
2005-07-03 22:45:48 +00:00
opal_condition_signal ( & orterun_globals . cond ) ;
2005-03-14 20:57:21 +00:00
break ;
2005-08-30 17:29:43 +00:00
case ORTE_PROC_STATE_AT_STG1 :
orte_totalview_init_after_spawn ( jobid ) ;
break ;
2006-02-08 17:40:11 +00:00
default :
opal_output ( 0 , " orterun: job state callback in unexpected state - jobid %lu, state 0x%04x \n " , jobid , state ) ;
break ;
2005-03-14 20:57:21 +00:00
}
2005-07-03 22:45:48 +00:00
OPAL_THREAD_UNLOCK ( & orterun_globals . lock ) ;
2005-03-14 20:57:21 +00:00
}
2005-03-31 19:39:02 +00:00
/*
2005-09-04 20:54:19 +00:00
* Fail - safe in the event the job hangs and doesn ' t
2005-03-31 19:39:02 +00:00
* cleanup correctly .
*/
2005-03-14 20:57:21 +00:00
static void exit_callback ( int fd , short event , void * arg )
{
2005-09-19 17:20:01 +00:00
OPAL_TRACE ( 1 ) ;
2005-09-20 17:09:11 +00:00
2005-07-28 21:17:48 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:abnormal-exit " ,
2005-08-27 13:32:25 +00:00
true , orterun_basename , orterun_basename ) ;
2005-08-26 20:36:11 +00:00
2005-09-10 23:22:37 +00:00
/* Remove the TERM and INT signal handlers */
opal_signal_del ( & term_handler ) ;
opal_signal_del ( & int_handler ) ;
2006-07-11 05:24:08 +00:00
# ifndef __WINDOWS__
2006-06-08 18:27:17 +00:00
/** Remove the USR signal handlers */
opal_signal_del ( & sigusr1_handler ) ;
opal_signal_del ( & sigusr2_handler ) ;
2006-07-11 05:24:08 +00:00
# endif /* __WINDOWS__ */
2006-06-08 18:27:17 +00:00
2005-08-26 20:36:11 +00:00
/* Trigger the normal exit conditions */
orterun_globals . exit = true ;
orterun_globals . exit_status = 1 ;
opal_condition_signal ( & orterun_globals . cond ) ;
2005-03-14 20:57:21 +00:00
}
2005-03-31 19:39:02 +00:00
/*
* Attempt to terminate the job and wait for callback indicating
2005-09-04 20:54:19 +00:00
* the job has been aborted .
2005-03-31 19:39:02 +00:00
*/
2006-06-08 18:27:17 +00:00
static void abort_signal_callback ( int fd , short flags , void * arg )
2005-03-14 20:57:21 +00:00
{
int ret ;
struct timeval tv = { 5 , 0 } ;
2005-07-03 23:09:55 +00:00
opal_event_t * event ;
2005-03-14 20:57:21 +00:00
static int signalled = 0 ;
2005-09-20 17:09:11 +00:00
2005-09-19 17:20:01 +00:00
OPAL_TRACE ( 1 ) ;
2005-09-20 17:09:11 +00:00
2005-03-14 20:57:21 +00:00
if ( 0 ! = signalled + + ) {
return ;
}
2006-06-26 18:21:45 +00:00
if ( ! orterun_globals . quiet ) {
fprintf ( stderr , " %s: killing job... " , orterun_basename ) ;
}
2005-03-14 20:57:21 +00:00
if ( jobid ! = ORTE_JOBID_MAX ) {
ret = orte_rmgr . terminate_job ( jobid ) ;
if ( ORTE_SUCCESS ! = ret ) {
jobid = ORTE_JOBID_MAX ;
}
}
2005-07-03 23:09:55 +00:00
if ( NULL ! = ( event = ( opal_event_t * ) malloc ( sizeof ( opal_event_t ) ) ) ) {
opal_evtimer_set ( event , exit_callback , NULL ) ;
opal_evtimer_add ( event , & tv ) ;
2005-03-14 20:57:21 +00:00
}
2006-07-18 14:42:27 +00:00
/* make the output a little prettier - move the prompt to its own line */
if ( ! orterun_globals . quiet ) {
fprintf ( stderr , " \n \n " ) ;
}
2005-03-14 20:57:21 +00:00
}
2006-06-08 18:27:17 +00:00
/**
* Pass user signals to the remote application processes
*/
2006-06-26 15:12:52 +00:00
static void signal_forward_callback ( int fd , short event , void * arg )
2006-06-08 18:27:17 +00:00
{
2006-08-23 02:35:00 +00:00
struct opal_event * signal = ( struct opal_event * ) arg ;
2006-06-26 15:12:52 +00:00
int signum , ret ;
2006-06-08 18:27:17 +00:00
OPAL_TRACE ( 1 ) ;
2006-06-26 15:12:52 +00:00
signum = OPAL_EVENT_SIGNAL ( signal ) ;
2006-06-26 18:21:45 +00:00
if ( ! orterun_globals . quiet ) {
fprintf ( stderr , " %s: Forwarding signal %d to job " ,
2006-06-26 15:12:52 +00:00
orterun_basename , signum ) ;
2006-06-26 18:21:45 +00:00
}
2006-06-08 18:27:17 +00:00
/** send the signal out to the processes */
2006-06-26 15:12:52 +00:00
if ( ORTE_SUCCESS ! = ( ret = orte_rmgr . signal_job ( jobid , signum ) ) ) {
fprintf ( stderr , " Signal %d could not be sent to the job (returned %d) " ,
signum , ret ) ;
2006-06-08 18:27:17 +00:00
}
}
2005-09-04 20:54:19 +00:00
static int init_globals ( void )
2005-03-14 20:57:21 +00:00
{
struct globals_t tmp = {
2006-07-10 21:25:33 +00:00
/* help = */ false ,
/* version = */ false ,
/* verbose = */ false ,
/* quiet = */ false ,
/* exit = */ false ,
/* no_wait = */ false ,
/* by_node = */ false ,
/* by_slot = */ false ,
/* no_oversub = */ false ,
/* debugger = */ false ,
/* no_local = */ false ,
/* num_procs = */ 0 ,
/* exit_status = */ 0 ,
/* hostfile = */ NULL ,
/* env_val = */ NULL ,
/* appfile = */ NULL ,
/* wdir = */ NULL ,
/* path = */ NULL
2005-03-14 20:57:21 +00:00
} ;
2005-03-18 23:58:36 +00:00
/* Only CONSTRUCT things once */
2005-09-04 20:54:19 +00:00
2005-03-18 23:58:36 +00:00
if ( ! globals_init ) {
2005-07-03 22:45:48 +00:00
OBJ_CONSTRUCT ( & orterun_globals . lock , opal_mutex_t ) ;
OBJ_CONSTRUCT ( & orterun_globals . cond , opal_condition_t ) ;
2005-03-18 23:58:36 +00:00
}
2006-07-10 21:25:33 +00:00
/* Reset the other fields every time */
2005-03-18 23:58:36 +00:00
2005-03-14 20:57:21 +00:00
orterun_globals = tmp ;
2005-03-18 23:58:36 +00:00
/* All done */
globals_init = true ;
2005-03-14 20:57:21 +00:00
return ORTE_SUCCESS ;
}
static int parse_globals ( int argc , char * argv [ ] )
{
2005-07-04 00:13:44 +00:00
opal_cmd_line_t cmd_line ;
2005-08-08 16:42:28 +00:00
int id , ret ;
2005-03-14 20:57:21 +00:00
/* Setup and parse the command line */
init_globals ( ) ;
2005-07-04 00:13:44 +00:00
opal_cmd_line_create ( & cmd_line , cmd_line_init ) ;
2005-08-02 18:52:12 +00:00
mca_base_cmd_line_setup ( & cmd_line ) ;
2006-02-12 01:33:29 +00:00
if ( ORTE_SUCCESS ! = ( ret = opal_cmd_line_parse ( & cmd_line , true ,
2005-07-28 21:17:48 +00:00
argc , argv ) ) ) {
return ret ;
}
2005-09-04 20:54:19 +00:00
2006-06-09 17:21:23 +00:00
/* print version if requested. Do this before check for help so
that - - version - - help works as one might expect . */
2006-06-22 19:48:27 +00:00
if ( orterun_globals . version & &
! ( 1 = = argc | | orterun_globals . help ) ) {
2006-06-09 17:21:23 +00:00
char * project_name = NULL ;
if ( 0 = = strcmp ( orterun_basename , " mpirun " ) ) {
project_name = " Open MPI " ;
} else {
project_name = " OpenRTE " ;
}
opal_show_help ( " help-orterun.txt " , " orterun:version " , false ,
2006-06-22 19:48:27 +00:00
orterun_basename , project_name , OPAL_VERSION ,
PACKAGE_BUGREPORT ) ;
2006-06-09 17:21:23 +00:00
/* if we were the only argument, exit */
if ( 2 = = argc ) exit ( 0 ) ;
}
2005-07-28 21:17:48 +00:00
/* Check for help request */
2005-09-04 20:54:19 +00:00
2005-04-12 16:01:30 +00:00
if ( 1 = = argc | | orterun_globals . help ) {
2005-03-14 20:57:21 +00:00
char * args = NULL ;
2006-06-22 19:48:27 +00:00
char * project_name = NULL ;
if ( 0 = = strcmp ( orterun_basename , " mpirun " ) ) {
project_name = " Open MPI " ;
} else {
project_name = " OpenRTE " ;
}
2005-07-04 00:13:44 +00:00
args = opal_cmd_line_get_usage_msg ( & cmd_line ) ;
2005-07-04 02:38:44 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:usage " , false ,
2006-06-22 19:48:27 +00:00
orterun_basename , project_name , OPAL_VERSION ,
orterun_basename , args ,
PACKAGE_BUGREPORT ) ;
2005-03-14 20:57:21 +00:00
free ( args ) ;
2005-09-04 20:54:19 +00:00
2005-03-14 20:57:21 +00:00
/* If someone asks for help, that should be all we do */
exit ( 0 ) ;
}
2005-11-20 16:06:53 +00:00
/* Do we want a user-level debugger? */
2005-10-05 10:24:34 +00:00
2005-11-20 16:06:53 +00:00
if ( orterun_globals . debugger ) {
orte_run_debugger ( orterun_basename , argc , argv ) ;
}
2005-10-05 10:24:34 +00:00
2005-06-06 13:43:20 +00:00
/* Allocate and map by node or by slot? Shortcut for setting an
MCA param . */
(copied from a mail that has a lengthy description of this commit)
I spoke with Tim about this the other day -- he gave me the green
light to go ahead with this, but it turned into a bigger job than I
thought it would be. I revamped how the default RAS scheduling and
round_robin RMAPS mapping occurs. The previous algorithms were pretty
brain dead, and ignored the "slots" and "max_slots" tokens in
hostfiles. I considered this a big enough problem to fix it for the
beta (because there is currently no way to control where processes are
launched on SMPs).
There's still some more bells and whistles that I'd like to implement,
but there's no hurry, and they can go on the trunk at any time. My
patches below are for what I considered "essential", and do the
following:
- honor the "slots" and "max-slots" tokens in the hostfile (and all
their synonyms), meaning that we allocate/map until we fill slots,
and if there are still more processes to allocate/map, we keep going
until we fill max-slots (i.e., only oversubscribe a node if we have
to).
- offer two different algorithms, currently supported by two new
options to orterun. Remember that there are two parts here -- slot
allocation and process mapping. Slot allocation controls how many
processes we'll be running on a node. After that decision has been
made, process mapping effectively controls where the ranks of
MPI_COMM_WORLD (MCW) are placed. Some of the examples given below
don't make sense unless you remember that there is a difference
between the two (which makes total sense, but you have to think
about it in terms of both things):
1. "-bynode": allocates/maps one process per node in a round-robin
fashion until all slots on the node are taken. If we still have more
processes after all slots are taken, then keep going until all
max-slots are taken. Examples:
- The hostfile:
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -bynode -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 2
vogon: MCW ranks 1, 3, 4, 5
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4
vogon: MCW ranks 1, 3, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until each
node's max_slots is hit, of course)
- orterun -bynode -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4, 6
vogon: MCW ranks 1, 3, 5, 7, 8, 9, 10, 11
2. "-byslot" (this is the default if you don't specify -bynode):
greedily takes all available slots on a node for a job before moving
on to the next node. If we still have processes to allocate/schedule,
then oversubscribe all nodes equally (i.e., go round robin on all
nodes until each node's max_slots is hit). Examples:
- The hostfile
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 1
vogon: MCW ranks 2, 3, 4, 5
- orterun -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2
vogon: MCW ranks 3, 4, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until max_slots
is hit)
- orterun -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2, 3
vogon: MCW ranks 4, 5, 6, 7, 8, 9, 10, 11
The above examples are fairly contrived, and it's not clear from them
that you can get different allocation answers in all cases (the
mapping differences are obvious). Consider the following allocation
example:
- The hostfile
eddie count=4
vogon count=4
earth count=4
deep-thought count=4
- orterun -np 8 -hostfile hostfile a.out
eddie: 4 slots will be allocated
vogon: 4 slots will be allocated
earth: no slots allocated
deep-thought: no slots allocated
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: 2 slots will be allocated
vogon: 2 slots will be allocated
earth: 2 slots will be allocated
deep-thought: 2 slots will be allocated
This commit was SVN r5894.
2005-05-31 16:36:53 +00:00
2005-06-06 13:43:20 +00:00
/* JMS To be changed post-beta to LAM's C/N command line notation */
2005-09-27 02:54:15 +00:00
/* Don't initialize the MCA parameter here unless we have to,
2005-10-07 22:24:52 +00:00
* since it really should be initialized in rmaps_base_open */
2005-09-27 02:54:15 +00:00
if ( orterun_globals . by_node | | orterun_globals . by_slot ) {
char * policy = NULL ;
2006-07-04 20:12:35 +00:00
id = mca_base_param_reg_string_name ( " rmaps " , " base_schedule_policy " ,
2005-10-07 22:24:52 +00:00
" Scheduling policy for RMAPS. [slot | node] " ,
2005-09-27 02:54:15 +00:00
false , false , " slot " , & policy ) ;
if ( orterun_globals . by_node ) {
orterun_globals . by_slot = false ;
mca_base_param_set_string ( id , " node " ) ;
} else {
orterun_globals . by_slot = true ;
mca_base_param_set_string ( id , " slot " ) ;
}
free ( policy ) ;
}
else {
/* Default */
(copied from a mail that has a lengthy description of this commit)
I spoke with Tim about this the other day -- he gave me the green
light to go ahead with this, but it turned into a bigger job than I
thought it would be. I revamped how the default RAS scheduling and
round_robin RMAPS mapping occurs. The previous algorithms were pretty
brain dead, and ignored the "slots" and "max_slots" tokens in
hostfiles. I considered this a big enough problem to fix it for the
beta (because there is currently no way to control where processes are
launched on SMPs).
There's still some more bells and whistles that I'd like to implement,
but there's no hurry, and they can go on the trunk at any time. My
patches below are for what I considered "essential", and do the
following:
- honor the "slots" and "max-slots" tokens in the hostfile (and all
their synonyms), meaning that we allocate/map until we fill slots,
and if there are still more processes to allocate/map, we keep going
until we fill max-slots (i.e., only oversubscribe a node if we have
to).
- offer two different algorithms, currently supported by two new
options to orterun. Remember that there are two parts here -- slot
allocation and process mapping. Slot allocation controls how many
processes we'll be running on a node. After that decision has been
made, process mapping effectively controls where the ranks of
MPI_COMM_WORLD (MCW) are placed. Some of the examples given below
don't make sense unless you remember that there is a difference
between the two (which makes total sense, but you have to think
about it in terms of both things):
1. "-bynode": allocates/maps one process per node in a round-robin
fashion until all slots on the node are taken. If we still have more
processes after all slots are taken, then keep going until all
max-slots are taken. Examples:
- The hostfile:
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -bynode -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 2
vogon: MCW ranks 1, 3, 4, 5
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4
vogon: MCW ranks 1, 3, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until each
node's max_slots is hit, of course)
- orterun -bynode -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4, 6
vogon: MCW ranks 1, 3, 5, 7, 8, 9, 10, 11
2. "-byslot" (this is the default if you don't specify -bynode):
greedily takes all available slots on a node for a job before moving
on to the next node. If we still have processes to allocate/schedule,
then oversubscribe all nodes equally (i.e., go round robin on all
nodes until each node's max_slots is hit). Examples:
- The hostfile
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 1
vogon: MCW ranks 2, 3, 4, 5
- orterun -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2
vogon: MCW ranks 3, 4, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until max_slots
is hit)
- orterun -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2, 3
vogon: MCW ranks 4, 5, 6, 7, 8, 9, 10, 11
The above examples are fairly contrived, and it's not clear from them
that you can get different allocation answers in all cases (the
mapping differences are obvious). Consider the following allocation
example:
- The hostfile
eddie count=4
vogon count=4
earth count=4
deep-thought count=4
- orterun -np 8 -hostfile hostfile a.out
eddie: 4 slots will be allocated
vogon: 4 slots will be allocated
earth: no slots allocated
deep-thought: no slots allocated
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: 2 slots will be allocated
vogon: 2 slots will be allocated
earth: 2 slots will be allocated
deep-thought: 2 slots will be allocated
This commit was SVN r5894.
2005-05-31 16:36:53 +00:00
orterun_globals . by_slot = true ;
}
2006-07-10 21:25:33 +00:00
/** Do we want to disallow oversubscription of nodes? */
id = mca_base_param_reg_int_name ( " rmaps " , " base_no_oversubscribe " ,
" If nonzero, do not allow oversubscription of processes on nodes. If zero (default), oversubscription is allowed. " ,
false , false , 0 , & ret ) ;
if ( orterun_globals . no_oversubscribe ) {
mca_base_param_set_int ( id , ( int ) true ) ;
} else {
mca_base_param_set_int ( id , ( int ) false ) ;
}
(copied from a mail that has a lengthy description of this commit)
I spoke with Tim about this the other day -- he gave me the green
light to go ahead with this, but it turned into a bigger job than I
thought it would be. I revamped how the default RAS scheduling and
round_robin RMAPS mapping occurs. The previous algorithms were pretty
brain dead, and ignored the "slots" and "max_slots" tokens in
hostfiles. I considered this a big enough problem to fix it for the
beta (because there is currently no way to control where processes are
launched on SMPs).
There's still some more bells and whistles that I'd like to implement,
but there's no hurry, and they can go on the trunk at any time. My
patches below are for what I considered "essential", and do the
following:
- honor the "slots" and "max-slots" tokens in the hostfile (and all
their synonyms), meaning that we allocate/map until we fill slots,
and if there are still more processes to allocate/map, we keep going
until we fill max-slots (i.e., only oversubscribe a node if we have
to).
- offer two different algorithms, currently supported by two new
options to orterun. Remember that there are two parts here -- slot
allocation and process mapping. Slot allocation controls how many
processes we'll be running on a node. After that decision has been
made, process mapping effectively controls where the ranks of
MPI_COMM_WORLD (MCW) are placed. Some of the examples given below
don't make sense unless you remember that there is a difference
between the two (which makes total sense, but you have to think
about it in terms of both things):
1. "-bynode": allocates/maps one process per node in a round-robin
fashion until all slots on the node are taken. If we still have more
processes after all slots are taken, then keep going until all
max-slots are taken. Examples:
- The hostfile:
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -bynode -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 2
vogon: MCW ranks 1, 3, 4, 5
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4
vogon: MCW ranks 1, 3, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until each
node's max_slots is hit, of course)
- orterun -bynode -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 2, 4, 6
vogon: MCW ranks 1, 3, 5, 7, 8, 9, 10, 11
2. "-byslot" (this is the default if you don't specify -bynode):
greedily takes all available slots on a node for a job before moving
on to the next node. If we still have processes to allocate/schedule,
then oversubscribe all nodes equally (i.e., go round robin on all
nodes until each node's max_slots is hit). Examples:
- The hostfile
eddie slots=2 max-slots=4
vogon slots=4 max-slots=8
- orterun -np 6 -hostfile hostfile a.out
eddie: MCW ranks 0, 1
vogon: MCW ranks 2, 3, 4, 5
- orterun -np 8 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2
vogon: MCW ranks 3, 4, 5, 6, 7
-> the algorithm oversubscribes all nodes "equally" (until max_slots
is hit)
- orterun -np 12 -hostfile hostfile a.out
eddie: MCW ranks 0, 1, 2, 3
vogon: MCW ranks 4, 5, 6, 7, 8, 9, 10, 11
The above examples are fairly contrived, and it's not clear from them
that you can get different allocation answers in all cases (the
mapping differences are obvious). Consider the following allocation
example:
- The hostfile
eddie count=4
vogon count=4
earth count=4
deep-thought count=4
- orterun -np 8 -hostfile hostfile a.out
eddie: 4 slots will be allocated
vogon: 4 slots will be allocated
earth: no slots allocated
deep-thought: no slots allocated
- orterun -bynode -np 8 -hostfile hostfile a.out
eddie: 2 slots will be allocated
vogon: 2 slots will be allocated
earth: 2 slots will be allocated
deep-thought: 2 slots will be allocated
This commit was SVN r5894.
2005-05-31 16:36:53 +00:00
2006-07-04 20:12:35 +00:00
/* Do we want to allow MPI applications on the same node as
mpirun ? */
id = mca_base_param_reg_int_name ( " rmaps " , " base_schedule_local " ,
" If nonzero, allow scheduling MPI applications on the same node as mpirun (default). If zero, do not schedule any MPI applications on the same node as mpirun " ,
false , false , 1 , & ret ) ;
if ( orterun_globals . no_local_schedule ) {
mca_base_param_set_int ( id , 0 ) ;
} else {
mca_base_param_set_int ( id , 1 ) ;
}
2005-03-14 20:57:21 +00:00
/* If we don't want to wait, we don't want to wait */
if ( orterun_globals . no_wait_for_job_completion ) {
wait_for_job_completion = false ;
}
OBJ_DESTRUCT ( & cmd_line ) ;
return ORTE_SUCCESS ;
}
static int parse_locals ( int argc , char * argv [ ] )
{
int i , rc , app_num ;
int temp_argc ;
2005-08-08 16:42:28 +00:00
char * * temp_argv , * * env ;
2005-03-14 20:57:21 +00:00
orte_app_context_t * app ;
bool made_app ;
2006-08-15 19:54:10 +00:00
orte_std_cntr_t j , size1 ;
2005-03-14 20:57:21 +00:00
/* Make the apps */
temp_argc = 0 ;
temp_argv = NULL ;
2005-07-04 00:13:44 +00:00
opal_argv_append ( & temp_argc , & temp_argv , argv [ 0 ] ) ;
2006-03-24 15:39:09 +00:00
/* Make the max size of the array be INT_MAX because we may be
parsing an app file , in which case we don ' t know how many
entries there will be . The max size of an orte_pointer_array
is only a safety net ; it only initially allocates block_size
entries ( 2 , in this case ) */
2006-03-23 17:55:25 +00:00
orte_pointer_array_init ( & apps_pa , 1 , INT_MAX , 2 ) ;
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
2005-08-08 16:42:28 +00:00
/* NOTE: This bogus env variable is necessary in the calls to
create_app ( ) , below . See comment immediately before the
create_app ( ) function for an explanation . */
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
env = NULL ;
2005-03-14 20:57:21 +00:00
for ( app_num = 0 , i = 1 ; i < argc ; + + i ) {
if ( 0 = = strcmp ( argv [ i ] , " : " ) ) {
/* Make an app with this argv */
2005-07-04 00:13:44 +00:00
if ( opal_argv_count ( temp_argv ) > 1 ) {
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
if ( NULL ! = env ) {
2005-07-04 00:13:44 +00:00
opal_argv_free ( env ) ;
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
env = NULL ;
}
2006-03-24 15:28:42 +00:00
app = NULL ;
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
rc = create_app ( temp_argc , temp_argv , & app , & made_app , & env ) ;
2006-03-23 16:53:11 +00:00
/** keep track of the number of apps - point this app_context to that index */
2005-03-14 20:57:21 +00:00
if ( ORTE_SUCCESS ! = rc ) {
/* Assume that the error message has already been
printed ; no need to cleanup - - we can just
exit */
exit ( 1 ) ;
}
if ( made_app ) {
2006-08-15 19:54:10 +00:00
orte_std_cntr_t dummy ;
2006-03-24 15:28:42 +00:00
app - > idx = app_num ;
+ + app_num ;
2005-07-03 04:02:01 +00:00
orte_pointer_array_add ( & dummy , apps_pa , app ) ;
2005-03-14 20:57:21 +00:00
}
2005-09-04 20:54:19 +00:00
2005-03-14 20:57:21 +00:00
/* Reset the temps */
2005-09-04 20:54:19 +00:00
2005-03-14 20:57:21 +00:00
temp_argc = 0 ;
temp_argv = NULL ;
2005-07-04 00:13:44 +00:00
opal_argv_append ( & temp_argc , & temp_argv , argv [ 0 ] ) ;
2005-03-14 20:57:21 +00:00
}
} else {
2005-07-04 00:13:44 +00:00
opal_argv_append ( & temp_argc , & temp_argv , argv [ i ] ) ;
2005-03-14 20:57:21 +00:00
}
}
2005-07-04 00:13:44 +00:00
if ( opal_argv_count ( temp_argv ) > 1 ) {
2006-03-24 15:28:42 +00:00
app = NULL ;
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
rc = create_app ( temp_argc , temp_argv , & app , & made_app , & env ) ;
2005-03-14 20:57:21 +00:00
if ( ORTE_SUCCESS ! = rc ) {
/* Assume that the error message has already been printed;
no need to cleanup - - we can just exit */
exit ( 1 ) ;
}
if ( made_app ) {
2006-08-15 19:54:10 +00:00
orte_std_cntr_t dummy ;
2006-03-24 15:28:42 +00:00
app - > idx = app_num ;
+ + app_num ;
2005-07-03 04:02:01 +00:00
orte_pointer_array_add ( & dummy , apps_pa , app ) ;
2005-03-14 20:57:21 +00:00
}
}
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
if ( NULL ! = env ) {
2005-07-04 00:13:44 +00:00
opal_argv_free ( env ) ;
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
}
2005-07-04 00:13:44 +00:00
opal_argv_free ( temp_argv ) ;
2005-03-14 20:57:21 +00:00
2005-08-08 16:42:28 +00:00
/* Once we've created all the apps, add the global MCA params to
each app ' s environment ( checking for duplicates , of
course - - yay opal_environ_merge ( ) ) . */
if ( NULL ! = global_mca_env ) {
size1 = orte_pointer_array_get_size ( apps_pa ) ;
/* Iterate through all the apps */
for ( j = 0 ; j < size1 ; + + j ) {
2005-09-04 20:54:19 +00:00
app = ( orte_app_context_t * )
2005-08-08 16:42:28 +00:00
orte_pointer_array_get_item ( apps_pa , j ) ;
if ( NULL ! = app ) {
/* Use handy utility function */
env = opal_environ_merge ( global_mca_env , app - > env ) ;
opal_argv_free ( app - > env ) ;
app - > env = env ;
}
}
}
/* Now take a subset of the MCA params and set them as MCA
overrides here in orterun ( so that when we orte_init ( ) later ,
all the components see these MCA params ) . Here ' s how we decide
which subset of the MCA params we set here in orterun :
1. If any global MCA params were set , use those
2. If no global MCA params were set and there was only one app ,
then use its app MCA params
3. Otherwise , don ' t set any
*/
env = NULL ;
if ( NULL ! = global_mca_env ) {
env = global_mca_env ;
} else {
if ( orte_pointer_array_get_size ( apps_pa ) > = 1 ) {
/* Remember that pointer_array's can be padded with NULL
entries ; so only use the app ' s env if there is exactly
1 non - NULL entry */
2005-09-04 20:54:19 +00:00
app = ( orte_app_context_t * )
2005-08-08 16:42:28 +00:00
orte_pointer_array_get_item ( apps_pa , 0 ) ;
if ( NULL ! = app ) {
env = app - > env ;
for ( j = 1 ; j < orte_pointer_array_get_size ( apps_pa ) ; + + j ) {
if ( NULL ! = orte_pointer_array_get_item ( apps_pa , j ) ) {
env = NULL ;
break ;
}
}
}
}
}
2005-09-04 20:54:19 +00:00
2005-08-08 16:42:28 +00:00
if ( NULL ! = env ) {
size1 = opal_argv_count ( env ) ;
for ( j = 0 ; j < size1 ; + + j ) {
putenv ( env [ j ] ) ;
}
}
2005-03-14 20:57:21 +00:00
/* All done */
return ORTE_SUCCESS ;
}
2005-08-08 16:42:28 +00:00
/*
* This function takes a " char ***app_env " parameter to handle the
* specific case :
*
* orterun - - mca foo bar - app appfile
*
* That is , we ' ll need to keep foo = bar , but the presence of the app
* file will cause an invocation of parse_appfile ( ) , which will cause
* one or more recursive calls back to create_app ( ) . Since the
* foo = bar value applies globally to all apps in the appfile , we need
* to pass in the " base " environment ( that contains the foo = bar value )
* when we parse each line in the appfile .
*
* This is really just a special case - - when we have a simple case like :
*
* orterun - - mca foo bar - np 4 hostname
*
* Then the upper - level function ( parse_locals ( ) ) calls create_app ( )
* with a NULL value for app_env , meaning that there is no " base "
* environment that the app needs to be created from .
*/
2005-03-14 20:57:21 +00:00
static int create_app ( int argc , char * argv [ ] , orte_app_context_t * * app_ptr ,
2005-08-08 16:42:28 +00:00
bool * made_app , char * * * app_env )
2005-03-14 20:57:21 +00:00
{
2005-07-04 00:13:44 +00:00
opal_cmd_line_t cmd_line ;
2005-03-14 20:57:21 +00:00
char cwd [ OMPI_PATH_MAX ] ;
2006-02-07 03:32:36 +00:00
int i , j , count , rc ;
2005-03-14 20:57:21 +00:00
char * param , * value , * value2 ;
orte_app_context_t * app = NULL ;
2005-11-03 18:15:47 +00:00
#if 0 /* Used only in the C/N notion case, remove to silence compiler warnings */
2006-08-15 19:54:10 +00:00
orte_std_cntr_t l , len ;
2005-11-03 18:15:47 +00:00
# endif
2005-04-19 05:45:25 +00:00
bool map_data = false , save_arg , cmd_line_made = false ;
2005-03-14 20:57:21 +00:00
int new_argc = 0 ;
char * * new_argv = NULL ;
* made_app = false ;
/* Pre-process the command line:
2005-09-04 20:54:19 +00:00
2005-03-14 20:57:21 +00:00
- convert C , cX , N , nX arguments to " -rawmap <id> <arg> " so
that the parser can pick it up nicely .
- convert - host to - rawmap < id > < arg >
- convert - arch to - rawmap < id > < arg >
Converting these to the same argument type will a ) simplify the
logic down below , and b ) allow us to preserve the ordering of
these arguments as the user specified them on the command
line . */
for ( i = 0 ; i < argc ; + + i ) {
map_data = false ;
save_arg = true ;
2005-11-03 18:15:47 +00:00
/* JJH To fix in the future
* Currently C / N notation is not supported so don ' t execute this check
2006-02-07 03:32:36 +00:00
* Bug : Make this context sensitive since it will not behave properly
2005-11-03 18:15:47 +00:00
* with the following argument set :
* $ orterun - np 2 - host c2 , c3 , c12 hostname
* Since it will see the hosts c2 , c3 , and c12 as C options instead
* of hostnames .
*/
if ( false ) { ; } /* Wrapper to preserve logic continuation while the below
is commented out */
#if 0
2005-03-14 20:57:21 +00:00
if ( 0 = = strcmp ( argv [ i ] , " C " ) | |
0 = = strcmp ( argv [ i ] , " N " ) ) {
map_data = true ;
2005-09-04 20:54:19 +00:00
}
2005-03-14 20:57:21 +00:00
2005-09-04 20:54:19 +00:00
/* Heuristic: if the string fits "[cn][0-9]+" or "[cn][0-9],",
2005-03-14 20:57:21 +00:00
then accept it as mapping data */
else if ( ' c ' = = argv [ i ] [ 0 ] | | ' n ' = = argv [ i ] [ 0 ] ) {
len = strlen ( argv [ i ] ) ;
if ( len > 1 ) {
for ( l = 1 ; l < len ; + + l ) {
if ( ' , ' = = argv [ i ] [ l ] ) {
map_data = true ;
break ;
} else if ( ! isdigit ( argv [ i ] [ l ] ) ) {
break ;
}
}
if ( l > = len ) {
map_data = true ;
}
}
}
2005-11-03 18:15:47 +00:00
# endif
2005-03-14 20:57:21 +00:00
2005-09-20 08:56:02 +00:00
#if 0
/* JMS commented out because we don't handle this in any
mapper */
2005-03-14 20:57:21 +00:00
/* Save -arch args */
else if ( 0 = = strcmp ( " -arch " , argv [ i ] ) ) {
char str [ 2 ] = { ' 0 ' + ORTE_APP_CONTEXT_MAP_ARCH , ' \0 ' } ;
2005-07-04 00:13:44 +00:00
opal_argv_append ( & new_argc , & new_argv , " -rawmap " ) ;
opal_argv_append ( & new_argc , & new_argv , str ) ;
2005-03-14 20:57:21 +00:00
save_arg = false ;
}
2005-09-20 08:56:02 +00:00
# endif
2005-03-14 20:57:21 +00:00
/* Save -host args */
2006-02-07 03:32:36 +00:00
else if ( 0 = = strcmp ( " --host " , argv [ i ] ) | |
0 = = strcmp ( " -host " , argv [ i ] ) | |
2005-09-22 16:08:40 +00:00
0 = = strcmp ( " -H " , argv [ i ] ) ) {
2005-03-14 20:57:21 +00:00
char str [ 2 ] = { ' 0 ' + ORTE_APP_CONTEXT_MAP_HOSTNAME , ' \0 ' } ;
2005-07-04 00:13:44 +00:00
opal_argv_append ( & new_argc , & new_argv , " -rawmap " ) ;
opal_argv_append ( & new_argc , & new_argv , str ) ;
2005-03-14 20:57:21 +00:00
save_arg = false ;
}
/* If this token was C/N map data, save it */
if ( map_data ) {
char str [ 2 ] = { ' 0 ' + ORTE_APP_CONTEXT_MAP_CN , ' \0 ' } ;
2005-07-04 00:13:44 +00:00
opal_argv_append ( & new_argc , & new_argv , " -rawmap " ) ;
opal_argv_append ( & new_argc , & new_argv , str ) ;
2005-03-14 20:57:21 +00:00
}
if ( save_arg ) {
2005-07-04 00:13:44 +00:00
opal_argv_append ( & new_argc , & new_argv , argv [ i ] ) ;
2005-03-14 20:57:21 +00:00
}
}
/* Parse application command line options. Add the -rawmap option
separately so that the user doesn ' t see it in the - - help
message . */
init_globals ( ) ;
2005-07-04 00:13:44 +00:00
opal_cmd_line_create ( & cmd_line , cmd_line_init ) ;
2005-03-18 03:43:59 +00:00
mca_base_cmd_line_setup ( & cmd_line ) ;
2005-03-14 20:57:21 +00:00
cmd_line_made = true ;
2005-07-04 00:13:44 +00:00
opal_cmd_line_make_opt3 ( & cmd_line , ' \0 ' , NULL , " rawmap " , 2 ,
2005-03-14 20:57:21 +00:00
" Hidden / internal parameter -- users should not use this! " ) ;
2005-07-04 00:13:44 +00:00
rc = opal_cmd_line_parse ( & cmd_line , true , new_argc , new_argv ) ;
opal_argv_free ( new_argv ) ;
2005-03-14 20:57:21 +00:00
new_argv = NULL ;
2006-02-12 01:33:29 +00:00
if ( ORTE_SUCCESS ! = rc ) {
2005-03-14 20:57:21 +00:00
goto cleanup ;
}
2005-08-08 16:42:28 +00:00
mca_base_cmd_line_process_args ( & cmd_line , app_env , & global_mca_env ) ;
2005-03-14 20:57:21 +00:00
/* Is there an appfile in here? */
if ( NULL ! = orterun_globals . appfile ) {
OBJ_DESTRUCT ( & cmd_line ) ;
2005-08-08 16:42:28 +00:00
return parse_appfile ( strdup ( orterun_globals . appfile ) , app_env ) ;
2005-03-14 20:57:21 +00:00
}
/* Setup application context */
app = OBJ_NEW ( orte_app_context_t ) ;
2006-02-07 03:32:36 +00:00
opal_cmd_line_get_tail ( & cmd_line , & count , & app - > argv ) ;
2005-03-14 20:57:21 +00:00
/* See if we have anything left */
2006-02-07 03:32:36 +00:00
if ( 0 = = count ) {
2005-07-04 02:38:44 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:executable-not-specified " ,
2005-04-12 16:01:30 +00:00
true , orterun_basename , orterun_basename ) ;
2005-03-14 20:57:21 +00:00
rc = ORTE_ERR_NOT_FOUND ;
goto cleanup ;
}
2005-04-09 01:26:17 +00:00
/* Grab all OMPI_* environment variables */
2005-03-14 20:57:21 +00:00
2005-08-08 16:42:28 +00:00
app - > env = opal_argv_copy ( * app_env ) ;
2005-03-14 20:57:21 +00:00
for ( i = 0 ; NULL ! = environ [ i ] ; + + i ) {
2005-04-06 01:58:30 +00:00
if ( 0 = = strncmp ( " OMPI_ " , environ [ i ] , 5 ) ) {
2005-07-04 00:13:44 +00:00
opal_argv_append_nosize ( & app - > env , environ [ i ] ) ;
2005-03-14 20:57:21 +00:00
}
}
/* Did the user request to export any environment variables? */
2005-07-04 00:13:44 +00:00
if ( opal_cmd_line_is_taken ( & cmd_line , " x " ) ) {
j = opal_cmd_line_get_ninsts ( & cmd_line , " x " ) ;
2005-03-14 20:57:21 +00:00
for ( i = 0 ; i < j ; + + i ) {
2005-07-04 00:13:44 +00:00
param = opal_cmd_line_get_param ( & cmd_line , " x " , i , 0 ) ;
2005-03-14 20:57:21 +00:00
if ( NULL ! = strchr ( param , ' = ' ) ) {
2005-07-04 00:13:44 +00:00
opal_argv_append_nosize ( & app - > env , param ) ;
2005-03-14 20:57:21 +00:00
} else {
value = getenv ( param ) ;
if ( NULL ! = value ) {
if ( NULL ! = strchr ( value , ' = ' ) ) {
2005-07-04 00:13:44 +00:00
opal_argv_append_nosize ( & app - > env , value ) ;
2005-03-14 20:57:21 +00:00
} else {
asprintf ( & value2 , " %s=%s " , param , value ) ;
2005-07-04 00:13:44 +00:00
opal_argv_append_nosize ( & app - > env , value2 ) ;
2005-05-12 21:44:23 +00:00
free ( value2 ) ;
2005-03-14 20:57:21 +00:00
}
} else {
2005-07-03 23:31:27 +00:00
opal_output ( 0 , " Warning: could not find environment variable \" %s \" \n " , param ) ;
2005-03-14 20:57:21 +00:00
}
}
}
}
/* Did the user request a specific path? */
if ( NULL ! = orterun_globals . path ) {
asprintf ( & value , " PATH=%s " , orterun_globals . path ) ;
2005-07-04 00:13:44 +00:00
opal_argv_append_nosize ( & app - > env , value ) ;
2005-03-14 20:57:21 +00:00
free ( value ) ;
}
/* Did the user request a specific wdir? */
if ( NULL ! = orterun_globals . wdir ) {
app - > cwd = strdup ( orterun_globals . wdir ) ;
2006-02-16 20:40:23 +00:00
app - > user_specified_cwd = true ;
2005-03-14 20:57:21 +00:00
} else {
getcwd ( cwd , sizeof ( cwd ) ) ;
app - > cwd = strdup ( cwd ) ;
2006-02-16 20:40:23 +00:00
app - > user_specified_cwd = false ;
2005-03-14 20:57:21 +00:00
}
2006-02-28 11:52:12 +00:00
/* Did the user specify a specific prefix for this app_context_t
or provide an absolute path name to argv [ 0 ] ? */
if ( opal_cmd_line_is_taken ( & cmd_line , " prefix " ) | |
' / ' = = argv [ 0 ] [ 0 ] ) {
2005-09-06 16:10:05 +00:00
size_t param_len ;
2006-02-28 14:44:40 +00:00
/* The --prefix option takes precedence over /path/to/orterun */
if ( opal_cmd_line_is_taken ( & cmd_line , " prefix " ) ) {
param = opal_cmd_line_get_param ( & cmd_line , " prefix " , 0 , 0 ) ;
} else {
2006-08-23 02:35:00 +00:00
char * tmp_basename = NULL ;
2006-02-28 14:44:40 +00:00
/* If they specified an absolute path, strip off the
/ bin / < exec_name > " and leave just the prefix */
2006-08-23 02:35:00 +00:00
param = opal_dirname ( argv [ 0 ] ) ;
2006-02-28 11:52:12 +00:00
/* Quick sanity check to ensure we got
something / bin / < exec_name > and that the installation
tree is at least more or less what we expect it to
be */
2006-08-23 02:35:00 +00:00
tmp_basename = opal_basename ( param ) ;
if ( 0 = = strcmp ( " bin " , tmp_basename ) ) {
char * tmp = param ;
param = opal_dirname ( tmp ) ;
free ( tmp ) ;
2006-02-28 11:52:12 +00:00
} else {
free ( param ) ;
param = NULL ;
2005-09-06 16:10:05 +00:00
}
2006-08-23 02:35:00 +00:00
free ( tmp_basename ) ;
2005-09-06 16:10:05 +00:00
}
2006-02-28 11:52:12 +00:00
if ( NULL ! = param ) {
/* "Parse" the param, aka remove superfluous path_sep "/". */
param_len = strlen ( param ) ;
2006-08-21 21:55:41 +00:00
while ( 0 = = strcmp ( OPAL_PATH_SEP , & ( param [ param_len - 1 ] ) ) ) {
2006-02-28 11:52:12 +00:00
param [ param_len - 1 ] = ' \0 ' ;
param_len - - ;
if ( 0 = = param_len ) {
opal_show_help ( " help-orterun.txt " , " orterun:empty-prefix " ,
true , orterun_basename , orterun_basename ) ;
return ORTE_ERR_FATAL ;
}
}
app - > prefix_dir = strdup ( param ) ;
}
2005-09-06 16:10:05 +00:00
}
2005-03-14 20:57:21 +00:00
/* Did the user request any mappings? They were all converted to
- - rawmap items , above . */
2005-07-04 00:13:44 +00:00
if ( opal_cmd_line_is_taken ( & cmd_line , " rawmap " ) ) {
j = opal_cmd_line_get_ninsts ( & cmd_line , " rawmap " ) ;
2006-08-23 02:35:00 +00:00
app - > map_data = ( orte_app_context_map_t * * ) malloc ( sizeof ( orte_app_context_map_t * ) * j ) ;
2005-03-14 20:57:21 +00:00
if ( NULL = = app - > map_data ) {
rc = ORTE_ERR_OUT_OF_RESOURCE ;
goto cleanup ;
}
app - > num_map = j ;
for ( i = 0 ; i < j ; + + i ) {
app - > map_data [ i ] = NULL ;
}
for ( i = 0 ; i < j ; + + i ) {
2005-07-04 00:13:44 +00:00
value = opal_cmd_line_get_param ( & cmd_line , " rawmap " , i , 0 ) ;
value2 = opal_cmd_line_get_param ( & cmd_line , " rawmap " , i , 1 ) ;
2005-03-14 20:57:21 +00:00
app - > map_data [ i ] = OBJ_NEW ( orte_app_context_map_t ) ;
if ( NULL = = app - > map_data [ i ] ) {
rc = ORTE_ERR_OUT_OF_RESOURCE ;
goto cleanup ;
}
app - > map_data [ i ] - > map_type = value [ 0 ] - ' 0 ' ;
app - > map_data [ i ] - > map_data = strdup ( value2 ) ;
2006-02-07 03:32:36 +00:00
/* map_data = true;
2005-10-07 22:24:52 +00:00
* JJH - This activates the C / N mapping stuff ,
* or at least allows us to pass the ' num_procs ' check below .
* since it is not implemented yet , leave commented . */
2005-03-14 20:57:21 +00:00
}
}
/* Get the numprocs */
app - > num_procs = orterun_globals . num_procs ;
2005-04-09 01:26:17 +00:00
2006-07-10 21:25:33 +00:00
/* If the user didn't specify the number of processes to run, then we
default to launching an app process using every slot . We can ' t do
anything about that here - we leave it to the RMAPS framework ' s
components to note this and deal with it later .
HOWEVER , we ONLY support this mode of operation if the number of
app_contexts is equal to ONE . If the user provides multiple applications ,
we simply must have more information - in this case , generate an
error .
*/
if ( app - > num_procs = = 0 ) {
have_zero_np = true ; /** flag that we have a zero_np situation */
2005-03-14 20:57:21 +00:00
}
2006-07-10 21:25:33 +00:00
if ( 0 < total_num_apps & & have_zero_np ) {
/** we have more than one app and a zero_np - that's no good.
* note that we have to do this as a two step logic check since
* the user may fail to specify num_procs for the first app , but
* then give us another application .
*/
opal_show_help ( " help-orterun.txt " , " orterun:multi-apps-and-zero-np " ,
true , orterun_basename , NULL ) ;
return ORTE_ERR_FATAL ;
}
total_num_apps + + ;
2006-02-16 20:40:23 +00:00
/* Do not try to find argv[0] here -- the starter is responsible
for that because it may not be relevant to try to find it on
the node where orterun is executing . So just strdup ( ) argv [ 0 ]
into app . */
2005-03-14 20:57:21 +00:00
2006-02-16 20:40:23 +00:00
app - > app = strdup ( app - > argv [ 0 ] ) ;
2005-03-14 20:57:21 +00:00
if ( NULL = = app - > app ) {
2006-02-16 20:40:23 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:call-failed " ,
true , orterun_basename , " library " , " strdup returned NULL " , errno ) ;
2005-03-14 20:57:21 +00:00
rc = ORTE_ERR_NOT_FOUND ;
goto cleanup ;
}
* app_ptr = app ;
app = NULL ;
* made_app = true ;
/* All done */
cleanup :
if ( NULL ! = app ) {
OBJ_RELEASE ( app ) ;
}
if ( NULL ! = new_argv ) {
2005-07-04 00:13:44 +00:00
opal_argv_free ( new_argv ) ;
2005-03-14 20:57:21 +00:00
}
if ( cmd_line_made ) {
OBJ_DESTRUCT ( & cmd_line ) ;
}
return rc ;
}
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
static int parse_appfile ( char * filename , char * * * env )
2005-03-14 20:57:21 +00:00
{
size_t i , len ;
FILE * fp ;
char line [ BUFSIZ ] ;
2006-03-23 17:55:25 +00:00
int rc , argc , app_num ;
2005-03-14 20:57:21 +00:00
char * * argv ;
orte_app_context_t * app ;
bool blank , made_app ;
char bogus [ ] = " bogus " ;
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
char * * tmp_env ;
2005-03-14 20:57:21 +00:00
/* Try to open the file */
fp = fopen ( filename , " r " ) ;
if ( NULL = = fp ) {
2005-07-04 02:38:44 +00:00
opal_show_help ( " help-orterun.txt " , " orterun:appfile-not-found " , true ,
2005-03-14 20:57:21 +00:00
filename ) ;
return ORTE_ERR_NOT_FOUND ;
}
/* Read in line by line */
line [ sizeof ( line ) - 1 ] = ' \0 ' ;
2006-03-23 17:55:25 +00:00
app_num = 0 ;
2005-03-14 20:57:21 +00:00
do {
/* We need a bogus argv[0] (because when argv comes in from
the command line , argv [ 0 ] is " orterun " , so the parsing
logic ignores it ) . So create one here rather than making
an argv and then pre - pending a new argv [ 0 ] ( which would be
rather inefficient ) . */
line [ 0 ] = ' \0 ' ;
strcat ( line , bogus ) ;
2005-09-04 20:54:19 +00:00
if ( NULL = = fgets ( line + sizeof ( bogus ) - 1 ,
2005-03-14 20:57:21 +00:00
sizeof ( line ) - sizeof ( bogus ) - 1 , fp ) ) {
break ;
}
2005-04-12 18:42:34 +00:00
/* Remove a trailing newline */
2005-03-14 20:57:21 +00:00
len = strlen ( line ) ;
2005-04-12 18:42:34 +00:00
if ( len > 0 & & ' \n ' = = line [ len - 1 ] ) {
line [ len - 1 ] = ' \0 ' ;
if ( len > 0 ) {
- - len ;
}
}
/* Remove comments */
2005-03-14 20:57:21 +00:00
for ( i = 0 ; i < len ; + + i ) {
if ( ' # ' = = line [ i ] ) {
line [ i ] = ' \0 ' ;
break ;
} else if ( i + 1 < len & & ' / ' = = line [ i ] & & ' / ' = = line [ i + 1 ] ) {
line [ i ] = ' \0 ' ;
break ;
}
}
/* Is this a blank line? */
len = strlen ( line ) ;
for ( blank = true , i = sizeof ( bogus ) ; i < len ; + + i ) {
if ( ! isspace ( line [ i ] ) ) {
blank = false ;
break ;
}
}
if ( blank ) {
continue ;
}
/* We got a line with *something* on it. So process it */
2005-07-04 00:13:44 +00:00
argv = opal_argv_split ( line , ' ' ) ;
argc = opal_argv_count ( argv ) ;
2005-03-14 20:57:21 +00:00
if ( argc > 0 ) {
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
2005-08-08 16:42:28 +00:00
/* Create a temporary env to use in the recursive call --
that is : don ' t disturb the original env so that we can
have a consistent global env . This allows for the
case :
2005-09-04 20:54:19 +00:00
orterun - - mca foo bar - - appfile file
2005-08-08 16:42:28 +00:00
where the " file " contains multiple apps . In this case ,
each app in " file " will get * only * foo = bar as the base
environment from which its specific environment is
constructed . */
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
if ( NULL ! = * env ) {
2005-07-04 00:13:44 +00:00
tmp_env = opal_argv_copy ( * env ) ;
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
if ( NULL = = tmp_env ) {
return ORTE_ERR_OUT_OF_RESOURCE ;
}
} else {
tmp_env = NULL ;
}
rc = create_app ( argc , argv , & app , & made_app , & tmp_env ) ;
2005-03-14 20:57:21 +00:00
if ( ORTE_SUCCESS ! = rc ) {
/* Assume that the error message has already been
printed ; no need to cleanup - - we can just exit */
exit ( 1 ) ;
}
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
if ( NULL ! = tmp_env ) {
2005-07-04 00:13:44 +00:00
opal_argv_free ( tmp_env ) ;
While waiting for fortran compiles...
Fixes for orterun in handling different MCA params for different
processes (reviewed by Brian):
- By design, if you run the following:
mpirun --mca foo aaa --mca foo bbb a.out
a.out will get a single MCA param for foo with value "aaa,bbb".
- However, if you specify multiple apps with different values for the
same MCA param, you should expect to get the different values for
each app. For example:
mpirun --mca foo aaa a.out : --mca foo bbb b.out
Should yield a.out with a "foo" param with value "aaa" and b.out
with a "foo" param with a value "bbb".
- This did not work -- both a.out and b.out would get a "foo" with
"aaa,bbb".
- This commit fixes this behavior -- now a.out will get aaa and b.out
will get bbb.
- Additionally, if you mix --mca and and app file, you can have
"global" params and per-line-in-the-appfile params. For example:
mpirun --mca foo zzzz --app appfile
where "appfile" contains:
-np 1 --mca bar aaa a.out
-np 1 --mca bar bbb b.out
In this case, a.out will get foo=zzzz and bar=aaa, and b.out will
get foo=zzzz and bar=bbb.
Spiffy.
Ok, fortran build is done... back to Fortran... sigh...
This commit was SVN r5710.
2005-05-13 14:36:36 +00:00
}
2005-03-14 20:57:21 +00:00
if ( made_app ) {
2006-08-15 19:54:10 +00:00
orte_std_cntr_t dummy ;
2006-03-24 15:28:42 +00:00
app - > idx = app_num ;
+ + app_num ;
2005-07-03 04:02:01 +00:00
orte_pointer_array_add ( & dummy , apps_pa , app ) ;
2005-03-14 20:57:21 +00:00
}
}
} while ( ! feof ( fp ) ) ;
fclose ( fp ) ;
/* All done */
free ( filename ) ;
return ORTE_SUCCESS ;
}