mirror of
https://gitee.com/milvus-io/milvus.git
synced 2025-12-28 22:45:26 +08:00
issue: https://github.com/milvus-io/milvus/issues/41636 --------- Signed-off-by: sunby <sunbingyi1992@gmail.com>
158 lines
4.1 KiB
C++
158 lines
4.1 KiB
C++
// Copyright (C) 2019-2020 Zilliz. All rights reserved.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance
|
|
// with the License. You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software distributed under the License
|
|
// is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
|
|
// or implied. See the License for the specific language governing permissions and limitations under the License.
|
|
|
|
#pragma once
|
|
|
|
#include <cstdio>
|
|
#include <string>
|
|
#include "common/EasyAssert.h"
|
|
#include "common/Types.h"
|
|
#include "fmt/core.h"
|
|
#include <fcntl.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
|
|
namespace milvus {
|
|
|
|
#define THROW_FILE_WRITE_ERROR(path) \
|
|
ThrowInfo(ErrorCode::FileWriteFailed, \
|
|
fmt::format("write data to file {} failed, error code {}", \
|
|
path, \
|
|
strerror(errno)));
|
|
|
|
class File {
|
|
public:
|
|
File(const File& file) = delete;
|
|
File(File&& file) noexcept {
|
|
fd_ = file.fd_;
|
|
file.fd_ = -1;
|
|
}
|
|
~File() {
|
|
if (fs_ != nullptr) {
|
|
fclose(fs_);
|
|
}
|
|
}
|
|
|
|
static File
|
|
Open(const std::string_view filepath, int flags) {
|
|
// using default buf size = 4096
|
|
return Open(filepath, flags, 4096);
|
|
}
|
|
|
|
static File
|
|
Open(const std::string_view filepath, int flags, size_t buf_size) {
|
|
int fd = open(filepath.data(), flags, S_IRUSR | S_IWUSR);
|
|
AssertInfo(fd != -1,
|
|
"failed to create mmap file {}: {}",
|
|
filepath,
|
|
strerror(errno));
|
|
FILE* fs = fdopen(fd, get_mode_from_flags(flags));
|
|
AssertInfo(fs != nullptr,
|
|
"failed to open file {}: {}",
|
|
filepath,
|
|
strerror(errno));
|
|
auto f = File(fd, fs, std::string(filepath));
|
|
// setup buffer size file stream will use
|
|
setvbuf(f.fs_, nullptr, _IOFBF, buf_size);
|
|
return f;
|
|
}
|
|
|
|
int
|
|
Descriptor() const {
|
|
return fd_;
|
|
}
|
|
|
|
std::string
|
|
Path() const {
|
|
return filepath_;
|
|
}
|
|
|
|
ssize_t
|
|
Write(const void* buf, size_t size) {
|
|
return write(fd_, buf, size);
|
|
}
|
|
|
|
template <typename T, std::enable_if_t<std::is_integral_v<T>, int> = 0>
|
|
ssize_t
|
|
WriteInt(T value) {
|
|
return write(fd_, &value, sizeof(value));
|
|
}
|
|
|
|
ssize_t
|
|
FWrite(const void* buf, size_t size) {
|
|
return fwrite(buf, sizeof(char), size, fs_);
|
|
}
|
|
|
|
template <typename T, std::enable_if_t<std::is_integral_v<T>, int> = 0>
|
|
ssize_t
|
|
FWriteInt(T value) {
|
|
return fwrite(&value, 1, sizeof(value), fs_);
|
|
}
|
|
|
|
int
|
|
FFlush() {
|
|
return fflush(fs_);
|
|
}
|
|
|
|
offset_t
|
|
Seek(offset_t offset, int whence) {
|
|
return lseek(fd_, offset, whence);
|
|
}
|
|
|
|
void
|
|
Close() {
|
|
fclose(fs_);
|
|
fs_ = nullptr;
|
|
fd_ = -1;
|
|
}
|
|
|
|
private:
|
|
static inline const char*
|
|
get_mode_from_flags(int flags) {
|
|
switch (flags) {
|
|
case O_RDONLY: {
|
|
return "rb";
|
|
}
|
|
case O_WRONLY: {
|
|
return flags & O_APPEND ? "ab" : "wb";
|
|
}
|
|
case O_RDWR: {
|
|
return flags & O_APPEND ? "ab+" : "wb+";
|
|
}
|
|
default: {
|
|
AssertInfo(false, "invalid file flags: {}", flags);
|
|
}
|
|
}
|
|
}
|
|
|
|
explicit File(int fd, FILE* fs, const std::string& filepath)
|
|
: fd_(fd), filepath_(filepath), fs_(fs) {
|
|
}
|
|
int fd_{-1};
|
|
FILE* fs_;
|
|
std::string filepath_;
|
|
};
|
|
|
|
class MmapFileRAII {
|
|
public:
|
|
MmapFileRAII(const std::string& filepath) : file_path_(filepath) {
|
|
}
|
|
~MmapFileRAII() {
|
|
if (!file_path_.empty()) {
|
|
unlink(file_path_.c_str());
|
|
}
|
|
}
|
|
|
|
private:
|
|
const std::string file_path_;
|
|
};
|
|
} // namespace milvus
|