Blame test/mpi/manual/singjoin.c

Packit Service c5cf8c
/* -*- Mode: C; c-basic-offset:4 ; indent-tabs-mode:nil ; -*- */
Packit Service c5cf8c
/*
Packit Service c5cf8c
 *
Packit Service c5cf8c
 *  (C) 2006 by Argonne National Laboratory.
Packit Service c5cf8c
 *      See COPYRIGHT in top-level directory.
Packit Service c5cf8c
 */
Packit Service c5cf8c
Packit Service c5cf8c
/* Test of MPI_Comm_join.  This should work even when each process
Packit Service c5cf8c
 * is started as a singleton process.
Packit Service c5cf8c
 *
Packit Service c5cf8c
 * To run
Packit Service c5cf8c
 *
Packit Service c5cf8c
 * Start the server:
Packit Service c5cf8c
 *
Packit Service c5cf8c
 * ./singjoin -s -p 54321
Packit Service c5cf8c
 *
Packit Service c5cf8c
 * In a separate shell, start the client:
Packit Service c5cf8c
 *
Packit Service c5cf8c
 * ./singjoin -c -p 54321
Packit Service c5cf8c
 *
Packit Service c5cf8c
 * where "54321" is some available port number.
Packit Service c5cf8c
 */
Packit Service c5cf8c
Packit Service c5cf8c
/* Include mpitestconf to determine if we can use h_addr or need h_addr_list */
Packit Service c5cf8c
#include "mpitestconf.h"
Packit Service c5cf8c
#include <stdio.h>
Packit Service c5cf8c
#include <stdlib.h>
Packit Service c5cf8c
#include <string.h>
Packit Service c5cf8c
#ifndef HAVE_WINDOWS_H
Packit Service c5cf8c
#include <strings.h>
Packit Service c5cf8c
#include <unistd.h>
Packit Service c5cf8c
#include <sys/types.h>
Packit Service c5cf8c
#include <sys/socket.h>
Packit Service c5cf8c
#include <netinet/in.h>
Packit Service c5cf8c
#include <netdb.h>
Packit Service c5cf8c
#else
Packit Service c5cf8c
#include <windows.h>
Packit Service c5cf8c
#endif
Packit Service c5cf8c
#include <mpi.h>
Packit Service c5cf8c
Packit Service c5cf8c
#ifndef HAVE_WINDOWS_H
Packit Service c5cf8c
#define SOCKET_FD_TYPE int
Packit Service c5cf8c
#define INVALID_SOCKET_FD -1
Packit Service c5cf8c
#else
Packit Service c5cf8c
#define SOCKET_FD_TYPE  SOCKET
Packit Service c5cf8c
#define INVALID_SOCKET_FD INVALID_SOCKET
Packit Service c5cf8c
#define bcopy(s1, s2, n)    memmove(s2, s1, n)
Packit Service c5cf8c
#endif
Packit Service c5cf8c
Packit Service c5cf8c
Packit Service c5cf8c
int parse_args(int argc, char **argv);
Packit Service c5cf8c
void usage(char *name);
Packit Service c5cf8c
SOCKET_FD_TYPE server_routine(int portnum);
Packit Service c5cf8c
SOCKET_FD_TYPE client_routine(int portnum);
Packit Service c5cf8c
Packit Service c5cf8c
Packit Service c5cf8c
int is_server = 0;
Packit Service c5cf8c
int is_client = 0;
Packit Service c5cf8c
int opt_port = 0;
Packit Service c5cf8c
Packit Service c5cf8c
/* open a socket, establish a connection with a client, and return the fd */
Packit Service c5cf8c
SOCKET_FD_TYPE server_routine(int portnum)
Packit Service c5cf8c
{
Packit Service c5cf8c
    SOCKET_FD_TYPE listenfd, peer_fd;
Packit Service c5cf8c
    int ret, peer_addr_size;
Packit Service c5cf8c
    struct sockaddr_in server_addr, peer_addr;
Packit Service c5cf8c
Packit Service c5cf8c
    listenfd = socket(AF_INET, SOCK_STREAM, 0);
Packit Service c5cf8c
    if (listenfd == INVALID_SOCKET_FD) {
Packit Service c5cf8c
        perror("server socket");
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
Packit Service c5cf8c
    memset(&server_addr, 0, sizeof(server_addr));
Packit Service c5cf8c
    server_addr.sin_family = AF_INET;
Packit Service c5cf8c
    server_addr.sin_addr.s_addr = INADDR_ANY;
Packit Service c5cf8c
    server_addr.sin_port = htons(portnum);
Packit Service c5cf8c
    ret = bind(listenfd, (struct sockaddr *) &server_addr, sizeof(server_addr));
Packit Service c5cf8c
    if (ret < 0) {
Packit Service c5cf8c
        perror("server binding");
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
Packit Service c5cf8c
    ret = listen(listenfd, 1024);
Packit Service c5cf8c
    if (ret < 0) {
Packit Service c5cf8c
        perror("server listen");
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
Packit Service c5cf8c
    peer_addr_size = sizeof(peer_addr);
Packit Service c5cf8c
    peer_fd = accept(listenfd, (struct sockaddr *) &peer_addr, &peer_addr_size);
Packit Service c5cf8c
Packit Service c5cf8c
    if (peer_fd < 0) {
Packit Service c5cf8c
        perror("server accept");
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    return peer_fd;
Packit Service c5cf8c
}
Packit Service c5cf8c
Packit Service c5cf8c
/* open a socket, establish a connection with a server, and return the fd */
Packit Service c5cf8c
SOCKET_FD_TYPE client_routine(int portnum)
Packit Service c5cf8c
{
Packit Service c5cf8c
    SOCKET_FD_TYPE sockfd;
Packit Service c5cf8c
    struct sockaddr_in server_addr;
Packit Service c5cf8c
    struct hostent *server_ent;
Packit Service c5cf8c
    char hostname[MPI_MAX_PROCESSOR_NAME];
Packit Service c5cf8c
    int hostnamelen;
Packit Service c5cf8c
Packit Service c5cf8c
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
Packit Service c5cf8c
    if (sockfd == INVALID_SOCKET_FD) {
Packit Service c5cf8c
        perror("client socket");
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
Packit Service c5cf8c
    /* Try to get the processor name from MPI */
Packit Service c5cf8c
    MPI_Get_processor_name(hostname, &hostnamelen);
Packit Service c5cf8c
    server_ent = gethostbyname(hostname);
Packit Service c5cf8c
    if (server_ent == NULL) {
Packit Service c5cf8c
        fprintf(stderr, "gethostbyname fails\n");
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
Packit Service c5cf8c
    memset(&server_addr, 0, sizeof(server_addr));
Packit Service c5cf8c
    server_addr.sin_family = AF_INET;
Packit Service c5cf8c
    server_addr.sin_port = htons(portnum);
Packit Service c5cf8c
    /* POSIX might define h_addr_list only and not define h_addr */
Packit Service c5cf8c
#ifdef HAVE_H_ADDR_LIST
Packit Service c5cf8c
    bcopy(server_ent->h_addr_list[0], (char *) &server_addr.sin_addr.s_addr, server_ent->h_length);
Packit Service c5cf8c
#else
Packit Service c5cf8c
    bcopy(server_ent->h_addr, (char *) &server_addr.sin_addr.s_addr, server_ent->h_length);
Packit Service c5cf8c
#endif
Packit Service c5cf8c
Packit Service c5cf8c
    if (connect(sockfd, (struct sockaddr *) &server_addr, sizeof(server_addr)) < 0) {
Packit Service c5cf8c
        perror("client connect");
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    return sockfd;
Packit Service c5cf8c
}
Packit Service c5cf8c
Packit Service c5cf8c
void usage(char *name)
Packit Service c5cf8c
{
Packit Service c5cf8c
    fprintf(stderr, "usage: %s [-s|-c] -p <PORT>\n", name);
Packit Service c5cf8c
    fprintf(stderr, "      specify one and only one of -s or -c\n");
Packit Service c5cf8c
    fprintf(stderr, "      port is mandatory\n");
Packit Service c5cf8c
}
Packit Service c5cf8c
Packit Service c5cf8c
#ifdef HAVE_WINDOWS_H
Packit Service c5cf8c
static int get_option(int argc, char **argv, int arg_cnt)
Packit Service c5cf8c
{
Packit Service c5cf8c
    if (argv == NULL) {
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    if (*argv == NULL) {
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    if (arg_cnt < 0 || arg_cnt >= argc) {
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    if (strlen(argv[arg_cnt]) < 2) {
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    return (*(argv[arg_cnt] + 1));
Packit Service c5cf8c
}
Packit Service c5cf8c
#endif
Packit Service c5cf8c
Packit Service c5cf8c
int parse_args(int argc, char **argv)
Packit Service c5cf8c
{
Packit Service c5cf8c
#ifndef HAVE_WINDOWS_H
Packit Service c5cf8c
    int c;
Packit Service c5cf8c
    extern char *optarg;
Packit Service c5cf8c
    while ((c = getopt(argc, argv, "csp:")) != -1) {
Packit Service c5cf8c
        switch (c) {
Packit Service c5cf8c
            case 's':
Packit Service c5cf8c
                is_server = 1;
Packit Service c5cf8c
                break;
Packit Service c5cf8c
            case 'c':
Packit Service c5cf8c
                is_client = 1;
Packit Service c5cf8c
                break;
Packit Service c5cf8c
            case 'p':
Packit Service c5cf8c
                opt_port = atoi(optarg);
Packit Service c5cf8c
                break;
Packit Service c5cf8c
            case '?':
Packit Service c5cf8c
            case ':':
Packit Service c5cf8c
            default:
Packit Service c5cf8c
                usage(argv[0]);
Packit Service c5cf8c
                return -1;
Packit Service c5cf8c
        }
Packit Service c5cf8c
    }
Packit Service c5cf8c
    if ((is_client == 0) && (is_server == 0)) {
Packit Service c5cf8c
        usage(argv[0]);
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    if (opt_port == 0) {
Packit Service c5cf8c
        usage(argv[0]);
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    return 0;
Packit Service c5cf8c
#else
Packit Service c5cf8c
    int arg_cnt = 1;
Packit Service c5cf8c
    int c;
Packit Service c5cf8c
    while ((c = get_option(argc, argv, arg_cnt++)) > 0) {
Packit Service c5cf8c
        switch (c) {
Packit Service c5cf8c
            case 's':
Packit Service c5cf8c
                is_server = 1;
Packit Service c5cf8c
                break;
Packit Service c5cf8c
            case 'c':
Packit Service c5cf8c
                is_client = 1;
Packit Service c5cf8c
                break;
Packit Service c5cf8c
            case 'p':
Packit Service c5cf8c
                opt_port = atoi(argv[arg_cnt++]);
Packit Service c5cf8c
                break;
Packit Service c5cf8c
            default:
Packit Service c5cf8c
                usage(argv[0]);
Packit Service c5cf8c
                return -1;
Packit Service c5cf8c
        }
Packit Service c5cf8c
    }
Packit Service c5cf8c
    if ((is_client == 0) && (is_server == 0)) {
Packit Service c5cf8c
        usage(argv[0]);
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    if (opt_port == 0) {
Packit Service c5cf8c
        usage(argv[0]);
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
    return 0;
Packit Service c5cf8c
#endif
Packit Service c5cf8c
}
Packit Service c5cf8c
Packit Service c5cf8c
Packit Service c5cf8c
int main(int argc, char **argv)
Packit Service c5cf8c
{
Packit Service c5cf8c
    int rank, size;
Packit Service c5cf8c
    SOCKET_FD_TYPE fd = INVALID_SOCKET_FD;
Packit Service c5cf8c
    MPI_Comm intercomm, intracomm;
Packit Service c5cf8c
Packit Service c5cf8c
    MPI_Init(&argc, &argv);
Packit Service c5cf8c
    MPI_Comm_size(MPI_COMM_WORLD, &size);
Packit Service c5cf8c
    if (size != 1) {
Packit Service c5cf8c
        fprintf(stderr, "This test requires that only one process be in each comm_world\n");
Packit Service c5cf8c
        MPI_Abort(MPI_COMM_WORLD, 1);
Packit Service c5cf8c
    }
Packit Service c5cf8c
    if (parse_args(argc, argv) == -1) {
Packit Service c5cf8c
        fprintf(stderr, "Unable to parse the command line arguments\n");
Packit Service c5cf8c
        MPI_Abort(MPI_COMM_WORLD, 1);
Packit Service c5cf8c
    }
Packit Service c5cf8c
Packit Service c5cf8c
    if (is_server) {
Packit Service c5cf8c
        fd = server_routine(opt_port);
Packit Service c5cf8c
Packit Service c5cf8c
    } else if (is_client) {
Packit Service c5cf8c
        fd = client_routine(opt_port);
Packit Service c5cf8c
    }
Packit Service c5cf8c
Packit Service c5cf8c
    if (fd == INVALID_SOCKET_FD) {
Packit Service c5cf8c
        return -1;
Packit Service c5cf8c
    }
Packit Service c5cf8c
#ifdef SINGLETON_KICK
Packit Service c5cf8c
/* #warning isn't standard C, so we comment out this directive */
Packit Service c5cf8c
/* #warning  using singleton workaround */
Packit Service c5cf8c
    {
Packit Service c5cf8c
        int *usize, aflag;
Packit Service c5cf8c
        MPI_Comm_get_attr(MPI_COMM_WORLD, MPI_UNIVERSE_SIZE, &usize, &aflag);
Packit Service c5cf8c
    }
Packit Service c5cf8c
#endif
Packit Service c5cf8c
Packit Service c5cf8c
    MPI_Comm_join(fd, &intercomm);
Packit Service c5cf8c
Packit Service c5cf8c
    if (is_server) {
Packit Service c5cf8c
        MPI_Send(MPI_BOTTOM, 0, MPI_INT, 0, 0, intercomm);
Packit Service c5cf8c
    } else {
Packit Service c5cf8c
        MPI_Recv(MPI_BOTTOM, 0, MPI_INT, 0, 0, intercomm, MPI_STATUS_IGNORE);
Packit Service c5cf8c
        printf("Completed receive on intercomm\n");
Packit Service c5cf8c
        fflush(stdout);
Packit Service c5cf8c
    }
Packit Service c5cf8c
Packit Service c5cf8c
    MPI_Intercomm_merge(intercomm, 0, &intracomm);
Packit Service c5cf8c
    MPI_Comm_rank(intracomm, &rank;;
Packit Service c5cf8c
    MPI_Comm_size(intracomm, &size);
Packit Service c5cf8c
Packit Service c5cf8c
    printf("[%d/%d] after Intercomm_merge\n", rank, size);
Packit Service c5cf8c
Packit Service c5cf8c
    MPI_Comm_free(&intracomm);
Packit Service c5cf8c
    MPI_Comm_disconnect(&intercomm);
Packit Service c5cf8c
    MPI_Finalize();
Packit Service c5cf8c
    return 0;
Packit Service c5cf8c
}