mpd/src/fs/io/FileReader.cxx

143 lines
3.0 KiB
C++
Raw Normal View History

2014-08-07 18:10:23 +02:00
/*
2015-01-01 19:48:13 +01:00
* Copyright (C) 2003-2015 The Music Player Daemon Project
2014-08-07 18:10:23 +02:00
* 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.
*/
#include "config.h"
#include "FileReader.hxx"
#include "fs/FileInfo.hxx"
2014-08-07 18:10:23 +02:00
#include "util/Error.hxx"
#ifdef WIN32
FileReader::FileReader(Path _path, Error &error)
:path(_path),
handle(CreateFile(path.c_str(), GENERIC_READ, FILE_SHARE_READ,
nullptr, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL,
nullptr))
{
if (handle == INVALID_HANDLE_VALUE) {
const auto path_utf8 = path.ToUTF8();
error.FormatLastError("Failed to open %s", path_utf8.c_str());
}
2014-08-07 18:10:23 +02:00
}
bool
FileReader::GetFileInfo(FileInfo &info, Error &error) const
{
assert(IsDefined());
return ::GetFileInfo(path, info, error);
}
2014-08-07 18:10:23 +02:00
size_t
FileReader::Read(void *data, size_t size, Error &error)
{
assert(IsDefined());
DWORD nbytes;
if (!ReadFile(handle, data, size, &nbytes, nullptr)) {
const auto path_utf8 = path.ToUTF8();
error.FormatLastError("Failed to read from %s",
path_utf8.c_str());
2014-08-07 18:10:23 +02:00
nbytes = 0;
}
return nbytes;
}
2015-03-03 14:29:36 +01:00
bool
FileReader::Seek(off_t offset, Error &error)
{
assert(IsDefined());
auto result = SetFilePointer(handle, offset, nullptr, FILE_BEGIN);
const bool success = result != INVALID_SET_FILE_POINTER;
if (!success)
error.SetLastError("Failed to seek");
return success;
}
2014-08-07 18:10:23 +02:00
void
FileReader::Close()
{
assert(IsDefined());
CloseHandle(handle);
}
#else
FileReader::FileReader(Path _path, Error &error)
:path(_path)
2014-08-07 18:10:23 +02:00
{
fd.OpenReadOnly(path.c_str());
if (!fd.IsDefined())
2014-08-07 18:10:23 +02:00
error.FormatErrno("Failed to open %s", path.c_str());
}
bool
FileReader::GetFileInfo(FileInfo &info, Error &error) const
{
assert(IsDefined());
const bool success = fstat(fd.Get(), &info.st) == 0;
if (!success)
error.FormatErrno("Failed to access %s",
path.ToUTF8().c_str());
return success;
}
2014-08-07 18:10:23 +02:00
size_t
FileReader::Read(void *data, size_t size, Error &error)
{
assert(IsDefined());
ssize_t nbytes = fd.Read(data, size);
2014-08-07 18:10:23 +02:00
if (nbytes < 0) {
error.FormatErrno("Failed to read from %s", path.c_str());
nbytes = 0;
}
return nbytes;
}
2015-03-03 14:29:36 +01:00
bool
FileReader::Seek(off_t offset, Error &error)
{
assert(IsDefined());
auto result = fd.Seek(offset);
2015-03-03 14:29:36 +01:00
const bool success = result >= 0;
if (!success)
error.SetErrno("Failed to seek");
return success;
}
2014-08-07 18:10:23 +02:00
void
FileReader::Close()
{
assert(IsDefined());
fd.Close();
2014-08-07 18:10:23 +02:00
}
#endif