forked from openvinotoolkit/openvino
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwin_mmap_object.cpp
151 lines (126 loc) · 4.47 KB
/
win_mmap_object.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
// Copyright (C) 2018-2024 Intel Corporation
// SPDX-License-Identifier: Apache-2.0
//
#include <stdexcept>
#include "openvino/util/file_util.hpp"
#include "openvino/util/mmap_object.hpp"
// clang-format-off
#ifndef NOMINMAX
# define NOMINMAX
#endif
#include <windows.h>
// clang-format-on
namespace ov {
class HandleHolder {
HANDLE m_handle = INVALID_HANDLE_VALUE;
void reset() {
if (m_handle != INVALID_HANDLE_VALUE) {
::CloseHandle(m_handle);
m_handle = INVALID_HANDLE_VALUE;
}
}
public:
explicit HandleHolder(HANDLE handle = INVALID_HANDLE_VALUE) : m_handle(handle) {}
HandleHolder(const HandleHolder&) = delete;
HandleHolder(HandleHolder&& other) noexcept : m_handle(other.m_handle) {
other.m_handle = INVALID_HANDLE_VALUE;
}
HandleHolder& operator=(const HandleHolder&) = delete;
HandleHolder& operator=(HandleHolder&& other) noexcept {
if (this == &other) {
return *this;
}
reset();
m_handle = other.m_handle;
other.m_handle = INVALID_HANDLE_VALUE;
return *this;
}
~HandleHolder() {
reset();
}
HANDLE get() const noexcept {
return m_handle;
}
};
class MapHolder : public ov::MappedMemory {
public:
MapHolder() = default;
~MapHolder() {
if (m_data) {
::UnmapViewOfFile(m_data);
}
}
void set(const std::string& path) {
// Note that file can't be changed (renamed/deleted) until it's unmapped. FILE_SHARE_DELETE flag allow
// rename/deletion, but it doesn't work with FAT32 filesystem (works on NTFS)
auto h = ::CreateFileA(path.c_str(), GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
map(path, h);
}
#ifdef OPENVINO_ENABLE_UNICODE_PATH_SUPPORT
void set(const std::wstring& path) {
// Note that file can't be changed (renamed/deleted) until it's unmapped. FILE_SHARE_DELETE flag allow
// rename/deletion, but it doesn't work with FAT32 filesystem (works on NTFS)
auto h = ::CreateFileW(path.c_str(), GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
map(ov::util::wstring_to_string(path), h);
}
#endif
char* data() noexcept override {
return static_cast<char*>(m_data);
}
size_t size() const noexcept override {
return m_size;
}
private:
void map(const std::string& path, HANDLE h) {
if (h == INVALID_HANDLE_VALUE) {
throw std::runtime_error("Can not open file " + path +
" for mapping. Ensure that file exists and has appropriate permissions");
}
m_handle = HandleHolder(h);
DWORD map_mode = FILE_MAP_READ;
DWORD access = PAGE_READONLY;
LARGE_INTEGER file_size_large;
if (::GetFileSizeEx(m_handle.get(), &file_size_large) == 0) {
throw std::runtime_error("Can not get file size for " + path);
}
m_size = static_cast<uint64_t>(file_size_large.QuadPart);
if (m_size > 0) {
m_mapping =
HandleHolder(::CreateFileMapping(m_handle.get(), 0, access, m_size >> 32, m_size & 0xffffffff, 0));
if (m_mapping.get() == INVALID_HANDLE_VALUE) {
throw std::runtime_error("Can not create file mapping for " + path);
}
m_data = ::MapViewOfFile(m_mapping.get(),
map_mode,
0, // offset_align >> 32,
0, // offset_align & 0xffffffff,
m_size);
if (!m_data) {
throw std::runtime_error("Can not create map view for " + path);
}
} else {
m_data = nullptr;
}
}
private:
void* m_data = nullptr;
size_t m_size = 0;
HandleHolder m_handle;
HandleHolder m_mapping;
};
std::shared_ptr<ov::MappedMemory> load_mmap_object(const std::string& path) {
auto holder = std::make_shared<MapHolder>();
holder->set(path);
return holder;
}
#ifdef OPENVINO_ENABLE_UNICODE_PATH_SUPPORT
std::shared_ptr<ov::MappedMemory> load_mmap_object(const std::wstring& path) {
auto holder = std::make_shared<MapHolder>();
holder->set(path);
return holder;
}
MmapStream::MmapStream(const std::wstring& path) : std::ifstream(path.data(), std::ios_base::binary) {
m_memory = ov::load_mmap_object(path);
}
#endif
} // namespace ov