1
1
openmpi/orte/mca/filem/rsh/filem_rsh_module.c
George Bosilca f2a6b9394f Deal with the include spree. Protect "environ" on Windows.
Some others minors modifications in order to make it
compile [again] on Windows.

This commit was SVN r14188.
2007-04-01 16:16:54 +00:00

439 строки
14 KiB
C

/*
* Copyright (c) 2004-2007 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"
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif /* HAVE_UNISTD_H */
#include "orte/orte_constants.h"
#include "opal/mca/mca.h"
#include "opal/mca/base/base.h"
#include "opal/util/output.h"
#include "opal/mca/base/mca_base_param.h"
#include "opal/util/output.h"
#include "opal/util/show_help.h"
#include "opal/util/argv.h"
#include "opal/util/opal_environ.h"
#include "orte/mca/gpr/gpr.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_base_rsh_copy(orte_filem_base_request_t *request, bool put_func);
static int orte_filem_rsh_query_remote_path(char **remote_ref, orte_process_name_t *proc, int *flag);
static void orte_filem_rsh_query_callback(int status,
orte_process_name_t* sender,
orte_buffer_t *buffer,
orte_rml_tag_t tag,
void* cbdata);
/*
* 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_base_rsh_put,
orte_filem_base_rsh_get,
orte_filem_base_rsh_rm
};
/*
* MCA Functions
*/
orte_filem_base_module_1_0_0_t *
orte_filem_rsh_component_query(int *priority)
{
opal_output_verbose(10, mca_filem_rsh_component.super.output_handle,
"filem:rsh: component_query()");
*priority = mca_filem_rsh_component.super.priority;
return &loc_module;
}
int orte_filem_rsh_module_init(void)
{
int ret;
opal_output_verbose(10, mca_filem_rsh_component.super.output_handle,
"filem:rsh: module_init()");
if( ORTE_SUCCESS != (ret = orte_filem_base_listener_init(orte_filem_rsh_query_callback) ) ) {
return ret;
}
return ORTE_SUCCESS;
}
int orte_filem_rsh_module_finalize(void)
{
opal_output_verbose(10, mca_filem_rsh_component.super.output_handle,
"filem:rsh: module_finalize()");
orte_filem_base_listener_cancel();
return ORTE_SUCCESS;
}
/******************
* Local functions
******************/
int orte_filem_base_rsh_put(orte_filem_base_request_t *request)
{
int ret;
if( ORTE_SUCCESS != (ret = orte_filem_base_rsh_copy(request, true) ) ) {
return ret;
}
return ORTE_SUCCESS;
}
int orte_filem_base_rsh_get(orte_filem_base_request_t *request)
{
int ret;
if( ORTE_SUCCESS != (ret = orte_filem_base_rsh_copy(request, false) ) ) {
return ret;
}
return ORTE_SUCCESS;
}
static int orte_filem_base_rsh_copy(orte_filem_base_request_t *request, bool put_func) {
int ret = ORTE_SUCCESS, exit_status = ORTE_SUCCESS;
char *command = NULL;
char *remote_machine = NULL;
char *remote_file = NULL;
char *dir_arg = NULL;
int fn, pn;
/* For each process: */
for( pn = 0; pn < request->num_procs; ++pn) {
/*
* Get the remote machine identifier from the process_name struct
*/
if( ORTE_SUCCESS != (ret = orte_filem_base_get_proc_node_name(&request->proc_name[pn], &remote_machine))) {
exit_status = ret;
goto cleanup;
}
/* For each file: */
for( fn = 0; fn < request->num_targets; ++fn) {
/*
* 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(request->remote_targets[fn]);
if( ORTE_SUCCESS != (ret = orte_filem_rsh_query_remote_path(&remote_file, &request->proc_name[pn], &request->target_flags[fn]) ) ) {
exit_status = ret;
goto cleanup;
}
/*
* Transfer the file or directory
*/
if(ORTE_FILEM_TYPE_DIR == request->target_flags[fn]) {
dir_arg = strdup(" -r ");
}
else if(ORTE_FILEM_TYPE_UNKNOWN == request->target_flags[fn]) {
continue;
}
else {
dir_arg = strdup("");
}
/*
* If this is the put() routine
*/
if( put_func ) {
asprintf(&command, "%s %s %s %s:%s > /dev/null",
mca_filem_rsh_component.cp_command,
dir_arg,
request->local_targets[fn],
remote_machine,
remote_file);
opal_output_verbose(20, mca_filem_rsh_component.super.output_handle,
"filem:rsh:put about to execute %s", command);
if( 0 > (ret = system(command) ) ) {
exit_status = ret;
goto cleanup;
}
}
/*
* ow it is the get() routine
*/
else {
asprintf(&command, "%s %s %s:%s %s > /dev/null",
mca_filem_rsh_component.cp_command,
dir_arg,
remote_machine,
remote_file,
request->local_targets[fn]);
opal_output_verbose(20, mca_filem_rsh_component.super.output_handle,
"filem:rsh:get about to execute %s", command);
/* TBP: This needs to be fixed at some point. If scp can not authorize, it
* returns an error code of 256
* JJH: Implement a bound on the loop, but still would like something better.
*/
{
int wait_limit = 30;
int wait_i = 0;
while(256 == (ret = system(command) ) ) {
sleep(1);
++wait_i;
if(wait_i >= wait_limit) {
opal_output(mca_filem_rsh_component.super.output_handle,
"filem:rsh:get %s failed after %d attempts and returned %d",
mca_filem_rsh_component.cp_command,
wait_limit,
ret);
break;
}
}
}
if( 0 != ret) {
opal_output(mca_filem_rsh_component.super.output_handle,
"filem:rsh:get %s failed and returned %d",
mca_filem_rsh_component.cp_command,
ret);
exit_status = ret;
goto cleanup;
}
}
/* 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;
}
}
/* a small bit of cleanup */
if(NULL != remote_machine) {
free(remote_machine);
remote_machine = NULL;
}
}
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 ret;
}
int orte_filem_base_rsh_rm(orte_filem_base_request_t *request)
{
int ret = ORTE_SUCCESS, exit_status = ORTE_SUCCESS;
char *command = NULL;
char *remote_machine = NULL;
char *remote_targets = NULL;
char *dir_arg = NULL;
char **remote_file_set = NULL;
char *tmp_file = NULL;
int fn, pn, argc = 0;
/* For each process: */
for( pn = 0; pn < request->num_procs; ++pn) {
/*
* Get the remote machine identifier from the process_name struct
*/
if( ORTE_SUCCESS != (ret = orte_filem_base_get_proc_node_name(&request->proc_name[pn], &remote_machine))) {
exit_status = ret;
goto cleanup;
}
/* For each file: */
for( fn = 0; fn < request->num_targets; ++fn) {
tmp_file = strdup(request->remote_targets[fn]);
/*
* 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.
*/
if( ORTE_SUCCESS != (ret = orte_filem_rsh_query_remote_path(&tmp_file, &request->proc_name[pn], &request->target_flags[fn]) ) ) {
exit_status = ret;
goto cleanup;
}
if(ORTE_FILEM_TYPE_UNKNOWN == request->target_flags[fn]) {
continue;
}
opal_argv_append(&argc, &remote_file_set, tmp_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 == request->target_flags[fn]) {
dir_arg = strdup(" -rf ");
}
}
}
if(NULL == dir_arg)
dir_arg = strdup(" -f ");
remote_targets = opal_argv_join(remote_file_set, ' ');
asprintf(&command, "%s %s rm %s %s > /dev/null",
mca_filem_rsh_component.remote_sh_command,
remote_machine,
dir_arg,
remote_targets);
opal_output_verbose(20, mca_filem_rsh_component.super.output_handle,
"filem:rsh:rm about to execute %s", command);
if( 0 > (ret = system(command) ) ) {
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;
}
}
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);
if( NULL != tmp_file)
free(tmp_file);
return ret;
}
/******************
* Local Functions
******************/
/*
* This function is paired with the orte_filem_rsh_query_callback() 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 /* Some debugging */
/* If it is an absolute path */
if( *remote_ref[0] == '/' ) {
*flag = ORTE_FILEM_TYPE_DIR;
return ORTE_SUCCESS;
}
#endif
/* Put our listener on hold */
orte_filem_base_listener_cancel();
/* Call the base function */
if( ORTE_SUCCESS != (ret = orte_filem_base_query_remote_path(remote_ref, peer, flag) ) ) {
return ret;
}
/* Reset the listener */
if( ORTE_SUCCESS != (ret = orte_filem_base_listener_init(orte_filem_rsh_query_callback) ) ) {
return ret;
}
return ORTE_SUCCESS;
}
/*
* This function is paired with the orte_filem_rsh_query_remote_path() function on the
* requesting machine.
* This function is responsible for:
* - Constructing the remote absolute path for the specified file/dir
* - Verify the existence of the file/dir
* - Determine if the specified file/dir is in fact a file or dir or unknown if not found.
*
*/
static void orte_filem_rsh_query_callback(int status,
orte_process_name_t* peer,
orte_buffer_t *buffer,
orte_rml_tag_t tag,
void* cbdata)
{
opal_output_verbose(10, mca_filem_rsh_component.super.output_handle,
"filem:rsh: query_callback([%lu,%lu,%lu] -> [%lu,%lu,%lu])",
ORTE_NAME_ARGS(orte_process_info.my_name),
ORTE_NAME_ARGS(peer));
/* Call the base callback function */
orte_filem_base_query_callback(status, peer, buffer, tag, cbdata);
/* Reset the listener */
orte_filem_base_listener_init(orte_filem_rsh_query_callback);
}