316 lines
11 KiB
C++
316 lines
11 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/BindingUtils.h"
|
|
#include "mozilla/dom/Document.h"
|
|
#include "mozilla/dom/TestFunctions.h"
|
|
#include "mozilla/dom/TestFunctionsBinding.h"
|
|
#include "mozilla/dom/WindowBinding.h"
|
|
#include "mozilla/dom/WrapperCachedNonISupportsTestInterface.h"
|
|
#include "mozilla/StringBuffer.h"
|
|
#include "mozITestInterfaceJS.h"
|
|
#include "nsComponentManagerUtils.h"
|
|
#include "nsGlobalWindowInner.h"
|
|
|
|
namespace mozilla::dom {
|
|
|
|
/* static */
|
|
UniquePtr<TestFunctions> TestFunctions::Constructor(GlobalObject& aGlobal) {
|
|
return MakeUnique<TestFunctions>();
|
|
}
|
|
|
|
/* static */
|
|
void TestFunctions::ThrowUncatchableException(GlobalObject& aGlobal,
|
|
ErrorResult& aRv) {
|
|
aRv.ThrowUncatchableException();
|
|
}
|
|
|
|
/* static */
|
|
Promise* TestFunctions::PassThroughPromise(GlobalObject& aGlobal,
|
|
Promise& aPromise) {
|
|
return &aPromise;
|
|
}
|
|
|
|
/* static */
|
|
already_AddRefed<Promise> TestFunctions::PassThroughCallbackPromise(
|
|
GlobalObject& aGlobal, PromiseReturner& aCallback, ErrorResult& aRv) {
|
|
return aCallback.Call(aRv);
|
|
}
|
|
|
|
void TestFunctions::SetStringData(const nsAString& aString) {
|
|
mStringData = aString;
|
|
}
|
|
|
|
void TestFunctions::GetStringDataAsAString(nsAString& aString) {
|
|
aString = mStringData;
|
|
}
|
|
|
|
void TestFunctions::GetStringDataAsAString(uint32_t aLength,
|
|
nsAString& aString) {
|
|
MOZ_RELEASE_ASSERT(aLength <= mStringData.Length(),
|
|
"Bogus test passing in a too-big length");
|
|
aString.Assign(mStringData.BeginReading(), aLength);
|
|
}
|
|
|
|
void TestFunctions::GetStringDataAsDOMString(const Optional<uint32_t>& aLength,
|
|
DOMString& aString) {
|
|
uint32_t length;
|
|
if (aLength.WasPassed()) {
|
|
length = aLength.Value();
|
|
MOZ_RELEASE_ASSERT(length <= mStringData.Length(),
|
|
"Bogus test passing in a too-big length");
|
|
} else {
|
|
length = mStringData.Length();
|
|
}
|
|
|
|
if (StringBuffer* buf = mStringData.GetStringBuffer()) {
|
|
aString.SetKnownLiveStringBuffer(buf, length);
|
|
return;
|
|
}
|
|
|
|
// We better have an empty mStringData; otherwise why did we not have a string
|
|
// buffer?
|
|
MOZ_RELEASE_ASSERT(length == 0, "Why no stringbuffer?");
|
|
// No need to do anything here; aString is already empty.
|
|
}
|
|
|
|
void TestFunctions::GetShortLiteralString(nsAString& aString) {
|
|
// JS inline strings can hold 2 * sizeof(void*) chars, which on 32-bit means 8
|
|
// chars. Return fewer than that.
|
|
aString.AssignLiteral(u"012345");
|
|
}
|
|
|
|
void TestFunctions::GetMediumLiteralString(nsAString& aString) {
|
|
// JS inline strings are at most 2 * sizeof(void*) chars, so at most 16 on
|
|
// 64-bit. FakeString can hold 63 chars in its inline buffer (plus the null
|
|
// terminator). Let's return 40 chars; that way if we ever move to 128-bit
|
|
// void* or something this test will still be valid.
|
|
aString.AssignLiteral(u"0123456789012345678901234567890123456789");
|
|
}
|
|
|
|
void TestFunctions::GetLongLiteralString(nsAString& aString) {
|
|
// Need more than 64 chars.
|
|
aString.AssignLiteral(
|
|
u"0123456789012345678901234567890123456789" // 40
|
|
"0123456789012345678901234567890123456789" // 80
|
|
);
|
|
}
|
|
|
|
void TestFunctions::GetStringbufferString(const nsAString& aInput,
|
|
nsAString& aRetval) {
|
|
// We have to be a bit careful: if aRetval is an autostring, if we just assign
|
|
// it won't cause stringbuffer allocation. So we have to round-trip through
|
|
// something that definitely causes a stringbuffer allocation.
|
|
nsString str;
|
|
// Can't use operator= here, because if aInput is a literal string then str
|
|
// would end up the same way.
|
|
str.Assign(aInput.BeginReading(), aInput.Length());
|
|
|
|
// Now we might end up hitting our external string cache and getting the wrong
|
|
// sort of external string, so replace the last char by a different value
|
|
// (replacing, not just appending, to preserve the length). If we have an
|
|
// empty string, our caller screwed up and there's not much we can do for
|
|
// them.
|
|
if (str.Length() > 1) {
|
|
char16_t last = str[str.Length() - 1];
|
|
str.Truncate(str.Length() - 1);
|
|
if (last == 'x') {
|
|
str.Append('y');
|
|
} else {
|
|
str.Append('x');
|
|
}
|
|
}
|
|
|
|
// Here we use operator= to preserve stringbufferness.
|
|
aRetval = str;
|
|
}
|
|
|
|
StringType TestFunctions::GetStringType(const nsAString& aString) {
|
|
if (aString.IsLiteral()) {
|
|
return StringType::Literal;
|
|
}
|
|
|
|
if (aString.GetStringBuffer()) {
|
|
return StringType::Stringbuffer;
|
|
}
|
|
|
|
if (aString.GetDataFlags() & nsAString::DataFlags::INLINE) {
|
|
return StringType::Inline;
|
|
}
|
|
|
|
return StringType::Other;
|
|
}
|
|
|
|
bool TestFunctions::StringbufferMatchesStored(const nsAString& aString) {
|
|
return aString.GetStringBuffer() &&
|
|
aString.GetStringBuffer() == mStringData.GetStringBuffer();
|
|
}
|
|
|
|
void TestFunctions::TestThrowNsresult(ErrorResult& aError) {
|
|
nsCOMPtr<mozITestInterfaceJS> impl =
|
|
do_CreateInstance("@mozilla.org/dom/test-interface-js;1");
|
|
aError = impl->TestThrowNsresult();
|
|
}
|
|
|
|
void TestFunctions::TestThrowNsresultFromNative(ErrorResult& aError) {
|
|
nsCOMPtr<mozITestInterfaceJS> impl =
|
|
do_CreateInstance("@mozilla.org/dom/test-interface-js;1");
|
|
aError = impl->TestThrowNsresultFromNative();
|
|
}
|
|
|
|
already_AddRefed<Promise> TestFunctions::ThrowToRejectPromise(
|
|
GlobalObject& aGlobal, ErrorResult& aError) {
|
|
aError.Throw(NS_ERROR_DOM_INVALID_STATE_ERR);
|
|
return nullptr;
|
|
}
|
|
|
|
int32_t TestFunctions::One() const { return 1; }
|
|
|
|
int32_t TestFunctions::Two() const { return 2; }
|
|
|
|
void TestFunctions::SetClampedNullableOctet(const Nullable<uint8_t>& aOctet) {
|
|
mClampedNullableOctet = aOctet;
|
|
}
|
|
|
|
Nullable<uint8_t> TestFunctions::GetClampedNullableOctet() const {
|
|
return mClampedNullableOctet;
|
|
}
|
|
|
|
void TestFunctions::SetEnforcedNullableOctet(const Nullable<uint8_t>& aOctet) {
|
|
mEnforcedNullableOctet = aOctet;
|
|
}
|
|
|
|
Nullable<uint8_t> TestFunctions::GetEnforcedNullableOctet() const {
|
|
return mEnforcedNullableOctet;
|
|
}
|
|
|
|
void TestFunctions::SetArrayBufferView(const ArrayBufferView& aBuffer) {}
|
|
|
|
void TestFunctions::GetArrayBufferView(JSContext* aCx,
|
|
JS::Handle<JSObject*> aObj,
|
|
JS::MutableHandle<JSObject*> aRetval,
|
|
ErrorResult& aError) {
|
|
aError.Throw(NS_ERROR_NOT_IMPLEMENTED);
|
|
}
|
|
|
|
void TestFunctions::SetAllowSharedArrayBufferView(
|
|
const ArrayBufferView& aBuffer) {}
|
|
|
|
void TestFunctions::GetAllowSharedArrayBufferView(
|
|
JSContext* aCx, JS::Handle<JSObject*> aObj,
|
|
JS::MutableHandle<JSObject*> aRetval, ErrorResult& aError) {
|
|
aError.Throw(NS_ERROR_NOT_IMPLEMENTED);
|
|
}
|
|
|
|
void TestFunctions::SetSequenceOfArrayBufferView(
|
|
const Sequence<ArrayBufferView>& aBuffers) {}
|
|
|
|
void TestFunctions::GetSequenceOfArrayBufferView(JSContext* aCx,
|
|
JS::Handle<JSObject*> aObj,
|
|
nsTArray<JSObject*>& aRetval,
|
|
ErrorResult& aError) {
|
|
aError.Throw(NS_ERROR_NOT_IMPLEMENTED);
|
|
}
|
|
|
|
void TestFunctions::SetSequenceOfAllowSharedArrayBufferView(
|
|
const Sequence<ArrayBufferView>& aBuffers) {}
|
|
|
|
void TestFunctions::GetSequenceOfAllowSharedArrayBufferView(
|
|
JSContext* aCx, JS::Handle<JSObject*> aObj, nsTArray<JSObject*>& aRetval,
|
|
ErrorResult& aError) {
|
|
aError.Throw(NS_ERROR_NOT_IMPLEMENTED);
|
|
}
|
|
|
|
void TestFunctions::SetArrayBuffer(const ArrayBuffer& aBuffer) {}
|
|
|
|
void TestFunctions::GetArrayBuffer(JSContext* aCx, JS::Handle<JSObject*> aObj,
|
|
JS::MutableHandle<JSObject*> aRetval,
|
|
ErrorResult& aError) {
|
|
aError.Throw(NS_ERROR_NOT_IMPLEMENTED);
|
|
}
|
|
|
|
void TestFunctions::SetAllowSharedArrayBuffer(const ArrayBuffer& aBuffer) {}
|
|
|
|
void TestFunctions::GetAllowSharedArrayBuffer(
|
|
JSContext* aCx, JS::Handle<JSObject*> aObj,
|
|
JS::MutableHandle<JSObject*> aRetval, ErrorResult& aError) {
|
|
aError.Throw(NS_ERROR_NOT_IMPLEMENTED);
|
|
}
|
|
|
|
void TestFunctions::SetSequenceOfArrayBuffer(
|
|
const Sequence<ArrayBuffer>& aBuffers) {}
|
|
|
|
void TestFunctions::GetSequenceOfArrayBuffer(JSContext* aCx,
|
|
JS::Handle<JSObject*> aObj,
|
|
nsTArray<JSObject*>& aRetval,
|
|
ErrorResult& aError) {
|
|
aError.Throw(NS_ERROR_NOT_IMPLEMENTED);
|
|
}
|
|
|
|
void TestFunctions::SetSequenceOfAllowSharedArrayBuffer(
|
|
const Sequence<ArrayBuffer>& aBuffers) {}
|
|
|
|
void TestFunctions::GetSequenceOfAllowSharedArrayBuffer(
|
|
JSContext* aCx, JS::Handle<JSObject*> aObj, nsTArray<JSObject*>& aRetval,
|
|
ErrorResult& aError) {
|
|
aError.Throw(NS_ERROR_NOT_IMPLEMENTED);
|
|
}
|
|
|
|
void TestFunctions::TestNotAllowShared(const ArrayBufferView& aBuffer) {}
|
|
|
|
void TestFunctions::TestNotAllowShared(const ArrayBuffer& aBuffer) {}
|
|
|
|
void TestFunctions::TestNotAllowShared(const nsAString& aBuffer) {}
|
|
|
|
void TestFunctions::TestAllowShared(const ArrayBufferView& aBuffer) {}
|
|
|
|
void TestFunctions::TestAllowShared(const ArrayBuffer& aBuffer) {}
|
|
|
|
void TestFunctions::TestDictWithAllowShared(
|
|
const DictWithAllowSharedBufferSource& aDict) {}
|
|
|
|
void TestFunctions::TestUnionOfBufferSource(
|
|
const ArrayBufferOrArrayBufferViewOrString& aUnion) {}
|
|
|
|
void TestFunctions::TestUnionOfAllowSharedBufferSource(
|
|
const AllowSharedBufferSource& aUnion) {}
|
|
|
|
void TestFunctions::TestUnionWithAllowShared(
|
|
const MaybeSharedInt8ArrayOrMaybeSharedInt16Array& aUnion) {}
|
|
|
|
bool TestFunctions::ObjectFromAboutBlank(JSContext* aCx, JSObject* aObj) {
|
|
// We purposefully don't use WindowOrNull here, because we want to
|
|
// demonstrate the incorrect behavior we get, not just fail some asserts.
|
|
RefPtr<nsGlobalWindowInner> win;
|
|
UNWRAP_MAYBE_CROSS_ORIGIN_OBJECT(Window, aObj, win, aCx);
|
|
if (!win) {
|
|
return false;
|
|
}
|
|
|
|
Document* doc = win->GetDoc();
|
|
if (!doc) {
|
|
return false;
|
|
}
|
|
|
|
return doc->GetDocumentURI()->GetSpecOrDefault().EqualsLiteral("about:blank");
|
|
}
|
|
|
|
WrapperCachedNonISupportsTestInterface*
|
|
TestFunctions::WrapperCachedNonISupportsObject() {
|
|
if (!mWrapperCachedNonISupportsTestInterface) {
|
|
mWrapperCachedNonISupportsTestInterface =
|
|
new WrapperCachedNonISupportsTestInterface();
|
|
}
|
|
return mWrapperCachedNonISupportsTestInterface;
|
|
}
|
|
|
|
bool TestFunctions::WrapObject(JSContext* aCx,
|
|
JS::Handle<JSObject*> aGivenProto,
|
|
JS::MutableHandle<JSObject*> aWrapper) {
|
|
return TestFunctions_Binding::Wrap(aCx, this, aGivenProto, aWrapper);
|
|
}
|
|
|
|
} // namespace mozilla::dom
|