mirror of
https://github.com/MariaDB/server.git
synced 2025-01-27 01:04:19 +01:00
7f7329f092
Apparently, invoking fcntl(fd, F_SETFL, O_DIRECT) will lead to unexpected behaviour on Linux bcachefs and possibly other file systems, depending on the operating system version. So, let us avoid doing that, and instead just attempt to pass the O_DIRECT flag to open(). This should make us compatible with NetBSD, IBM AIX, as well as Solaris and its derivatives. This fix does not change the fact that we had only implemented innodb_log_file_buffering=OFF on systems where we can determine the physical block size (typically 512 or 4096 bytes). Currently, those operating systems are Linux and Microsoft Windows. HAVE_FCNTL_DIRECT, os_file_set_nocache(): Remove. OS_FILE_OVERWRITE, OS_FILE_CREATE_PATH: Remove (never used parameters). os_file_log_buffered(), os_file_log_maybe_unbuffered(): Helper functions. os_file_create_simple_func(): When applicable, initially attempt to open files in O_DIRECT mode. os_file_create_func(): When applicable, initially attempt to open files in O_DIRECT mode. For type==OS_LOG_FILE && create_mode != OS_FILE_CREATE we will first invoke stat(2) on the file name to find out if the size is compatible with O_DIRECT. If create_mode == OS_FILE_CREATE, we will invoke fstat(2) on the created log file afterwards, and may close and reopen the file in O_DIRECT mode if applicable. create_temp_file(): Support O_DIRECT. This is only used if O_TMPFILE is available and innodb_disable_sort_file_cache=ON (non-default value). Notably, that setting never worked on Microsoft Windows. row_merge_file_create_mode(): Split from row_merge_file_create_low(). Create a temporary file in the specified mode. Reviewed by: Vladislav Vaintroub
206 lines
6.1 KiB
C
206 lines
6.1 KiB
C
/* Copyright (c) 2000, 2013, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1335 USA */
|
|
|
|
#include "mysys_priv.h"
|
|
#include <m_string.h>
|
|
#include "my_static.h"
|
|
#include "mysys_err.h"
|
|
#include <errno.h>
|
|
#ifdef HAVE_PATHS_H
|
|
#include <paths.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_MKOSTEMP
|
|
#define mkstemp(A) mkostemp(A, O_CLOEXEC)
|
|
#endif
|
|
|
|
/*
|
|
@brief
|
|
Create a temporary file with unique name in a given directory
|
|
|
|
@details
|
|
create_temp_file
|
|
to pointer to buffer where temporary filename will be stored
|
|
dir directory where to create the file
|
|
prefix prefix the filename with this
|
|
mode Flags to use for my_create/my_open
|
|
MyFlags Magic flags
|
|
|
|
@return
|
|
File descriptor of opened file if success
|
|
-1 and sets errno if fails.
|
|
|
|
@note
|
|
The behaviour of this function differs a lot between
|
|
implementation, it's main use is to generate a file with
|
|
a name that does not already exist.
|
|
|
|
When passing MY_TEMPORARY flag in MyFlags the file is automatically deleted
|
|
|
|
"mode" bits that always must be used for newly created files with
|
|
unique file names (O_EXCL | O_TRUNC | O_CREAT | O_RDWR) are added
|
|
automatically, and shouldn't be specified by the caller.
|
|
|
|
The implementation using mkstemp should be considered the
|
|
reference implementation when adding a new or modifying an
|
|
existing one
|
|
|
|
*/
|
|
|
|
File create_temp_file(char *to, const char *dir, const char *prefix,
|
|
int mode, myf MyFlags)
|
|
{
|
|
File file= -1;
|
|
|
|
DBUG_ENTER("create_temp_file");
|
|
DBUG_PRINT("enter", ("dir: %s, prefix: %s", dir ? dir : "(null)", prefix));
|
|
#if !defined _WIN32 && defined O_DIRECT
|
|
DBUG_ASSERT((mode & (O_EXCL | O_TRUNC | O_CREAT | O_RDWR | O_DIRECT)) == 0);
|
|
#else
|
|
DBUG_ASSERT((mode & (O_EXCL | O_TRUNC | O_CREAT | O_RDWR)) == 0);
|
|
#endif
|
|
|
|
mode|= O_TRUNC | O_CREAT | O_RDWR; /* not O_EXCL, see Windows code below */
|
|
|
|
#ifdef _WIN32
|
|
{
|
|
TCHAR path_buf[MAX_PATH-14];
|
|
/*
|
|
Use GetTempPath to determine path for temporary files.
|
|
This is because the documentation for GetTempFileName
|
|
has the following to say about this parameter:
|
|
"If this parameter is NULL, the function fails."
|
|
*/
|
|
if (!dir)
|
|
{
|
|
if(GetTempPath(sizeof(path_buf), path_buf) > 0)
|
|
dir = path_buf;
|
|
}
|
|
/*
|
|
Use GetTempFileName to generate a unique filename, create
|
|
the file and release it's handle
|
|
- uses up to the first three letters from prefix
|
|
*/
|
|
if (GetTempFileName(dir, prefix, 0, to) == 0)
|
|
DBUG_RETURN(-1);
|
|
|
|
DBUG_PRINT("info", ("name: %s", to));
|
|
|
|
if (MyFlags & MY_TEMPORARY)
|
|
mode|= O_SHORT_LIVED | O_TEMPORARY;
|
|
|
|
/*
|
|
Open the file without O_EXCL flag
|
|
since the file has already been created by GetTempFileName
|
|
*/
|
|
if ((file= my_open(to, mode, MyFlags)) < 0)
|
|
{
|
|
/* Open failed, remove the file created by GetTempFileName */
|
|
int tmp= my_errno;
|
|
(void) my_delete(to, MYF(0));
|
|
my_errno= tmp;
|
|
}
|
|
}
|
|
#elif defined(HAVE_MKSTEMP)
|
|
if (!dir && ! (dir =getenv("TMPDIR")))
|
|
dir= DEFAULT_TMPDIR;
|
|
#ifdef O_TMPFILE
|
|
{
|
|
static int O_TMPFILE_works= 1;
|
|
|
|
if ((MyFlags & MY_TEMPORARY) && O_TMPFILE_works)
|
|
{
|
|
/*
|
|
explicitly don't use O_EXCL here has it has a different
|
|
meaning with O_TMPFILE
|
|
*/
|
|
const int flags= (mode & ~O_CREAT) | O_TMPFILE | O_CLOEXEC;
|
|
const mode_t open_mode= S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP;
|
|
# ifdef O_DIRECT
|
|
static int O_TMPFILE_works_with_O_DIRECT= O_DIRECT;
|
|
const int try_O_DIRECT= mode & O_TMPFILE_works_with_O_DIRECT;
|
|
if (try_O_DIRECT)
|
|
file= open(dir, flags | O_DIRECT, open_mode);
|
|
else
|
|
# endif
|
|
file= open(dir, flags, open_mode);
|
|
|
|
if (file >= 0)
|
|
{
|
|
# ifdef O_DIRECT
|
|
O_TMPFILE_works:
|
|
# endif
|
|
my_snprintf(to, FN_REFLEN, "%s/#sql/fd=%d", dir, file);
|
|
file=my_register_filename(file, to, FILE_BY_O_TMPFILE,
|
|
EE_CANTCREATEFILE, MyFlags);
|
|
}
|
|
# ifdef O_DIRECT
|
|
else if (errno == EINVAL && try_O_DIRECT)
|
|
{
|
|
file= open(dir, flags, open_mode);
|
|
if (file >= 0)
|
|
{
|
|
O_TMPFILE_works_with_O_DIRECT= 0;
|
|
goto O_TMPFILE_works;
|
|
}
|
|
}
|
|
# endif
|
|
else if (errno == EOPNOTSUPP || errno == EINVAL)
|
|
{
|
|
my_printf_error(EE_CANTCREATEFILE, "O_TMPFILE is not supported on %s "
|
|
"(disabling future attempts)",
|
|
MYF(ME_NOTE | ME_ERROR_LOG_ONLY), dir);
|
|
O_TMPFILE_works= 0;
|
|
}
|
|
}
|
|
}
|
|
if (file == -1)
|
|
#endif /* O_TMPFILE */
|
|
{
|
|
char prefix_buff[30];
|
|
uint pfx_len;
|
|
File org_file;
|
|
|
|
pfx_len= (uint) (strmov(strnmov(prefix_buff,
|
|
prefix ? prefix : "tmp.",
|
|
sizeof(prefix_buff)-7),"XXXXXX") -
|
|
prefix_buff);
|
|
if (strlen(dir)+ pfx_len > FN_REFLEN-2)
|
|
{
|
|
errno=my_errno= ENAMETOOLONG;
|
|
DBUG_RETURN(file);
|
|
}
|
|
strmov(convert_dirname(to,dir,NullS),prefix_buff);
|
|
org_file=mkstemp(to);
|
|
if (org_file >= 0 && (MyFlags & MY_TEMPORARY))
|
|
(void) my_delete(to, MYF(MY_WME));
|
|
file=my_register_filename(org_file, to, FILE_BY_MKSTEMP,
|
|
EE_CANTCREATEFILE, MyFlags);
|
|
/* If we didn't manage to register the name, remove the temp file */
|
|
if (org_file >= 0 && file < 0)
|
|
{
|
|
int tmp=my_errno;
|
|
close(org_file);
|
|
(void) my_delete(to, MYF(MY_WME));
|
|
my_errno=tmp;
|
|
}
|
|
}
|
|
#else
|
|
#error No implementation found for create_temp_file
|
|
#endif
|
|
if (file >= 0)
|
|
statistic_increment(my_tmp_file_created,&THR_LOCK_open);
|
|
DBUG_RETURN(file);
|
|
}
|