All Classes Files Functions Enumerations Enumerator Pages
file_handle.hpp
1 /*
2  * Copyright (c) 2021-2025, NVIDIA CORPORATION.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 #pragma once
17 
18 #include <sys/stat.h>
19 #include <sys/types.h>
20 
21 #include <cstddef>
22 #include <cstdlib>
23 #include <stdexcept>
24 #include <system_error>
25 #include <utility>
26 
27 #include <kvikio/buffer.hpp>
28 #include <kvikio/cufile/config.hpp>
29 #include <kvikio/defaults.hpp>
30 #include <kvikio/error.hpp>
31 #include <kvikio/parallel_operation.hpp>
32 #include <kvikio/posix_io.hpp>
33 #include <kvikio/shim/cufile.hpp>
34 #include <kvikio/stream.hpp>
35 #include <kvikio/utils.hpp>
36 
37 namespace kvikio {
38 
44 class FileHandle {
45  private:
46  // We use two file descriptors, one opened with the O_DIRECT flag and one without.
47  int _fd_direct_on{-1};
48  int _fd_direct_off{-1};
49  bool _initialized{false};
50  CompatMode _compat_mode{CompatMode::AUTO};
51  mutable std::size_t _nbytes{0}; // The size of the underlying file, zero means unknown.
52  CUfileHandle_t _handle{};
53 
63  bool is_compat_mode_preferred_for_async(CompatMode requested_compat_mode);
64 
65  public:
66  static constexpr mode_t m644 = S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH;
67  FileHandle() noexcept = default;
68 
85  FileHandle(const std::string& file_path,
86  const std::string& flags = "r",
87  mode_t mode = m644,
88  CompatMode compat_mode = defaults::compat_mode());
89 
93  FileHandle(const FileHandle&) = delete;
94  FileHandle& operator=(FileHandle const&) = delete;
95  FileHandle(FileHandle&& o) noexcept;
96  FileHandle& operator=(FileHandle&& o) noexcept;
97  ~FileHandle() noexcept;
98 
104  [[nodiscard]] bool closed() const noexcept;
105 
109  void close() noexcept;
110 
119  [[nodiscard]] CUfileHandle_t handle();
120 
130  [[nodiscard]] int fd() const noexcept;
131 
141  [[nodiscard]] int fd_open_flags() const;
142 
150  [[nodiscard]] std::size_t nbytes() const;
151 
182  std::size_t read(void* devPtr_base,
183  std::size_t size,
184  std::size_t file_offset,
185  std::size_t devPtr_offset,
186  bool sync_default_stream = true);
187 
219  std::size_t write(const void* devPtr_base,
220  std::size_t size,
221  std::size_t file_offset,
222  std::size_t devPtr_offset,
223  bool sync_default_stream = true);
224 
255  std::future<std::size_t> pread(void* buf,
256  std::size_t size,
257  std::size_t file_offset = 0,
258  std::size_t task_size = defaults::task_size(),
259  std::size_t gds_threshold = defaults::gds_threshold(),
260  bool sync_default_stream = true);
261 
292  std::future<std::size_t> pwrite(const void* buf,
293  std::size_t size,
294  std::size_t file_offset = 0,
295  std::size_t task_size = defaults::task_size(),
296  std::size_t gds_threshold = defaults::gds_threshold(),
297  bool sync_default_stream = true);
298 
333  void read_async(void* devPtr_base,
334  std::size_t* size_p,
335  off_t* file_offset_p,
336  off_t* devPtr_offset_p,
337  ssize_t* bytes_read_p,
338  CUstream stream);
339 
365  [[nodiscard]] StreamFuture read_async(void* devPtr_base,
366  std::size_t size,
367  off_t file_offset = 0,
368  off_t devPtr_offset = 0,
369  CUstream stream = nullptr);
370 
406  void write_async(void* devPtr_base,
407  std::size_t* size_p,
408  off_t* file_offset_p,
409  off_t* devPtr_offset_p,
410  ssize_t* bytes_written_p,
411  CUstream stream);
412 
438  [[nodiscard]] StreamFuture write_async(void* devPtr_base,
439  std::size_t size,
440  off_t file_offset = 0,
441  off_t devPtr_offset = 0,
442  CUstream stream = nullptr);
443 
453  [[nodiscard]] bool is_compat_mode_preferred() const noexcept;
454 
465  [[nodiscard]] bool is_compat_mode_preferred_for_async() const noexcept;
466 };
467 
468 } // namespace kvikio
Handle of an open file registered with cufile.
Definition: file_handle.hpp:44
void close() noexcept
Deregister the file and close the two files.
FileHandle(const FileHandle &)=delete
FileHandle support move semantic but isn't copyable.
std::future< std::size_t > pread(void *buf, std::size_t size, std::size_t file_offset=0, std::size_t task_size=defaults::task_size(), std::size_t gds_threshold=defaults::gds_threshold(), bool sync_default_stream=true)
Reads specified bytes from the file into the device or host memory in parallel.
void read_async(void *devPtr_base, std::size_t *size_p, off_t *file_offset_p, off_t *devPtr_offset_p, ssize_t *bytes_read_p, CUstream stream)
Reads specified bytes from the file into the device memory asynchronously.
CUfileHandle_t handle()
Get the underlying cuFile file handle.
int fd_open_flags() const
Get the flags of one of the file descriptors (see open(2))
std::size_t write(const void *devPtr_base, std::size_t size, std::size_t file_offset, std::size_t devPtr_offset, bool sync_default_stream=true)
Writes specified bytes from the device memory into the file.
bool is_compat_mode_preferred_for_async() const noexcept
Returns true if the compatibility mode is expected to be ON for the asynchronous I/O on this file.
bool is_compat_mode_preferred() const noexcept
Returns true if the compatibility mode is expected to be ON for this file.
bool closed() const noexcept
Whether the file is closed according to its initialization status.
FileHandle(const std::string &file_path, const std::string &flags="r", mode_t mode=m644, CompatMode compat_mode=defaults::compat_mode())
Construct a file handle from a file path.
int fd() const noexcept
Get one of the file descriptors.
void write_async(void *devPtr_base, std::size_t *size_p, off_t *file_offset_p, off_t *devPtr_offset_p, ssize_t *bytes_written_p, CUstream stream)
Writes specified bytes from the device memory into the file asynchronously.
std::future< std::size_t > pwrite(const void *buf, std::size_t size, std::size_t file_offset=0, std::size_t task_size=defaults::task_size(), std::size_t gds_threshold=defaults::gds_threshold(), bool sync_default_stream=true)
Writes specified bytes from device or host memory into the file in parallel.
std::size_t nbytes() const
Get the file size.
std::size_t read(void *devPtr_base, std::size_t size, std::size_t file_offset, std::size_t devPtr_offset, bool sync_default_stream=true)
Reads specified bytes from the file into the device memory.
Future of an asynchronous IO operation.
Definition: stream.hpp:46
Singleton class of default values used throughout KvikIO.
Definition: defaults.hpp:88
static CompatMode compat_mode()
Return whether the KvikIO library is running in compatibility mode or not.
CompatMode
I/O compatibility mode.
Definition: defaults.hpp:38