#include "net/base/file_stream_context.h"
#include "arkweb/build/features/features.h"
#include <utility>
#include "base/files/file_path.h"
#include "base/functional/bind.h"
#include "base/location.h"
#include "base/logging.h"
#include "base/task/task_runner.h"
#include "base/threading/thread_restrictions.h"
#include "base/values.h"
#include "build/build_config.h"
#include "net/base/net_errors.h"
#if BUILDFLAG(IS_MAC)
#include "net/base/apple/guarded_fd.h"
#endif
#if BUILDFLAG(IS_OHOS) && BUILDFLAG(ARKWEB_FILE_UPLOAD)
#include "arkweb/chromium_ext/base/datashare_uri_utils.h"
#endif
#if BUILDFLAG(IS_WIN)
#include <windows.h>
#endif
namespace net {
namespace {
void CallInt64ToInt(CompletionOnceCallback callback, int64_t result) {
std::move(callback).Run(static_cast<int>(result));
}
}
FileStream::Context::IOResult::IOResult()
: result(OK),
os_error(0) {
}
FileStream::Context::IOResult::IOResult(int64_t result,
logging::SystemErrorCode os_error)
: result(result), os_error(os_error) {
}
FileStream::Context::IOResult FileStream::Context::IOResult::FromOSError(
logging::SystemErrorCode os_error) {
return IOResult(MapSystemError(os_error), os_error);
}
FileStream::Context::OpenResult::OpenResult() = default;
FileStream::Context::OpenResult::OpenResult(base::File file,
IOResult error_code)
: file(std::move(file)), error_code(error_code) {}
FileStream::Context::OpenResult::OpenResult(OpenResult&& other)
: file(std::move(other.file)), error_code(other.error_code) {}
FileStream::Context::OpenResult& FileStream::Context::OpenResult::operator=(
OpenResult&& other) {
file = std::move(other.file);
error_code = other.error_code;
return *this;
}
void FileStream::Context::Orphan() {
DCHECK(!orphaned_);
orphaned_ = true;
if (!async_in_progress_) {
CloseAndDelete();
} else if (file_.IsValid()) {
#if BUILDFLAG(IS_WIN)
CancelIo(file_.GetPlatformFile());
#endif
}
}
void FileStream::Context::Open(const base::FilePath& path,
int open_flags,
CompletionOnceCallback callback) {
DCHECK(!async_in_progress_);
bool posted = task_runner_->PostTaskAndReplyWithResult(
FROM_HERE,
base::BindOnce(&Context::OpenFileImpl, base::Unretained(this), path,
open_flags),
base::BindOnce(&Context::OnOpenCompleted, base::Unretained(this),
std::move(callback)));
DCHECK(posted);
async_in_progress_ = true;
}
void FileStream::Context::Close(CompletionOnceCallback callback) {
DCHECK(!async_in_progress_);
bool posted = task_runner_->PostTaskAndReplyWithResult(
FROM_HERE,
base::BindOnce(&Context::CloseFileImpl, base::Unretained(this)),
base::BindOnce(&Context::OnAsyncCompleted, base::Unretained(this),
IntToInt64(std::move(callback))));
DCHECK(posted);
async_in_progress_ = true;
}
void FileStream::Context::Seek(int64_t offset,
Int64CompletionOnceCallback callback) {
DCHECK(!async_in_progress_);
if (offset < 0) {
std::move(callback).Run(net::ERR_INVALID_ARGUMENT);
return;
}
bool posted = task_runner_->PostTaskAndReplyWithResult(
FROM_HERE,
base::BindOnce(&Context::SeekFileImpl, base::Unretained(this), offset),
base::BindOnce(&Context::OnAsyncCompleted, base::Unretained(this),
std::move(callback)));
DCHECK(posted);
async_in_progress_ = true;
}
void FileStream::Context::GetFileInfo(base::File::Info* file_info,
CompletionOnceCallback callback) {
task_runner_->PostTaskAndReplyWithResult(
FROM_HERE,
base::BindOnce(&Context::GetFileInfoImpl, base::Unretained(this),
base::Unretained(file_info)),
base::BindOnce(&Context::OnAsyncCompleted, base::Unretained(this),
IntToInt64(std::move(callback))));
async_in_progress_ = true;
}
void FileStream::Context::Flush(CompletionOnceCallback callback) {
DCHECK(!async_in_progress_);
bool posted = task_runner_->PostTaskAndReplyWithResult(
FROM_HERE,
base::BindOnce(&Context::FlushFileImpl, base::Unretained(this)),
base::BindOnce(&Context::OnAsyncCompleted, base::Unretained(this),
IntToInt64(std::move(callback))));
DCHECK(posted);
async_in_progress_ = true;
}
bool FileStream::Context::IsOpen() const {
return file_.IsValid();
}
FileStream::Context::OpenResult FileStream::Context::OpenFileImpl(
const base::FilePath& path, int open_flags) {
#if BUILDFLAG(IS_POSIX)
open_flags &= ~base::File::FLAG_ASYNC;
#endif
#if BUILDFLAG(IS_WIN)
open_flags |= base::File::FLAG_WIN_SHARE_DELETE;
#endif
base::File file(path, open_flags);
#if BUILDFLAG(IS_OHOS) && BUILDFLAG(ARKWEB_FILE_UPLOAD)
if (path.IsDataShareUri()) {
file = base::OpenDatashareUriForRead(path);
} else {
open_flags |= base::File::FLAG_WIN_SHARE_DELETE;
file.Initialize(path, open_flags);
}
#endif
if (!file.IsValid()) {
return OpenResult(base::File(),
IOResult::FromOSError(logging::GetLastSystemErrorCode()));
}
return OpenResult(std::move(file), IOResult(OK, 0));
}
FileStream::Context::IOResult FileStream::Context::GetFileInfoImpl(
base::File::Info* file_info) {
bool result = file_.GetInfo(file_info);
if (!result)
return IOResult::FromOSError(logging::GetLastSystemErrorCode());
return IOResult(OK, 0);
}
FileStream::Context::IOResult FileStream::Context::CloseFileImpl() {
#if BUILDFLAG(IS_MAC)
if (file_.IsValid()) {
guardid_t guardid = reinterpret_cast<guardid_t>(this);
PCHECK(change_fdguard_np(file_.GetPlatformFile(), &guardid,
GUARD_CLOSE | GUARD_DUP,
nullptr, 0,
nullptr) == 0);
}
#endif
file_.Close();
return IOResult(OK, 0);
}
FileStream::Context::IOResult FileStream::Context::FlushFileImpl() {
if (file_.Flush())
return IOResult(OK, 0);
return IOResult::FromOSError(logging::GetLastSystemErrorCode());
}
void FileStream::Context::OnOpenCompleted(CompletionOnceCallback callback,
OpenResult open_result) {
file_ = std::move(open_result.file);
if (file_.IsValid() && !orphaned_)
OnFileOpened();
#if BUILDFLAG(IS_MAC)
if (file_.IsValid()) {
guardid_t guardid = reinterpret_cast<guardid_t>(this);
PCHECK(change_fdguard_np(file_.GetPlatformFile(), nullptr,
0, &guardid,
GUARD_CLOSE | GUARD_DUP,
nullptr) == 0);
}
#endif
OnAsyncCompleted(IntToInt64(std::move(callback)), open_result.error_code);
}
void FileStream::Context::CloseAndDelete() {
DCHECK(!async_in_progress_);
if (file_.IsValid()) {
bool posted = task_runner_.get()->PostTask(
FROM_HERE, base::BindOnce(base::IgnoreResult(&Context::CloseFileImpl),
base::Owned(this)));
DCHECK(posted);
} else {
delete this;
}
}
Int64CompletionOnceCallback FileStream::Context::IntToInt64(
CompletionOnceCallback callback) {
return base::BindOnce(&CallInt64ToInt, std::move(callback));
}
void FileStream::Context::OnAsyncCompleted(Int64CompletionOnceCallback callback,
const IOResult& result) {
async_in_progress_ = false;
if (orphaned_) {
CloseAndDelete();
} else {
std::move(callback).Run(result.result);
}
}
}