diff options
author | hcpp <hcpp@ydb.tech> | 2023-11-08 12:09:41 +0300 |
---|---|---|
committer | hcpp <hcpp@ydb.tech> | 2023-11-08 12:56:14 +0300 |
commit | a361f5b98b98b44ea510d274f6769164640dd5e1 (patch) | |
tree | c47c80962c6e2e7b06798238752fd3da0191a3f6 /contrib/libs/libmysql_r/mysys/my_sync.cc | |
parent | 9478806fde1f4d40bd5a45e7cbe77237dab613e9 (diff) | |
download | ydb-a361f5b98b98b44ea510d274f6769164640dd5e1.tar.gz |
metrics have been added
Diffstat (limited to 'contrib/libs/libmysql_r/mysys/my_sync.cc')
-rw-r--r-- | contrib/libs/libmysql_r/mysys/my_sync.cc | 191 |
1 files changed, 191 insertions, 0 deletions
diff --git a/contrib/libs/libmysql_r/mysys/my_sync.cc b/contrib/libs/libmysql_r/mysys/my_sync.cc new file mode 100644 index 0000000000..ce9b5318a9 --- /dev/null +++ b/contrib/libs/libmysql_r/mysys/my_sync.cc @@ -0,0 +1,191 @@ +/* Copyright (c) 2003, 2017, 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, version 2.0, + as published by the Free Software Foundation. + + This program is also distributed with certain software (including + but not limited to OpenSSL) that is licensed under separate terms, + as designated in a particular file or component or in included license + documentation. The authors of MySQL hereby grant you an additional + permission to link the program and your derivative works with the + separately licensed software that they have included with MySQL. + + Without limiting anything contained in the foregoing, this file, + which is part of C Driver for MySQL (Connector/C), is also subject to the + Universal FOSS Exception, version 1.0, a copy of which can be found at + http://oss.oracle.com/licenses/universal-foss-exception. + + 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, version 2.0, 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-1301 USA */ + +/** + @file mysys/my_sync.cc +*/ + +#include "my_config.h" + +#include <errno.h> +#include <fcntl.h> +#include <stddef.h> +#ifdef HAVE_UNISTD_H +#include <unistd.h> +#endif + +#include "my_compiler.h" +#include "my_dbug.h" +#include "my_inttypes.h" +#include "my_io.h" +#include "my_sys.h" +#include "my_thread_local.h" +#include "mysys_err.h" +#if defined(_WIN32) +#include "mysys/mysys_priv.h" +#endif + +static void (*before_sync_wait)(void) = 0; +static void (*after_sync_wait)(void) = 0; + +void thr_set_sync_wait_callback(void (*before_wait)(void), + void (*after_wait)(void)) { + before_sync_wait = before_wait; + after_sync_wait = after_wait; +} + +/* + Sync data in file to disk + + SYNOPSIS + my_sync() + fd File descritor to sync + my_flags Flags (now only MY_WME is supported) + + NOTE + If file system supports its, only file data is synced, not inode data. + + MY_IGNORE_BADFD is useful when fd is "volatile" - not protected by a + mutex. In this case by the time of fsync(), fd may be already closed by + another thread, or even reassigned to a different file. With this flag - + MY_IGNORE_BADFD - such a situation will not be considered an error. + (which is correct behaviour, if we know that the other thread synced the + file before closing) + + RETURN + 0 ok + -1 error +*/ + +int my_sync(File fd, myf my_flags) { + int res; + DBUG_ENTER("my_sync"); + DBUG_PRINT("my", ("Fd: %d my_flags: %d", fd, my_flags)); + + if (before_sync_wait) (*before_sync_wait)(); + do { +#if defined(HAVE_FDATASYNC) && defined(HAVE_DECL_FDATASYNC) + res = fdatasync(fd); +#elif defined(HAVE_FSYNC) + res = fsync(fd); +#elif defined(_WIN32) + res = my_win_fsync(fd); +#else +#error Cannot find a way to sync a file, durability in danger + res = 0; /* No sync (strange OS) */ +#endif + } while (res == -1 && errno == EINTR); + + if (res) { + int er = errno; + set_my_errno(er); + if (!er) set_my_errno(-1); /* Unknown error */ + if (after_sync_wait) (*after_sync_wait)(); + if ((my_flags & MY_IGNORE_BADFD) && + (er == EBADF || er == EINVAL || er == EROFS +#ifdef __APPLE__ + || er == ENOTSUP +#endif + )) { + DBUG_PRINT("info", ("ignoring errno %d", er)); + res = 0; + } else if (my_flags & MY_WME) { + char errbuf[MYSYS_STRERROR_SIZE]; + my_error(EE_SYNC, MYF(0), my_filename(fd), my_errno(), + my_strerror(errbuf, sizeof(errbuf), my_errno())); + } + } else { + if (after_sync_wait) (*after_sync_wait)(); + } + DBUG_RETURN(res); +} /* my_sync */ + +/* + Force directory information to disk. + + SYNOPSIS + my_sync_dir() + dir_name the name of the directory + my_flags flags (MY_WME etc) + + RETURN + 0 if ok, !=0 if error +*/ + +int my_sync_dir(const char *dir_name MY_ATTRIBUTE((unused)), + myf my_flags MY_ATTRIBUTE((unused))) { +/* + Only Linux is known to need an explicit sync of the directory to make sure a + file creation/deletion/renaming in(from,to) this directory durable. +*/ +#ifdef __linux__ + static const char cur_dir_name[] = {FN_CURLIB, 0}; + File dir_fd; + int res = 0; + const char *correct_dir_name; + DBUG_ENTER("my_sync_dir"); + DBUG_PRINT("my", ("Dir: '%s' my_flags: %d", dir_name, my_flags)); + /* Sometimes the path does not contain an explicit directory */ + correct_dir_name = (dir_name[0] == 0) ? cur_dir_name : dir_name; + /* + Syncing a dir may give EINVAL on tmpfs on Linux, which is ok. + EIO on the other hand is very important. Hence MY_IGNORE_BADFD. + */ + if ((dir_fd = my_open(correct_dir_name, O_RDONLY, MYF(my_flags))) >= 0) { + if (my_sync(dir_fd, MYF(my_flags | MY_IGNORE_BADFD))) res = 2; + if (my_close(dir_fd, MYF(my_flags))) res = 3; + } else + res = 1; + DBUG_RETURN(res); +#else + return 0; +#endif +} + +/* + Force directory information to disk. + + SYNOPSIS + my_sync_dir_by_file() + file_name the name of a file in the directory + my_flags flags (MY_WME etc) + + RETURN + 0 if ok, !=0 if error +*/ + +int my_sync_dir_by_file(const char *file_name MY_ATTRIBUTE((unused)), + myf my_flags MY_ATTRIBUTE((unused))) { +#ifdef __linux__ + char dir_name[FN_REFLEN]; + size_t dir_name_length; + dirname_part(dir_name, file_name, &dir_name_length); + return my_sync_dir(dir_name, my_flags); +#else + return 0; +#endif +} |