Blame netwerk/protocol/ftp/FTPChannelParent.cpp

Packit f0b94e
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
Packit f0b94e
/* vim: set sw=2 ts=8 et tw=80 : */
Packit f0b94e
Packit f0b94e
/* This Source Code Form is subject to the terms of the Mozilla Public
Packit f0b94e
 * License, v. 2.0. If a copy of the MPL was not distributed with this
Packit f0b94e
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
Packit f0b94e
Packit f0b94e
#include "mozilla/net/FTPChannelParent.h"
Packit f0b94e
#include "nsStringStream.h"
Packit f0b94e
#include "mozilla/net/ChannelEventQueue.h"
Packit f0b94e
#include "mozilla/dom/TabParent.h"
Packit f0b94e
#include "nsFTPChannel.h"
Packit f0b94e
#include "nsNetCID.h"
Packit f0b94e
#include "nsNetUtil.h"
Packit f0b94e
#include "nsQueryObject.h"
Packit f0b94e
#include "nsFtpProtocolHandler.h"
Packit f0b94e
#include "nsIAuthPrompt.h"
Packit f0b94e
#include "nsIAuthPromptProvider.h"
Packit f0b94e
#include "nsIEncodedChannel.h"
Packit f0b94e
#include "nsIHttpChannelInternal.h"
Packit f0b94e
#include "nsIForcePendingChannel.h"
Packit f0b94e
#include "mozilla/ipc/IPCStreamUtils.h"
Packit f0b94e
#include "mozilla/ipc/URIUtils.h"
Packit f0b94e
#include "mozilla/Unused.h"
Packit f0b94e
#include "SerializedLoadContext.h"
Packit f0b94e
#include "nsIContentPolicy.h"
Packit f0b94e
#include "mozilla/ipc/BackgroundUtils.h"
Packit f0b94e
#include "mozilla/LoadInfo.h"
Packit f0b94e
#include "mozilla/dom/ContentParent.h"
Packit f0b94e
Packit f0b94e
using namespace mozilla::dom;
Packit f0b94e
using namespace mozilla::ipc;
Packit f0b94e
Packit f0b94e
#undef LOG
Packit f0b94e
#define LOG(args) MOZ_LOG(gFTPLog, mozilla::LogLevel::Debug, args)
Packit f0b94e
Packit f0b94e
namespace mozilla {
Packit f0b94e
namespace net {
Packit f0b94e
Packit f0b94e
FTPChannelParent::FTPChannelParent(const PBrowserOrId& aIframeEmbedding,
Packit f0b94e
                                   nsILoadContext* aLoadContext,
Packit f0b94e
                                   PBOverrideStatus aOverrideStatus)
Packit f0b94e
    : mIPCClosed(false),
Packit f0b94e
      mLoadContext(aLoadContext),
Packit f0b94e
      mPBOverride(aOverrideStatus),
Packit f0b94e
      mStatus(NS_OK),
Packit f0b94e
      mDivertingFromChild(false),
Packit f0b94e
      mDivertedOnStartRequest(false),
Packit f0b94e
      mSuspendedForDiversion(false),
Packit f0b94e
      mUseUTF8(false) {
Packit f0b94e
  nsIProtocolHandler* handler;
Packit f0b94e
  CallGetService(NS_NETWORK_PROTOCOL_CONTRACTID_PREFIX "ftp", &handler);
Packit f0b94e
  MOZ_ASSERT(handler, "no ftp handler");
Packit f0b94e
Packit f0b94e
  if (aIframeEmbedding.type() == PBrowserOrId::TPBrowserParent) {
Packit f0b94e
    mTabParent =
Packit f0b94e
        static_cast<dom::TabParent*>(aIframeEmbedding.get_PBrowserParent());
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  mEventQ = new ChannelEventQueue(static_cast<nsIParentChannel*>(this));
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
FTPChannelParent::~FTPChannelParent() { gFtpHandler->Release(); }
Packit f0b94e
Packit f0b94e
void FTPChannelParent::ActorDestroy(ActorDestroyReason why) {
Packit f0b94e
  // We may still have refcount>0 if the channel hasn't called OnStopRequest
Packit f0b94e
  // yet, but we must not send any more msgs to child.
Packit f0b94e
  mIPCClosed = true;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
// FTPChannelParent::nsISupports
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
Packit f0b94e
NS_IMPL_ISUPPORTS(FTPChannelParent, nsIStreamListener, nsIParentChannel,
Packit f0b94e
                  nsIInterfaceRequestor, nsIRequestObserver,
Packit f0b94e
                  nsIChannelEventSink, nsIFTPChannelParentInternal)
Packit f0b94e
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
// FTPChannelParent::PFTPChannelParent
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
// FTPChannelParent methods
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
Packit f0b94e
bool FTPChannelParent::Init(const FTPChannelCreationArgs& aArgs) {
Packit f0b94e
  switch (aArgs.type()) {
Packit f0b94e
    case FTPChannelCreationArgs::TFTPChannelOpenArgs: {
Packit f0b94e
      const FTPChannelOpenArgs& a = aArgs.get_FTPChannelOpenArgs();
Packit f0b94e
      return DoAsyncOpen(a.uri(), a.startPos(), a.entityID(), a.uploadStream(),
Packit f0b94e
                         a.loadInfo());
Packit f0b94e
    }
Packit f0b94e
    case FTPChannelCreationArgs::TFTPChannelConnectArgs: {
Packit f0b94e
      const FTPChannelConnectArgs& cArgs = aArgs.get_FTPChannelConnectArgs();
Packit f0b94e
      return ConnectChannel(cArgs.channelId());
Packit f0b94e
    }
Packit f0b94e
    default:
Packit f0b94e
      NS_NOTREACHED("unknown open type");
Packit f0b94e
      return false;
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
bool FTPChannelParent::DoAsyncOpen(const URIParams& aURI,
Packit f0b94e
                                   const uint64_t& aStartPos,
Packit f0b94e
                                   const nsCString& aEntityID,
Packit f0b94e
                                   const OptionalIPCStream& aUploadStream,
Packit f0b94e
                                   const OptionalLoadInfoArgs& aLoadInfoArgs) {
Packit f0b94e
  nsresult rv;
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIURI> uri = DeserializeURI(aURI);
Packit f0b94e
  if (!uri) return false;
Packit f0b94e
Packit f0b94e
#ifdef DEBUG
Packit f0b94e
  LOG(("FTPChannelParent DoAsyncOpen [this=%p uri=%s]\n", this,
Packit f0b94e
       uri->GetSpecOrDefault().get()));
Packit f0b94e
#endif
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIIOService> ios(do_GetIOService(&rv));
Packit f0b94e
  if (NS_FAILED(rv)) {
Packit f0b94e
    return SendFailedAsyncOpen(rv);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsILoadInfo> loadInfo;
Packit f0b94e
  rv = mozilla::ipc::LoadInfoArgsToLoadInfo(aLoadInfoArgs,
Packit f0b94e
                                            getter_AddRefs(loadInfo));
Packit f0b94e
  if (NS_FAILED(rv)) {
Packit f0b94e
    return SendFailedAsyncOpen(rv);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  OriginAttributes attrs;
Packit f0b94e
  rv = loadInfo->GetOriginAttributes(&attrs);
Packit f0b94e
  if (NS_FAILED(rv)) {
Packit f0b94e
    return SendFailedAsyncOpen(rv);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIChannel> chan;
Packit f0b94e
  rv = NS_NewChannelInternal(getter_AddRefs(chan), uri, loadInfo, nullptr,
Packit f0b94e
                             nullptr, nullptr, nsIRequest::LOAD_NORMAL, ios);
Packit f0b94e
Packit f0b94e
  if (NS_FAILED(rv)) return SendFailedAsyncOpen(rv);
Packit f0b94e
Packit f0b94e
  mChannel = chan;
Packit f0b94e
Packit f0b94e
  // later on mChannel may become an HTTP channel (we'll be redirected to one
Packit f0b94e
  // if we're using a proxy), but for now this is safe
Packit f0b94e
  nsFtpChannel* ftpChan = static_cast<nsFtpChannel*>(mChannel.get());
Packit f0b94e
Packit f0b94e
  if (mPBOverride != kPBOverride_Unset) {
Packit f0b94e
    ftpChan->SetPrivate(mPBOverride == kPBOverride_Private ? true : false);
Packit f0b94e
  }
Packit f0b94e
  rv = ftpChan->SetNotificationCallbacks(this);
Packit f0b94e
  if (NS_FAILED(rv)) return SendFailedAsyncOpen(rv);
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIInputStream> upload = DeserializeIPCStream(aUploadStream);
Packit f0b94e
  if (upload) {
Packit f0b94e
    // contentType and contentLength are ignored
Packit f0b94e
    rv = ftpChan->SetUploadStream(upload, EmptyCString(), 0);
Packit f0b94e
    if (NS_FAILED(rv)) return SendFailedAsyncOpen(rv);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  rv = ftpChan->ResumeAt(aStartPos, aEntityID);
Packit f0b94e
  if (NS_FAILED(rv)) return SendFailedAsyncOpen(rv);
Packit f0b94e
Packit f0b94e
  if (loadInfo && loadInfo->GetEnforceSecurity()) {
Packit f0b94e
    rv = ftpChan->AsyncOpen2(this);
Packit f0b94e
  } else {
Packit f0b94e
    rv = ftpChan->AsyncOpen(this, nullptr);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (NS_FAILED(rv)) return SendFailedAsyncOpen(rv);
Packit f0b94e
Packit f0b94e
  return true;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
bool FTPChannelParent::ConnectChannel(const uint32_t& channelId) {
Packit f0b94e
  nsresult rv;
Packit f0b94e
Packit f0b94e
  LOG(("Looking for a registered channel [this=%p, id=%d]", this, channelId));
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIChannel> channel;
Packit f0b94e
  rv = NS_LinkRedirectChannels(channelId, this, getter_AddRefs(channel));
Packit f0b94e
  if (NS_SUCCEEDED(rv)) mChannel = channel;
Packit f0b94e
Packit f0b94e
  LOG(("  found channel %p, rv=%08" PRIx32, mChannel.get(),
Packit f0b94e
       static_cast<uint32_t>(rv)));
Packit f0b94e
Packit f0b94e
  return true;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
mozilla::ipc::IPCResult FTPChannelParent::RecvCancel(const nsresult& status) {
Packit f0b94e
  if (mChannel) mChannel->Cancel(status);
Packit f0b94e
Packit f0b94e
  return IPC_OK();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
mozilla::ipc::IPCResult FTPChannelParent::RecvSuspend() {
Packit f0b94e
  if (mChannel) {
Packit f0b94e
    SuspendChannel();
Packit f0b94e
  }
Packit f0b94e
  return IPC_OK();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
mozilla::ipc::IPCResult FTPChannelParent::RecvResume() {
Packit f0b94e
  if (mChannel) {
Packit f0b94e
    ResumeChannel();
Packit f0b94e
  }
Packit f0b94e
  return IPC_OK();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
class FTPDivertDataAvailableEvent : public MainThreadChannelEvent {
Packit f0b94e
 public:
Packit f0b94e
  FTPDivertDataAvailableEvent(FTPChannelParent* aParent, const nsCString& data,
Packit f0b94e
                              const uint64_t& offset, const uint32_t& count)
Packit f0b94e
      : mParent(aParent), mData(data), mOffset(offset), mCount(count) {}
Packit f0b94e
Packit f0b94e
  void Run() override {
Packit f0b94e
    mParent->DivertOnDataAvailable(mData, mOffset, mCount);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
 private:
Packit f0b94e
  FTPChannelParent* mParent;
Packit f0b94e
  nsCString mData;
Packit f0b94e
  uint64_t mOffset;
Packit f0b94e
  uint32_t mCount;
Packit f0b94e
};
Packit f0b94e
Packit f0b94e
mozilla::ipc::IPCResult FTPChannelParent::RecvDivertOnDataAvailable(
Packit f0b94e
    const nsCString& data, const uint64_t& offset, const uint32_t& count) {
Packit f0b94e
  if (NS_WARN_IF(!mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(mDivertingFromChild,
Packit f0b94e
               "Cannot RecvDivertOnDataAvailable if diverting is not set!");
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
    return IPC_FAIL_NO_REASON(this);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Drop OnDataAvailables if the parent was canceled already.
Packit f0b94e
  if (NS_FAILED(mStatus)) {
Packit f0b94e
    return IPC_OK();
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  mEventQ->RunOrEnqueue(
Packit f0b94e
      new FTPDivertDataAvailableEvent(this, data, offset, count));
Packit f0b94e
  return IPC_OK();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void FTPChannelParent::DivertOnDataAvailable(const nsCString& data,
Packit f0b94e
                                             const uint64_t& offset,
Packit f0b94e
                                             const uint32_t& count) {
Packit f0b94e
  LOG(("FTPChannelParent::DivertOnDataAvailable [this=%p]\n", this));
Packit f0b94e
Packit f0b94e
  if (NS_WARN_IF(!mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(mDivertingFromChild,
Packit f0b94e
               "Cannot DivertOnDataAvailable if diverting is not set!");
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Drop OnDataAvailables if the parent was canceled already.
Packit f0b94e
  if (NS_FAILED(mStatus)) {
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIInputStream> stringStream;
Packit f0b94e
  nsresult rv = NS_NewByteInputStream(getter_AddRefs(stringStream), data.get(),
Packit f0b94e
                                      count, NS_ASSIGNMENT_DEPEND);
Packit f0b94e
  if (NS_FAILED(rv)) {
Packit f0b94e
    if (mChannel) {
Packit f0b94e
      mChannel->Cancel(rv);
Packit f0b94e
    }
Packit f0b94e
    mStatus = rv;
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  AutoEventEnqueuer ensureSerialDispatch(mEventQ);
Packit f0b94e
Packit f0b94e
  rv = OnDataAvailable(mChannel, nullptr, stringStream, offset, count);
Packit f0b94e
Packit f0b94e
  stringStream->Close();
Packit f0b94e
  if (NS_FAILED(rv)) {
Packit f0b94e
    if (mChannel) {
Packit f0b94e
      mChannel->Cancel(rv);
Packit f0b94e
    }
Packit f0b94e
    mStatus = rv;
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
class FTPDivertStopRequestEvent : public MainThreadChannelEvent {
Packit f0b94e
 public:
Packit f0b94e
  FTPDivertStopRequestEvent(FTPChannelParent* aParent,
Packit f0b94e
                            const nsresult& statusCode)
Packit f0b94e
      : mParent(aParent), mStatusCode(statusCode) {}
Packit f0b94e
Packit f0b94e
  void Run() override { mParent->DivertOnStopRequest(mStatusCode); }
Packit f0b94e
Packit f0b94e
 private:
Packit f0b94e
  FTPChannelParent* mParent;
Packit f0b94e
  nsresult mStatusCode;
Packit f0b94e
};
Packit f0b94e
Packit f0b94e
mozilla::ipc::IPCResult FTPChannelParent::RecvDivertOnStopRequest(
Packit f0b94e
    const nsresult& statusCode) {
Packit f0b94e
  if (NS_WARN_IF(!mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(mDivertingFromChild,
Packit f0b94e
               "Cannot RecvDivertOnStopRequest if diverting is not set!");
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
    return IPC_FAIL_NO_REASON(this);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  mEventQ->RunOrEnqueue(new FTPDivertStopRequestEvent(this, statusCode));
Packit f0b94e
  return IPC_OK();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void FTPChannelParent::DivertOnStopRequest(const nsresult& statusCode) {
Packit f0b94e
  LOG(("FTPChannelParent::DivertOnStopRequest [this=%p]\n", this));
Packit f0b94e
Packit f0b94e
  if (NS_WARN_IF(!mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(mDivertingFromChild,
Packit f0b94e
               "Cannot DivertOnStopRequest if diverting is not set!");
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Honor the channel's status even if the underlying transaction completed.
Packit f0b94e
  nsresult status = NS_FAILED(mStatus) ? mStatus : statusCode;
Packit f0b94e
Packit f0b94e
  // Reset fake pending status in case OnStopRequest has already been called.
Packit f0b94e
  if (mChannel) {
Packit f0b94e
    nsCOMPtr<nsIForcePendingChannel> forcePendingIChan =
Packit f0b94e
        do_QueryInterface(mChannel);
Packit f0b94e
    if (forcePendingIChan) {
Packit f0b94e
      forcePendingIChan->ForcePending(false);
Packit f0b94e
    }
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  AutoEventEnqueuer ensureSerialDispatch(mEventQ);
Packit f0b94e
  OnStopRequest(mChannel, nullptr, status);
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
class FTPDivertCompleteEvent : public MainThreadChannelEvent {
Packit f0b94e
 public:
Packit f0b94e
  explicit FTPDivertCompleteEvent(FTPChannelParent* aParent)
Packit f0b94e
      : mParent(aParent) {}
Packit f0b94e
Packit f0b94e
  void Run() override { mParent->DivertComplete(); }
Packit f0b94e
Packit f0b94e
 private:
Packit f0b94e
  FTPChannelParent* mParent;
Packit f0b94e
};
Packit f0b94e
Packit f0b94e
mozilla::ipc::IPCResult FTPChannelParent::RecvDivertComplete() {
Packit f0b94e
  if (NS_WARN_IF(!mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(mDivertingFromChild,
Packit f0b94e
               "Cannot RecvDivertComplete if diverting is not set!");
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
    return IPC_FAIL_NO_REASON(this);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  mEventQ->RunOrEnqueue(new FTPDivertCompleteEvent(this));
Packit f0b94e
  return IPC_OK();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void FTPChannelParent::DivertComplete() {
Packit f0b94e
  LOG(("FTPChannelParent::DivertComplete [this=%p]\n", this));
Packit f0b94e
Packit f0b94e
  if (NS_WARN_IF(!mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(mDivertingFromChild,
Packit f0b94e
               "Cannot DivertComplete if diverting is not set!");
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsresult rv = ResumeForDiversion();
Packit f0b94e
  if (NS_WARN_IF(NS_FAILED(rv))) {
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
// FTPChannelParent::nsIRequestObserver
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::OnStartRequest(nsIRequest* aRequest, nsISupports* aContext) {
Packit f0b94e
  LOG(("FTPChannelParent::OnStartRequest [this=%p]\n", this));
Packit f0b94e
Packit f0b94e
  if (mDivertingFromChild) {
Packit f0b94e
    MOZ_RELEASE_ASSERT(mDivertToListener,
Packit f0b94e
                       "Cannot divert if listener is unset!");
Packit f0b94e
    return mDivertToListener->OnStartRequest(aRequest, aContext);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIChannel> chan = do_QueryInterface(aRequest);
Packit f0b94e
  MOZ_ASSERT(chan);
Packit f0b94e
  NS_ENSURE_TRUE(chan, NS_ERROR_UNEXPECTED);
Packit f0b94e
Packit f0b94e
  // Send down any permissions which are relevant to this URL if we are
Packit f0b94e
  // performing a document load.
Packit f0b94e
  if (!mIPCClosed) {
Packit f0b94e
    PContentParent* pcp = Manager()->Manager();
Packit f0b94e
    MOZ_ASSERT(pcp, "We should have a manager if our IPC isn't closed");
Packit f0b94e
    DebugOnly<nsresult> rv =
Packit f0b94e
        static_cast<ContentParent*>(pcp)
Packit f0b94e
            ->AboutToLoadHttpFtpWyciwygDocumentForChild(chan);
Packit f0b94e
    MOZ_ASSERT(NS_SUCCEEDED(rv));
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  int64_t contentLength;
Packit f0b94e
  chan->GetContentLength(&contentLength);
Packit f0b94e
  nsCString contentType;
Packit f0b94e
  chan->GetContentType(contentType);
Packit f0b94e
Packit f0b94e
  nsCString entityID;
Packit f0b94e
  nsCOMPtr<nsIResumableChannel> resChan = do_QueryInterface(aRequest);
Packit f0b94e
  MOZ_ASSERT(
Packit f0b94e
      resChan);  // both FTP and HTTP should implement nsIResumableChannel
Packit f0b94e
  if (resChan) {
Packit f0b94e
    resChan->GetEntityID(entityID);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  PRTime lastModified = 0;
Packit f0b94e
  nsCOMPtr<nsIFTPChannel> ftpChan = do_QueryInterface(aRequest);
Packit f0b94e
  if (ftpChan) {
Packit f0b94e
    ftpChan->GetLastModifiedTime(&lastModified);
Packit f0b94e
  }
Packit f0b94e
  nsCOMPtr<nsIHttpChannelInternal> httpChan = do_QueryInterface(aRequest);
Packit f0b94e
  if (httpChan) {
Packit f0b94e
    Unused << httpChan->GetLastModifiedTime(&lastModified);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  URIParams uriparam;
Packit f0b94e
  nsCOMPtr<nsIURI> uri;
Packit f0b94e
  chan->GetURI(getter_AddRefs(uri));
Packit f0b94e
  SerializeURI(uri, uriparam);
Packit f0b94e
Packit f0b94e
  if (mIPCClosed || !SendOnStartRequest(mStatus, contentLength, contentType,
Packit f0b94e
                                        lastModified, entityID, uriparam)) {
Packit f0b94e
    return NS_ERROR_UNEXPECTED;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::OnStopRequest(nsIRequest* aRequest, nsISupports* aContext,
Packit f0b94e
                                nsresult aStatusCode) {
Packit f0b94e
  LOG(("FTPChannelParent::OnStopRequest: [this=%p status=%" PRIu32 "]\n", this,
Packit f0b94e
       static_cast<uint32_t>(aStatusCode)));
Packit f0b94e
Packit f0b94e
  if (mDivertingFromChild) {
Packit f0b94e
    MOZ_RELEASE_ASSERT(mDivertToListener,
Packit f0b94e
                       "Cannot divert if listener is unset!");
Packit f0b94e
    return mDivertToListener->OnStopRequest(aRequest, aContext, aStatusCode);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (mIPCClosed || !SendOnStopRequest(aStatusCode, mErrorMsg, mUseUTF8)) {
Packit f0b94e
    return NS_ERROR_UNEXPECTED;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
// FTPChannelParent::nsIStreamListener
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::OnDataAvailable(nsIRequest* aRequest, nsISupports* aContext,
Packit f0b94e
                                  nsIInputStream* aInputStream,
Packit f0b94e
                                  uint64_t aOffset, uint32_t aCount) {
Packit f0b94e
  LOG(("FTPChannelParent::OnDataAvailable [this=%p]\n", this));
Packit f0b94e
Packit f0b94e
  if (mDivertingFromChild) {
Packit f0b94e
    MOZ_RELEASE_ASSERT(mDivertToListener,
Packit f0b94e
                       "Cannot divert if listener is unset!");
Packit f0b94e
    return mDivertToListener->OnDataAvailable(aRequest, aContext, aInputStream,
Packit f0b94e
                                              aOffset, aCount);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsCString data;
Packit f0b94e
  nsresult rv = NS_ReadInputStreamToString(aInputStream, data, aCount);
Packit f0b94e
  if (NS_FAILED(rv)) return rv;
Packit f0b94e
Packit f0b94e
  if (mIPCClosed || !SendOnDataAvailable(mStatus, data, aOffset, aCount))
Packit f0b94e
    return NS_ERROR_UNEXPECTED;
Packit f0b94e
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
// FTPChannelParent::nsIParentChannel
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::SetParentListener(HttpChannelParentListener* aListener) {
Packit f0b94e
  // Do not need ptr to HttpChannelParentListener.
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::NotifyTrackingProtectionDisabled() {
Packit f0b94e
  // One day, this should probably be filled in.
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::NotifyTrackingResource() {
Packit f0b94e
  // One day, this should probably be filled in.
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::SetClassifierMatchedInfo(const nsACString& aList,
Packit f0b94e
                                           const nsACString& aProvider,
Packit f0b94e
                                           const nsACString& aFullHash) {
Packit f0b94e
  // One day, this should probably be filled in.
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::Delete() {
Packit f0b94e
  if (mIPCClosed || !SendDeleteSelf()) return NS_ERROR_UNEXPECTED;
Packit f0b94e
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
// FTPChannelParent::nsIInterfaceRequestor
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::GetInterface(const nsIID& uuid, void** result) {
Packit f0b94e
  if (uuid.Equals(NS_GET_IID(nsIAuthPromptProvider)) ||
Packit f0b94e
      uuid.Equals(NS_GET_IID(nsISecureBrowserUI))) {
Packit f0b94e
    if (mTabParent) {
Packit f0b94e
      return mTabParent->QueryInterface(uuid, result);
Packit f0b94e
    }
Packit f0b94e
  } else if (uuid.Equals(NS_GET_IID(nsIAuthPrompt)) ||
Packit f0b94e
             uuid.Equals(NS_GET_IID(nsIAuthPrompt2))) {
Packit f0b94e
    nsCOMPtr<nsIAuthPromptProvider> provider(do_QueryObject(mTabParent));
Packit f0b94e
    if (provider) {
Packit f0b94e
      return provider->GetAuthPrompt(nsIAuthPromptProvider::PROMPT_NORMAL, uuid,
Packit f0b94e
                                     result);
Packit f0b94e
    }
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Only support nsILoadContext if child channel's callbacks did too
Packit f0b94e
  if (uuid.Equals(NS_GET_IID(nsILoadContext)) && mLoadContext) {
Packit f0b94e
    nsCOMPtr<nsILoadContext> copy = mLoadContext;
Packit f0b94e
    copy.forget(result);
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  return QueryInterface(uuid, result);
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
nsresult FTPChannelParent::SuspendChannel() {
Packit f0b94e
  nsCOMPtr<nsIChannelWithDivertableParentListener> chan =
Packit f0b94e
      do_QueryInterface(mChannel);
Packit f0b94e
  if (chan) {
Packit f0b94e
    return chan->SuspendInternal();
Packit f0b94e
  } else {
Packit f0b94e
    return mChannel->Suspend();
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
nsresult FTPChannelParent::ResumeChannel() {
Packit f0b94e
  nsCOMPtr<nsIChannelWithDivertableParentListener> chan =
Packit f0b94e
      do_QueryInterface(mChannel);
Packit f0b94e
  if (chan) {
Packit f0b94e
    return chan->ResumeInternal();
Packit f0b94e
  } else {
Packit f0b94e
    return mChannel->Resume();
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
// FTPChannelParent::ADivertableParentChannel
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
nsresult FTPChannelParent::SuspendForDiversion() {
Packit f0b94e
  MOZ_ASSERT(mChannel);
Packit f0b94e
  if (NS_WARN_IF(mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(!mDivertingFromChild, "Already suspended for diversion!");
Packit f0b94e
    return NS_ERROR_UNEXPECTED;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Try suspending the channel. Allow it to fail, since OnStopRequest may have
Packit f0b94e
  // been called and thus the channel may not be pending.
Packit f0b94e
  nsresult rv = SuspendChannel();
Packit f0b94e
  MOZ_ASSERT(NS_SUCCEEDED(rv) || rv == NS_ERROR_NOT_AVAILABLE);
Packit f0b94e
  mSuspendedForDiversion = NS_SUCCEEDED(rv);
Packit f0b94e
Packit f0b94e
  // Once this is set, no more OnStart/OnData/OnStop callbacks should be sent
Packit f0b94e
  // to the child.
Packit f0b94e
  mDivertingFromChild = true;
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIChannelWithDivertableParentListener> chan =
Packit f0b94e
      do_QueryInterface(mChannel);
Packit f0b94e
  if (chan) {
Packit f0b94e
    chan->MessageDiversionStarted(this);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
/* private, supporting function for ADivertableParentChannel */
Packit f0b94e
nsresult FTPChannelParent::ResumeForDiversion() {
Packit f0b94e
  MOZ_ASSERT(mChannel);
Packit f0b94e
  MOZ_ASSERT(mDivertToListener);
Packit f0b94e
  if (NS_WARN_IF(!mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(mDivertingFromChild,
Packit f0b94e
               "Cannot ResumeForDiversion if not diverting!");
Packit f0b94e
    return NS_ERROR_UNEXPECTED;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIChannelWithDivertableParentListener> chan =
Packit f0b94e
      do_QueryInterface(mChannel);
Packit f0b94e
  if (chan) {
Packit f0b94e
    chan->MessageDiversionStop();
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (mSuspendedForDiversion) {
Packit f0b94e
    nsresult rv = ResumeChannel();
Packit f0b94e
    if (NS_WARN_IF(NS_FAILED(rv))) {
Packit f0b94e
      FailDiversion(NS_ERROR_UNEXPECTED, true);
Packit f0b94e
      return rv;
Packit f0b94e
    }
Packit f0b94e
    mSuspendedForDiversion = false;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Delete() will tear down IPDL, but ref from underlying nsFTPChannel will
Packit f0b94e
  // keep us alive if there's more data to be delivered to listener.
Packit f0b94e
  if (NS_WARN_IF(NS_FAILED(Delete()))) {
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
    return NS_ERROR_UNEXPECTED;
Packit f0b94e
  }
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
nsresult FTPChannelParent::SuspendMessageDiversion() {
Packit f0b94e
  // This only need to suspend message queue.
Packit f0b94e
  mEventQ->Suspend();
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
nsresult FTPChannelParent::ResumeMessageDiversion() {
Packit f0b94e
  // This only need to resumes message queue.
Packit f0b94e
  mEventQ->Resume();
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
nsresult FTPChannelParent::CancelDiversion() {
Packit f0b94e
  // Only HTTP channels can have child-process-sourced-data that's long-lived
Packit f0b94e
  // so this isn't currently relevant for FTP channels and there is nothing to
Packit f0b94e
  // do.
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void FTPChannelParent::DivertTo(nsIStreamListener* aListener) {
Packit f0b94e
  MOZ_ASSERT(aListener);
Packit f0b94e
  if (NS_WARN_IF(!mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(mDivertingFromChild,
Packit f0b94e
               "Cannot DivertTo new listener if diverting is not set!");
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (NS_WARN_IF(mIPCClosed || !SendFlushedForDiversion())) {
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  mDivertToListener = aListener;
Packit f0b94e
Packit f0b94e
  // Call OnStartRequest and SendDivertMessages asynchronously to avoid
Packit f0b94e
  // reentering client context.
Packit f0b94e
  NS_DispatchToCurrentThread(
Packit f0b94e
      NewRunnableMethod("net::FTPChannelParent::StartDiversion", this,
Packit f0b94e
                        &FTPChannelParent::StartDiversion));
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void FTPChannelParent::StartDiversion() {
Packit f0b94e
  if (NS_WARN_IF(!mDivertingFromChild)) {
Packit f0b94e
    MOZ_ASSERT(mDivertingFromChild,
Packit f0b94e
               "Cannot StartDiversion if diverting is not set!");
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Fake pending status in case OnStopRequest has already been called.
Packit f0b94e
  if (mChannel) {
Packit f0b94e
    nsCOMPtr<nsIForcePendingChannel> forcePendingIChan =
Packit f0b94e
        do_QueryInterface(mChannel);
Packit f0b94e
    if (forcePendingIChan) {
Packit f0b94e
      forcePendingIChan->ForcePending(true);
Packit f0b94e
    }
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  {
Packit f0b94e
    AutoEventEnqueuer ensureSerialDispatch(mEventQ);
Packit f0b94e
    // Call OnStartRequest for the "DivertTo" listener.
Packit f0b94e
    nsresult rv = OnStartRequest(mChannel, nullptr);
Packit f0b94e
    if (NS_FAILED(rv)) {
Packit f0b94e
      if (mChannel) {
Packit f0b94e
        mChannel->Cancel(rv);
Packit f0b94e
      }
Packit f0b94e
      mStatus = rv;
Packit f0b94e
      return;
Packit f0b94e
    }
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // After OnStartRequest has been called, tell FTPChannelChild to divert the
Packit f0b94e
  // OnDataAvailables and OnStopRequest to this FTPChannelParent.
Packit f0b94e
  if (NS_WARN_IF(mIPCClosed || !SendDivertMessages())) {
Packit f0b94e
    FailDiversion(NS_ERROR_UNEXPECTED);
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
class FTPFailDiversionEvent : public Runnable {
Packit f0b94e
 public:
Packit f0b94e
  FTPFailDiversionEvent(FTPChannelParent* aChannelParent, nsresult aErrorCode,
Packit f0b94e
                        bool aSkipResume)
Packit f0b94e
      : Runnable("net::FTPFailDiversionEvent"),
Packit f0b94e
        mChannelParent(aChannelParent),
Packit f0b94e
        mErrorCode(aErrorCode),
Packit f0b94e
        mSkipResume(aSkipResume) {
Packit f0b94e
    MOZ_RELEASE_ASSERT(aChannelParent);
Packit f0b94e
    MOZ_RELEASE_ASSERT(NS_FAILED(aErrorCode));
Packit f0b94e
  }
Packit f0b94e
  NS_IMETHOD Run() override {
Packit f0b94e
    mChannelParent->NotifyDiversionFailed(mErrorCode, mSkipResume);
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
 private:
Packit f0b94e
  RefPtr<FTPChannelParent> mChannelParent;
Packit f0b94e
  nsresult mErrorCode;
Packit f0b94e
  bool mSkipResume;
Packit f0b94e
};
Packit f0b94e
Packit f0b94e
void FTPChannelParent::FailDiversion(nsresult aErrorCode, bool aSkipResume) {
Packit f0b94e
  MOZ_RELEASE_ASSERT(NS_FAILED(aErrorCode));
Packit f0b94e
  MOZ_RELEASE_ASSERT(mDivertingFromChild);
Packit f0b94e
  MOZ_RELEASE_ASSERT(mDivertToListener);
Packit f0b94e
  MOZ_RELEASE_ASSERT(mChannel);
Packit f0b94e
Packit f0b94e
  NS_DispatchToCurrentThread(
Packit f0b94e
      new FTPFailDiversionEvent(this, aErrorCode, aSkipResume));
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void FTPChannelParent::NotifyDiversionFailed(nsresult aErrorCode,
Packit f0b94e
                                             bool aSkipResume) {
Packit f0b94e
  MOZ_RELEASE_ASSERT(NS_FAILED(aErrorCode));
Packit f0b94e
  MOZ_RELEASE_ASSERT(mDivertingFromChild);
Packit f0b94e
  MOZ_RELEASE_ASSERT(mDivertToListener);
Packit f0b94e
  MOZ_RELEASE_ASSERT(mChannel);
Packit f0b94e
Packit f0b94e
  mChannel->Cancel(aErrorCode);
Packit f0b94e
  nsCOMPtr<nsIForcePendingChannel> forcePendingIChan =
Packit f0b94e
      do_QueryInterface(mChannel);
Packit f0b94e
  if (forcePendingIChan) {
Packit f0b94e
    forcePendingIChan->ForcePending(false);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  bool isPending = false;
Packit f0b94e
  nsresult rv = mChannel->IsPending(&isPending);
Packit f0b94e
  MOZ_RELEASE_ASSERT(NS_SUCCEEDED(rv));
Packit f0b94e
Packit f0b94e
  // Resume only we suspended earlier.
Packit f0b94e
  if (mSuspendedForDiversion) {
Packit f0b94e
    ResumeChannel();
Packit f0b94e
  }
Packit f0b94e
  // Channel has already sent OnStartRequest to the child, so ensure that we
Packit f0b94e
  // call it here if it hasn't already been called.
Packit f0b94e
  if (!mDivertedOnStartRequest) {
Packit f0b94e
    nsCOMPtr<nsIForcePendingChannel> forcePendingIChan =
Packit f0b94e
        do_QueryInterface(mChannel);
Packit f0b94e
    if (forcePendingIChan) {
Packit f0b94e
      forcePendingIChan->ForcePending(true);
Packit f0b94e
    }
Packit f0b94e
    mDivertToListener->OnStartRequest(mChannel, nullptr);
Packit f0b94e
Packit f0b94e
    if (forcePendingIChan) {
Packit f0b94e
      forcePendingIChan->ForcePending(false);
Packit f0b94e
    }
Packit f0b94e
  }
Packit f0b94e
  // If the channel is pending, it will call OnStopRequest itself; otherwise, do
Packit f0b94e
  // it here.
Packit f0b94e
  if (!isPending) {
Packit f0b94e
    mDivertToListener->OnStopRequest(mChannel, nullptr, aErrorCode);
Packit f0b94e
  }
Packit f0b94e
  mDivertToListener = nullptr;
Packit f0b94e
  mChannel = nullptr;
Packit f0b94e
Packit f0b94e
  if (!mIPCClosed) {
Packit f0b94e
    Unused << SendDeleteSelf();
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
// FTPChannelParent::nsIChannelEventSink
Packit f0b94e
//-----------------------------------------------------------------------------
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::AsyncOnChannelRedirect(
Packit f0b94e
    nsIChannel* oldChannel, nsIChannel* newChannel, uint32_t redirectFlags,
Packit f0b94e
    nsIAsyncVerifyRedirectCallback* callback) {
Packit f0b94e
  nsCOMPtr<nsIFTPChannel> ftpChan = do_QueryInterface(newChannel);
Packit f0b94e
  if (!ftpChan) {
Packit f0b94e
    // when FTP is set to use HTTP proxying, we wind up getting redirected to an
Packit f0b94e
    // HTTP channel.
Packit f0b94e
    nsCOMPtr<nsIHttpChannel> httpChan = do_QueryInterface(newChannel);
Packit f0b94e
    if (!httpChan) return NS_ERROR_UNEXPECTED;
Packit f0b94e
  }
Packit f0b94e
  mChannel = newChannel;
Packit f0b94e
  callback->OnRedirectVerifyCallback(NS_OK);
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
FTPChannelParent::SetErrorMsg(const char* aMsg, bool aUseUTF8) {
Packit f0b94e
  mErrorMsg = aMsg;
Packit f0b94e
  mUseUTF8 = aUseUTF8;
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
//---------------------
Packit f0b94e
}  // namespace net
Packit f0b94e
}  // namespace mozilla