2011-04-25 02:30:54 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
|
|
* vim: set ts=8 sw=2 et tw=80:
|
|
|
|
*
|
2012-05-21 11:12:37 +00:00
|
|
|
* 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/. */
|
2011-04-25 02:30:54 +00:00
|
|
|
|
|
|
|
#include "nsStructuredCloneContainer.h"
|
|
|
|
|
|
|
|
#include "nsCOMPtr.h"
|
2014-08-05 16:18:30 +00:00
|
|
|
#include "nsIGlobalObject.h"
|
2011-04-25 02:30:54 +00:00
|
|
|
#include "nsIVariant.h"
|
2013-03-17 07:55:15 +00:00
|
|
|
#include "nsIXPConnect.h"
|
2011-04-25 02:30:54 +00:00
|
|
|
#include "nsServiceManagerUtils.h"
|
|
|
|
#include "nsContentUtils.h"
|
2013-08-17 22:50:18 +00:00
|
|
|
#include "jsapi.h"
|
2014-08-05 16:18:30 +00:00
|
|
|
#include "jsfriendapi.h"
|
2013-08-20 06:43:47 +00:00
|
|
|
#include "js/StructuredClone.h"
|
2014-08-05 16:18:30 +00:00
|
|
|
#include "xpcpublic.h"
|
2011-12-28 08:13:38 +00:00
|
|
|
|
|
|
|
#include "mozilla/Base64.h"
|
2014-08-05 16:18:30 +00:00
|
|
|
#include "mozilla/dom/ScriptSettings.h"
|
2011-12-28 08:13:38 +00:00
|
|
|
|
|
|
|
using namespace mozilla;
|
2011-04-25 02:30:54 +00:00
|
|
|
|
|
|
|
NS_IMPL_ADDREF(nsStructuredCloneContainer)
|
|
|
|
NS_IMPL_RELEASE(nsStructuredCloneContainer)
|
|
|
|
|
|
|
|
NS_INTERFACE_MAP_BEGIN(nsStructuredCloneContainer)
|
|
|
|
NS_INTERFACE_MAP_ENTRY(nsIStructuredCloneContainer)
|
|
|
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
|
|
|
NS_INTERFACE_MAP_END
|
|
|
|
|
|
|
|
nsStructuredCloneContainer::nsStructuredCloneContainer()
|
2012-07-30 14:20:58 +00:00
|
|
|
: mData(nullptr), mSize(0), mVersion(0)
|
2011-04-25 02:30:54 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
nsStructuredCloneContainer::~nsStructuredCloneContainer()
|
|
|
|
{
|
|
|
|
free(mData);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
2014-08-05 16:18:30 +00:00
|
|
|
nsStructuredCloneContainer::InitFromJSVal(JS::Handle<JS::Value> aData)
|
2011-04-25 02:30:54 +00:00
|
|
|
{
|
|
|
|
NS_ENSURE_STATE(!mData);
|
|
|
|
|
2012-07-30 14:20:58 +00:00
|
|
|
uint64_t* jsBytes = nullptr;
|
2014-08-05 16:18:30 +00:00
|
|
|
bool success = false;
|
|
|
|
if (aData.isPrimitive()) {
|
|
|
|
// |aData| is a primitive, so the structured clone algorithm won't run
|
|
|
|
// script and we can just use AutoJSAPI.
|
|
|
|
dom::AutoJSAPI jsapi;
|
|
|
|
jsapi.Init();
|
|
|
|
success = JS_WriteStructuredClone(jsapi.cx(), aData, &jsBytes, &mSize,
|
|
|
|
nullptr, nullptr,
|
|
|
|
JS::UndefinedHandleValue);
|
|
|
|
} else {
|
|
|
|
// |aData| is an object and the structured clone algorithm can run script as
|
|
|
|
// part of the "own" "deep clone" sub-steps, so we need an AutoEntryScript.
|
|
|
|
// http://www.whatwg.org/specs/web-apps/current-work/#internal-structured-cloning-algorithm
|
|
|
|
nsIGlobalObject* nativeGlobal =
|
2014-09-21 19:31:53 +00:00
|
|
|
xpc::NativeGlobal(js::GetGlobalForObjectCrossCompartment(&aData.toObject()));
|
2014-08-05 16:18:30 +00:00
|
|
|
dom::AutoEntryScript aes(nativeGlobal);
|
|
|
|
success = JS_WriteStructuredClone(aes.cx(), aData, &jsBytes, &mSize,
|
|
|
|
nullptr, nullptr,
|
|
|
|
JS::UndefinedHandleValue);
|
|
|
|
}
|
2011-04-25 02:30:54 +00:00
|
|
|
NS_ENSURE_STATE(success);
|
|
|
|
NS_ENSURE_STATE(jsBytes);
|
|
|
|
|
|
|
|
// Copy jsBytes into our own buffer.
|
Bug 714332 - Fix uint64 types usage in dom/workers and dom/base; r=bent
Use uint64_t where appropriate for mData, fails to build on OpenBSD
otherwise with wrong casts/no matching template.
dom/base/nsStructuredCloneContainer.cpp:96: error: invalid conversion from 'PRUint64*' to 'uint64_t*'
dom/base/nsStructuredCloneContainer.cpp:131: error: invalid conversion from 'PRUint64*' to 'uint64_t*'
dom/workers/WorkerPrivate.cpp:822: error: no matching function for call to 'JSAutoStructuredCloneBuffer::steal(uint64**, size_t*)'
../../dist/include/jsapi.h:4641: note: candidates are: void
JSAutoStructuredCloneBuffer::steal(uint64_t**, size_t*, uint32_t*) <near
match>
dom/workers/WorkerPrivate.cpp:833: error: no matching function for call to 'JSAutoStructuredCloneBuffer::adopt(uint64*&, size_t&)'
../../dist/include/jsapi.h:4634: note: candidates are: void
JSAutoStructuredCloneBuffer::adopt(uint64_t*, size_t, uint32_t) <near
match>
dom/workers/WorkerPrivate.cpp:853: error: no matching function for call to 'JSAutoStructuredCloneBuffer::steal(uint64**, size_t*)'
../../dist/include/jsapi.h:4641: note: candidates are: void
JSAutoStructuredCloneBuffer::steal(uint64_t**, size_t*, uint32_t*) <near
match>
2012-01-02 18:08:14 +00:00
|
|
|
mData = (uint64_t*) malloc(mSize);
|
2011-04-25 02:30:54 +00:00
|
|
|
if (!mData) {
|
|
|
|
mSize = 0;
|
|
|
|
mVersion = 0;
|
2011-05-16 20:36:12 +00:00
|
|
|
|
2013-11-15 04:42:34 +00:00
|
|
|
JS_ClearStructuredClone(jsBytes, mSize, nullptr, nullptr);
|
2011-04-25 02:30:54 +00:00
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
mVersion = JS_STRUCTURED_CLONE_VERSION;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(mData, jsBytes, mSize);
|
2011-05-16 20:36:12 +00:00
|
|
|
|
2013-11-15 04:42:34 +00:00
|
|
|
JS_ClearStructuredClone(jsBytes, mSize, nullptr, nullptr);
|
2011-04-25 02:30:54 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsStructuredCloneContainer::InitFromBase64(const nsAString &aData,
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t aFormatVersion,
|
2011-04-25 02:30:54 +00:00
|
|
|
JSContext *aCx)
|
|
|
|
{
|
|
|
|
NS_ENSURE_STATE(!mData);
|
|
|
|
|
|
|
|
NS_ConvertUTF16toUTF8 data(aData);
|
|
|
|
|
2012-09-02 02:35:17 +00:00
|
|
|
nsAutoCString binaryData;
|
2011-12-28 08:13:38 +00:00
|
|
|
nsresult rv = Base64Decode(data, binaryData);
|
2011-04-25 02:30:54 +00:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
// Copy the string's data into our own buffer.
|
Bug 714332 - Fix uint64 types usage in dom/workers and dom/base; r=bent
Use uint64_t where appropriate for mData, fails to build on OpenBSD
otherwise with wrong casts/no matching template.
dom/base/nsStructuredCloneContainer.cpp:96: error: invalid conversion from 'PRUint64*' to 'uint64_t*'
dom/base/nsStructuredCloneContainer.cpp:131: error: invalid conversion from 'PRUint64*' to 'uint64_t*'
dom/workers/WorkerPrivate.cpp:822: error: no matching function for call to 'JSAutoStructuredCloneBuffer::steal(uint64**, size_t*)'
../../dist/include/jsapi.h:4641: note: candidates are: void
JSAutoStructuredCloneBuffer::steal(uint64_t**, size_t*, uint32_t*) <near
match>
dom/workers/WorkerPrivate.cpp:833: error: no matching function for call to 'JSAutoStructuredCloneBuffer::adopt(uint64*&, size_t&)'
../../dist/include/jsapi.h:4634: note: candidates are: void
JSAutoStructuredCloneBuffer::adopt(uint64_t*, size_t, uint32_t) <near
match>
dom/workers/WorkerPrivate.cpp:853: error: no matching function for call to 'JSAutoStructuredCloneBuffer::steal(uint64**, size_t*)'
../../dist/include/jsapi.h:4641: note: candidates are: void
JSAutoStructuredCloneBuffer::steal(uint64_t**, size_t*, uint32_t*) <near
match>
2012-01-02 18:08:14 +00:00
|
|
|
mData = (uint64_t*) malloc(binaryData.Length());
|
2011-04-25 02:30:54 +00:00
|
|
|
NS_ENSURE_STATE(mData);
|
|
|
|
memcpy(mData, binaryData.get(), binaryData.Length());
|
|
|
|
|
|
|
|
mSize = binaryData.Length();
|
|
|
|
mVersion = aFormatVersion;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsStructuredCloneContainer::DeserializeToVariant(JSContext *aCx,
|
|
|
|
nsIVariant **aData)
|
|
|
|
{
|
|
|
|
NS_ENSURE_STATE(mData);
|
|
|
|
NS_ENSURE_ARG_POINTER(aData);
|
2012-07-30 14:20:58 +00:00
|
|
|
*aData = nullptr;
|
2011-04-25 02:30:54 +00:00
|
|
|
|
2013-04-11 22:52:10 +00:00
|
|
|
// Deserialize to a JS::Value.
|
2013-05-04 07:52:57 +00:00
|
|
|
JS::Rooted<JS::Value> jsStateObj(aCx);
|
2013-08-08 22:53:04 +00:00
|
|
|
bool hasTransferable = false;
|
2011-09-29 06:19:26 +00:00
|
|
|
bool success = JS_ReadStructuredClone(aCx, mData, mSize, mVersion,
|
2013-10-23 00:18:32 +00:00
|
|
|
&jsStateObj, nullptr, nullptr) &&
|
2012-10-03 22:19:22 +00:00
|
|
|
JS_StructuredCloneHasTransferables(mData, mSize,
|
|
|
|
&hasTransferable);
|
|
|
|
// We want to be sure that mData doesn't contain transferable objects
|
|
|
|
MOZ_ASSERT(!hasTransferable);
|
|
|
|
NS_ENSURE_STATE(success && !hasTransferable);
|
2011-04-25 02:30:54 +00:00
|
|
|
|
2013-04-11 22:52:10 +00:00
|
|
|
// Now wrap the JS::Value as an nsIVariant.
|
2011-04-25 02:30:54 +00:00
|
|
|
nsCOMPtr<nsIVariant> varStateObj;
|
|
|
|
nsCOMPtr<nsIXPConnect> xpconnect = do_GetService(nsIXPConnect::GetCID());
|
|
|
|
NS_ENSURE_STATE(xpconnect);
|
2014-01-09 17:39:36 +00:00
|
|
|
xpconnect->JSValToVariant(aCx, jsStateObj, getter_AddRefs(varStateObj));
|
2011-04-25 02:30:54 +00:00
|
|
|
NS_ENSURE_STATE(varStateObj);
|
|
|
|
|
2013-08-10 06:47:59 +00:00
|
|
|
NS_ADDREF(*aData = varStateObj);
|
2011-04-25 02:30:54 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsStructuredCloneContainer::GetDataAsBase64(nsAString &aOut)
|
|
|
|
{
|
|
|
|
NS_ENSURE_STATE(mData);
|
|
|
|
aOut.Truncate();
|
|
|
|
|
2012-09-02 02:35:17 +00:00
|
|
|
nsAutoCString binaryData(reinterpret_cast<char*>(mData), mSize);
|
|
|
|
nsAutoCString base64Data;
|
2011-12-28 08:13:38 +00:00
|
|
|
nsresult rv = Base64Encode(binaryData, base64Data);
|
2011-04-25 02:30:54 +00:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
aOut.Assign(NS_ConvertASCIItoUTF16(base64Data));
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
2012-08-22 15:56:38 +00:00
|
|
|
nsStructuredCloneContainer::GetSerializedNBytes(uint64_t *aSize)
|
2011-04-25 02:30:54 +00:00
|
|
|
{
|
|
|
|
NS_ENSURE_STATE(mData);
|
|
|
|
NS_ENSURE_ARG_POINTER(aSize);
|
|
|
|
|
2012-08-22 15:56:38 +00:00
|
|
|
// mSize is a size_t, while aSize is a uint64_t. We rely on an implicit cast
|
2012-10-11 23:38:04 +00:00
|
|
|
// here so that we'll get a compile error if a size_t-to-uint64_t cast is
|
2011-04-25 02:30:54 +00:00
|
|
|
// narrowing.
|
|
|
|
*aSize = mSize;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
2012-08-22 15:56:38 +00:00
|
|
|
nsStructuredCloneContainer::GetFormatVersion(uint32_t *aFormatVersion)
|
2011-04-25 02:30:54 +00:00
|
|
|
{
|
|
|
|
NS_ENSURE_STATE(mData);
|
|
|
|
NS_ENSURE_ARG_POINTER(aFormatVersion);
|
|
|
|
*aFormatVersion = mVersion;
|
|
|
|
return NS_OK;
|
|
|
|
}
|