2017-02-27 11:32:40 +01:00
|
|
|
#include "pch.h"
|
|
|
|
#include "ppltasks.h"
|
2017-03-02 17:07:12 +01:00
|
|
|
#include "base/logging.h"
|
|
|
|
#include "file/file_util.h"
|
|
|
|
#include "thread/threadutil.h"
|
2017-02-27 11:32:40 +01:00
|
|
|
#include "StorageFileLoader.h"
|
2017-03-02 17:07:12 +01:00
|
|
|
#include "UWPUtil.h"
|
2017-02-27 11:32:40 +01:00
|
|
|
|
|
|
|
using namespace Concurrency;
|
|
|
|
using namespace Windows::Storage;
|
|
|
|
using namespace Windows::Storage::Streams;
|
|
|
|
|
2017-03-13 10:31:58 +01:00
|
|
|
static std::mutex initMutex;
|
2017-03-03 22:48:05 +01:00
|
|
|
|
2017-02-27 11:32:40 +01:00
|
|
|
StorageFileLoader::StorageFileLoader(Windows::Storage::StorageFile ^file) {
|
2017-02-28 01:47:13 +01:00
|
|
|
file_ = file;
|
2017-03-02 17:07:12 +01:00
|
|
|
path_ = FromPlatformString(file_->Path);
|
|
|
|
thread_.reset(new std::thread([this]() { this->threadfunc(); }));
|
2017-02-27 11:32:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
StorageFileLoader::~StorageFileLoader() {
|
|
|
|
active_ = false;
|
2017-03-02 17:07:12 +01:00
|
|
|
operationRequested_ = false;
|
|
|
|
cond_.notify_all();
|
|
|
|
thread_->join();
|
2017-02-27 11:32:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void StorageFileLoader::threadfunc() {
|
2017-03-02 17:07:12 +01:00
|
|
|
setCurrentThreadName("StorageFileLoader");
|
|
|
|
|
|
|
|
initMutex.lock();
|
|
|
|
|
|
|
|
auto opentask = create_task(file_->OpenReadAsync()).then([this](IRandomAccessStreamWithContentType ^stream) {
|
|
|
|
stream_ = stream;
|
|
|
|
active_ = true;
|
|
|
|
});
|
|
|
|
|
|
|
|
try {
|
|
|
|
opentask.wait();
|
|
|
|
}
|
|
|
|
catch (const std::exception& e) {
|
|
|
|
operationFailed_ = true;
|
|
|
|
// TODO: What do we do?
|
|
|
|
const char *what = e.what();
|
|
|
|
ILOG("%s", what);
|
|
|
|
}
|
2017-03-03 22:48:05 +01:00
|
|
|
catch (Platform::COMException ^e) {
|
|
|
|
|
|
|
|
}
|
2017-03-02 17:07:12 +01:00
|
|
|
|
|
|
|
auto sizetask = create_task(file_->GetBasicPropertiesAsync()).then([this](Windows::Storage::FileProperties::BasicProperties ^props) {
|
|
|
|
size_ = props->Size;
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
sizetask.wait();
|
|
|
|
}
|
|
|
|
catch (const std::exception& e) {
|
|
|
|
const char *what = e.what();
|
|
|
|
ILOG("%s", what);
|
|
|
|
}
|
2017-03-03 22:48:05 +01:00
|
|
|
catch (Platform::COMException ^e) {
|
|
|
|
std::string what = FromPlatformString(e->ToString());
|
2017-03-05 01:39:26 +01:00
|
|
|
ILOG("%s", what.c_str());
|
2017-03-03 22:48:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
initMutex.unlock();
|
2017-03-02 17:07:12 +01:00
|
|
|
|
2017-02-27 11:32:40 +01:00
|
|
|
std::unique_lock<std::mutex> lock(mutex_);
|
|
|
|
while (active_) {
|
2017-03-02 17:07:12 +01:00
|
|
|
if (!operationRequested_) {
|
|
|
|
cond_.wait(lock);
|
|
|
|
}
|
|
|
|
if (operationRequested_) {
|
|
|
|
switch (operation_.type) {
|
|
|
|
case OpType::READ_AT: {
|
|
|
|
Streams::Buffer ^buf = ref new Streams::Buffer(operation_.size);
|
|
|
|
operationFailed_ = false;
|
|
|
|
stream_->Seek(operation_.offset);
|
|
|
|
auto task = create_task(stream_->ReadAsync(buf, operation_.size, Streams::InputStreamOptions::None));
|
|
|
|
Streams::IBuffer ^output = nullptr;
|
|
|
|
try {
|
|
|
|
task.wait();
|
|
|
|
output = task.get();
|
|
|
|
} catch (const std::exception& e) {
|
|
|
|
operationFailed_ = true;
|
|
|
|
const char *what = e.what();
|
|
|
|
ILOG("%s", what);
|
|
|
|
}
|
|
|
|
operationRequested_ = false;
|
|
|
|
std::unique_lock<std::mutex> lock(mutexResponse_);
|
|
|
|
response_.buffer = output;
|
|
|
|
responseAvailable_ = true;
|
|
|
|
condResponse_.notify_one();
|
2017-02-27 11:32:40 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
2017-03-02 17:07:12 +01:00
|
|
|
ELOG("Unknown operation");
|
|
|
|
operationRequested_ = false;
|
2017-02-27 11:32:40 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool StorageFileLoader::Exists() {
|
2017-03-02 17:07:12 +01:00
|
|
|
return file_ != nullptr;
|
2017-02-27 11:32:40 +01:00
|
|
|
}
|
2017-03-02 17:07:12 +01:00
|
|
|
|
2017-02-27 11:32:40 +01:00
|
|
|
bool StorageFileLoader::ExistsFast() {
|
2017-03-02 17:07:12 +01:00
|
|
|
return file_ != nullptr;
|
2017-02-27 11:32:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool StorageFileLoader::IsDirectory() {
|
2017-02-28 01:47:13 +01:00
|
|
|
return (file_->Attributes & Windows::Storage::FileAttributes::Directory) != Windows::Storage::FileAttributes::Normal;
|
2017-02-27 11:32:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
s64 StorageFileLoader::FileSize() {
|
2017-03-02 17:07:12 +01:00
|
|
|
EnsureOpen();
|
2017-02-28 01:47:13 +01:00
|
|
|
if (size_ == -1)
|
|
|
|
__debugbreak(); // crude race condition detection
|
|
|
|
return size_;
|
2017-02-27 11:32:40 +01:00
|
|
|
}
|
|
|
|
|
2017-02-28 01:47:13 +01:00
|
|
|
std::string StorageFileLoader::Path() const {
|
2017-03-02 17:07:12 +01:00
|
|
|
return path_;
|
2017-02-28 01:47:13 +01:00
|
|
|
}
|
2017-02-27 11:32:40 +01:00
|
|
|
|
2017-02-28 01:47:13 +01:00
|
|
|
std::string StorageFileLoader::Extension() {
|
2017-03-02 17:07:12 +01:00
|
|
|
return "." + getFileExtension(path_);
|
2017-02-28 01:47:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void StorageFileLoader::EnsureOpen() {
|
2017-03-02 17:07:12 +01:00
|
|
|
// UGLY!
|
|
|
|
while (!thread_)
|
|
|
|
Sleep(100);
|
2017-02-28 01:47:13 +01:00
|
|
|
while (size_ == -1)
|
|
|
|
Sleep(100);
|
|
|
|
}
|
2017-02-27 11:32:40 +01:00
|
|
|
|
|
|
|
size_t StorageFileLoader::ReadAt(s64 absolutePos, size_t bytes, size_t count, void *data, Flags flags) {
|
2017-02-28 01:47:13 +01:00
|
|
|
EnsureOpen();
|
2017-03-09 11:01:30 +01:00
|
|
|
if (operationRequested_ || responseAvailable_)
|
2017-03-02 17:07:12 +01:00
|
|
|
Crash();
|
2017-02-28 01:47:13 +01:00
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(mutex_);
|
2017-03-02 17:07:12 +01:00
|
|
|
operation_.type = OpType::READ_AT;
|
|
|
|
operation_.offset = absolutePos;
|
|
|
|
operation_.size = (int64_t)(bytes * count);
|
|
|
|
operationRequested_ = true;
|
2017-02-28 01:47:13 +01:00
|
|
|
cond_.notify_one();
|
|
|
|
}
|
2017-03-02 17:07:12 +01:00
|
|
|
|
2017-02-28 01:47:13 +01:00
|
|
|
// OK, now wait for response...
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> responseLock(mutexResponse_);
|
2017-03-09 11:01:30 +01:00
|
|
|
while (!responseAvailable_) {
|
|
|
|
condResponse_.wait(responseLock);
|
|
|
|
}
|
|
|
|
if (operationFailed_) {
|
|
|
|
return 0;
|
|
|
|
}
|
2017-03-02 17:07:12 +01:00
|
|
|
DataReader ^rd = DataReader::FromBuffer(response_.buffer);
|
|
|
|
size_t len = response_.buffer->Length;
|
|
|
|
Platform::Array<uint8_t> ^bytearray = ref new Platform::Array<uint8_t>((unsigned int)len);
|
|
|
|
rd->ReadBytes(bytearray);
|
|
|
|
memcpy(data, bytearray->Data, len);
|
|
|
|
responseAvailable_ = false;
|
|
|
|
response_.buffer = nullptr;
|
|
|
|
return len / bytes;
|
2017-02-28 01:47:13 +01:00
|
|
|
}
|
2017-02-27 11:32:40 +01:00
|
|
|
}
|
2017-03-02 17:07:12 +01:00
|
|
|
|
|
|
|
FileLoader *StorageFileLoaderFactory::ConstructFileLoader(const std::string &filename) {
|
|
|
|
return file_ ? new StorageFileLoader(file_) : nullptr;
|
|
|
|
}
|