2014-12-30 13:56:05 +08:00
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
* Copyright by The HDF Group. *
|
|
|
|
* All rights reserved. *
|
|
|
|
* *
|
|
|
|
* This file is part of HDF5. The full HDF5 copyright notice, including *
|
|
|
|
* terms governing use, modification, and redistribution, is contained in *
|
2017-04-18 03:32:16 +08:00
|
|
|
* the COPYING file, which can be found at the root of the source code *
|
2021-02-17 22:52:36 +08:00
|
|
|
* distribution tree, or in https://www.hdfgroup.org/licenses. *
|
2017-04-18 03:32:16 +08:00
|
|
|
* If you do not have access to either file, you may request a copy from *
|
|
|
|
* help@hdfgroup.org. *
|
2014-12-30 13:56:05 +08:00
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
/* changes:
|
|
|
|
* rename pio_timer.c as io_timer.c;
|
2021-12-07 22:27:29 +08:00
|
|
|
* Removed pio_perf.h so that it is not dependent on it;
|
2014-12-30 13:56:05 +08:00
|
|
|
* Removed set_timer_type() and get_timer_type() since no one calls them;
|
|
|
|
* Merged sio_timer.c into io_timer.c;
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Purpose:
|
|
|
|
*
|
|
|
|
* This is a module of useful timing functions for performance testing.
|
|
|
|
*/
|
|
|
|
|
2015-01-06 05:47:34 +08:00
|
|
|
#include "H5private.h"
|
2014-12-30 13:56:05 +08:00
|
|
|
|
|
|
|
#include "io_timer.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The number to divide the tv_usec field with to get a nice decimal to add to
|
|
|
|
* the number of seconds.
|
|
|
|
*/
|
2020-09-30 22:27:10 +08:00
|
|
|
#define MICROSECOND 1000000.0F
|
2014-12-30 13:56:05 +08:00
|
|
|
|
|
|
|
/* global variables */
|
2020-09-30 22:27:10 +08:00
|
|
|
io_time_t *timer_g; /* timer: global for stub functions */
|
2014-12-30 13:56:05 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Function: sub_time
|
|
|
|
* Purpose: Struct two time values, and return the difference, in microseconds
|
|
|
|
*
|
|
|
|
* Note that the function assumes that a > b
|
|
|
|
* Programmer: Leon Arber, 1/27/06
|
|
|
|
*/
|
2020-09-30 22:27:10 +08:00
|
|
|
static double
|
|
|
|
sub_time(struct timeval *a, struct timeval *b)
|
2014-12-30 13:56:05 +08:00
|
|
|
{
|
2020-09-30 22:27:10 +08:00
|
|
|
return (((double)a->tv_sec + ((double)a->tv_usec) / (double)MICROSECOND) -
|
|
|
|
((double)b->tv_sec + ((double)b->tv_usec) / (double)MICROSECOND));
|
2014-12-30 13:56:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Function: io_time_new
|
|
|
|
* Purpose: Build us a brand, spankin', new performance time object.
|
|
|
|
* The object is a black box to the user. They just tell us
|
|
|
|
* what type of timer they want (MPI_CLOCK for MPI_Wtime or
|
|
|
|
* SYS_CLOCK for system time).
|
|
|
|
* Return: Pointer to io_time object
|
|
|
|
* Programmer: Bill Wendling, 01. October 2001
|
|
|
|
*/
|
|
|
|
io_time_t *
|
|
|
|
io_time_new(clock_type type)
|
|
|
|
{
|
2015-01-06 05:47:34 +08:00
|
|
|
io_time_t *pt = (io_time_t *)HDcalloc(1, sizeof(struct io_time_t));
|
2014-12-30 13:56:05 +08:00
|
|
|
|
|
|
|
/* set global timer variable */
|
|
|
|
timer_g = pt;
|
|
|
|
|
|
|
|
pt->type = type;
|
|
|
|
return pt;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Function: io_time_destroy
|
|
|
|
* Purpose: Remove the memory allocated for the io_time object. Only
|
|
|
|
* need to call on a pointer allocated with the ``io_time_new''
|
|
|
|
* function.
|
|
|
|
* Return: Nothing
|
|
|
|
* Programmer: Bill Wendling, 01. October 2001
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
io_time_destroy(io_time_t *pt)
|
|
|
|
{
|
|
|
|
HDfree(pt);
|
|
|
|
/* reset the global timer pointer too. */
|
|
|
|
timer_g = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* no one is calling set_timer_type or get_timer_type ???*/
|
|
|
|
/*
|
|
|
|
* Function: set_timer_type
|
|
|
|
* Purpose: Set the type of the timer to either MPI_CLOCK or SYS_CLOCK.
|
|
|
|
* This really only needs to be called if you didn't construct a
|
|
|
|
* timer with the pio_timer_new function (shame!).
|
|
|
|
* Return: Nothing
|
|
|
|
* Programmer: Bill Wendling, 04. October 2001
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
set_timer_type(io_time_t *pt, clock_type type)
|
|
|
|
{
|
|
|
|
pt->type = type;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Function: get_timer_type
|
|
|
|
* Purpose: Get the type of the timer.
|
|
|
|
* Return: MPI_CLOCK or SYS_CLOCK.
|
|
|
|
* Programmer: Bill Wendling, 04. October 2001
|
|
|
|
*/
|
|
|
|
clock_type
|
|
|
|
get_timer_type(io_time_t *pt)
|
|
|
|
{
|
|
|
|
return pt->type;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
2019-09-20 04:50:49 +08:00
|
|
|
* Function: io_time_set
|
2014-12-30 13:56:05 +08:00
|
|
|
* Purpose: Set the time in a ``io_time_t'' object.
|
|
|
|
* Return: Pointer to the passed in ``io_time_t'' object if SUCCEED; Null otherwise.
|
|
|
|
* Programmer: Bill Wendling, 01. October 2001
|
|
|
|
*/
|
|
|
|
io_time_t *
|
2019-09-20 04:50:49 +08:00
|
|
|
io_time_set(io_time_t *pt, timer_type t, int start_stop)
|
2014-12-30 13:56:05 +08:00
|
|
|
{
|
|
|
|
/* sanity check */
|
2019-09-20 04:50:49 +08:00
|
|
|
HDassert(pt);
|
2014-12-30 13:56:05 +08:00
|
|
|
|
2020-09-30 22:27:10 +08:00
|
|
|
switch (pt->type) {
|
2014-12-30 13:56:05 +08:00
|
|
|
#ifdef H5_HAVE_PARALLEL
|
2020-09-30 22:27:10 +08:00
|
|
|
case MPI_CLOCK:
|
|
|
|
if (start_stop == TSTART) {
|
|
|
|
pt->mpi_timer[t] = MPI_Wtime();
|
|
|
|
|
|
|
|
/* When we start the timer for HDF5_FINE_WRITE_FIXED_DIMS or HDF5_FINE_READ_FIXED_DIMS
|
|
|
|
* we compute the time it took to only open the file */
|
|
|
|
if (t == HDF5_FINE_WRITE_FIXED_DIMS)
|
|
|
|
pt->total_time[HDF5_FILE_WRITE_OPEN] +=
|
|
|
|
pt->mpi_timer[t] - pt->mpi_timer[HDF5_GROSS_WRITE_FIXED_DIMS];
|
|
|
|
else if (t == HDF5_FINE_READ_FIXED_DIMS)
|
|
|
|
pt->total_time[HDF5_FILE_READ_OPEN] +=
|
|
|
|
pt->mpi_timer[t] - pt->mpi_timer[HDF5_GROSS_READ_FIXED_DIMS];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
pt->total_time[t] += MPI_Wtime() - pt->mpi_timer[t];
|
|
|
|
pt->mpi_timer[t] = MPI_Wtime();
|
|
|
|
|
|
|
|
/* When we stop the timer for HDF5_GROSS_WRITE_FIXED_DIMS or HDF5_GROSS_READ_FIXED_DIMS
|
|
|
|
* we compute the time it took to close the file after the last read/write finished */
|
|
|
|
if (t == HDF5_GROSS_WRITE_FIXED_DIMS)
|
|
|
|
pt->total_time[HDF5_FILE_WRITE_CLOSE] +=
|
|
|
|
pt->mpi_timer[t] - pt->mpi_timer[HDF5_FINE_WRITE_FIXED_DIMS];
|
|
|
|
else if (t == HDF5_GROSS_READ_FIXED_DIMS)
|
|
|
|
pt->total_time[HDF5_FILE_READ_CLOSE] +=
|
|
|
|
pt->mpi_timer[t] - pt->mpi_timer[HDF5_FINE_READ_FIXED_DIMS];
|
|
|
|
}
|
|
|
|
break;
|
2015-09-01 03:04:23 +08:00
|
|
|
#else
|
2020-09-30 22:27:10 +08:00
|
|
|
case MPI_CLOCK:
|
|
|
|
HDfprintf(stderr, "MPI clock set in serial library\n");
|
|
|
|
return NULL;
|
2014-12-30 13:56:05 +08:00
|
|
|
#endif /* H5_HAVE_PARALLEL */
|
2020-09-30 22:27:10 +08:00
|
|
|
case SYS_CLOCK:
|
2014-12-30 13:56:05 +08:00
|
|
|
if (start_stop == TSTART) {
|
|
|
|
HDgettimeofday(&pt->sys_timer[t], NULL);
|
|
|
|
|
2020-09-30 22:27:10 +08:00
|
|
|
/* When we start the timer for HDF5_FINE_WRITE_FIXED_DIMS or HDF5_FINE_READ_FIXED_DIMS
|
|
|
|
* we compute the time it took to only open the file */
|
|
|
|
if (t == HDF5_FINE_WRITE_FIXED_DIMS)
|
|
|
|
pt->total_time[HDF5_FILE_WRITE_OPEN] +=
|
|
|
|
sub_time(&(pt->sys_timer[t]), &(pt->sys_timer[HDF5_GROSS_WRITE_FIXED_DIMS]));
|
|
|
|
else if (t == HDF5_FINE_READ_FIXED_DIMS)
|
|
|
|
pt->total_time[HDF5_FILE_READ_OPEN] +=
|
|
|
|
sub_time(&(pt->sys_timer[t]), &(pt->sys_timer[HDF5_GROSS_READ_FIXED_DIMS]));
|
|
|
|
}
|
|
|
|
else {
|
2014-12-30 13:56:05 +08:00
|
|
|
struct timeval sys_t;
|
|
|
|
|
|
|
|
HDgettimeofday(&sys_t, NULL);
|
|
|
|
pt->total_time[t] += sub_time(&sys_t, &(pt->sys_timer[t]));
|
|
|
|
|
2020-09-30 22:27:10 +08:00
|
|
|
/* When we stop the timer for HDF5_GROSS_WRITE_FIXED_DIMS or HDF5_GROSS_READ_FIXED_DIMS
|
|
|
|
* we compute the time it took to close the file after the last read/write finished */
|
|
|
|
if (t == HDF5_GROSS_WRITE_FIXED_DIMS)
|
|
|
|
pt->total_time[HDF5_FILE_WRITE_CLOSE] +=
|
|
|
|
sub_time(&(pt->sys_timer[t]), &(pt->sys_timer[HDF5_FINE_WRITE_FIXED_DIMS]));
|
|
|
|
else if (t == HDF5_GROSS_READ_FIXED_DIMS)
|
|
|
|
pt->total_time[HDF5_FILE_READ_CLOSE] +=
|
|
|
|
sub_time(&(pt->sys_timer[t]), &(pt->sys_timer[HDF5_FINE_READ_FIXED_DIMS]));
|
2014-12-30 13:56:05 +08:00
|
|
|
}
|
2020-09-30 22:27:10 +08:00
|
|
|
break;
|
2015-09-01 03:04:23 +08:00
|
|
|
|
2020-09-30 22:27:10 +08:00
|
|
|
default:
|
|
|
|
HDfprintf(stderr, "Unknown time clock type (%d)\n", pt->type);
|
|
|
|
return NULL;
|
2015-09-01 03:04:23 +08:00
|
|
|
} /* end switch */
|
2014-12-30 13:56:05 +08:00
|
|
|
|
2015-01-06 05:47:34 +08:00
|
|
|
#if 0
|
|
|
|
/* this does not belong here. Need fix in h5perf code when set_time() is called. -AKC- */
|
2014-12-30 13:56:05 +08:00
|
|
|
debug_start_stop_time(pt, t, start_stop);
|
2015-01-06 05:47:34 +08:00
|
|
|
#endif
|
2014-12-30 13:56:05 +08:00
|
|
|
|
|
|
|
return pt;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2019-09-20 04:50:49 +08:00
|
|
|
* Function: io_time_get
|
2014-12-30 13:56:05 +08:00
|
|
|
* Purpose: Get the time from a ``io_time_t'' object.
|
|
|
|
* Return: The number of seconds as a DOUBLE.
|
|
|
|
* Programmer: Bill Wendling, 01. October 2001
|
|
|
|
*/
|
2016-06-29 05:53:48 +08:00
|
|
|
H5_ATTR_PURE double
|
2019-09-20 04:50:49 +08:00
|
|
|
io_time_get(io_time_t *pt, timer_type t)
|
2014-12-30 13:56:05 +08:00
|
|
|
{
|
2019-09-20 04:50:49 +08:00
|
|
|
/* sanity check */
|
|
|
|
HDassert(pt);
|
|
|
|
|
2014-12-30 13:56:05 +08:00
|
|
|
return pt->total_time[t];
|
|
|
|
}
|