/* * Copyright (c) 2004-2009 The Trustees of Indiana University. * All rights reserved. * Copyright (c) 2004-2005 The Trustees of the University of Tennessee. * All rights reserved. * Copyright (c) 2004-2005 High Performance Computing Center Stuttgart, * University of Stuttgart. All rights reserved. * Copyright (c) 2004-2005 The Regents of the University of California. * All rights reserved. * $COPYRIGHT$ * * Additional copyrights may follow * * $HEADER$ */ /* * */ #include "orte_config.h" #ifdef HAVE_STRING_H #include #endif #include #include #include #include #ifdef HAVE_UNISTD_H #include #endif /* HAVE_UNISTD_H */ #include "opal/mca/mca.h" #include "opal/mca/base/base.h" #include "opal/mca/base/mca_base_param.h" #include "opal/event/event.h" #include "orte/constants.h" #include "orte/util/show_help.h" #include "opal/util/argv.h" #include "opal/util/output.h" #include "opal/util/opal_environ.h" #include "opal/util/basename.h" #include "opal/threads/mutex.h" #include "opal/threads/condition.h" #include "orte/util/name_fns.h" #include "orte/util/proc_info.h" #include "orte/runtime/orte_globals.h" #include "orte/runtime/orte_wait.h" #include "orte/mca/errmgr/errmgr.h" #include "orte/mca/rml/rml_types.h" #include "orte/mca/filem/filem.h" #include "orte/mca/filem/base/base.h" #include "filem_rsh.h" /********** * Local Function and Variable Declarations **********/ static int orte_filem_rsh_start_copy(orte_filem_base_request_t *request); static int orte_filem_rsh_start_rm(orte_filem_base_request_t *request); static int orte_filem_rsh_start_command(orte_filem_base_process_set_t *proc_set, orte_filem_base_file_set_t *file_set, char * command, orte_filem_base_request_t *request, int index); static int start_child(char * command, orte_filem_base_request_t *request, int index); static int orte_filem_rsh_query_remote_path(char **remote_ref, orte_process_name_t *proc, int *flag); static void filem_rsh_waitpid_cb(pid_t pid, int status, void* cbdata); /* Permission to send functionality */ static int orte_filem_rsh_permission_listener_init(orte_rml_buffer_callback_fn_t rml_cbfunc); static int orte_filem_rsh_permission_listener_cancel(void); static void orte_filem_rsh_permission_callback(int status, orte_process_name_t* sender, opal_buffer_t *buffer, orte_rml_tag_t tag, void* cbdata); static int orte_filem_rsh_permission_ask(orte_process_name_t* sender, int num_sends); static int permission_send_done(orte_process_name_t* sender, int num_avail, int32_t exit_status, char * local_target, char * remote_target, char * command); static int permission_send_num_allowed(orte_process_name_t* sender, int num_allowed); /************* * Local work pool structure *************/ int cur_num_incomming = 0; int cur_num_outgoing = 0; static bool work_pool_all_done = false; static opal_mutex_t work_pool_lock; static opal_condition_t work_pool_cond; /* * work_pool_waiting: * - processes that are waiting for my permission to put() to me */ opal_list_t work_pool_waiting; /* * work_pool_held: * - requests that are held before asking permission to reduce load */ opal_list_t work_pool_held; /* * work_pool_pending: * - put requests waiting on permission to send to peer */ opal_list_t work_pool_pending; /* * work_pool_active: * - put requests currently sending */ opal_list_t work_pool_active; struct orte_filem_rsh_work_pool_item_t { /** This is an object, so must have a super */ opal_list_item_t super; /** Command to exec */ char * command; /** Pointer to FileM Request */ orte_filem_base_request_t *request; /** Index into the request */ int index; /** Process Set */ orte_filem_base_process_set_t proc_set; /** File Set */ orte_filem_base_file_set_t file_set; /** If this item is active */ bool active; }; typedef struct orte_filem_rsh_work_pool_item_t orte_filem_rsh_work_pool_item_t; OBJ_CLASS_DECLARATION(orte_filem_rsh_work_pool_item_t); void orte_filem_rsh_work_pool_construct(orte_filem_rsh_work_pool_item_t *obj); void orte_filem_rsh_work_pool_destruct( orte_filem_rsh_work_pool_item_t *obj); OBJ_CLASS_INSTANCE(orte_filem_rsh_work_pool_item_t, opal_list_item_t, orte_filem_rsh_work_pool_construct, orte_filem_rsh_work_pool_destruct); void orte_filem_rsh_work_pool_construct(orte_filem_rsh_work_pool_item_t *obj) { obj->command = NULL; obj->request = NULL; obj->index = 0; OBJ_CONSTRUCT(&(obj->proc_set), orte_filem_base_process_set_t); OBJ_CONSTRUCT(&(obj->file_set), orte_filem_base_file_set_t); obj->active = false; } void orte_filem_rsh_work_pool_destruct( orte_filem_rsh_work_pool_item_t *obj) { if( NULL != obj->command ) { free(obj->command); obj->command = NULL; } if( NULL != obj->request ) { OBJ_RELEASE(obj->request); obj->request = NULL; } obj->index = 0; OBJ_DESTRUCT(&(obj->proc_set)); OBJ_DESTRUCT(&(obj->file_set)); obj->active = false; } /* * Rsh module */ static orte_filem_base_module_t loc_module = { /** Initialization Function */ orte_filem_rsh_module_init, /** Finalization Function */ orte_filem_rsh_module_finalize, orte_filem_rsh_put, orte_filem_rsh_put_nb, orte_filem_rsh_get, orte_filem_rsh_get_nb, orte_filem_rsh_rm, orte_filem_rsh_rm_nb, orte_filem_rsh_wait, orte_filem_rsh_wait_all }; /* * MCA Functions */ int orte_filem_rsh_component_query(mca_base_module_t **module, int *priority) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: component_query()")); *priority = mca_filem_rsh_component.super.priority; *module = (mca_base_module_t *)&loc_module; return ORTE_SUCCESS; } int orte_filem_rsh_module_init(void) { int ret; orte_filem_base_is_active = false; OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: module_init()")); /* * Allocate the work pools */ OBJ_CONSTRUCT(&work_pool_waiting, opal_list_t); OBJ_CONSTRUCT(&work_pool_held, opal_list_t); OBJ_CONSTRUCT(&work_pool_pending, opal_list_t); OBJ_CONSTRUCT(&work_pool_active, opal_list_t); OBJ_CONSTRUCT(&work_pool_lock, opal_mutex_t); OBJ_CONSTRUCT(&work_pool_cond, opal_condition_t); work_pool_all_done = false; /* * Start the listener for permission */ if( ORTE_SUCCESS != (ret = orte_filem_rsh_permission_listener_init(orte_filem_rsh_permission_callback) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh:init Failed to start listener\n"); return ret; } /* start the base receive */ if (ORTE_SUCCESS != (ret = orte_filem_base_comm_start())) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh:init Failed to start base receive\n"); return ret; } return ORTE_SUCCESS; } int orte_filem_rsh_module_finalize(void) { opal_list_item_t *item = NULL; OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: module_finalize()")); /* * Make sure all active requests are completed */ if( orte_filem_base_is_active ) { while(0 < opal_list_get_size(&work_pool_active) ) { opal_progress(); } } orte_filem_base_is_active = false; /* * Stop the listeners */ orte_filem_rsh_permission_listener_cancel(); /* * Stop the base receive */ orte_filem_base_comm_stop(); /* * Deallocate the work pools */ while( NULL != (item = opal_list_remove_first(&work_pool_waiting)) ) { OBJ_RELEASE(item); } OBJ_DESTRUCT(&work_pool_waiting); while( NULL != (item = opal_list_remove_first(&work_pool_held)) ) { OBJ_RELEASE(item); } OBJ_DESTRUCT(&work_pool_held); while( NULL != (item = opal_list_remove_first(&work_pool_pending)) ) { OBJ_RELEASE(item); } OBJ_DESTRUCT(&work_pool_pending); while( NULL != (item = opal_list_remove_first(&work_pool_active)) ) { OBJ_RELEASE(item); } OBJ_DESTRUCT(&work_pool_active); OBJ_DESTRUCT(&work_pool_lock); OBJ_DESTRUCT(&work_pool_cond); return ORTE_SUCCESS; } /****************** * Local functions ******************/ int orte_filem_rsh_put(orte_filem_base_request_t *request) { int ret, exit_status = ORTE_SUCCESS; orte_filem_base_is_active = true; if( ORTE_SUCCESS != (ret = orte_filem_base_prepare_request(request, ORTE_FILEM_MOVE_TYPE_PUT) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: put(): Failed to prepare the request structure (%d)", ret); exit_status = ret; goto cleanup; } if( ORTE_SUCCESS != (ret = orte_filem_rsh_start_copy(request) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: put(): Failed to post the request (%d)", ret); exit_status = ret; goto cleanup; } if( ORTE_SUCCESS != (ret = orte_filem_rsh_wait(request)) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: put(): Failed to wait on the request (%d)", ret); exit_status = ret; goto cleanup; } cleanup: if( 0 < opal_list_get_size(&work_pool_active) ) { orte_filem_base_is_active = true; } else { orte_filem_base_is_active = false; } return exit_status; } int orte_filem_rsh_put_nb(orte_filem_base_request_t *request) { int ret, exit_status = ORTE_SUCCESS; orte_filem_base_is_active = true; if( ORTE_SUCCESS != (ret = orte_filem_base_prepare_request(request, ORTE_FILEM_MOVE_TYPE_PUT) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: put(): Failed to prepare the request structure (%d)", ret); exit_status = ret; goto cleanup; } if( ORTE_SUCCESS != (ret = orte_filem_rsh_start_copy(request) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: put(): Failed to post the request (%d)", ret); exit_status = ret; goto cleanup; } cleanup: if( 0 < opal_list_get_size(&work_pool_active) ) { orte_filem_base_is_active = true; } else { orte_filem_base_is_active = false; } return exit_status; } int orte_filem_rsh_get(orte_filem_base_request_t *request) { int ret, exit_status = ORTE_SUCCESS; orte_filem_base_is_active = true; if( ORTE_SUCCESS != (ret = orte_filem_base_prepare_request(request, ORTE_FILEM_MOVE_TYPE_GET) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: get(): Failed to prepare the request structure (%d)", ret); exit_status = ret; goto cleanup; } if( ORTE_SUCCESS != (ret = orte_filem_rsh_start_copy(request) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: get(): Failed to post the request (%d)", ret); exit_status = ret; goto cleanup; } if( ORTE_SUCCESS != (ret = orte_filem_rsh_wait(request)) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: get(): Failed to wait on the request (%d)", ret); exit_status = ret; goto cleanup; } cleanup: if( 0 < opal_list_get_size(&work_pool_active) ) { orte_filem_base_is_active = true; } else { orte_filem_base_is_active = false; } return exit_status; } int orte_filem_rsh_get_nb(orte_filem_base_request_t *request) { int ret, exit_status = ORTE_SUCCESS; orte_filem_base_is_active = true; if( ORTE_SUCCESS != (ret = orte_filem_base_prepare_request(request, ORTE_FILEM_MOVE_TYPE_GET) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: get(): Failed to prepare the request structure (%d)", ret); exit_status = ret; goto cleanup; } if( ORTE_SUCCESS != (ret = orte_filem_rsh_start_copy(request) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: get(): Failed to post the request (%d)", ret); exit_status = ret; goto cleanup; } cleanup: if( 0 < opal_list_get_size(&work_pool_active) ) { orte_filem_base_is_active = true; } else { orte_filem_base_is_active = false; } return exit_status; } int orte_filem_rsh_rm(orte_filem_base_request_t *request) { int ret = ORTE_SUCCESS, exit_status = ORTE_SUCCESS; orte_filem_base_is_active = true; if( ORTE_SUCCESS != (ret = orte_filem_base_prepare_request(request, ORTE_FILEM_MOVE_TYPE_RM) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: rm(): Failed to prepare on the request (%d)", ret); exit_status = ret; goto cleanup; } if( ORTE_SUCCESS != (ret = orte_filem_rsh_start_rm(request) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: rm(): Failed to start the request (%d)", ret); exit_status = ret; goto cleanup; } if( ORTE_SUCCESS != (ret = orte_filem_rsh_wait(request)) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: rm(): Failed to wait on the request (%d)", ret); exit_status = ret; goto cleanup; } cleanup: if( 0 < opal_list_get_size(&work_pool_active) ) { orte_filem_base_is_active = true; } else { orte_filem_base_is_active = false; } return exit_status; } int orte_filem_rsh_rm_nb(orte_filem_base_request_t *request) { int ret = ORTE_SUCCESS, exit_status = ORTE_SUCCESS; orte_filem_base_is_active = true; if( ORTE_SUCCESS != (ret = orte_filem_base_prepare_request(request, ORTE_FILEM_MOVE_TYPE_RM) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: rm_nb(): Failed to prepare on the request (%d)", ret); exit_status = ret; goto cleanup; } if( ORTE_SUCCESS != (ret = orte_filem_rsh_start_rm(request) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: rm_nb(): Failed to start on the request (%d)", ret); exit_status = ret; goto cleanup; } cleanup: if( 0 < opal_list_get_size(&work_pool_active) ) { orte_filem_base_is_active = true; } else { orte_filem_base_is_active = false; } return exit_status; } int orte_filem_rsh_wait(orte_filem_base_request_t *request) { int exit_status = ORTE_SUCCESS; orte_filem_rsh_work_pool_item_t *wp_item = NULL; opal_list_item_t *item = NULL; int i; int num_finished = 0; bool found_match = false; /* * Stage 0: A pass to see if the request is already done */ for(i = 0; i < request->num_mv; ++i) { if( request->is_done[i] == true && request->is_active[i] == true) { ++num_finished; } } while(num_finished < request->num_mv ) { /* * Stage 1: Complete all active requests */ for(i = 0; i < request->num_mv; ++i) { /* If the child is still executing (active) then continue * checking other children */ if( request->is_done[i] == false && request->is_active[i] == true) { continue; } /* If the child is done executing (!active), but has not been * collected then do so */ else if( request->is_done[i] == true && request->is_active[i] == false) { /* * Find the reference in the active pool */ found_match = false; for (item = opal_list_get_first( &work_pool_active); item != opal_list_get_end( &work_pool_active); item = opal_list_get_next( item) ) { wp_item = (orte_filem_rsh_work_pool_item_t *)item; if(request == wp_item->request && i == wp_item->index) { found_match = true; break; } } /* If no match then assume on the pending list, and continue */ if( !found_match ) { continue; } OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: wait(): Transfer complete. Cleanup\n")); opal_list_remove_item(&work_pool_active, item); /* Mark as fully done [active = true, done = true] * It does not make complete sense to call this 'active' when * it is obviously not, but this is a state [true, true] that * should only be reached if the transfer has finished * completely. */ request->is_done[i] = true; request->is_active[i] = true; /* Tell peer we are finished with a send */ permission_send_done(&(wp_item->proc_set.source), 1, request->exit_status[i], wp_item->file_set.local_target, wp_item->file_set.remote_target, wp_item->command); OBJ_RELEASE(wp_item); wp_item = NULL; ++num_finished; } } /* * Wait for a child to complete */ if( num_finished < request->num_mv ) { OPAL_THREAD_LOCK(&work_pool_lock); opal_condition_wait(&work_pool_cond, &work_pool_lock); OPAL_THREAD_UNLOCK(&work_pool_lock); } } /* * Stage 2: Determine the return value */ for(i = 0; i < request->num_mv; ++i) { if( request->exit_status[i] < 0 ) { exit_status = request->exit_status[i]; } } if( 0 < opal_list_get_size(&work_pool_active) ) { orte_filem_base_is_active = true; } else { orte_filem_base_is_active = false; } return exit_status; } int orte_filem_rsh_wait_all(opal_list_t * request_list) { int ret = ORTE_SUCCESS, exit_status = ORTE_SUCCESS; opal_list_item_t *item = NULL; for (item = opal_list_get_first( request_list); item != opal_list_get_end( request_list); item = opal_list_get_next( item) ) { orte_filem_base_request_t *request = (orte_filem_base_request_t *) item; if( ORTE_SUCCESS != (ret = orte_filem_rsh_wait(request)) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: wait_all(): Wait failed (%d)", ret); exit_status = ret; goto cleanup; } } cleanup: if( 0 < opal_list_get_size(&work_pool_active) ) { orte_filem_base_is_active = true; } else { orte_filem_base_is_active = false; } return exit_status; } /************************** * Support functions **************************/ static int orte_filem_rsh_start_copy(orte_filem_base_request_t *request) { int ret = ORTE_SUCCESS, exit_status = ORTE_SUCCESS; opal_list_item_t *f_item = NULL; opal_list_item_t *p_item = NULL; char *remote_machine = NULL; char *remote_file = NULL; char *command = NULL; char *dir_arg = NULL; int cur_index = 0; /* For each file pair */ for (f_item = opal_list_get_first( &request->file_sets); f_item != opal_list_get_end( &request->file_sets); f_item = opal_list_get_next( f_item) ) { orte_filem_base_file_set_t * f_set = (orte_filem_base_file_set_t*)f_item; /* For each process set */ for (p_item = opal_list_get_first( &request->process_sets); p_item != opal_list_get_end( &request->process_sets); p_item = opal_list_get_next( p_item) ) { orte_filem_base_process_set_t * p_set = (orte_filem_base_process_set_t*)p_item; /* * If the source and sink are the same, then this is a local operation * Further if the files are the same, then nothing to do */ if (OPAL_EQUAL == orte_util_compare_name_fields(ORTE_NS_CMP_ALL, &p_set->source, &p_set->sink) ) { if( 0 == strncmp(f_set->local_target, f_set->remote_target, strlen(f_set->remote_target) ) ) { request->is_done[cur_index] = true; request->is_active[cur_index] = true; request->exit_status[cur_index] = 0; goto continue_set; } } if( request->movement_type == ORTE_FILEM_MOVE_TYPE_PUT ) { /* * The file should exist if we are going to put it somewhere else */ if( 0 != access(f_set->local_target, R_OK) ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): %s -> %s: Error: Cannot move file %s to %s. Does not exist at source\n", ORTE_NAME_PRINT(&p_set->source), ORTE_NAME_PRINT(&p_set->sink), f_set->local_target, f_set->remote_target)); orte_show_help("help-orte-filem-rsh.txt", "orte-filem-rsh:get-file-not-exist", true, f_set->local_target, orte_process_info.nodename); request->is_done[cur_index] = true; request->is_active[cur_index] = true; request->exit_status[cur_index] = -1; goto continue_set; } } /* Do not check a local get() operation, to help supress the warnings from the HNP */ else if (OPAL_EQUAL != orte_util_compare_name_fields(ORTE_NS_CMP_ALL, &p_set->source, &p_set->sink) ) { char *base = NULL; asprintf(&base, "%s/%s", f_set->local_target, opal_basename(f_set->remote_target)); /* * The file should not exist if we are getting a file with the * same name since we do not want to overwrite the filename * without the users consent. */ if( 0 == access(base, R_OK) ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): %s -> %s: Error: Cannot move file %s to %s. Already exists at destination (%s)\n", ORTE_NAME_PRINT(&p_set->source), ORTE_NAME_PRINT(&p_set->sink), f_set->remote_target, f_set->local_target, base)); orte_show_help("help-orte-filem-rsh.txt", "orte-filem-rsh:get-file-exists", true, f_set->local_target, orte_process_info.nodename); free(base); base = NULL; request->is_done[cur_index] = true; request->is_active[cur_index] = true; request->exit_status[cur_index] = -1; goto continue_set; } free(base); base = NULL; } if( request->movement_type == ORTE_FILEM_MOVE_TYPE_PUT ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): %s -> %s: Moving file %s %s to %s %s\n", ORTE_NAME_PRINT(&p_set->source), ORTE_NAME_PRINT(&p_set->sink), (f_set->local_hint == ORTE_FILEM_HINT_SHARED ? "(S)" : ""), f_set->local_target, (f_set->remote_hint == ORTE_FILEM_HINT_SHARED ? "(S)" : ""), f_set->remote_target)); } else { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): %s -> %s: Moving file %s %s to %s %s\n", ORTE_NAME_PRINT(&p_set->source), ORTE_NAME_PRINT(&p_set->sink), (f_set->remote_hint == ORTE_FILEM_HINT_SHARED ? "(S)" : ""), f_set->remote_target, (f_set->local_hint == ORTE_FILEM_HINT_SHARED ? "(S)" : ""), f_set->local_target)); } /* * Get the remote machine identifier from the process_name struct */ OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): %s -> %s: Get node name.\n", ORTE_NAME_PRINT(&p_set->source), ORTE_NAME_PRINT(&p_set->sink))); if( ORTE_SUCCESS != (ret = orte_filem_base_get_proc_node_name(&p_set->source, &remote_machine))) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): Get Node Name failed (%d)", ret); exit_status = ret; goto cleanup; } OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): %s -> %s: Got node name: %s\n", ORTE_NAME_PRINT(&p_set->source), ORTE_NAME_PRINT(&p_set->sink), remote_machine)); /* * Fix the remote_filename. * If it is an absolute path, then assume it is valid for the remote server * ow then we must construct the correct path. */ OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): %s -> %s: Query remote path (%s).\n", ORTE_NAME_PRINT(&p_set->source), ORTE_NAME_PRINT(&p_set->sink), f_set->remote_target)); remote_file = strdup(f_set->remote_target); if( ORTE_SUCCESS != (ret = orte_filem_rsh_query_remote_path(&remote_file, &p_set->source, &f_set->target_flag) ) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): Query Remote Path failed (%d)", ret); exit_status = ret; goto cleanup; } OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): %s -> %s: Remote path (%s) is (%s).\n", ORTE_NAME_PRINT(&p_set->source), ORTE_NAME_PRINT(&p_set->sink), f_set->remote_target, remote_file)); /* * Transfer the file or directory */ if(ORTE_FILEM_TYPE_DIR == f_set->target_flag) { dir_arg = strdup(" -r "); } else if(ORTE_FILEM_TYPE_UNKNOWN == f_set->target_flag) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: copy(): Error: File type unknown (%s)", f_set->remote_target); request->is_done[cur_index] = true; request->is_active[cur_index] = true; request->exit_status[cur_index] = -1; goto continue_set; } else { dir_arg = strdup(""); } /* * If this is the put() routine */ if( request->movement_type == ORTE_FILEM_MOVE_TYPE_PUT ) { /* Use a local 'cp' when able */ if(f_set->remote_hint == ORTE_FILEM_HINT_SHARED ) { asprintf(&command, "%s %s %s %s ", mca_filem_rsh_component.cp_local_command, dir_arg, f_set->local_target, remote_file); } else { asprintf(&command, "%s %s %s %s:%s ", mca_filem_rsh_component.cp_command, dir_arg, f_set->local_target, remote_machine, remote_file); } OPAL_OUTPUT_VERBOSE((17, mca_filem_rsh_component.super.output_handle, "filem:rsh:put about to execute [%s]", command)); if( ORTE_SUCCESS != (ret = orte_filem_rsh_start_command(p_set, f_set, command, request, cur_index)) ) { exit_status = ret; goto cleanup; } } /* * ow it is the get() routine */ else { /* Use a local 'cp' when able */ if(f_set->local_hint == ORTE_FILEM_HINT_SHARED ) { asprintf(&command, "%s %s %s %s %s %s ", mca_filem_rsh_component.remote_sh_command, remote_machine, mca_filem_rsh_component.cp_local_command, dir_arg, remote_file, f_set->local_target); } else { asprintf(&command, "%s %s %s:%s %s ", mca_filem_rsh_component.cp_command, dir_arg, remote_machine, remote_file, f_set->local_target); } OPAL_OUTPUT_VERBOSE((17, mca_filem_rsh_component.super.output_handle, "filem:rsh:get about to execute [%s]", command)); if( ORTE_SUCCESS != (ret = orte_filem_rsh_start_command(p_set, f_set, command, request, cur_index)) ) { exit_status = ret; goto cleanup; } } continue_set: /* A small bit of cleanup */ if( NULL != dir_arg) { free(dir_arg); dir_arg = NULL; } if( NULL != remote_file) { free(remote_file); remote_file = NULL; } if(NULL != remote_machine) { free(remote_machine); remote_machine = NULL; } ++cur_index; } /* For each process set */ } /* For each file pair */ cleanup: if( NULL != command ) free(command); if( NULL != remote_machine) free(remote_machine); if( NULL != dir_arg) free(dir_arg); if( NULL != remote_file) free(remote_file); return exit_status; } static int orte_filem_rsh_start_rm(orte_filem_base_request_t *request) { int ret = ORTE_SUCCESS, exit_status = ORTE_SUCCESS; opal_list_item_t *f_item = NULL; opal_list_item_t *p_item = NULL; char *command = NULL; char *remote_machine = NULL; char *remote_targets = NULL; char *remote_file = NULL; char *dir_arg = NULL; char **remote_file_set = NULL; int argc = 0; int cur_index = 0; /* For each process set */ for (p_item = opal_list_get_first( &request->process_sets); p_item != opal_list_get_end( &request->process_sets); p_item = opal_list_get_next( p_item) ) { orte_filem_base_process_set_t * p_set = (orte_filem_base_process_set_t*)p_item; /* * Get the remote machine identifier from the process_name struct */ if( ORTE_SUCCESS != (ret = orte_filem_base_get_proc_node_name(&p_set->source, &remote_machine))) { exit_status = ret; goto cleanup; } /* For each file pair */ for (f_item = opal_list_get_first( &request->file_sets); f_item != opal_list_get_end( &request->file_sets); f_item = opal_list_get_next( f_item) ) { orte_filem_base_file_set_t * f_set = (orte_filem_base_file_set_t*)f_item; /* * Fix the remote_filename. * If it is an absolute path, then assume it is valid for the remote server * ow then we must construct the correct path. */ remote_file = strdup(f_set->remote_target); if( ORTE_SUCCESS != (ret = orte_filem_rsh_query_remote_path(&remote_file, &p_set->source, &f_set->target_flag) ) ) { exit_status = ret; goto cleanup; } if(ORTE_FILEM_TYPE_UNKNOWN == f_set->target_flag) { continue; } opal_argv_append(&argc, &remote_file_set, remote_file); /* * If we are removing a directory in the mix, then we * need the recursive argument. */ if(NULL == dir_arg) { if(ORTE_FILEM_TYPE_DIR == f_set->target_flag) { dir_arg = strdup(" -rf "); } } } /* All File Pairs */ if(NULL == dir_arg) { dir_arg = strdup(" -f "); } remote_targets = opal_argv_join(remote_file_set, ' '); asprintf(&command, "%s %s rm %s %s ", mca_filem_rsh_component.remote_sh_command, remote_machine, dir_arg, remote_targets); OPAL_OUTPUT_VERBOSE((15, mca_filem_rsh_component.super.output_handle, "filem:rsh:rm about to execute [%s]", command)); if( ORTE_SUCCESS != (ret = orte_filem_rsh_start_command(p_set, NULL, command, request, cur_index)) ) { exit_status = ret; goto cleanup; } /* A small bit of cleanup */ if( NULL != dir_arg) { free(dir_arg); dir_arg = NULL; } if( NULL != remote_targets) { free(remote_targets); remote_targets = NULL; } if( NULL != remote_file_set) { opal_argv_free(remote_file_set); remote_file_set = NULL; } if(NULL != remote_machine) { free(remote_machine); remote_machine = NULL; } ++cur_index; } /* Process set */ cleanup: if( NULL != command ) free(command); if( NULL != remote_machine) free(remote_machine); if( NULL != dir_arg) free(dir_arg); if( NULL != remote_targets) free(remote_targets); if( NULL != remote_file_set) opal_argv_free(remote_file_set); return exit_status; } /****************** * Local Functions ******************/ /****************************** * Work Pool functions ******************************/ static int orte_filem_rsh_start_command(orte_filem_base_process_set_t *proc_set, orte_filem_base_file_set_t *file_set, char * command, orte_filem_base_request_t *request, int index) { orte_filem_rsh_work_pool_item_t *wp_item = NULL; int ret; /* Construct a work pool item */ wp_item = OBJ_NEW(orte_filem_rsh_work_pool_item_t); /* Copy the Process Set */ if( NULL != proc_set ) { wp_item->proc_set.source.jobid = proc_set->source.jobid; wp_item->proc_set.source.vpid = proc_set->source.vpid; wp_item->proc_set.sink.jobid = proc_set->sink.jobid; wp_item->proc_set.sink.vpid = proc_set->sink.vpid; } /* Copy the File Set */ if( NULL != file_set ) { wp_item->file_set.local_target = strdup(file_set->local_target); wp_item->file_set.remote_target = strdup(file_set->remote_target); wp_item->file_set.target_flag = file_set->target_flag; } OBJ_RETAIN(request); wp_item->command = strdup(command); wp_item->request = request; wp_item->index = index; if( orte_filem_rsh_max_outgoing > 0 && cur_num_outgoing >= orte_filem_rsh_max_outgoing ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: wait(): *** Hold send from proc %s (%d of %d)", ORTE_NAME_PRINT(&(wp_item->proc_set.source)), cur_num_outgoing, orte_filem_rsh_max_outgoing)); /* * - put the request on the held list, since we only allow 2 active filem ops at a time */ opal_list_append(&work_pool_held, &(wp_item->super)); } else { ++cur_num_outgoing; /* * - put the request on the pending list * - wait for the peer to tell us that it can receive */ opal_list_append(&work_pool_pending, &(wp_item->super)); /* * Ask for permission to send this file so we do not overwhelm the peer */ OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: start_command(): Ask permission to send from proc %s (%d of %d)", ORTE_NAME_PRINT(&(proc_set->source)), cur_num_outgoing, orte_filem_rsh_max_outgoing)); if( ORTE_SUCCESS != (ret = orte_filem_rsh_permission_ask(&(proc_set->source), 1)) ) { return ret; } } return ORTE_SUCCESS; } /****************************** * Child process start and wait functions ******************************/ static int start_child(char * command, orte_filem_base_request_t *request, int index) { char **argv = NULL; int status, ret; OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: start_child(): Starting the command [%s]", command)); /* fork() -> done = false, active = true */ request->is_done[index] = false; request->is_active[index] = true; request->exit_status[index] = fork(); if( request->exit_status[index] == 0 ) { /* Child */ /* Redirect stdout to /dev/null */ freopen( "/dev/null", "w", stdout); argv = opal_argv_split(command, ' '); status = execvp(argv[0], argv); opal_output(0, "filem:rsh:start_child Failed to exec child [%s] status = %d\n", command, status); exit(ORTE_ERROR); } else if( request->exit_status[index] > 0 ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: start_child(): Started Child %d Running command [%s]", request->exit_status[index], command)); /* * Register a callback for when this process exits */ if( ORTE_SUCCESS != (ret = orte_wait_cb(request->exit_status[index], filem_rsh_waitpid_cb, NULL) ) ) { opal_output(0, "filem:rsh: start_child(): Failed to register a waitpid callback for child [%d] executing the command [%s]\n", request->exit_status[index], command); return ret; } } else { return ORTE_ERROR; } return ORTE_SUCCESS; } static void filem_rsh_waitpid_cb(pid_t pid, int status, void* cbdata) { int ret; orte_filem_rsh_work_pool_item_t *wp_item = NULL; orte_filem_base_request_t *request; opal_list_item_t *item = NULL; int index; OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: waitpid_cb(): Pid %d finished with status [%d].\n", pid, status)); /* * Find this pid in the active queue */ OPAL_THREAD_LOCK(&work_pool_lock); for (item = opal_list_get_first( &work_pool_active); item != opal_list_get_end( &work_pool_active); item = opal_list_get_next( item) ) { wp_item = (orte_filem_rsh_work_pool_item_t *)item; request = wp_item->request; index = wp_item->index; if( request->is_done[index] == false && request->exit_status[index] == pid ) { request->exit_status[index] = status; /* waitpid() -> done = true, active = false */ request->is_done[index] = true; request->is_active[index] = false; OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: waitpid_cb(): Marked pid %d as complete [status = %d].\n", pid, status)); break; } } --cur_num_outgoing; /* * If we are holding any requests, start them */ if( opal_list_get_size(&work_pool_held) > 0 ) { item = opal_list_remove_first(&work_pool_held); wp_item = (orte_filem_rsh_work_pool_item_t *)item; ++cur_num_outgoing; /* * - put the request on the pending list * - wait for the peer to tell us that it can receive */ opal_list_append(&work_pool_pending, &(wp_item->super)); /* * Ask for permission to send this file so we do not overwhelm the peer */ OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: start_command(): Ask permission to send from proc %s (*** Activate Held)", ORTE_NAME_PRINT(&(wp_item->proc_set.source)))); if( ORTE_SUCCESS != (ret = orte_filem_rsh_permission_ask(&(wp_item->proc_set.source), 1)) ) { opal_output(0, "ERROR: Failed to ask permission!\n"); } } /* * Signal in case anyone is waiting for a child to finish. */ opal_condition_signal(&work_pool_cond); OPAL_THREAD_UNLOCK(&work_pool_lock); } /****************************** * Path resolution functions ******************************/ /* * This function is paired with the filem_base_process_get_remote_path_cmd() function on the remote machine */ static int orte_filem_rsh_query_remote_path(char **remote_ref, orte_process_name_t *peer, int *flag) { int ret; #if 0 /* An optimization if we are guarenteed that this remote files exists. * Then the 'scp -r' option will work with both files and directories. * JJH: For general correctness disable this piece of code. */ if( *remote_ref[0] == '/' ) { *flag = ORTE_FILEM_TYPE_DIR; return ORTE_SUCCESS; } #endif /* Call the base function */ if( ORTE_SUCCESS != (ret = orte_filem_base_get_remote_path(remote_ref, peer, flag) ) ) { return ret; } return ORTE_SUCCESS; } /****************************** * Permission functions ******************************/ static int orte_filem_rsh_permission_listener_init(orte_rml_buffer_callback_fn_t rml_cbfunc) { int ret; if( ORTE_SUCCESS != (ret = orte_rml.recv_buffer_nb(ORTE_NAME_WILDCARD, ORTE_RML_TAG_FILEM_RSH, ORTE_RML_PERSISTENT, rml_cbfunc, NULL)) ) { opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: listener_init: Failed to register the receive callback (%d)", ret); return ret; } return ORTE_SUCCESS; } static int orte_filem_rsh_permission_listener_cancel(void) { int ret; if( ORTE_SUCCESS != (ret = orte_rml.recv_cancel(ORTE_NAME_WILDCARD, ORTE_RML_TAG_FILEM_RSH) ) ) { #if 0 opal_output(mca_filem_rsh_component.super.output_handle, "filem:rsh: listener_cancel: Failed to deregister the receive callback (%d)", ret); #endif return ret; } return ORTE_SUCCESS; } static void orte_filem_rsh_permission_callback(int status, orte_process_name_t* sender, opal_buffer_t *buffer, orte_rml_tag_t tag, void* cbdata) { orte_filem_rsh_work_pool_item_t *wp_item = NULL; opal_list_item_t *item = NULL; int ret; orte_std_cntr_t n; int num_req, num_allowed = 0; int perm_flag, i; int32_t peer_status = 0; OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: permission_callback(? ?): Peer %s ...", ORTE_NAME_PRINT(sender))); /* * Receive the flag indicating if this is: * - Asking for permission (ORTE_FILEM_RSH_ASK) * - Allowing us to send (ORTE_FILEM_RSH_ALLOW) */ n = 1; if (ORTE_SUCCESS != (ret = opal_dss.unpack(buffer, &perm_flag, &n, OPAL_INT))) { goto cleanup; } /* Asking for permission to send */ if( ORTE_FILEM_RSH_ASK == perm_flag ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: permission_callback(ASK): Peer %s Asking permission to send [Used %d of %d]", ORTE_NAME_PRINT(sender), cur_num_incomming, orte_filem_rsh_max_incomming)); /* * Receive the requested amount */ n = 1; if (ORTE_SUCCESS != (ret = opal_dss.unpack(buffer, &num_req, &n, OPAL_INT))) { goto cleanup; } /* * Determine how many we can allow * if none then put a request on the waiting list * ow tell the peer to start sending now. * Send back number allowed to be started */ if( orte_filem_rsh_max_incomming > 0 && orte_filem_rsh_max_incomming < cur_num_incomming + 1) { /* Add to the waiting list */ OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: permission_callback(ASK): Add Peer %s request to waiting list", ORTE_NAME_PRINT(sender))); wp_item = OBJ_NEW(orte_filem_rsh_work_pool_item_t); wp_item->proc_set.source.jobid = sender->jobid; wp_item->proc_set.source.vpid = sender->vpid; opal_list_append(&work_pool_waiting, &(wp_item->super)); } /* Start the transfer immediately */ else { num_allowed = 1; cur_num_incomming += 1; OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: permission_callback(ASK): Respond to Peer %s with %d", ORTE_NAME_PRINT(sender), num_allowed)); permission_send_num_allowed(sender, num_allowed); } } /* Allowing us to start some number of sends */ else if( ORTE_FILEM_RSH_ALLOW == perm_flag ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: permission_callback(ALLOW): Peer %s Allowing me to send", ORTE_NAME_PRINT(sender))); /* * Receive the allowed transmit amount */ n = 1; if (ORTE_SUCCESS != (ret = opal_dss.unpack(buffer, &num_req, &n, OPAL_INT))) { goto cleanup; } /* * For each alloacted spot for transmit * - Get a pending request directed at this peer * - Start the pending request */ for(i = 0; i < num_req; ++i ) { if( 0 >= opal_list_get_size(&work_pool_pending) ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: permission_callback(ALLOW): No more pending sends to peer %s...", ORTE_NAME_PRINT(sender))); break; } for (item = opal_list_get_first( &work_pool_pending); item != opal_list_get_end( &work_pool_pending); item = opal_list_get_next( item) ) { wp_item = (orte_filem_rsh_work_pool_item_t *)item; if(sender->jobid == wp_item->proc_set.source.jobid && sender->vpid == wp_item->proc_set.source.vpid ) { opal_list_remove_item( &work_pool_pending, item); break; } } if( item == opal_list_get_end(&work_pool_pending) ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: permission_callback(ALLOW): Unable to find message on the pending list\n")); } OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: permission_callback(ALLOW): Starting to send to peer %s... (# pending = %d)", ORTE_NAME_PRINT(sender), (int)opal_list_get_size(&work_pool_pending))); wp_item->active = true; opal_list_append(&work_pool_active, &(wp_item->super)); if( ORTE_SUCCESS != (ret = start_child(wp_item->command, wp_item->request, wp_item->index)) ) { goto cleanup; } } } /* Peer said they are done sending one or more files */ else if( ORTE_FILEM_RSH_DONE == perm_flag ) { OPAL_OUTPUT_VERBOSE((10, mca_filem_rsh_component.super.output_handle, "filem:rsh: permission_callback(DONE): Peer %s is done sending to me", ORTE_NAME_PRINT(sender))); /* * Receive the number of open slots */ n = 1; if (ORTE_SUCCESS != (ret = opal_dss.unpack(buffer, &num_req, &n, OPAL_INT))) { goto cleanup; } cur_num_incomming -= num_req; /* * Receive the exit status */ n = 1; if (ORTE_SUCCESS != (ret = opal_dss.unpack(buffer, &peer_status, &n, OPAL_INT32))) { goto cleanup; } if( peer_status != 0 ) { char * local_target = NULL; char * remote_target = NULL; char * remote_cmd = NULL; n = 1; if (ORTE_SUCCESS != (ret = opal_dss.unpack(buffer, &local_target, &n, OPAL_STRING))) { ORTE_ERROR_LOG(ret); goto cleanup; } n = 1; if (ORTE_SUCCESS != (ret = opal_dss.unpack(buffer, &remote_target, &n, OPAL_STRING))) { ORTE_ERROR_LOG(ret); goto cleanup; } n = 1; if (ORTE_SUCCESS != (ret = opal_dss.unpack(buffer, &remote_cmd, &n, OPAL_STRING))) { ORTE_ERROR_LOG(ret); goto cleanup; } orte_show_help("help-orte-filem-rsh.txt", "orte-filem-rsh:remote-get-failed", true, ORTE_NAME_PRINT(sender), peer_status, local_target, remote_target, remote_cmd); free(local_target); free(remote_target); free(remote_cmd); } /* * For each open slot, notify a waiting peer that it may send */ for(i = 0; i < num_req; ++i ) { item = opal_list_get_first( &work_pool_waiting); if( item != opal_list_get_end( &work_pool_waiting) ) { wp_item = (orte_filem_rsh_work_pool_item_t *)item; num_allowed = 1; cur_num_incomming += 1; opal_list_remove_item(&work_pool_waiting, item); permission_send_num_allowed(&(wp_item->proc_set.source), num_allowed); OBJ_RELEASE(wp_item); } } } cleanup: return; } static int orte_filem_rsh_permission_ask(orte_process_name_t* source, int num_sends) { int ret, exit_status = ORTE_SUCCESS; opal_buffer_t loc_buffer; int perm_flag = ORTE_FILEM_RSH_ASK; OBJ_CONSTRUCT(&loc_buffer, opal_buffer_t); if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &perm_flag, 1, OPAL_INT))) { exit_status = ret; goto cleanup; } if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &num_sends, 1, OPAL_INT))) { exit_status = ret; goto cleanup; } if (0 > (ret = orte_rml.send_buffer(source, &loc_buffer, ORTE_RML_TAG_FILEM_RSH, 0))) { exit_status = ret; goto cleanup; } cleanup: OBJ_DESTRUCT(&loc_buffer); return exit_status; } static int permission_send_done(orte_process_name_t* peer, int num_avail, int32_t status, char * local_target, char * remote_target, char * command) { int ret, exit_status = ORTE_SUCCESS; opal_buffer_t loc_buffer; int perm_flag = ORTE_FILEM_RSH_DONE; OBJ_CONSTRUCT(&loc_buffer, opal_buffer_t); if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &perm_flag, 1, OPAL_INT))) { exit_status = ret; goto cleanup; } if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &num_avail, 1, OPAL_INT))) { exit_status = ret; goto cleanup; } if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &status, 1, OPAL_INT32))) { exit_status = ret; goto cleanup; } if( status != 0 ) { if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &local_target, 1, OPAL_STRING))) { exit_status = ret; goto cleanup; } if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &remote_target, 1, OPAL_STRING))) { exit_status = ret; goto cleanup; } if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &command, 1, OPAL_STRING))) { exit_status = ret; goto cleanup; } } if (0 > (ret = orte_rml.send_buffer(peer, &loc_buffer, ORTE_RML_TAG_FILEM_RSH, 0))) { exit_status = ret; goto cleanup; } cleanup: OBJ_DESTRUCT(&loc_buffer); return exit_status; } static int permission_send_num_allowed(orte_process_name_t* peer, int num_allowed) { int ret, exit_status = ORTE_SUCCESS; opal_buffer_t loc_buffer; int perm_flag = ORTE_FILEM_RSH_ALLOW; OBJ_CONSTRUCT(&loc_buffer, opal_buffer_t); if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &perm_flag, 1, OPAL_INT))) { exit_status = ret; goto cleanup; } if (ORTE_SUCCESS != (ret = opal_dss.pack(&loc_buffer, &num_allowed, 1, OPAL_INT))) { exit_status = ret; goto cleanup; } if (0 > (ret = orte_rml.send_buffer(peer, &loc_buffer, ORTE_RML_TAG_FILEM_RSH, 0))) { exit_status = ret; goto cleanup; } cleanup: OBJ_DESTRUCT(&loc_buffer); return exit_status; }