зеркало из https://github.com/mozilla/gecko-dev.git
194 строки
5.2 KiB
C++
194 строки
5.2 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "mozilla/dom/FileSystemRequestParent.h"
|
|
#include "mozilla/dom/PFileSystemParams.h"
|
|
|
|
#include "GetDirectoryListingTask.h"
|
|
#include "GetFileOrDirectoryTask.h"
|
|
|
|
#include "mozilla/dom/ContentParent.h"
|
|
#include "mozilla/dom/FileSystemBase.h"
|
|
#include "mozilla/dom/FileSystemSecurity.h"
|
|
#include "mozilla/ipc/BackgroundParent.h"
|
|
#include "mozilla/Unused.h"
|
|
#include "nsProxyRelease.h"
|
|
|
|
using namespace mozilla::ipc;
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
FileSystemRequestParent::FileSystemRequestParent()
|
|
: mDestroyed(false)
|
|
{
|
|
AssertIsOnBackgroundThread();
|
|
}
|
|
|
|
FileSystemRequestParent::~FileSystemRequestParent()
|
|
{
|
|
AssertIsOnBackgroundThread();
|
|
}
|
|
|
|
#define FILESYSTEM_REQUEST_PARENT_DISPATCH_ENTRY(name) \
|
|
case FileSystemParams::TFileSystem##name##Params: { \
|
|
const FileSystem##name##Params& p = aParams; \
|
|
mFileSystem = new OSFileSystemParent(p.filesystem()); \
|
|
MOZ_ASSERT(mFileSystem); \
|
|
mTask = name##TaskParent::Create(mFileSystem, p, this, rv); \
|
|
if (NS_WARN_IF(rv.Failed())) { \
|
|
rv.SuppressException(); \
|
|
return false; \
|
|
} \
|
|
break; \
|
|
}
|
|
|
|
bool
|
|
FileSystemRequestParent::Initialize(const FileSystemParams& aParams)
|
|
{
|
|
AssertIsOnBackgroundThread();
|
|
|
|
ErrorResult rv;
|
|
|
|
switch (aParams.type()) {
|
|
|
|
FILESYSTEM_REQUEST_PARENT_DISPATCH_ENTRY(GetDirectoryListing)
|
|
FILESYSTEM_REQUEST_PARENT_DISPATCH_ENTRY(GetFileOrDirectory)
|
|
FILESYSTEM_REQUEST_PARENT_DISPATCH_ENTRY(GetFiles)
|
|
|
|
default: {
|
|
MOZ_CRASH("not reached");
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (NS_WARN_IF(!mTask || !mFileSystem)) {
|
|
// Should never reach here.
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
namespace {
|
|
|
|
class CheckPermissionRunnable final : public Runnable
|
|
{
|
|
public:
|
|
CheckPermissionRunnable(already_AddRefed<ContentParent> aParent,
|
|
FileSystemRequestParent* aActor,
|
|
FileSystemTaskParentBase* aTask,
|
|
const nsAString& aPath)
|
|
: mContentParent(aParent)
|
|
, mActor(aActor)
|
|
, mTask(aTask)
|
|
, mPath(aPath)
|
|
, mBackgroundEventTarget(NS_GetCurrentThread())
|
|
{
|
|
AssertIsInMainProcess();
|
|
AssertIsOnBackgroundThread();
|
|
|
|
MOZ_ASSERT(mContentParent);
|
|
MOZ_ASSERT(mActor);
|
|
MOZ_ASSERT(mTask);
|
|
MOZ_ASSERT(mBackgroundEventTarget);
|
|
}
|
|
|
|
NS_IMETHOD
|
|
Run() override
|
|
{
|
|
if (NS_IsMainThread()) {
|
|
auto raii = mozilla::MakeScopeExit([&] { mContentParent = nullptr; });
|
|
|
|
|
|
if (!mozilla::Preferences::GetBool("dom.filesystem.pathcheck.disabled", false)) {
|
|
RefPtr<FileSystemSecurity> fss = FileSystemSecurity::Get();
|
|
if (NS_WARN_IF(!fss ||
|
|
!fss->ContentProcessHasAccessTo(mContentParent->ChildID(),
|
|
mPath))) {
|
|
mContentParent->KillHard("This path is not allowed.");
|
|
return NS_OK;
|
|
}
|
|
}
|
|
|
|
return mBackgroundEventTarget->Dispatch(this, NS_DISPATCH_NORMAL);
|
|
}
|
|
|
|
AssertIsOnBackgroundThread();
|
|
|
|
// It can happen that this actor has been destroyed in the meantime we were
|
|
// on the main-thread.
|
|
if (!mActor->Destroyed()) {
|
|
mTask->Start();
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
~CheckPermissionRunnable()
|
|
{
|
|
NS_ProxyRelease(mBackgroundEventTarget, mActor.forget());
|
|
}
|
|
|
|
RefPtr<ContentParent> mContentParent;
|
|
RefPtr<FileSystemRequestParent> mActor;
|
|
RefPtr<FileSystemTaskParentBase> mTask;
|
|
const nsString mPath;
|
|
|
|
nsCOMPtr<nsIEventTarget> mBackgroundEventTarget;
|
|
};
|
|
|
|
} // anonymous
|
|
|
|
void
|
|
FileSystemRequestParent::Start()
|
|
{
|
|
AssertIsInMainProcess();
|
|
AssertIsOnBackgroundThread();
|
|
|
|
MOZ_ASSERT(!mDestroyed);
|
|
MOZ_ASSERT(mFileSystem);
|
|
MOZ_ASSERT(mTask);
|
|
|
|
nsAutoString path;
|
|
if (NS_WARN_IF(NS_FAILED(mTask->GetTargetPath(path)))) {
|
|
Unused << Send__delete__(this, FileSystemErrorResponse(NS_ERROR_DOM_SECURITY_ERR));
|
|
return;
|
|
}
|
|
|
|
RefPtr<ContentParent> parent = BackgroundParent::GetContentParent(Manager());
|
|
|
|
// If the ContentParent is null we are dealing with a same-process actor.
|
|
if (!parent) {
|
|
mTask->Start();
|
|
return;
|
|
}
|
|
|
|
RefPtr<Runnable> runnable =
|
|
new CheckPermissionRunnable(parent.forget(), this, mTask, path);
|
|
NS_DispatchToMainThread(runnable);
|
|
}
|
|
|
|
void
|
|
FileSystemRequestParent::ActorDestroy(ActorDestroyReason aWhy)
|
|
{
|
|
AssertIsOnBackgroundThread();
|
|
MOZ_ASSERT(!mDestroyed);
|
|
|
|
if (!mFileSystem) {
|
|
return;
|
|
}
|
|
|
|
mFileSystem->Shutdown();
|
|
mFileSystem = nullptr;
|
|
mTask = nullptr;
|
|
mDestroyed = true;
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|