File: get_elements.c

package info (click to toggle)
openmpi 1.6.5-9.1
  • links: PTS, VCS
  • area: main
  • in suites: jessie-kfreebsd
  • size: 91,652 kB
  • sloc: ansic: 408,966; cpp: 44,454; sh: 27,828; makefile: 10,486; asm: 3,882; python: 1,239; lex: 805; perl: 549; csh: 253; fortran: 232; f90: 126; tcl: 12
file content (120 lines) | stat: -rw-r--r-- 4,469 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
/*
 * Copyright (c) 2004-2007 The Trustees of Indiana University and Indiana
 *                         University Research and Technology
 *                         Corporation.  All rights reserved.
 * Copyright (c) 2004-2010 The University of Tennessee and The University
 *                         of Tennessee Research Foundation.  All rights
 *                         reserved.
 * Copyright (c) 2004-2008 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 "ompi_config.h"
#include <stdio.h>
#include <limits.h>

#include "ompi/mpi/c/bindings.h"
#include "ompi/runtime/params.h"
#include "ompi/communicator/communicator.h"
#include "ompi/errhandler/errhandler.h"
#include "ompi/datatype/ompi_datatype.h"
#include "ompi/memchecker.h"

#if OPAL_HAVE_WEAK_SYMBOLS && OMPI_PROFILING_DEFINES
#pragma weak MPI_Get_elements = PMPI_Get_elements
#endif

#if OMPI_PROFILING_DEFINES
#include "ompi/mpi/c/profile/defines.h"
#endif

static const char FUNC_NAME[] = "MPI_Get_elements";


int MPI_Get_elements(MPI_Status *status, MPI_Datatype datatype, int *count) 
{
    size_t size, internal_count;
    int i;

    OPAL_CR_NOOP_PROGRESS();

    MEMCHECKER(
               if (status != MPI_STATUSES_IGNORE) {
                   /*
                    * Before checking the complete status, we need to reset the definedness
                    * of the MPI_ERROR-field (single-completion calls wait/test).
                    */
                   opal_memchecker_base_mem_defined(&status->MPI_ERROR, sizeof(int));
                   memchecker_status(status);
                   memchecker_datatype(datatype);
               }
               );

    if (MPI_PARAM_CHECK) {
        int err = MPI_SUCCESS;
        OMPI_ERR_INIT_FINALIZE(FUNC_NAME);
        if (NULL == status || MPI_STATUSES_IGNORE == status || 
            MPI_STATUS_IGNORE == status || NULL == count) {
            err = MPI_ERR_ARG;
        } else if (NULL == datatype || MPI_DATATYPE_NULL == datatype) {
            err = MPI_ERR_TYPE;
        } else {
            OMPI_CHECK_DATATYPE_FOR_RECV(err, datatype, 1);
        }
        OMPI_ERRHANDLER_CHECK(err, MPI_COMM_WORLD, err, FUNC_NAME);
    }

    *count = 0;
    if( ompi_datatype_type_size( datatype, &size ) == MPI_SUCCESS ) {
        if( size == 0 ) {
            /* If the size of the datatype is zero let's return a count of zero */
            return MPI_SUCCESS;
        }
        internal_count = status->_ucount / size;    /* how many full types? */
        size = status->_ucount - internal_count * size;  /* leftover bytes */
        /* if basic type we should return the same result as MPI_Get_count */
        if( ompi_datatype_is_predefined(datatype) ) {
            if( size != 0 ) {  /* no leftover is supported for predefined types */
                *count = MPI_UNDEFINED;
            }
            goto more_than_int_elements;
        }
        if( internal_count != 0 ) {
            size_t total = 0;  /* count the basic elements in the datatype */
            for( i = 4; i < OPAL_DATATYPE_MAX_PREDEFINED; i++ ) {
                total += datatype->super.btypes[i];
            }
            internal_count = total * internal_count;
        }
        if( size > 0 ) {
            /* If there are any leftover bytes, compute the number of predefined
             * types in the datatype that can fit in these bytes.
             */
            if( (i = ompi_datatype_get_element_count( datatype, size )) != -1 )
                internal_count += i;
            else {
                *count = MPI_UNDEFINED;
                return MPI_SUCCESS;
            }
        }
        goto more_than_int_elements;
    }
    return OMPI_ERRHANDLER_INVOKE(MPI_COMM_WORLD, MPI_ERR_ARG, FUNC_NAME);
 more_than_int_elements:
    if( internal_count > ((size_t)INT_MAX) ) {
        /* We have more elements that we can represent with a signed int, and therefore
         * we're outside the standard here. I don't see what should we report back
         * here to make it useful. So, let's return an untouched *count and trigger
         * an MPI_ERR_TRUNCATE.
         */
        return MPI_ERR_TRUNCATE;
    }
    *count = (int)internal_count;
    return MPI_SUCCESS;
}