aboutsummaryrefslogblamecommitdiffstatshomepage
path: root/src/file.cpp
blob: 4f1d2a94bdb3247bd0f71362c9ace0e58143dcf1 (plain) (tree)
1
2
3
4
5
6
7
8
9







                                                                 
                          

                          

                  
                  
                    




                  







                                                        








                                                     






                                                                















                                                   
                                                                          




                                                    
                                                   










                                                                   
                                

 







                                                            

              
                                            


                                                                     
                                              
                                                                     

 
                                                          


                                                                                
                                                            


                                                                                















                                                                      























                                                                              
                     
// Copyright (c) 2020 Egor Tensin <Egor.Tensin@gmail.com>
// This file is part of the "winapi-common" project.
// For details, see https://github.com/egor-tensin/winapi-common.
// Distributed under the MIT License.

#include <winapi/error.hpp>
#include <winapi/file.hpp>
#include <winapi/handle.hpp>
#include <winapi/path.hpp>
#include <winapi/utf8.hpp>

#include <cstddef>
#include <cstdint>
#include <cstring>
#include <stdexcept>
#include <string>

namespace winapi {
namespace {

std::wstring to_system_path(const std::string& path) {
    return widen(path);
}

std::wstring to_system_path(const CanonicalPath& path) {
    return widen(R"(\\?\)" + path.get());
}

struct CreateFileParams {
    static CreateFileParams read() {
        CreateFileParams params;
        params.dwDesiredAccess = GENERIC_READ;
        params.dwShareMode = FILE_SHARE_READ;
        params.dwCreationDisposition = OPEN_EXISTING;
        return params;
    }

    static CreateFileParams read_attributes() {
        auto params = read();
        params.dwDesiredAccess = FILE_READ_ATTRIBUTES;
        params.dwShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE;
        return params;
    }

    static CreateFileParams write() {
        CreateFileParams params;
        params.dwDesiredAccess = GENERIC_WRITE;
        params.dwShareMode = FILE_SHARE_READ;
        params.dwCreationDisposition = OPEN_ALWAYS;
        return params;
    }

    DWORD dwDesiredAccess = 0;
    DWORD dwShareMode = 0;
    DWORD dwCreationDisposition = 0;

private:
    CreateFileParams() = default;
};

File open_file(const std::wstring& path, const CreateFileParams& params) {
    SECURITY_ATTRIBUTES attributes;
    std::memset(&attributes, 0, sizeof(attributes));
    attributes.nLength = sizeof(attributes);
    attributes.bInheritHandle = TRUE;

    const auto handle = ::CreateFileW(path.c_str(),
                                      params.dwDesiredAccess,
                                      params.dwShareMode,
                                      &attributes,
                                      params.dwCreationDisposition,
                                      FILE_ATTRIBUTE_NORMAL,
                                      NULL);

    if (handle == INVALID_HANDLE_VALUE) {
        throw error::windows(GetLastError(), "CreateFileW");
    }

    return File{Handle{handle}};
}

void remove_file(const std::wstring& path) {
    const auto ret = ::DeleteFileW(path.c_str());

    if (!ret) {
        throw error::windows(GetLastError(), "DeleteFileW");
    }
}

} // namespace

File File::open_r(const std::string& path) {
    return open_file(to_system_path(path), CreateFileParams::read());
}

File File::open_r(const CanonicalPath& path) {
    return open_file(to_system_path(path), CreateFileParams::read());
}

File File::open_read_attributes(const std::string& path) {
    return open_file(to_system_path(path), CreateFileParams::read_attributes());
}

File File::open_read_attributes(const CanonicalPath& path) {
    return open_file(to_system_path(path), CreateFileParams::read_attributes());
}

File File::open_w(const std::string& path) {
    return open_file(to_system_path(path), CreateFileParams::write());
}

File File::open_w(const CanonicalPath& path) {
    return open_file(to_system_path(path), CreateFileParams::write());
}

void File::remove(const std::string& path) {
    remove_file(to_system_path(path));
}

void File::remove(const CanonicalPath& path) {
    remove_file(to_system_path(path));
}

std::size_t File::get_size() const {
    LARGE_INTEGER size;

    if (!GetFileSizeEx(get(), &size))
        throw error::windows(GetLastError(), "GetFileSizeEx");

    if (size.QuadPart < 0 || size.QuadPart > SIZE_MAX)
        throw std::runtime_error{"invalid file size"};
    return static_cast<std::size_t>(size.QuadPart);
}

bool operator==(const FILE_ID_128& a, const FILE_ID_128& b) {
    return 0 == std::memcmp(a.Identifier, b.Identifier, sizeof(a.Identifier));
}

File::ID File::query_id() const {
    FILE_ID_INFO id;

    if (!GetFileInformationByHandleEx(get(), FileIdInfo, &id, sizeof(id)))
        throw error::windows(GetLastError(), "GetFileInformationByHandleEx");

    return {id};
}

} // namespace winapi