FileInfo.hxx 3.24 KB
Newer Older
1
/*
2
 * Copyright 2003-2018 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
 * http://www.musicpd.org
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

#ifndef MPD_FS_FILE_INFO_HXX
#define MPD_FS_FILE_INFO_HXX

#include "Path.hxx"
24
#include "system/Error.hxx"
25

26
#ifdef _WIN32
27 28
#include <fileapi.h>
#else
29
#include <sys/stat.h>
30 31
#endif

32 33 34 35
#include <chrono>

#include <stdint.h>

36
#ifdef _WIN32
37

38
static constexpr uint64_t
39 40 41 42 43 44 45 46 47 48 49 50
ConstructUint64(DWORD lo, DWORD hi)
{
	return uint64_t(lo) | (uint64_t(hi) << 32);
}

static constexpr time_t
FileTimeToTimeT(FILETIME ft)
{
	return (ConstructUint64(ft.dwLowDateTime, ft.dwHighDateTime)
		- 116444736000000000) / 10000000;
}

51 52 53 54 55 56 57
static std::chrono::system_clock::time_point
FileTimeToChrono(FILETIME ft)
{
	// TODO: eliminate the time_t roundtrip, preserve sub-second resolution
	return std::chrono::system_clock::from_time_t(FileTimeToTimeT(ft));
}

58
#endif
59 60 61 62

class FileInfo {
	friend bool GetFileInfo(Path path, FileInfo &info,
				bool follow_symlinks);
63
	friend class FileReader;
64

65
#ifdef _WIN32
66 67
	WIN32_FILE_ATTRIBUTE_DATA data;
#else
68
	struct stat st;
69
#endif
70 71

public:
72 73 74 75
	FileInfo() = default;

	FileInfo(Path path, bool follow_symlinks=true) {
		if (!GetFileInfo(path, *this, follow_symlinks)) {
76
#ifdef _WIN32
77 78 79 80 81 82 83 84 85
			throw FormatLastError("Failed to access %s",
					      path.ToUTF8().c_str());
#else
			throw FormatErrno("Failed to access %s",
					  path.ToUTF8().c_str());
#endif
		}
	}

86
	bool IsRegular() const {
87
#ifdef _WIN32
88 89
		return (data.dwFileAttributes &
			(FILE_ATTRIBUTE_DIRECTORY|FILE_ATTRIBUTE_DEVICE)) == 0;
90
#else
91
		return S_ISREG(st.st_mode);
92
#endif
93 94 95
	}

	bool IsDirectory() const {
96
#ifdef _WIN32
97 98
		return data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY;
#else
99
		return S_ISDIR(st.st_mode);
100
#endif
101 102 103
	}

	uint64_t GetSize() const {
104
#ifdef _WIN32
105 106
		return ConstructUint64(data.nFileSizeLow, data.nFileSizeHigh);
#else
107
		return st.st_size;
108
#endif
109 110
	}

111
	std::chrono::system_clock::time_point GetModificationTime() const {
112
#ifdef _WIN32
113
		return FileTimeToChrono(data.ftLastWriteTime);
114
#else
115
		return std::chrono::system_clock::from_time_t(st.st_mtime);
116
#endif
117 118
	}

119
#ifndef _WIN32
120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
	uid_t GetUid() const {
		return st.st_uid;
	}

	mode_t GetMode() const {
		return st.st_mode;
	}

	dev_t GetDevice() const {
		return st.st_dev;
	}

	ino_t GetInode() const {
		return st.st_ino;
	}
#endif
};

inline bool
GetFileInfo(Path path, FileInfo &info, bool follow_symlinks=true)
{
141
#ifdef _WIN32
142
	(void)follow_symlinks;
143 144
	return GetFileAttributesEx(path.c_str(), GetFileExInfoStandard,
				   &info.data);
145 146 147 148 149 150 151 152 153
#else
	int ret = follow_symlinks
		? stat(path.c_str(), &info.st)
		: lstat(path.c_str(), &info.st);
	return ret == 0;
#endif
}

#endif