mirror of
https://github.com/denoland/rusty_v8.git
synced 2025-01-13 17:40:23 -05:00
Add non-static external onebyte string (#1511)
* Rename existing method
* Enable getting external onebyte resource + fix encoding
* Enable creating an external onebyte string that's freed by v8
* Add basic test
* Boxed str, fix as_str, doc comments
* Cleanup binding + fix adjust memory
* Use bindgen + raw api
* Rm unnecessary cast
* Rm unnecessary lines
* Revert "Use bindgen + raw api"
This reverts commit 8ea00476b6
.
* Add raw api, cleanup
* Fix unnecessary conversion
* Address comments
* Trying to retrigger CI
This commit is contained in:
parent
04dd49291a
commit
6bac39b579
4 changed files with 215 additions and 18 deletions
|
@ -1071,6 +1071,35 @@ v8__String__GetExternalStringResourceBase(const v8::String& self,
|
|||
return self.GetExternalStringResourceBase(encoding_out);
|
||||
}
|
||||
|
||||
class ExternalOneByteString : public v8::String::ExternalOneByteStringResource {
|
||||
public:
|
||||
using RustDestroy = void (*)(char*, size_t);
|
||||
ExternalOneByteString(char* data, size_t length, RustDestroy rustDestroy,
|
||||
v8::Isolate* isolate)
|
||||
: data_(data),
|
||||
length_(length),
|
||||
rustDestroy_(rustDestroy),
|
||||
isolate_(isolate) {
|
||||
isolate_->AdjustAmountOfExternalAllocatedMemory(
|
||||
static_cast<int64_t>(length_));
|
||||
}
|
||||
~ExternalOneByteString() override {
|
||||
(*rustDestroy_)(data_, length_);
|
||||
isolate_->AdjustAmountOfExternalAllocatedMemory(
|
||||
-static_cast<int64_t>(-length_));
|
||||
}
|
||||
|
||||
const char* data() const override { return data_; }
|
||||
|
||||
size_t length() const override { return length_; }
|
||||
|
||||
private:
|
||||
char* data_;
|
||||
const size_t length_;
|
||||
RustDestroy rustDestroy_;
|
||||
v8::Isolate* isolate_;
|
||||
};
|
||||
|
||||
class ExternalStaticOneByteStringResource
|
||||
: public v8::String::ExternalOneByteStringResource {
|
||||
public:
|
||||
|
@ -1109,7 +1138,7 @@ class ExternalConstOneByteStringResource
|
|||
const int _length;
|
||||
};
|
||||
|
||||
const v8::String* v8__String__NewExternalOneByte(
|
||||
const v8::String* v8__String__NewExternalOneByteConst(
|
||||
v8::Isolate* isolate, v8::String::ExternalOneByteStringResource* resource) {
|
||||
return maybe_local_to_ptr(v8::String::NewExternalOneByte(isolate, resource));
|
||||
}
|
||||
|
@ -1121,6 +1150,23 @@ const v8::String* v8__String__NewExternalOneByteStatic(v8::Isolate* isolate,
|
|||
isolate, new ExternalStaticOneByteStringResource(data, length)));
|
||||
}
|
||||
|
||||
const v8::String* v8__String__NewExternalOneByte(
|
||||
v8::Isolate* isolate, char* data, int length,
|
||||
ExternalOneByteString::RustDestroy rustDestroy) {
|
||||
return maybe_local_to_ptr(v8::String::NewExternalOneByte(
|
||||
isolate, new ExternalOneByteString(data, length, rustDestroy, isolate)));
|
||||
}
|
||||
|
||||
const char* v8__ExternalOneByteStringResource__data(
|
||||
v8::String::ExternalOneByteStringResource* self) {
|
||||
return self->data();
|
||||
}
|
||||
|
||||
size_t v8__ExternalOneByteStringResource__length(
|
||||
v8::String::ExternalOneByteStringResource* self) {
|
||||
return self->length();
|
||||
}
|
||||
|
||||
class ExternalStaticStringResource : public v8::String::ExternalStringResource {
|
||||
public:
|
||||
ExternalStaticStringResource(const uint16_t* data, int length)
|
||||
|
|
|
@ -146,6 +146,7 @@ pub use script::ScriptOrigin;
|
|||
pub use script_compiler::CachedData;
|
||||
pub use snapshot::FunctionCodeHandling;
|
||||
pub use snapshot::StartupData;
|
||||
pub use string::Encoding;
|
||||
pub use string::NewStringType;
|
||||
pub use string::OneByteConst;
|
||||
pub use string::WriteOptions;
|
||||
|
|
157
src/string.rs
157
src/string.rs
|
@ -77,9 +77,9 @@ extern "C" {
|
|||
fn v8__String__GetExternalStringResourceBase(
|
||||
this: *const String,
|
||||
encoding: *mut Encoding,
|
||||
) -> *mut ExternalOneByteStringResourceBase;
|
||||
) -> *mut ExternalStringResourceBase;
|
||||
|
||||
fn v8__String__NewExternalOneByte(
|
||||
fn v8__String__NewExternalOneByteConst(
|
||||
isolate: *mut Isolate,
|
||||
onebyte_const: *const OneByteConst,
|
||||
) -> *const String;
|
||||
|
@ -90,6 +90,13 @@ extern "C" {
|
|||
length: int,
|
||||
) -> *const String;
|
||||
|
||||
fn v8__String__NewExternalOneByte(
|
||||
isolate: *mut Isolate,
|
||||
buffer: *mut char,
|
||||
length: size_t,
|
||||
free: extern "C" fn(*mut char, size_t),
|
||||
) -> *const String;
|
||||
|
||||
fn v8__String__NewExternalTwoByteStatic(
|
||||
isolate: *mut Isolate,
|
||||
buffer: *const u16,
|
||||
|
@ -103,20 +110,63 @@ extern "C" {
|
|||
#[allow(dead_code)]
|
||||
fn v8__String__IsOneByte(this: *const String) -> bool;
|
||||
fn v8__String__ContainsOnlyOneByte(this: *const String) -> bool;
|
||||
fn v8__ExternalOneByteStringResource__data(
|
||||
this: *const ExternalOneByteStringResource,
|
||||
) -> *const char;
|
||||
fn v8__ExternalOneByteStringResource__length(
|
||||
this: *const ExternalOneByteStringResource,
|
||||
) -> size_t;
|
||||
}
|
||||
|
||||
#[derive(PartialEq, Debug)]
|
||||
#[repr(C)]
|
||||
pub enum Encoding {
|
||||
Unknown = 0,
|
||||
OneByte = 1,
|
||||
TwoByte = 2,
|
||||
Unknown = 0x1,
|
||||
TwoByte = 0x2,
|
||||
OneByte = 0x8,
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct ExternalStringResource(Opaque);
|
||||
|
||||
#[repr(C)]
|
||||
pub struct ExternalOneByteStringResourceBase(Opaque);
|
||||
pub struct ExternalStringResourceBase(Opaque);
|
||||
|
||||
#[repr(C)]
|
||||
/// An external, one-byte string resource.
|
||||
/// This corresponds with `v8::String::ExternalOneByteStringResource`.
|
||||
pub struct ExternalOneByteStringResource(Opaque);
|
||||
|
||||
impl ExternalOneByteStringResource {
|
||||
/// Returns a pointer to the data owned by this resource.
|
||||
/// This pointer is valid as long as the resource is alive.
|
||||
/// The data is guaranteed to be ASCII.
|
||||
pub fn data(&self) -> *const char {
|
||||
unsafe { v8__ExternalOneByteStringResource__data(self) }
|
||||
}
|
||||
|
||||
/// Returns the length of the data owned by this resource.
|
||||
pub fn length(&self) -> usize {
|
||||
unsafe { v8__ExternalOneByteStringResource__length(self) }
|
||||
}
|
||||
|
||||
/// Returns the data owned by this resource as a string slice.
|
||||
/// The data is guaranteed to be ASCII.
|
||||
pub fn as_str(&self) -> &str {
|
||||
let len = self.length();
|
||||
if len == 0 {
|
||||
""
|
||||
} else {
|
||||
// SAFETY: We know this is ASCII and length > 0
|
||||
unsafe {
|
||||
std::str::from_utf8_unchecked(std::slice::from_raw_parts(
|
||||
self.data().cast(),
|
||||
len,
|
||||
))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
#[derive(Copy, Clone, Debug)]
|
||||
|
@ -512,8 +562,8 @@ impl String {
|
|||
}
|
||||
}
|
||||
|
||||
// Creates a v8::String from a `&'static OneByteConst`
|
||||
// which is guaranteed to be Latin-1 or ASCII.
|
||||
/// Creates a v8::String from a `&'static OneByteConst`
|
||||
/// which is guaranteed to be Latin-1 or ASCII.
|
||||
#[inline(always)]
|
||||
pub fn new_from_onebyte_const<'s>(
|
||||
scope: &mut HandleScope<'s, ()>,
|
||||
|
@ -521,13 +571,13 @@ impl String {
|
|||
) -> Option<Local<'s, String>> {
|
||||
unsafe {
|
||||
scope.cast_local(|sd| {
|
||||
v8__String__NewExternalOneByte(sd.get_isolate_ptr(), onebyte_const)
|
||||
v8__String__NewExternalOneByteConst(sd.get_isolate_ptr(), onebyte_const)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// Creates a v8::String from a `&'static [u8]`,
|
||||
// must be Latin-1 or ASCII, not UTF-8 !
|
||||
/// Creates a v8::String from a `&'static [u8]`,
|
||||
/// must be Latin-1 or ASCII, not UTF-8!
|
||||
#[inline(always)]
|
||||
pub fn new_external_onebyte_static<'s>(
|
||||
scope: &mut HandleScope<'s, ()>,
|
||||
|
@ -545,7 +595,54 @@ impl String {
|
|||
}
|
||||
}
|
||||
|
||||
// Creates a v8::String from a `&'static [u16]`.
|
||||
/// Creates a `v8::String` from owned bytes.
|
||||
/// The bytes must be Latin-1 or ASCII.
|
||||
/// V8 will take ownership of the buffer and free it when the string is garbage collected.
|
||||
#[inline(always)]
|
||||
pub fn new_external_onebyte<'s>(
|
||||
scope: &mut HandleScope<'s, ()>,
|
||||
buffer: Box<[u8]>,
|
||||
) -> Option<Local<'s, String>> {
|
||||
let buffer_len = buffer.len();
|
||||
unsafe {
|
||||
scope.cast_local(|sd| {
|
||||
v8__String__NewExternalOneByte(
|
||||
sd.get_isolate_ptr(),
|
||||
Box::into_raw(buffer).cast::<char>(),
|
||||
buffer_len,
|
||||
free_rust_external_onebyte,
|
||||
)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
/// Creates a `v8::String` from owned bytes, length, and a custom destructor.
|
||||
/// The bytes must be Latin-1 or ASCII.
|
||||
/// V8 will take ownership of the buffer and free it when the string is garbage collected.
|
||||
///
|
||||
/// SAFETY: `buffer` must be owned (valid for the lifetime of the string), and
|
||||
/// `destructor` must be a valid function pointer that can free the buffer.
|
||||
/// The destructor will be called with the buffer and length when the string is garbage collected.
|
||||
#[inline(always)]
|
||||
pub unsafe fn new_external_onebyte_raw<'s>(
|
||||
scope: &mut HandleScope<'s, ()>,
|
||||
buffer: *mut char,
|
||||
buffer_len: usize,
|
||||
destructor: extern "C" fn(*mut char, usize),
|
||||
) -> Option<Local<'s, String>> {
|
||||
unsafe {
|
||||
scope.cast_local(|sd| {
|
||||
v8__String__NewExternalOneByte(
|
||||
sd.get_isolate_ptr(),
|
||||
buffer,
|
||||
buffer_len,
|
||||
destructor,
|
||||
)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
/// Creates a v8::String from a `&'static [u16]`.
|
||||
#[inline(always)]
|
||||
pub fn new_external_twobyte_static<'s>(
|
||||
scope: &mut HandleScope<'s, ()>,
|
||||
|
@ -563,18 +660,37 @@ impl String {
|
|||
}
|
||||
}
|
||||
|
||||
// Get the ExternalStringResource for an external string.
|
||||
//
|
||||
// Returns None if is_external() doesn't return true.
|
||||
/// Get the ExternalStringResource for an external string.
|
||||
///
|
||||
/// Returns None if is_external() doesn't return true.
|
||||
pub fn get_external_string_resource(
|
||||
&self,
|
||||
) -> Option<NonNull<ExternalStringResource>> {
|
||||
NonNull::new(unsafe { v8__String__GetExternalStringResource(self) })
|
||||
}
|
||||
|
||||
/// Get the ExternalOneByteStringResource for an external one-byte string.
|
||||
///
|
||||
/// Returns None if is_external_onebyte() doesn't return true.
|
||||
pub fn get_external_onebyte_string_resource(
|
||||
&self,
|
||||
) -> Option<NonNull<ExternalOneByteStringResource>> {
|
||||
let (base, encoding) = self.get_external_string_resource_base();
|
||||
let base = base?;
|
||||
if encoding != Encoding::OneByte {
|
||||
return None;
|
||||
}
|
||||
|
||||
Some(base.cast())
|
||||
}
|
||||
|
||||
/// Get the ExternalStringResourceBase for an external string.
|
||||
/// Note this is just the base class, and isn't very useful on its own.
|
||||
/// You'll want to downcast to one of its subclasses, for instance
|
||||
/// with `get_external_onebyte_string_resource`.
|
||||
pub fn get_external_string_resource_base(
|
||||
&self,
|
||||
) -> (Option<NonNull<ExternalOneByteStringResourceBase>>, Encoding) {
|
||||
) -> (Option<NonNull<ExternalStringResourceBase>>, Encoding) {
|
||||
let mut encoding = Encoding::Unknown;
|
||||
(
|
||||
NonNull::new(unsafe {
|
||||
|
@ -805,3 +921,12 @@ impl String {
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub extern "C" fn free_rust_external_onebyte(s: *mut char, len: usize) {
|
||||
unsafe {
|
||||
let slice = std::slice::from_raw_parts_mut(s, len);
|
||||
|
||||
// Drop the slice
|
||||
drop(Box::from_raw(slice));
|
||||
}
|
||||
}
|
||||
|
|
|
@ -7839,10 +7839,35 @@ fn private() {
|
|||
}
|
||||
|
||||
#[test]
|
||||
fn bigint() {
|
||||
fn external_onebyte_string() {
|
||||
let _setup_guard = setup::parallel_test();
|
||||
let isolate = &mut v8::Isolate::new(Default::default());
|
||||
let scope = &mut v8::HandleScope::new(isolate);
|
||||
|
||||
let input = "hello";
|
||||
let s = v8::String::new_external_onebyte(
|
||||
scope,
|
||||
Box::<str>::from(input).into_boxed_bytes(),
|
||||
)
|
||||
.unwrap();
|
||||
|
||||
assert!(s.is_external_onebyte());
|
||||
assert_eq!(s.utf8_length(scope), 5);
|
||||
|
||||
let one_byte =
|
||||
unsafe { &*s.get_external_onebyte_string_resource().unwrap().as_ptr() };
|
||||
|
||||
assert_eq!(one_byte.length(), 5);
|
||||
|
||||
assert_eq!(one_byte.as_str(), "hello");
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn bigint() {
|
||||
let _setup_guard: setup::SetupGuard<std::sync::RwLockReadGuard<'_, ()>> =
|
||||
setup::parallel_test();
|
||||
let isolate = &mut v8::Isolate::new(Default::default());
|
||||
let scope = &mut v8::HandleScope::new(isolate);
|
||||
let context = v8::Context::new(scope);
|
||||
let scope = &mut v8::ContextScope::new(scope, context);
|
||||
|
||||
|
|
Loading…
Reference in a new issue