0
0
Fork 0
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:
Nathan Whitaker 2024-06-28 15:39:23 -07:00 committed by GitHub
parent 04dd49291a
commit 6bac39b579
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 215 additions and 18 deletions

View file

@ -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)

View file

@ -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;

View file

@ -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));
}
}

View file

@ -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);