2023-02-25 17:39:39 +08:00
|
|
|
#include "VxMemoryMappedFile.hpp"
|
2023-08-25 21:57:22 +08:00
|
|
|
#include "../VTEncoding.hpp"
|
2023-02-10 16:17:17 +08:00
|
|
|
|
2023-02-26 21:48:03 +08:00
|
|
|
namespace LibCmo::VxMath {
|
2023-02-10 16:17:17 +08:00
|
|
|
|
2023-09-17 10:38:46 +08:00
|
|
|
VxMemoryMappedFile::VxMemoryMappedFile(CKSTRING u8_filepath) :
|
2024-08-22 10:57:53 +08:00
|
|
|
// Initialize members
|
2024-08-17 20:43:27 +08:00
|
|
|
#if YYCC_OS == YYCC_OS_WINDOWS
|
2024-08-22 10:57:53 +08:00
|
|
|
// Initialize Windows specific.
|
2023-02-12 21:03:36 +08:00
|
|
|
m_hFile(NULL), m_hFileMapping(NULL), m_hFileMapView(NULL),
|
2024-08-22 10:57:53 +08:00
|
|
|
m_dwFileSize(),
|
2023-02-12 21:03:36 +08:00
|
|
|
#else
|
2024-08-22 10:57:53 +08:00
|
|
|
// Initialize Linux specific.
|
2023-03-04 14:08:16 +08:00
|
|
|
m_hFile(-1), m_offFileSize(0), m_pFileAddr((void*)-1),
|
2023-02-12 21:03:36 +08:00
|
|
|
#endif
|
2024-08-22 10:57:53 +08:00
|
|
|
// Initialize common members.
|
2023-02-10 16:17:17 +08:00
|
|
|
m_szFilePath(),
|
2023-02-12 21:03:36 +08:00
|
|
|
m_bIsValid(false), m_pMemoryMappedFileBase(nullptr), m_cbFile(0u) {
|
2023-02-10 16:17:17 +08:00
|
|
|
|
2024-08-22 10:57:53 +08:00
|
|
|
// Setup file path first
|
|
|
|
if (u8_filepath == nullptr) return;
|
|
|
|
m_szFilePath = u8_filepath;
|
2023-02-10 16:17:17 +08:00
|
|
|
|
2024-08-22 10:57:53 +08:00
|
|
|
// Do real mapping work according to different platform.
|
2024-08-17 20:43:27 +08:00
|
|
|
#if YYCC_OS == YYCC_OS_WINDOWS
|
2023-02-15 21:03:26 +08:00
|
|
|
|
2024-08-22 10:57:53 +08:00
|
|
|
// Parse file name to wchar_t
|
|
|
|
std::wstring w_filename;
|
|
|
|
if (!YYCC::EncodingHelper::UTF8ToWchar(m_szFilePath, w_filename))
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Open file
|
|
|
|
this->m_hFile = ::CreateFileW(
|
|
|
|
w_filename.c_str(),
|
2023-02-12 21:03:36 +08:00
|
|
|
GENERIC_READ,
|
|
|
|
0, // do not share
|
|
|
|
NULL, // no security
|
|
|
|
OPEN_EXISTING,
|
|
|
|
FILE_ATTRIBUTE_NORMAL, // no attr
|
|
|
|
NULL // no template
|
2023-02-10 16:17:17 +08:00
|
|
|
);
|
2023-02-12 21:03:36 +08:00
|
|
|
if (this->m_hFile == INVALID_HANDLE_VALUE) {
|
|
|
|
return;
|
|
|
|
}
|
2023-02-10 16:17:17 +08:00
|
|
|
|
2024-08-22 10:57:53 +08:00
|
|
|
// Get size and check its range.
|
|
|
|
if (!(::GetFileSizeEx(this->m_hFile, &m_dwFileSize))) {
|
2023-02-12 21:03:36 +08:00
|
|
|
CloseHandle(this->m_hFile);
|
|
|
|
return;
|
|
|
|
}
|
2024-08-22 10:57:53 +08:00
|
|
|
if (m_dwFileSize.HighPart != 0) {
|
|
|
|
CloseHandle(this->m_hFile);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_cbFile = m_dwFileSize.LowPart;
|
2023-02-12 21:03:36 +08:00
|
|
|
|
2024-08-22 10:57:53 +08:00
|
|
|
// Open mapping
|
2023-02-12 21:03:36 +08:00
|
|
|
this->m_hFileMapping = CreateFileMappingW(
|
|
|
|
this->m_hFile,
|
|
|
|
NULL, // default security
|
|
|
|
PAGE_READONLY,
|
|
|
|
0, 0, // expand to file size
|
|
|
|
NULL // no name
|
2023-02-10 16:17:17 +08:00
|
|
|
);
|
2023-02-12 21:03:36 +08:00
|
|
|
if (this->m_hFileMapping == NULL) {
|
|
|
|
CloseHandle(this->m_hFile);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-22 10:57:53 +08:00
|
|
|
// Open map view
|
2023-02-12 21:03:36 +08:00
|
|
|
this->m_hFileMapView = MapViewOfFile(
|
|
|
|
this->m_hFileMapping,
|
|
|
|
FILE_MAP_READ,
|
|
|
|
0, 0, //no offset
|
|
|
|
0 // expand to full file size
|
|
|
|
);
|
|
|
|
if (this->m_hFileMapView == NULL) {
|
|
|
|
CloseHandle(m_hFileMapping);
|
|
|
|
CloseHandle(m_hFile);
|
|
|
|
}
|
2024-08-22 10:57:53 +08:00
|
|
|
// Set base address
|
2023-02-12 21:03:36 +08:00
|
|
|
m_pMemoryMappedFileBase = m_hFileMapView;
|
2024-08-22 10:57:53 +08:00
|
|
|
|
2023-02-12 21:03:36 +08:00
|
|
|
#else
|
2023-03-04 14:08:16 +08:00
|
|
|
// create file
|
|
|
|
// we do not need provide mode_t, because is served for new created file.
|
|
|
|
// we are opening a existed file.
|
2024-08-22 10:57:53 +08:00
|
|
|
this->m_hFile = open(YYCC::EncodingHelper::ToOrdinary(m_szFilePath.c_str()), O_RDONLY);
|
2023-03-04 14:08:16 +08:00
|
|
|
if (m_hFile == -1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// calculate file size
|
|
|
|
struct stat sb;
|
|
|
|
int err = fstat(m_hFile, &sb);
|
|
|
|
// if failed or not a regular file, exit
|
|
|
|
if (err == -1 || (sb.st_mode & S_IFMT) != S_IFREG) {
|
|
|
|
close(m_hFile);
|
|
|
|
return;
|
|
|
|
}
|
2024-08-22 10:57:53 +08:00
|
|
|
// Setup size and check its range
|
2023-03-04 14:08:16 +08:00
|
|
|
this->m_offFileSize = sb.st_size;
|
2024-08-22 10:57:53 +08:00
|
|
|
if (this->m_offFileSize > static_cast<off_t>(std::numeric_limits<CKDWORD>::max())) {
|
|
|
|
close(m_hFile);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_cbFile = static_cast<CKDWORD>(this->m_offFileSize);
|
2023-03-04 14:08:16 +08:00
|
|
|
|
|
|
|
// map file
|
|
|
|
this->m_pFileAddr = mmap(
|
|
|
|
nullptr, // request linux distribute one
|
|
|
|
this->m_offFileSize, // map the full file
|
|
|
|
PROT_READ, // only for reading
|
|
|
|
MAP_PRIVATE,
|
|
|
|
this->m_hFile,
|
|
|
|
0 // no offset
|
|
|
|
);
|
|
|
|
if (this->m_pFileAddr == MAP_FAILED) {
|
|
|
|
close(m_hFile);
|
|
|
|
return;
|
|
|
|
}
|
2024-08-22 10:57:53 +08:00
|
|
|
// set base address
|
2023-03-04 14:08:16 +08:00
|
|
|
m_pMemoryMappedFileBase = m_pFileAddr;
|
2024-08-22 10:57:53 +08:00
|
|
|
|
2023-03-04 14:08:16 +08:00
|
|
|
#endif
|
2023-02-10 16:17:17 +08:00
|
|
|
|
|
|
|
// set valid
|
|
|
|
this->m_bIsValid = true;
|
|
|
|
}
|
|
|
|
|
2024-12-28 22:09:26 +08:00
|
|
|
VxMemoryMappedFile::~VxMemoryMappedFile() {
|
2023-02-12 21:03:36 +08:00
|
|
|
if (this->m_bIsValid) {
|
|
|
|
// only success mapping need free
|
|
|
|
this->m_bIsValid = false;
|
|
|
|
m_cbFile = 0;
|
|
|
|
m_pMemoryMappedFileBase = nullptr;
|
2023-02-10 16:17:17 +08:00
|
|
|
|
2024-08-17 20:43:27 +08:00
|
|
|
#if YYCC_OS == YYCC_OS_WINDOWS
|
2023-02-12 21:03:36 +08:00
|
|
|
UnmapViewOfFile(this->m_hFileMapView);
|
|
|
|
CloseHandle(m_hFileMapping);
|
|
|
|
CloseHandle(m_hFile);
|
|
|
|
#else
|
2023-03-04 14:08:16 +08:00
|
|
|
munmap(this->m_pFileAddr, this->m_offFileSize);
|
|
|
|
close(m_hFile);
|
2023-02-12 21:03:36 +08:00
|
|
|
#endif
|
|
|
|
}
|
2023-02-10 16:17:17 +08:00
|
|
|
}
|
|
|
|
|
2024-08-22 10:57:53 +08:00
|
|
|
const void* VxMemoryMappedFile::GetBase() const {
|
|
|
|
if (!this->IsValid())
|
|
|
|
throw RuntimeException("Try to get file base address on an invalid VxMemoryMappedFile");
|
|
|
|
return this->m_pMemoryMappedFileBase;
|
|
|
|
}
|
|
|
|
|
|
|
|
CKDWORD VxMemoryMappedFile::GetFileSize() const {
|
|
|
|
if (!this->IsValid())
|
|
|
|
throw RuntimeException("Try to get file size on an invalid VxMemoryMappedFile");
|
|
|
|
return this->m_cbFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VxMemoryMappedFile::IsValid() const {
|
|
|
|
return this->m_bIsValid;
|
|
|
|
}
|
2023-02-11 15:29:51 +08:00
|
|
|
|
2023-02-10 16:17:17 +08:00
|
|
|
}
|