aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/libs/libmysql_r/mysys/my_sync.cc
diff options
context:
space:
mode:
authorhcpp <hcpp@ydb.tech>2023-11-08 12:09:41 +0300
committerhcpp <hcpp@ydb.tech>2023-11-08 12:56:14 +0300
commita361f5b98b98b44ea510d274f6769164640dd5e1 (patch)
treec47c80962c6e2e7b06798238752fd3da0191a3f6 /contrib/libs/libmysql_r/mysys/my_sync.cc
parent9478806fde1f4d40bd5a45e7cbe77237dab613e9 (diff)
downloadydb-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.cc191
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
+}