// Copyright 2010 the V8 project authors. All rights reserved.
|
// Use of this source code is governed by a BSD-style license that can be
|
// found in the LICENSE file.
|
|
#include "src/extensions/externalize-string-extension.h"
|
|
#include "src/api-inl.h"
|
#include "src/handles.h"
|
#include "src/isolate.h"
|
#include "src/objects-inl.h"
|
|
namespace v8 {
|
namespace internal {
|
|
template <typename Char, typename Base>
|
class SimpleStringResource : public Base {
|
public:
|
// Takes ownership of |data|.
|
SimpleStringResource(Char* data, size_t length)
|
: data_(data),
|
length_(length) {}
|
|
virtual ~SimpleStringResource() { delete[] data_; }
|
|
virtual const Char* data() const { return data_; }
|
|
virtual size_t length() const { return length_; }
|
|
private:
|
Char* const data_;
|
const size_t length_;
|
};
|
|
|
typedef SimpleStringResource<char, v8::String::ExternalOneByteStringResource>
|
SimpleOneByteStringResource;
|
typedef SimpleStringResource<uc16, v8::String::ExternalStringResource>
|
SimpleTwoByteStringResource;
|
|
const char* const ExternalizeStringExtension::kSource =
|
"native function externalizeString();"
|
"native function isOneByteString();"
|
"function x() { return 1; }";
|
|
v8::Local<v8::FunctionTemplate>
|
ExternalizeStringExtension::GetNativeFunctionTemplate(
|
v8::Isolate* isolate, v8::Local<v8::String> str) {
|
if (strcmp(*v8::String::Utf8Value(isolate, str), "externalizeString") == 0) {
|
return v8::FunctionTemplate::New(isolate,
|
ExternalizeStringExtension::Externalize);
|
} else {
|
DCHECK_EQ(strcmp(*v8::String::Utf8Value(isolate, str), "isOneByteString"),
|
0);
|
return v8::FunctionTemplate::New(isolate,
|
ExternalizeStringExtension::IsOneByte);
|
}
|
}
|
|
|
void ExternalizeStringExtension::Externalize(
|
const v8::FunctionCallbackInfo<v8::Value>& args) {
|
if (args.Length() < 1 || !args[0]->IsString()) {
|
args.GetIsolate()->ThrowException(
|
v8::String::NewFromUtf8(
|
args.GetIsolate(),
|
"First parameter to externalizeString() must be a string.",
|
NewStringType::kNormal).ToLocalChecked());
|
return;
|
}
|
bool force_two_byte = false;
|
if (args.Length() >= 2) {
|
if (args[1]->IsBoolean()) {
|
force_two_byte =
|
args[1]
|
->BooleanValue(args.GetIsolate()->GetCurrentContext())
|
.FromJust();
|
} else {
|
args.GetIsolate()->ThrowException(
|
v8::String::NewFromUtf8(
|
args.GetIsolate(),
|
"Second parameter to externalizeString() must be a boolean.",
|
NewStringType::kNormal).ToLocalChecked());
|
return;
|
}
|
}
|
bool result = false;
|
Handle<String> string = Utils::OpenHandle(*args[0].As<v8::String>());
|
if (!string->SupportsExternalization()) {
|
args.GetIsolate()->ThrowException(
|
v8::String::NewFromUtf8(args.GetIsolate(),
|
"string does not support externalization.",
|
NewStringType::kNormal)
|
.ToLocalChecked());
|
return;
|
}
|
if (string->IsOneByteRepresentation() && !force_two_byte) {
|
uint8_t* data = new uint8_t[string->length()];
|
String::WriteToFlat(*string, data, 0, string->length());
|
SimpleOneByteStringResource* resource = new SimpleOneByteStringResource(
|
reinterpret_cast<char*>(data), string->length());
|
result = Utils::ToLocal(string)->MakeExternal(resource);
|
if (!result) delete resource;
|
} else {
|
uc16* data = new uc16[string->length()];
|
String::WriteToFlat(*string, data, 0, string->length());
|
SimpleTwoByteStringResource* resource = new SimpleTwoByteStringResource(
|
data, string->length());
|
result = Utils::ToLocal(string)->MakeExternal(resource);
|
if (!result) delete resource;
|
}
|
if (!result) {
|
args.GetIsolate()->ThrowException(
|
v8::String::NewFromUtf8(args.GetIsolate(),
|
"externalizeString() failed.",
|
NewStringType::kNormal).ToLocalChecked());
|
return;
|
}
|
}
|
|
|
void ExternalizeStringExtension::IsOneByte(
|
const v8::FunctionCallbackInfo<v8::Value>& args) {
|
if (args.Length() != 1 || !args[0]->IsString()) {
|
args.GetIsolate()->ThrowException(
|
v8::String::NewFromUtf8(
|
args.GetIsolate(),
|
"isOneByteString() requires a single string argument.",
|
NewStringType::kNormal).ToLocalChecked());
|
return;
|
}
|
bool is_one_byte =
|
Utils::OpenHandle(*args[0].As<v8::String>())->IsOneByteRepresentation();
|
args.GetReturnValue().Set(is_one_byte);
|
}
|
|
} // namespace internal
|
} // namespace v8
|