2009-09-09 06:31:35 +00:00
|
|
|
/* -*- Mode: C++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 8 -*- */
|
2009-09-09 14:04:09 +00:00
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
2009-09-09 06:31:35 +00:00
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is Mozilla Plugin App.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Benjamin Smedberg <benjamin@smedbergs.us>
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2009
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
2009-09-09 14:04:09 +00:00
|
|
|
#include "BrowserStreamChild.h"
|
2009-09-09 06:31:35 +00:00
|
|
|
#include "PluginInstanceChild.h"
|
2009-09-23 23:00:23 +00:00
|
|
|
#include "StreamNotifyChild.h"
|
2009-09-09 06:31:35 +00:00
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace plugins {
|
|
|
|
|
2009-09-09 14:04:09 +00:00
|
|
|
BrowserStreamChild::BrowserStreamChild(PluginInstanceChild* instance,
|
|
|
|
const nsCString& url,
|
|
|
|
const uint32_t& length,
|
|
|
|
const uint32_t& lastmodified,
|
2010-03-06 21:03:05 +00:00
|
|
|
StreamNotifyChild* notifyData,
|
2009-09-09 14:04:09 +00:00
|
|
|
const nsCString& headers,
|
|
|
|
const nsCString& mimeType,
|
|
|
|
const bool& seekable,
|
|
|
|
NPError* rv,
|
|
|
|
uint16_t* stype)
|
2009-09-09 06:31:35 +00:00
|
|
|
: mInstance(instance)
|
2010-03-06 21:03:05 +00:00
|
|
|
, mStreamStatus(kStreamOpen)
|
|
|
|
, mDestroyPending(NOT_DESTROYED)
|
|
|
|
, mNotifyPending(false)
|
2011-10-19 14:46:24 +00:00
|
|
|
, mStreamAsFilePending(false)
|
2010-03-06 21:03:05 +00:00
|
|
|
, mInstanceDying(false)
|
2010-02-23 21:45:00 +00:00
|
|
|
, mState(CONSTRUCTING)
|
2009-09-09 06:31:35 +00:00
|
|
|
, mURL(url)
|
|
|
|
, mHeaders(headers)
|
2010-03-06 21:03:05 +00:00
|
|
|
, mStreamNotify(notifyData)
|
|
|
|
, mDeliveryTracker(this)
|
2009-09-09 06:31:35 +00:00
|
|
|
{
|
2009-12-18 22:22:51 +00:00
|
|
|
PLUGIN_LOG_DEBUG(("%s (%s, %i, %i, %p, %s, %s)", FULLFUNCTION,
|
|
|
|
url.get(), length, lastmodified, (void*) notifyData,
|
|
|
|
headers.get(), mimeType.get()));
|
|
|
|
|
2009-10-08 19:11:13 +00:00
|
|
|
AssertPluginThread();
|
|
|
|
|
2009-09-09 06:31:35 +00:00
|
|
|
memset(&mStream, 0, sizeof(mStream));
|
2009-09-22 21:06:00 +00:00
|
|
|
mStream.ndata = static_cast<AStream*>(this);
|
2009-11-25 15:34:59 +00:00
|
|
|
mStream.url = NullableStringGet(mURL);
|
2009-09-09 06:31:35 +00:00
|
|
|
mStream.end = length;
|
|
|
|
mStream.lastmodified = lastmodified;
|
2009-11-25 15:34:59 +00:00
|
|
|
mStream.headers = NullableStringGet(mHeaders);
|
2010-03-06 21:03:05 +00:00
|
|
|
if (notifyData)
|
|
|
|
mStream.notifyData = notifyData->mClosure;
|
2010-01-12 23:06:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NPError
|
|
|
|
BrowserStreamChild::StreamConstructed(
|
|
|
|
const nsCString& mimeType,
|
|
|
|
const bool& seekable,
|
|
|
|
uint16_t* stype)
|
|
|
|
{
|
|
|
|
NPError rv = NPERR_NO_ERROR;
|
2009-09-09 06:31:35 +00:00
|
|
|
|
2010-01-26 17:05:54 +00:00
|
|
|
*stype = NP_NORMAL;
|
2010-01-12 23:06:18 +00:00
|
|
|
rv = mInstance->mPluginIface->newstream(
|
2009-11-25 15:34:59 +00:00
|
|
|
&mInstance->mData, const_cast<char*>(NullableStringGet(mimeType)),
|
|
|
|
&mStream, seekable, stype);
|
2010-02-23 21:45:00 +00:00
|
|
|
if (rv != NPERR_NO_ERROR) {
|
|
|
|
mState = DELETING;
|
2010-03-06 21:03:05 +00:00
|
|
|
mStreamNotify = NULL;
|
2010-02-23 21:45:00 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
mState = ALIVE;
|
2010-03-06 21:03:05 +00:00
|
|
|
|
|
|
|
if (mStreamNotify)
|
|
|
|
mStreamNotify->SetAssociatedStream(this);
|
2010-02-23 21:45:00 +00:00
|
|
|
}
|
2010-01-12 23:06:18 +00:00
|
|
|
|
|
|
|
return rv;
|
2009-09-09 06:31:35 +00:00
|
|
|
}
|
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
BrowserStreamChild::~BrowserStreamChild()
|
|
|
|
{
|
|
|
|
NS_ASSERTION(!mStreamNotify, "Should have nulled it by now!");
|
|
|
|
}
|
|
|
|
|
2009-09-17 23:09:20 +00:00
|
|
|
bool
|
2010-02-23 21:45:00 +00:00
|
|
|
BrowserStreamChild::RecvWrite(const int32_t& offset,
|
|
|
|
const Buffer& data,
|
|
|
|
const uint32_t& newlength)
|
2009-09-09 06:31:35 +00:00
|
|
|
{
|
2009-12-18 22:22:51 +00:00
|
|
|
PLUGIN_LOG_DEBUG_FUNCTION;
|
|
|
|
|
2009-10-08 19:11:13 +00:00
|
|
|
AssertPluginThread();
|
|
|
|
|
2010-02-23 21:45:00 +00:00
|
|
|
if (ALIVE != mState)
|
|
|
|
NS_RUNTIMEABORT("Unexpected state: received data after NPP_DestroyStream?");
|
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
if (kStreamOpen != mStreamStatus)
|
2009-09-17 23:09:20 +00:00
|
|
|
return true;
|
2009-09-09 06:31:35 +00:00
|
|
|
|
|
|
|
mStream.end = newlength;
|
|
|
|
|
2010-02-23 21:45:00 +00:00
|
|
|
NS_ASSERTION(data.Length() > 0, "Empty data");
|
2009-09-09 06:31:35 +00:00
|
|
|
|
2010-02-23 21:45:00 +00:00
|
|
|
PendingData* newdata = mPendingData.AppendElement();
|
|
|
|
newdata->offset = offset;
|
|
|
|
newdata->data = data;
|
|
|
|
newdata->curpos = 0;
|
2009-09-22 21:06:00 +00:00
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
EnsureDeliveryPending();
|
2009-09-09 06:31:35 +00:00
|
|
|
|
2009-09-17 23:09:20 +00:00
|
|
|
return true;
|
2009-09-09 06:31:35 +00:00
|
|
|
}
|
|
|
|
|
2009-12-03 08:16:14 +00:00
|
|
|
bool
|
2011-10-19 14:46:24 +00:00
|
|
|
BrowserStreamChild::RecvNPP_StreamAsFile(const nsCString& fname)
|
2010-02-23 21:45:00 +00:00
|
|
|
{
|
2010-02-25 11:00:56 +00:00
|
|
|
PLUGIN_LOG_DEBUG(("%s (fname=%s)", FULLFUNCTION, fname.get()));
|
2010-02-23 21:45:00 +00:00
|
|
|
|
2010-02-25 11:00:56 +00:00
|
|
|
AssertPluginThread();
|
2010-02-23 21:45:00 +00:00
|
|
|
|
2010-02-23 21:45:00 +00:00
|
|
|
if (ALIVE != mState)
|
|
|
|
NS_RUNTIMEABORT("Unexpected state: received file after NPP_DestroyStream?");
|
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
if (kStreamOpen != mStreamStatus)
|
2010-02-25 11:00:56 +00:00
|
|
|
return true;
|
2010-02-23 21:45:00 +00:00
|
|
|
|
2011-10-19 14:46:24 +00:00
|
|
|
mStreamAsFilePending = true;
|
|
|
|
mStreamAsFileName = fname;
|
|
|
|
EnsureDeliveryPending();
|
|
|
|
|
2010-02-23 21:45:00 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-03-06 16:08:19 +00:00
|
|
|
bool
|
2010-02-23 21:45:00 +00:00
|
|
|
BrowserStreamChild::RecvNPP_DestroyStream(const NPReason& reason)
|
|
|
|
{
|
|
|
|
PLUGIN_LOG_DEBUG_METHOD;
|
|
|
|
|
|
|
|
if (ALIVE != mState)
|
|
|
|
NS_RUNTIMEABORT("Unexpected state: recevied NPP_DestroyStream twice?");
|
|
|
|
|
|
|
|
mState = DYING;
|
2010-03-06 21:03:05 +00:00
|
|
|
mDestroyPending = DESTROY_PENDING;
|
|
|
|
if (NPRES_DONE != reason)
|
|
|
|
mStreamStatus = reason;
|
2010-02-23 21:45:00 +00:00
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
EnsureDeliveryPending();
|
2010-02-23 21:45:00 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
BrowserStreamChild::Recv__delete__()
|
2010-03-06 16:08:19 +00:00
|
|
|
{
|
|
|
|
AssertPluginThread();
|
2010-02-23 21:45:00 +00:00
|
|
|
|
|
|
|
if (DELETING != mState)
|
|
|
|
NS_RUNTIMEABORT("Bad state, not DELETING");
|
|
|
|
|
2010-03-06 16:08:19 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-09-22 21:06:00 +00:00
|
|
|
NPError
|
|
|
|
BrowserStreamChild::NPN_RequestRead(NPByteRange* aRangeList)
|
|
|
|
{
|
2009-12-18 22:22:51 +00:00
|
|
|
PLUGIN_LOG_DEBUG_FUNCTION;
|
|
|
|
|
2009-10-08 19:11:13 +00:00
|
|
|
AssertPluginThread();
|
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
if (ALIVE != mState || kStreamOpen != mStreamStatus)
|
2010-02-23 21:45:00 +00:00
|
|
|
return NPERR_GENERIC_ERROR;
|
|
|
|
|
2009-09-22 21:06:00 +00:00
|
|
|
IPCByteRanges ranges;
|
|
|
|
for (; aRangeList; aRangeList = aRangeList->next) {
|
|
|
|
IPCByteRange br = {aRangeList->offset, aRangeList->length};
|
|
|
|
ranges.push_back(br);
|
|
|
|
}
|
|
|
|
|
|
|
|
NPError result;
|
|
|
|
CallNPN_RequestRead(ranges, &result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2010-03-06 21:02:31 +00:00
|
|
|
void
|
|
|
|
BrowserStreamChild::NPN_DestroyStream(NPReason reason)
|
|
|
|
{
|
2010-03-06 21:03:05 +00:00
|
|
|
mStreamStatus = reason;
|
2010-03-06 21:02:31 +00:00
|
|
|
if (ALIVE == mState)
|
2010-03-06 21:03:05 +00:00
|
|
|
SendNPN_DestroyStream(reason);
|
2010-03-06 21:02:31 +00:00
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
EnsureDeliveryPending();
|
2010-03-06 21:02:31 +00:00
|
|
|
}
|
|
|
|
|
2010-03-06 16:08:19 +00:00
|
|
|
void
|
2010-03-06 21:03:05 +00:00
|
|
|
BrowserStreamChild::EnsureDeliveryPending()
|
2009-09-21 14:51:35 +00:00
|
|
|
{
|
2010-03-06 21:03:05 +00:00
|
|
|
MessageLoop::current()->PostTask(FROM_HERE,
|
|
|
|
mDeliveryTracker.NewRunnableMethod(&BrowserStreamChild::Deliver));
|
|
|
|
}
|
2009-10-08 19:11:13 +00:00
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
void
|
|
|
|
BrowserStreamChild::Deliver()
|
|
|
|
{
|
|
|
|
while (kStreamOpen == mStreamStatus && mPendingData.Length()) {
|
|
|
|
if (DeliverPendingData() && kStreamOpen == mStreamStatus) {
|
|
|
|
SetSuspendedTimer();
|
|
|
|
return;
|
2010-02-23 21:45:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ClearSuspendedTimer();
|
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
NS_ASSERTION(kStreamOpen != mStreamStatus || 0 == mPendingData.Length(),
|
|
|
|
"Exit out of the data-delivery loop with pending data");
|
|
|
|
mPendingData.Clear();
|
2010-03-06 21:02:31 +00:00
|
|
|
|
2011-10-19 14:46:24 +00:00
|
|
|
// NPP_StreamAsFile() is documented (at MDN) to be called "when the stream
|
|
|
|
// is complete" -- i.e. after all calls to NPP_WriteReady() and NPP_Write()
|
|
|
|
// have finished. We make these calls asynchronously (from
|
|
|
|
// DeliverPendingData()). So we need to make sure all the "pending data"
|
|
|
|
// has been "delivered" before calling NPP_StreamAsFile() (also
|
|
|
|
// asynchronously). Doing this resolves bug 687610, bug 670036 and possibly
|
|
|
|
// also other bugs.
|
|
|
|
if (mStreamAsFilePending) {
|
2011-11-21 18:40:19 +00:00
|
|
|
if (mStreamStatus == kStreamOpen)
|
|
|
|
mInstance->mPluginIface->asfile(&mInstance->mData, &mStream,
|
|
|
|
mStreamAsFileName.get());
|
2011-10-19 14:46:24 +00:00
|
|
|
mStreamAsFilePending = false;
|
|
|
|
}
|
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
if (DESTROY_PENDING == mDestroyPending) {
|
|
|
|
mDestroyPending = DESTROYED;
|
2010-03-06 21:02:31 +00:00
|
|
|
if (mState != DYING)
|
|
|
|
NS_RUNTIMEABORT("mDestroyPending but state not DYING");
|
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
NS_ASSERTION(NPRES_DONE != mStreamStatus, "Success status set too early!");
|
|
|
|
if (kStreamOpen == mStreamStatus)
|
|
|
|
mStreamStatus = NPRES_DONE;
|
2010-03-06 21:02:31 +00:00
|
|
|
|
|
|
|
(void) mInstance->mPluginIface
|
2010-03-06 21:03:05 +00:00
|
|
|
->destroystream(&mInstance->mData, &mStream, mStreamStatus);
|
|
|
|
}
|
|
|
|
if (DESTROYED == mDestroyPending && mNotifyPending) {
|
|
|
|
NS_ASSERTION(mStreamNotify, "mDestroyPending but no mStreamNotify?");
|
|
|
|
|
|
|
|
mNotifyPending = false;
|
|
|
|
mStreamNotify->NPP_URLNotify(mStreamStatus);
|
|
|
|
delete mStreamNotify;
|
|
|
|
mStreamNotify = NULL;
|
|
|
|
}
|
|
|
|
if (DYING == mState && DESTROYED == mDestroyPending
|
|
|
|
&& !mStreamNotify && !mInstanceDying) {
|
2010-03-06 21:02:31 +00:00
|
|
|
SendStreamDestroyed();
|
|
|
|
mState = DELETING;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-06 21:03:05 +00:00
|
|
|
bool
|
|
|
|
BrowserStreamChild::DeliverPendingData()
|
|
|
|
{
|
|
|
|
if (mState != ALIVE && mState != DYING)
|
|
|
|
NS_RUNTIMEABORT("Unexpected state");
|
|
|
|
|
|
|
|
NS_ASSERTION(mPendingData.Length(), "Called from Deliver with empty pending");
|
|
|
|
|
2010-05-12 18:12:27 +00:00
|
|
|
while (mPendingData[0].curpos < static_cast<int32_t>(mPendingData[0].data.Length())) {
|
2010-03-06 21:03:05 +00:00
|
|
|
int32_t r = mInstance->mPluginIface->writeready(&mInstance->mData, &mStream);
|
|
|
|
if (kStreamOpen != mStreamStatus)
|
|
|
|
return false;
|
|
|
|
if (0 == r) // plugin wants to suspend delivery
|
|
|
|
return true;
|
|
|
|
|
|
|
|
r = mInstance->mPluginIface->write(
|
|
|
|
&mInstance->mData, &mStream,
|
|
|
|
mPendingData[0].offset + mPendingData[0].curpos, // offset
|
|
|
|
mPendingData[0].data.Length() - mPendingData[0].curpos, // length
|
|
|
|
const_cast<char*>(mPendingData[0].data.BeginReading() + mPendingData[0].curpos));
|
|
|
|
if (kStreamOpen != mStreamStatus)
|
|
|
|
return false;
|
|
|
|
if (0 == r)
|
|
|
|
return true;
|
|
|
|
if (r < 0) { // error condition
|
|
|
|
NPN_DestroyStream(NPRES_NETWORK_ERR);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
mPendingData[0].curpos += r;
|
|
|
|
}
|
|
|
|
mPendingData.RemoveElementAt(0);
|
|
|
|
return false;
|
|
|
|
}
|
2010-03-06 21:02:31 +00:00
|
|
|
|
2010-02-23 21:45:00 +00:00
|
|
|
void
|
|
|
|
BrowserStreamChild::SetSuspendedTimer()
|
|
|
|
{
|
|
|
|
if (mSuspendedTimer.IsRunning())
|
2010-03-06 16:08:19 +00:00
|
|
|
return;
|
2010-02-23 21:45:00 +00:00
|
|
|
mSuspendedTimer.Start(
|
2010-03-06 21:03:05 +00:00
|
|
|
base::TimeDelta::FromMilliseconds(100), // 100ms copied from Mozilla plugin host
|
|
|
|
this, &BrowserStreamChild::Deliver);
|
2010-02-23 21:45:00 +00:00
|
|
|
}
|
2009-09-21 14:51:35 +00:00
|
|
|
|
2010-02-23 21:45:00 +00:00
|
|
|
void
|
|
|
|
BrowserStreamChild::ClearSuspendedTimer()
|
|
|
|
{
|
|
|
|
mSuspendedTimer.Stop();
|
2009-09-21 14:51:35 +00:00
|
|
|
}
|
|
|
|
|
2009-09-09 06:31:35 +00:00
|
|
|
} /* namespace plugins */
|
|
|
|
} /* namespace mozilla */
|