15
15
Place, Suite 330, Boston, MA 02111-1307 USA
17
17
*****************************************************************************/
19
/******************************************************
18
/***********************************************************************
20
Copyright (c) 1995, 2009, Innobase Oy. All Rights Reserved.
21
Copyright (c) 2009, Percona Inc.
23
Portions of this file contain modifications contributed and copyrighted
24
by Percona Inc.. Those modifications are
25
gratefully acknowledged and are described briefly in the InnoDB
26
documentation. The contributions by Percona Inc. are incorporated with
27
their permission, and subject to the conditions contained in the file
30
This program is free software; you can redistribute it and/or modify it
31
under the terms of the GNU General Public License as published by the
32
Free Software Foundation; version 2 of the License.
34
This program is distributed in the hope that it will be useful, but
35
WITHOUT ANY WARRANTY; without even the implied warranty of
36
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
37
Public License for more details.
39
You should have received a copy of the GNU General Public License along
40
with this program; if not, write to the Free Software Foundation, Inc.,
41
59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
43
***********************************************************************/
45
/**************************************************//**
20
47
The interface to the operating system file i/o primitives
22
49
Created 10/21/1995 Heikki Tuuri
23
50
*******************************************************/
25
52
#include "os0file.h"
27
#include "os0thread.h"
28
53
#include "ut0mem.h"
29
54
#include "srv0srv.h"
30
55
#include "srv0start.h"
31
56
#include "fil0fil.h"
32
57
#include "buf0buf.h"
34
#if defined(UNIV_HOTBACKUP) && defined(__WIN__)
58
#ifndef UNIV_HOTBACKUP
60
# include "os0thread.h"
61
#else /* !UNIV_HOTBACKUP */
35
63
/* Add includes for the _stat() call to compile on Windows */
36
#include <sys/types.h>
39
#endif /* UNIV_HOTBACKUP */
64
# include <sys/types.h>
65
# include <sys/stat.h>
68
#endif /* !UNIV_HOTBACKUP */
41
70
/* This specifies the file permissions InnoDB uses when it creates files in
42
71
Unix; the value of os_innodb_umask is initialized in ha_innodb.cc to
75
/** Umask for creating files */
46
76
UNIV_INTERN ulint os_innodb_umask
47
77
= S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP;
79
/** Umask for creating files */
49
80
UNIV_INTERN ulint os_innodb_umask = 0;
65
97
/* In simulated aio, merge at most this many consecutive i/os */
66
98
#define OS_AIO_MERGE_N_CONSECUTIVE 64
68
/* If this flag is TRUE, then we will use the native aio of the
100
/** If this flag is TRUE, then we will use the native aio of the
69
101
OS (provided we compiled Innobase with it in), otherwise we will
70
102
use simulated aio we build below with threads */
72
104
UNIV_INTERN ibool os_aio_use_native_aio = FALSE;
106
/** Flag: enable debug printout for asynchronous i/o */
74
107
UNIV_INTERN ibool os_aio_print_debug = FALSE;
76
/* The aio array slot structure */
109
/** The asynchronous i/o array slot structure */
77
110
typedef struct os_aio_slot_struct os_aio_slot_t;
112
/** The asynchronous i/o array slot structure */
79
113
struct os_aio_slot_struct{
80
ibool is_read; /* TRUE if a read operation */
81
ulint pos; /* index of the slot in the aio
114
ibool is_read; /*!< TRUE if a read operation */
115
ulint pos; /*!< index of the slot in the aio
83
ibool reserved; /* TRUE if this slot is reserved */
84
time_t reservation_time;/* time when reserved */
85
ulint len; /* length of the block to read or
117
ibool reserved; /*!< TRUE if this slot is reserved */
118
time_t reservation_time;/*!< time when reserved */
119
ulint len; /*!< length of the block to read or
87
byte* buf; /* buffer used in i/o */
88
ulint type; /* OS_FILE_READ or OS_FILE_WRITE */
89
ulint offset; /* 32 low bits of file offset in
121
byte* buf; /*!< buffer used in i/o */
122
ulint type; /*!< OS_FILE_READ or OS_FILE_WRITE */
123
ulint offset; /*!< 32 low bits of file offset in
91
ulint offset_high; /* 32 high bits of file offset */
92
os_file_t file; /* file where to read or write */
93
const char* name; /* file name or path */
94
ibool io_already_done;/* used only in simulated aio:
125
ulint offset_high; /*!< 32 high bits of file offset */
126
os_file_t file; /*!< file where to read or write */
127
const char* name; /*!< file name or path */
128
ibool io_already_done;/*!< used only in simulated aio:
95
129
TRUE if the physical i/o already
96
130
made and only the slot message
97
131
needs to be passed to the caller
98
132
of os_aio_simulated_handle */
99
fil_node_t* message1; /* message which is given by the */
100
void* message2; /* the requester of an aio operation
133
fil_node_t* message1; /*!< message which is given by the */
134
void* message2; /*!< the requester of an aio operation
101
135
and which can be used to identify
102
136
which pending aio operation was
104
138
#ifdef WIN_ASYNC_IO
105
os_event_t event; /* event object we need in the
139
os_event_t event; /*!< event object we need in the
106
140
OVERLAPPED struct */
107
OVERLAPPED control; /* Windows control block for the
141
OVERLAPPED control; /*!< Windows control block for the
112
/* The aio array structure */
146
/** The asynchronous i/o array structure */
113
147
typedef struct os_aio_array_struct os_aio_array_t;
149
/** The asynchronous i/o array structure */
115
150
struct os_aio_array_struct{
116
os_mutex_t mutex; /* the mutex protecting the aio array */
117
os_event_t not_full; /* The event which is set to the signaled
118
state when there is space in the aio
119
outside the ibuf segment */
120
os_event_t is_empty; /* The event which is set to the signaled
121
state when there are no pending i/os
123
ulint n_slots; /* Total number of slots in the aio array.
124
This must be divisible by n_threads. */
125
ulint n_segments;/* Number of segments in the aio array of
126
pending aio requests. A thread can wait
127
separately for any one of the segments. */
128
ulint n_reserved;/* Number of reserved slots in the
129
aio array outside the ibuf segment */
130
os_aio_slot_t* slots; /* Pointer to the slots in the array */
151
os_mutex_t mutex; /*!< the mutex protecting the aio array */
153
/*!< The event which is set to the
154
signaled state when there is space in
155
the aio outside the ibuf segment */
157
/*!< The event which is set to the
158
signaled state when there are no
159
pending i/os in this array */
160
ulint n_slots;/*!< Total number of slots in the aio
161
array. This must be divisible by
164
/*!< Number of segments in the aio
165
array of pending aio requests. A
166
thread can wait separately for any one
169
/*!< Number of reserved slots in the
170
aio array outside the ibuf segment */
171
os_aio_slot_t* slots; /*!< Pointer to the slots in the array */
132
173
os_native_event_t* native_events;
133
/* Pointer to an array of OS native event
134
handles where we copied the handles from
135
slots, in the same order. This can be used
136
in WaitForMultipleObjects; used only in
174
/*!< Pointer to an array of OS native
175
event handles where we copied the
176
handles from slots, in the same
177
order. This can be used in
178
WaitForMultipleObjects; used only in
141
/* Array of events used in simulated aio */
183
/** Array of events used in simulated aio */
142
184
static os_event_t* os_aio_segment_wait_events = NULL;
144
/* The aio arrays for non-ibuf i/o and ibuf i/o, as well as sync aio. These
145
are NULL when the module has not yet been initialized. */
146
static os_aio_array_t* os_aio_read_array = NULL;
147
static os_aio_array_t* os_aio_write_array = NULL;
148
static os_aio_array_t* os_aio_ibuf_array = NULL;
149
static os_aio_array_t* os_aio_log_array = NULL;
150
static os_aio_array_t* os_aio_sync_array = NULL;
186
/** The aio arrays for non-ibuf i/o and ibuf i/o, as well as sync aio. These
187
are NULL when the module has not yet been initialized. @{ */
188
static os_aio_array_t* os_aio_read_array = NULL; /*!< Reads */
189
static os_aio_array_t* os_aio_write_array = NULL; /*!< Writes */
190
static os_aio_array_t* os_aio_ibuf_array = NULL; /*!< Insert buffer */
191
static os_aio_array_t* os_aio_log_array = NULL; /*!< Redo log */
192
static os_aio_array_t* os_aio_sync_array = NULL; /*!< Synchronous I/O */
195
/** Number of asynchronous I/O segments. Set by os_aio_init(). */
152
196
static ulint os_aio_n_segments = ULINT_UNDEFINED;
154
/* If the following is TRUE, read i/o handler threads try to
198
/** If the following is TRUE, read i/o handler threads try to
155
199
wait until a batch of new read requests have been posted */
156
200
static ibool os_aio_recommend_sleep_for_read_threads = FALSE;
201
#endif /* !UNIV_HOTBACKUP */
158
203
UNIV_INTERN ulint os_n_file_reads = 0;
159
204
UNIV_INTERN ulint os_bytes_read_since_printout = 0;
428
/********************************************************************
429
Does error handling when a file operation fails. */
477
/****************************************************************//**
478
Does error handling when a file operation fails.
479
@return TRUE if we should retry the operation */
432
482
os_file_handle_error(
433
483
/*=================*/
434
/* out: TRUE if we should retry the
436
const char* name, /* in: name of a file or NULL */
437
const char* operation)/* in: operation */
484
const char* name, /*!< in: name of a file or NULL */
485
const char* operation)/*!< in: operation */
439
487
/* exit in case of unknown error */
440
488
return(os_file_handle_error_cond_exit(name, operation, TRUE));
443
/********************************************************************
444
Does error handling when a file operation fails. */
491
/****************************************************************//**
492
Does error handling when a file operation fails.
493
@return TRUE if we should retry the operation */
447
496
os_file_handle_error_no_exit(
448
497
/*=========================*/
449
/* out: TRUE if we should retry the
451
const char* name, /* in: name of a file or NULL */
452
const char* operation)/* in: operation */
498
const char* name, /*!< in: name of a file or NULL */
499
const char* operation)/*!< in: operation */
454
501
/* don't exit in case of unknown error */
455
502
return(os_file_handle_error_cond_exit(name, operation, FALSE));
516
/***************************************************************************
564
/***********************************************************************//**
517
565
Creates a temporary file. This function is like tmpfile(3), but
518
566
the temporary file is created in the MySQL temporary directory.
519
567
On Netware, this function is like tmpfile(3), because the C run-time
520
library of Netware does not expose the delete-on-close flag. */
568
library of Netware does not expose the delete-on-close flag.
569
@return temporary file handle, or NULL on error */
523
572
os_file_create_tmpfile(void)
524
573
/*========================*/
525
/* out: temporary file handle, or NULL on error */
527
#ifdef UNIV_HOTBACKUP
533
576
FILE* file = tmpfile();
534
# else /* __NETWARE__ */
577
#else /* __NETWARE__ */
535
578
FILE* file = NULL;
536
579
int fd = innobase_mysql_tmpfile();
539
582
file = fdopen(fd, "w+b");
541
# endif /* __NETWARE__ */
584
#endif /* __NETWARE__ */
544
587
ut_print_timestamp(stderr);
546
589
" InnoDB: Error: unable to create temporary file;"
547
590
" errno: %d\n", errno);
552
# endif /* !__NETWARE__ */
595
#endif /* !__NETWARE__ */
556
#endif /* UNIV_HOTBACKUP */
600
#endif /* !UNIV_HOTBACKUP */
559
/***************************************************************************
602
/***********************************************************************//**
560
603
The os_file_opendir() function opens a directory stream corresponding to the
561
604
directory named by the dirname argument. The directory stream is positioned
562
605
at the first entry. In both Unix and Windows we automatically skip the '.'
563
and '..' items at the start of the directory listing. */
606
and '..' items at the start of the directory listing.
607
@return directory stream, NULL if error */
568
/* out: directory stream, NULL if
570
const char* dirname, /* in: directory name; it must not
612
const char* dirname, /*!< in: directory name; it must not
571
613
contain a trailing '\' or '/' */
572
ibool error_is_fatal) /* in: TRUE if we should treat an
614
ibool error_is_fatal) /*!< in: TRUE if we should treat an
573
615
error as a fatal error; if we try to
574
616
open symlinks then we do not wish a
575
617
fatal error if it happens not to be
851
/********************************************************************
852
A simple function to open or create a file. */
890
/****************************************************************//**
891
A simple function to open or create a file.
892
@return own: handle to the file, not defined if error, error number
893
can be retrieved with os_file_get_last_error */
855
896
os_file_create_simple(
856
897
/*==================*/
857
/* out, own: handle to the file, not defined
858
if error, error number can be retrieved with
859
os_file_get_last_error */
860
const char* name, /* in: name of the file or path as a
898
const char* name, /*!< in: name of the file or path as a
861
899
null-terminated string */
862
ulint create_mode,/* in: OS_FILE_OPEN if an existing file is
900
ulint create_mode,/*!< in: OS_FILE_OPEN if an existing file is
863
901
opened (if does not exist, error), or
864
902
OS_FILE_CREATE if a new file is created
865
903
(if exists, error), or
866
904
OS_FILE_CREATE_PATH if new file
867
905
(if exists, error) and subdirectories along
868
906
its path are created (if needed)*/
869
ulint access_type,/* in: OS_FILE_READ_ONLY or
907
ulint access_type,/*!< in: OS_FILE_READ_ONLY or
870
908
OS_FILE_READ_WRITE */
871
ibool* success)/* out: TRUE if succeed, FALSE if error */
909
ibool* success)/*!< out: TRUE if succeed, FALSE if error */
990
1028
#endif /* __WIN__ */
993
/********************************************************************
994
A simple function to open or create a file. */
1031
/****************************************************************//**
1032
A simple function to open or create a file.
1033
@return own: handle to the file, not defined if error, error number
1034
can be retrieved with os_file_get_last_error */
997
1037
os_file_create_simple_no_error_handling(
998
1038
/*====================================*/
999
/* out, own: handle to the file, not defined
1000
if error, error number can be retrieved with
1001
os_file_get_last_error */
1002
const char* name, /* in: name of the file or path as a
1039
const char* name, /*!< in: name of the file or path as a
1003
1040
null-terminated string */
1004
ulint create_mode,/* in: OS_FILE_OPEN if an existing file
1041
ulint create_mode,/*!< in: OS_FILE_OPEN if an existing file
1005
1042
is opened (if does not exist, error), or
1006
1043
OS_FILE_CREATE if a new file is created
1007
1044
(if exists, error) */
1008
ulint access_type,/* in: OS_FILE_READ_ONLY,
1045
ulint access_type,/*!< in: OS_FILE_READ_ONLY,
1009
1046
OS_FILE_READ_WRITE, or
1010
1047
OS_FILE_READ_ALLOW_DELETE; the last option is
1011
1048
used by a backup program reading the file */
1012
ibool* success)/* out: TRUE if succeed, FALSE if error */
1049
ibool* success)/*!< out: TRUE if succeed, FALSE if error */
1015
1052
os_file_t file;
2286
/***********************************************************************
2323
/*******************************************************************//**
2287
2324
Requests a synchronous positioned read operation. This function does not do
2288
any error handling. In case of error it returns FALSE. */
2325
any error handling. In case of error it returns FALSE.
2326
@return TRUE if request was successful, FALSE if fail */
2291
2329
os_file_read_no_error_handling(
2292
2330
/*===========================*/
2293
/* out: TRUE if request was
2294
successful, FALSE if fail */
2295
os_file_t file, /* in: handle to a file */
2296
void* buf, /* in: buffer where to read */
2297
ulint offset, /* in: least significant 32 bits of file
2331
os_file_t file, /*!< in: handle to a file */
2332
void* buf, /*!< in: buffer where to read */
2333
ulint offset, /*!< in: least significant 32 bits of file
2298
2334
offset where to read */
2299
ulint offset_high, /* in: most significant 32 bits of
2335
ulint offset_high, /*!< in: most significant 32 bits of
2301
ulint n) /* in: number of bytes to read */
2337
ulint n) /*!< in: number of bytes to read */
2404
2440
str[flen] = '\0';
2407
/***********************************************************************
2408
Requests a synchronous write operation. */
2443
/*******************************************************************//**
2444
Requests a synchronous write operation.
2445
@return TRUE if request was successful, FALSE if fail */
2413
/* out: TRUE if request was
2414
successful, FALSE if fail */
2415
const char* name, /* in: name of the file or path as a
2450
const char* name, /*!< in: name of the file or path as a
2416
2451
null-terminated string */
2417
os_file_t file, /* in: handle to a file */
2418
const void* buf, /* in: buffer from which to write */
2419
ulint offset, /* in: least significant 32 bits of file
2452
os_file_t file, /*!< in: handle to a file */
2453
const void* buf, /*!< in: buffer from which to write */
2454
ulint offset, /*!< in: least significant 32 bits of file
2420
2455
offset where to write */
2421
ulint offset_high, /* in: most significant 32 bits of
2456
ulint offset_high, /*!< in: most significant 32 bits of
2423
ulint n) /* in: number of bytes to write */
2458
ulint n) /*!< in: number of bytes to write */
2852
2881
return(success);
2855
/********************************************************************
2856
Returns a pointer to the nth slot in the aio array. */
2884
#ifndef UNIV_HOTBACKUP
2885
/****************************************************************//**
2886
Returns a pointer to the nth slot in the aio array.
2887
@return pointer to slot */
2859
2890
os_aio_array_get_nth_slot(
2860
2891
/*======================*/
2861
/* out: pointer to slot */
2862
os_aio_array_t* array, /* in: aio array */
2863
ulint index) /* in: index of the slot */
2892
os_aio_array_t* array, /*!< in: aio array */
2893
ulint index) /*!< in: index of the slot */
2865
2895
ut_a(index < array->n_slots);
2867
2897
return((array->slots) + index);
2870
/****************************************************************************
2871
Creates an aio wait array. */
2900
/************************************************************************//**
2901
Creates an aio wait array.
2902
@return own: aio array */
2873
2904
os_aio_array_t*
2874
2905
os_aio_array_create(
2875
2906
/*================*/
2876
/* out, own: aio array */
2877
ulint n, /* in: maximum number of pending aio operations
2907
ulint n, /*!< in: maximum number of pending aio operations
2878
2908
allowed; n must be divisible by n_segments */
2879
ulint n_segments) /* in: number of segments in the aio array */
2909
ulint n_segments) /*!< in: number of segments in the aio array */
2881
2911
os_aio_array_t* array;
2924
/****************************************************************************
2925
Initializes the asynchronous io system. Calls also os_io_init_simple.
2926
Creates a separate aio array for
2927
non-ibuf read and write, a third aio array for the ibuf i/o, with just one
2928
segment, two aio arrays for log reads and writes with one segment, and a
2929
synchronous aio array of the specified size. The combined number of segments
2930
in the three first aio arrays is the parameter n_segments given to the
2931
function. The caller must create an i/o handler thread for each segment in
2932
the four first arrays, but not for the sync aio array. */
2954
/***********************************************************************
2955
Initializes the asynchronous io system. Creates one array each for ibuf
2956
and log i/o. Also creates one array each for read and write where each
2957
array is divided logically into n_read_segs and n_write_segs
2958
respectively. The caller must create an i/o handler thread for each
2959
segment in these arrays. This function also creates the sync array.
2960
No i/o handler thread needs to be created for that */
2937
ulint n, /* in: maximum number of pending aio operations
2938
allowed; n must be divisible by n_segments */
2939
ulint n_segments, /* in: combined number of segments in the four
2940
first aio arrays; must be >= 4 */
2941
ulint n_slots_sync) /* in: number of slots in the sync aio array */
2965
ulint n_per_seg, /*<! in: maximum number of pending aio
2966
operations allowed per segment */
2967
ulint n_read_segs, /*<! in: number of reader threads */
2968
ulint n_write_segs, /*<! in: number of writer threads */
2969
ulint n_slots_sync) /*<! in: number of slots in the sync aio
2973
ulint n_segments = 2 + n_read_segs + n_write_segs;
2948
ut_ad(n % n_segments == 0);
2949
2975
ut_ad(n_segments >= 4);
2951
2977
os_io_init_simple();
3126
3148
return(segment);
3129
/***********************************************************************
3151
/*******************************************************************//**
3130
3152
Requests for a slot in the aio array. If no slot is available, waits until
3131
not_full-event becomes signaled. */
3153
not_full-event becomes signaled.
3154
@return pointer to slot */
3134
3157
os_aio_array_reserve_slot(
3135
3158
/*======================*/
3136
/* out: pointer to slot */
3137
ulint type, /* in: OS_FILE_READ or OS_FILE_WRITE */
3138
os_aio_array_t* array, /* in: aio array */
3139
fil_node_t* message1,/* in: message to be passed along with
3140
the aio operation */
3141
void* message2,/* in: message to be passed along with
3142
the aio operation */
3143
os_file_t file, /* in: file handle */
3144
const char* name, /* in: name of the file or path as a
3159
ulint type, /*!< in: OS_FILE_READ or OS_FILE_WRITE */
3160
os_aio_array_t* array, /*!< in: aio array */
3161
fil_node_t* message1,/*!< in: message to be passed along with
3162
the aio operation */
3163
void* message2,/*!< in: message to be passed along with
3164
the aio operation */
3165
os_file_t file, /*!< in: file handle */
3166
const char* name, /*!< in: name of the file or path as a
3145
3167
null-terminated string */
3146
void* buf, /* in: buffer where to read or from which
3168
void* buf, /*!< in: buffer where to read or from which
3148
ulint offset, /* in: least significant 32 bits of file
3150
ulint offset_high, /* in: most significant 32 bits of
3152
ulint len) /* in: length of the block to read or write */
3170
ulint offset, /*!< in: least significant 32 bits of file
3172
ulint offset_high, /*!< in: most significant 32 bits of
3174
ulint len) /*!< in: length of the block to read or write */
3154
3176
os_aio_slot_t* slot;
3155
3177
#ifdef WIN_ASYNC_IO
3156
3178
OVERLAPPED* control;
3181
ulint slots_per_seg;
3184
/* No need of a mutex. Only reading constant fields */
3185
slots_per_seg = array->n_slots / array->n_segments;
3187
/* We attempt to keep adjacent blocks in the same local
3188
segment. This can help in merging IO requests when we are
3189
doing simulated AIO */
3190
local_seg = (offset >> (UNIV_PAGE_SIZE_SHIFT + 6))
3191
% array->n_segments;
3160
3194
os_mutex_enter(array->mutex);
3362
3407
because i/os are not actually handled until
3363
3408
all have been posted: use with great
3365
const char* name, /* in: name of the file or path as a
3410
const char* name, /*!< in: name of the file or path as a
3366
3411
null-terminated string */
3367
os_file_t file, /* in: handle to a file */
3368
void* buf, /* in: buffer where to read or from which
3412
os_file_t file, /*!< in: handle to a file */
3413
void* buf, /*!< in: buffer where to read or from which
3370
ulint offset, /* in: least significant 32 bits of file
3415
ulint offset, /*!< in: least significant 32 bits of file
3371
3416
offset where to read or write */
3372
ulint offset_high, /* in: most significant 32 bits of
3417
ulint offset_high, /*!< in: most significant 32 bits of
3374
ulint n, /* in: number of bytes to read or write */
3375
fil_node_t* message1,/* in: messages for the aio handler (these
3376
can be used to identify a completed aio
3377
operation); if mode is OS_AIO_SYNC, these
3419
ulint n, /*!< in: number of bytes to read or write */
3420
fil_node_t* message1,/*!< in: message for the aio handler
3421
(can be used to identify a completed
3422
aio operation); ignored if mode is
3424
void* message2)/*!< in: message for the aio handler
3425
(can be used to identify a completed
3426
aio operation); ignored if mode is
3381
3429
os_aio_array_t* array;
3382
3430
os_aio_slot_t* slot;
3640
/**************************************************************************
3688
/**********************************************************************//**
3641
3689
Does simulated aio. This function should be called by an i/o-handler
3691
@return TRUE if the aio operation succeeded */
3645
3694
os_aio_simulated_handle(
3646
3695
/*====================*/
3647
/* out: TRUE if the aio operation succeeded */
3648
ulint global_segment, /* in: the number of the segment in the aio
3696
ulint global_segment, /*!< in: the number of the segment in the aio
3649
3697
arrays to wait for; segment 0 is the ibuf
3650
3698
i/o thread, segment 1 the log i/o thread,
3651
3699
then follow the non-ibuf read threads, and as
3652
3700
the last are the non-ibuf write threads */
3653
fil_node_t**message1, /* out: the messages passed with the aio
3701
fil_node_t**message1, /*!< out: the messages passed with the aio
3654
3702
request; note that also in the case where
3655
3703
the aio operation failed, these output
3656
3704
parameters are valid and can be used to
3657
3705
restart the operation, for example */
3658
3706
void** message2,
3659
ulint* type) /* out: OS_FILE_WRITE or ..._READ */
3707
ulint* type) /*!< out: OS_FILE_WRITE or ..._READ */
3661
3709
os_aio_array_t* array;