1
/*****************************************************************************
3
Copyright (c) 1994, 2009, Innobase Oy. All Rights Reserved.
4
Copyright (c) 2009, Sun Microsystems, Inc.
6
Portions of this file contain modifications contributed and copyrighted by
7
Sun Microsystems, Inc. Those modifications are gratefully acknowledged and
8
are described briefly in the InnoDB documentation. The contributions by
9
Sun Microsystems are incorporated with their permission, and subject to the
10
conditions contained in the file COPYING.Sun_Microsystems.
12
This program is free software; you can redistribute it and/or modify it under
13
the terms of the GNU General Public License as published by the Free Software
14
Foundation; version 2 of the License.
16
This program is distributed in the hope that it will be useful, but WITHOUT
17
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
18
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
20
You should have received a copy of the GNU General Public License along with
21
this program; if not, write to the Free Software Foundation, Inc., 59 Temple
22
Place, Suite 330, Boston, MA 02111-1307 USA
24
*****************************************************************************/
26
/***************************************************************//**
28
Various utilities for Innobase.
30
Created 5/11/1994 Heikki Tuuri
31
********************************************************************/
43
#ifndef UNIV_HOTBACKUP
45
# if defined(BUILD_DRIZZLE)
46
# include "drizzled/common.h"
47
# if TIME_WITH_SYS_TIME
48
# include <sys/time.h>
52
# include <sys/time.h>
58
# include "ha_prototypes.h"
59
# include "mysql_com.h" /* NAME_LEN */
61
#endif /* UNIV_HOTBACKUP */
65
/** A constant to prevent the compiler from optimizing ut_delay() away. */
66
UNIV_INTERN ibool ut_always_false = FALSE;
69
/*****************************************************************//**
70
NOTE: The Windows epoch starts from 1601/01/01 whereas the Unix
71
epoch starts from 1970/1/1. For selection of constant see:
72
http://support.microsoft.com/kb/167296/ */
73
#define WIN_TO_UNIX_DELTA_USEC ((ib_int64_t) 11644473600000000ULL)
76
/*****************************************************************//**
77
This is the Windows version of gettimeofday(2).
78
@return 0 if all OK else -1 */
83
struct timeval* tv, /*!< out: Values are relative to Unix epoch */
84
void* tz) /*!< in: not used */
94
GetSystemTimeAsFileTime(&ft);
96
tm = (ib_int64_t) ft.dwHighDateTime << 32;
97
tm |= ft.dwLowDateTime;
99
ut_a(tm >= 0); /* If tm wraps over to negative, the quotient / 10
102
tm /= 10; /* Convert from 100 nsec periods to usec */
104
/* If we don't convert to the Unix epoch the value for
105
struct timeval::tv_sec will overflow.*/
106
tm -= WIN_TO_UNIX_DELTA_USEC;
108
tv->tv_sec = (long) (tm / 1000000L);
109
tv->tv_usec = (long) (tm % 1000000L);
114
/** An alias for gettimeofday(2). On Microsoft Windows, we have to
115
reimplement this function. */
116
#define ut_gettimeofday gettimeofday
119
/********************************************************//**
120
Gets the high 32 bits in a ulint. That is makes a shift >> 32,
121
but since there seem to be compiler bugs in both gcc and Visual C++,
122
we do this by a special conversion.
128
ulint a) /*!< in: ulint */
139
/**********************************************************//**
140
Returns system time. We do not specify the format of the time returned:
141
the only way to manipulate it is to use the function ut_difftime.
142
@return system time */
151
/**********************************************************//**
153
Upon successful completion, the value 0 is returned; otherwise the
154
value -1 is returned and the global variable errno is set to indicate the
156
@return 0 on success, -1 otherwise */
161
ulint* sec, /*!< out: seconds since the Epoch */
162
ulint* ms) /*!< out: microseconds since the Epoch+*sec */
166
int errno_gettimeofday;
169
for (i = 0; i < 10; i++) {
171
ret = ut_gettimeofday(&tv, NULL);
174
errno_gettimeofday = errno;
175
ut_print_timestamp(stderr);
176
fprintf(stderr, " InnoDB: gettimeofday(): %s\n",
177
strerror(errno_gettimeofday));
178
os_thread_sleep(100000); /* 0.1 sec */
179
errno = errno_gettimeofday;
186
*sec = (ulint) tv.tv_sec;
187
*ms = (ulint) tv.tv_usec;
193
/**********************************************************//**
194
Returns the number of microseconds since epoch. Similar to
195
time(3), the return value is also stored in *tloc, provided
196
that tloc is non-NULL.
197
@return us since epoch */
202
ullint* tloc) /*!< out: us since epoch, if non-NULL */
207
ut_gettimeofday(&tv, NULL);
209
us = (ullint) tv.tv_sec * 1000000 + tv.tv_usec;
218
/**********************************************************//**
219
Returns the difference of two times in seconds.
220
@return time2 - time1 expressed in seconds */
225
ib_time_t time2, /*!< in: time */
226
ib_time_t time1) /*!< in: time */
228
return(difftime(time2, time1));
231
/**********************************************************//**
232
Prints a timestamp to a file. */
237
FILE* file) /*!< in: file where to print */
242
GetLocalTime(&cal_tm);
244
fprintf(file,"%02d%02d%02d %2d:%02d:%02d",
245
(int)cal_tm.wYear % 100,
250
(int)cal_tm.wSecond);
253
struct tm* cal_tm_ptr;
258
#ifdef HAVE_LOCALTIME_R
259
localtime_r(&tm, &cal_tm);
260
cal_tm_ptr = &cal_tm;
262
cal_tm_ptr = localtime(&tm);
264
fprintf(file,"%02d%02d%02d %2d:%02d:%02d",
265
cal_tm_ptr->tm_year % 100,
266
cal_tm_ptr->tm_mon + 1,
274
/**********************************************************//**
275
Sprintfs a timestamp to a buffer, 13..14 chars plus terminating NUL. */
278
ut_sprintf_timestamp(
279
/*=================*/
280
char* buf) /*!< in: buffer where to sprintf */
285
GetLocalTime(&cal_tm);
287
sprintf(buf, "%02d%02d%02d %2d:%02d:%02d",
288
(int)cal_tm.wYear % 100,
293
(int)cal_tm.wSecond);
296
struct tm* cal_tm_ptr;
301
#ifdef HAVE_LOCALTIME_R
302
localtime_r(&tm, &cal_tm);
303
cal_tm_ptr = &cal_tm;
305
cal_tm_ptr = localtime(&tm);
307
sprintf(buf, "%02d%02d%02d %2d:%02d:%02d",
308
cal_tm_ptr->tm_year % 100,
309
cal_tm_ptr->tm_mon + 1,
317
#ifdef UNIV_HOTBACKUP
318
/**********************************************************//**
319
Sprintfs a timestamp to a buffer with no spaces and with ':' characters
323
ut_sprintf_timestamp_without_extra_chars(
324
/*=====================================*/
325
char* buf) /*!< in: buffer where to sprintf */
330
GetLocalTime(&cal_tm);
332
sprintf(buf, "%02d%02d%02d_%2d_%02d_%02d",
333
(int)cal_tm.wYear % 100,
338
(int)cal_tm.wSecond);
341
struct tm* cal_tm_ptr;
346
#ifdef HAVE_LOCALTIME_R
347
localtime_r(&tm, &cal_tm);
348
cal_tm_ptr = &cal_tm;
350
cal_tm_ptr = localtime(&tm);
352
sprintf(buf, "%02d%02d%02d_%2d_%02d_%02d",
353
cal_tm_ptr->tm_year % 100,
354
cal_tm_ptr->tm_mon + 1,
362
/**********************************************************//**
363
Returns current year, month, day. */
366
ut_get_year_month_day(
367
/*==================*/
368
ulint* year, /*!< out: current year */
369
ulint* month, /*!< out: month */
370
ulint* day) /*!< out: day */
375
GetLocalTime(&cal_tm);
377
*year = (ulint)cal_tm.wYear;
378
*month = (ulint)cal_tm.wMonth;
379
*day = (ulint)cal_tm.wDay;
382
struct tm* cal_tm_ptr;
387
#ifdef HAVE_LOCALTIME_R
388
localtime_r(&tm, &cal_tm);
389
cal_tm_ptr = &cal_tm;
391
cal_tm_ptr = localtime(&tm);
393
*year = (ulint)cal_tm_ptr->tm_year + 1900;
394
*month = (ulint)cal_tm_ptr->tm_mon + 1;
395
*day = (ulint)cal_tm_ptr->tm_mday;
398
#endif /* UNIV_HOTBACKUP */
400
#ifndef UNIV_HOTBACKUP
401
/*************************************************************//**
402
Runs an idle loop on CPU. The argument gives the desired delay
403
in microseconds on 100 MHz Pentium + Visual C++.
404
@return dummy value */
409
ulint delay) /*!< in: delay in microseconds on 100 MHz Pentium */
415
for (i = 0; i < delay * 50; i++) {
420
if (ut_always_false) {
421
ut_always_false = (ibool) j;
426
#endif /* !UNIV_HOTBACKUP */
428
/*************************************************************//**
429
Prints the contents of a memory buffer in hex and ascii. */
434
FILE* file, /*!< in: file where to print */
435
const void* buf, /*!< in: memory buffer */
436
ulint len) /*!< in: length of the buffer */
441
UNIV_MEM_ASSERT_RW(buf, len);
443
fprintf(file, " len %lu; hex ", len);
445
for (data = (const byte*)buf, i = 0; i < len; i++) {
446
fprintf(file, "%02lx", (ulong)*data++);
449
fputs("; asc ", file);
451
data = (const byte*)buf;
453
for (i = 0; i < len; i++) {
454
int c = (int) *data++;
455
putc(isprint(c) ? c : ' ', file);
461
/*************************************************************//**
462
Calculates fast the number rounded up to the nearest power of 2.
463
@return first power of 2 which is >= n */
468
ulint n) /*!< in: number != 0 */
483
/**********************************************************************//**
484
Outputs a NUL-terminated file name, quoted with apostrophes. */
489
FILE* f, /*!< in: output stream */
490
const char* name) /*!< in: name to print */
508
#ifndef UNIV_HOTBACKUP
509
/**********************************************************************//**
510
Outputs a fixed-length string, quoted as an SQL identifier.
511
If the string contains a slash '/', the string will be
512
output as two identifiers separated by a period (.),
513
as in SQL database_name.identifier. */
518
FILE* f, /*!< in: output stream */
519
trx_t* trx, /*!< in: transaction */
520
ibool table_id,/*!< in: TRUE=print a table name,
521
FALSE=print other identifier */
522
const char* name) /*!< in: name to print */
524
ut_print_namel(f, trx, table_id, name, strlen(name));
527
/**********************************************************************//**
528
Outputs a fixed-length string, quoted as an SQL identifier.
529
If the string contains a slash '/', the string will be
530
output as two identifiers separated by a period (.),
531
as in SQL database_name.identifier. */
536
FILE* f, /*!< in: output stream */
537
trx_t* trx, /*!< in: transaction (NULL=no quotes) */
538
ibool table_id,/*!< in: TRUE=print a table name,
539
FALSE=print other identifier */
540
const char* name, /*!< in: name to print */
541
ulint namelen)/*!< in: length of name */
543
/* 2 * NAME_LEN for database and table name,
544
and some slack for the #mysql50# prefix and quotes */
545
char buf[3 * NAME_LEN];
548
bufend = innobase_convert_name(buf, sizeof buf,
550
trx ? trx->mysql_thd : NULL,
553
ssize_t ret= fwrite(buf, 1, bufend - buf, f);
554
assert(ret==bufend-buf);
557
/**********************************************************************//**
563
FILE* dest, /*!< in: output file */
564
FILE* src) /*!< in: input file to be appended to output */
566
long len = ftell(src);
571
size_t maxs = len < (long) sizeof buf
574
size_t size = fread(buf, 1, maxs, src);
575
size_t ret= fwrite(buf, 1, size, dest);
583
#endif /* !UNIV_HOTBACKUP */
587
/**********************************************************************//**
588
A substitute for snprintf(3), formatted output conversion into
590
@return number of characters that would have been printed if the size
591
were unlimited, not including the terminating '\0'. */
596
char* str, /*!< out: string */
597
size_t size, /*!< in: str size */
598
const char* fmt, /*!< in: format */
599
...) /*!< in: format values */
608
res = _vscprintf(fmt, ap1);
612
_vsnprintf(str, size, fmt, ap2);
614
if ((size_t) res >= size) {
615
str[size - 1] = '\0';