rw_deno_core/
ops_builtin.rs

1// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
2use crate::error::format_file_name;
3use crate::error::type_error;
4use crate::io::AdaptiveBufferStrategy;
5use crate::io::BufMutView;
6use crate::io::BufView;
7use crate::io::ResourceId;
8use crate::op2;
9use crate::ops_builtin_types;
10use crate::ops_builtin_v8;
11use crate::CancelHandle;
12use crate::JsBuffer;
13use crate::Op;
14use crate::OpDecl;
15use crate::OpState;
16use crate::Resource;
17use anyhow::Error;
18use bytes::BytesMut;
19use serde_v8::ByteString;
20use std::cell::RefCell;
21use std::io::stderr;
22use std::io::stdout;
23use std::io::Write;
24use std::rc::Rc;
25
26macro_rules! builtin_ops {
27  ( $($op:ident $(:: $sub:ident)*),* ) => {
28    pub const BUILTIN_OPS: &'static [OpDecl] = &[
29      $( $op $(:: $sub)*::DECL, )*
30    ];
31  }
32}
33
34builtin_ops! {
35  op_close,
36  op_try_close,
37  op_print,
38  op_resources,
39  op_wasm_streaming_feed,
40  op_wasm_streaming_set_url,
41  op_void_sync,
42  op_error_async,
43  op_error_async_deferred,
44  op_void_async,
45  op_void_async_deferred,
46  op_add,
47  op_add_async,
48  op_read,
49  op_read_all,
50  op_write,
51  op_read_sync,
52  op_write_sync,
53  op_write_all,
54  op_write_type_error,
55  op_shutdown,
56  op_format_file_name,
57  op_str_byte_length,
58  op_panic,
59  op_cancel_handle,
60  op_encode_binary_string,
61  op_is_terminal,
62  ops_builtin_types::op_is_any_array_buffer,
63  ops_builtin_types::op_is_arguments_object,
64  ops_builtin_types::op_is_array_buffer,
65  ops_builtin_types::op_is_array_buffer_view,
66  ops_builtin_types::op_is_async_function,
67  ops_builtin_types::op_is_big_int_object,
68  ops_builtin_types::op_is_boolean_object,
69  ops_builtin_types::op_is_boxed_primitive,
70  ops_builtin_types::op_is_data_view,
71  ops_builtin_types::op_is_date,
72  ops_builtin_types::op_is_generator_function,
73  ops_builtin_types::op_is_generator_object,
74  ops_builtin_types::op_is_map,
75  ops_builtin_types::op_is_map_iterator,
76  ops_builtin_types::op_is_module_namespace_object,
77  ops_builtin_types::op_is_native_error,
78  ops_builtin_types::op_is_number_object,
79  ops_builtin_types::op_is_promise,
80  ops_builtin_types::op_is_proxy,
81  ops_builtin_types::op_is_reg_exp,
82  ops_builtin_types::op_is_set,
83  ops_builtin_types::op_is_set_iterator,
84  ops_builtin_types::op_is_shared_array_buffer,
85  ops_builtin_types::op_is_string_object,
86  ops_builtin_types::op_is_symbol_object,
87  ops_builtin_types::op_is_typed_array,
88  ops_builtin_types::op_is_weak_map,
89  ops_builtin_types::op_is_weak_set,
90  ops_builtin_v8::op_add_main_module_handler,
91  ops_builtin_v8::op_set_handled_promise_rejection_handler,
92  ops_builtin_v8::op_timer_queue,
93  ops_builtin_v8::op_timer_queue_system,
94  ops_builtin_v8::op_timer_queue_immediate,
95  ops_builtin_v8::op_timer_cancel,
96  ops_builtin_v8::op_timer_ref,
97  ops_builtin_v8::op_timer_unref,
98  ops_builtin_v8::op_ref_op,
99  ops_builtin_v8::op_unref_op,
100  ops_builtin_v8::op_lazy_load_esm,
101  ops_builtin_v8::op_run_microtasks,
102  ops_builtin_v8::op_has_tick_scheduled,
103  ops_builtin_v8::op_set_has_tick_scheduled,
104  ops_builtin_v8::op_eval_context,
105  ops_builtin_v8::op_queue_microtask,
106  ops_builtin_v8::op_encode,
107  ops_builtin_v8::op_decode,
108  ops_builtin_v8::op_serialize,
109  ops_builtin_v8::op_deserialize,
110  ops_builtin_v8::op_set_promise_hooks,
111  ops_builtin_v8::op_get_promise_details,
112  ops_builtin_v8::op_get_proxy_details,
113  ops_builtin_v8::op_get_non_index_property_names,
114  ops_builtin_v8::op_get_constructor_name,
115  ops_builtin_v8::op_memory_usage,
116  ops_builtin_v8::op_set_wasm_streaming_callback,
117  ops_builtin_v8::op_abort_wasm_streaming,
118  ops_builtin_v8::op_destructure_error,
119  ops_builtin_v8::op_dispatch_exception,
120  ops_builtin_v8::op_op_names,
121  ops_builtin_v8::op_apply_source_map,
122  ops_builtin_v8::op_apply_source_map_filename,
123  ops_builtin_v8::op_current_user_call_site,
124  ops_builtin_v8::op_set_format_exception_callback,
125  ops_builtin_v8::op_event_loop_has_more_work,
126  ops_builtin_v8::op_arraybuffer_was_detached,
127  ops_builtin_v8::op_leak_tracing_enable,
128  ops_builtin_v8::op_leak_tracing_submit,
129  ops_builtin_v8::op_leak_tracing_get_all,
130  ops_builtin_v8::op_leak_tracing_get
131}
132
133#[op2(fast)]
134pub fn op_panic(#[string] message: String) {
135  eprintln!("JS PANIC: {}", message);
136  panic!("JS PANIC: {}", message);
137}
138
139/// Return map of resources with id as key
140/// and string representation as value.
141#[op2]
142#[serde]
143pub fn op_resources(state: &mut OpState) -> Vec<(ResourceId, String)> {
144  state
145    .resource_table
146    .names()
147    .map(|(rid, name)| (rid, name.to_string()))
148    .collect()
149}
150
151#[op2(fast)]
152fn op_add(a: i32, b: i32) -> i32 {
153  a + b
154}
155
156#[op2(async)]
157pub async fn op_add_async(a: i32, b: i32) -> i32 {
158  a + b
159}
160
161#[op2(fast)]
162pub fn op_void_sync() {}
163
164#[op2(async)]
165pub async fn op_void_async() {}
166
167#[op2(async)]
168pub async fn op_error_async() -> Result<(), Error> {
169  Err(Error::msg("error"))
170}
171
172#[op2(async(deferred), fast)]
173pub async fn op_error_async_deferred() -> Result<(), Error> {
174  Err(Error::msg("error"))
175}
176
177#[op2(async(deferred), fast)]
178pub async fn op_void_async_deferred() {}
179
180/// Remove a resource from the resource table.
181#[op2(fast)]
182pub fn op_close(
183  state: Rc<RefCell<OpState>>,
184  #[smi] rid: ResourceId,
185) -> Result<(), Error> {
186  let resource = state.borrow_mut().resource_table.take_any(rid)?;
187  resource.close();
188  Ok(())
189}
190
191/// Try to remove a resource from the resource table. If there is no resource
192/// with the specified `rid`, this is a no-op.
193#[op2(fast)]
194pub fn op_try_close(state: Rc<RefCell<OpState>>, #[smi] rid: ResourceId) {
195  if let Ok(resource) = state.borrow_mut().resource_table.take_any(rid) {
196    resource.close();
197  }
198}
199
200/// Builtin utility to print to stdout/stderr
201#[op2(fast)]
202pub fn op_print(#[string] msg: &str, is_err: bool) -> Result<(), Error> {
203  if is_err {
204    stderr().write_all(msg.as_bytes())?;
205    stderr().flush().unwrap();
206  } else {
207    stdout().write_all(msg.as_bytes())?;
208    stdout().flush().unwrap();
209  }
210  Ok(())
211}
212
213pub struct WasmStreamingResource(pub(crate) RefCell<v8::WasmStreaming>);
214
215impl Resource for WasmStreamingResource {
216  fn close(self: Rc<Self>) {
217    // At this point there are no clones of Rc<WasmStreamingResource> on the
218    // resource table, and no one should own a reference outside of the stack.
219    // Therefore, we can be sure `self` is the only reference.
220    if let Ok(wsr) = Rc::try_unwrap(self) {
221      wsr.0.into_inner().finish();
222    } else {
223      panic!("Couldn't consume WasmStreamingResource.");
224    }
225  }
226}
227
228/// Feed bytes to WasmStreamingResource.
229#[op2(fast)]
230pub fn op_wasm_streaming_feed(
231  state: Rc<RefCell<OpState>>,
232  #[smi] rid: ResourceId,
233  #[buffer] bytes: &[u8],
234) -> Result<(), Error> {
235  let wasm_streaming = state
236    .borrow_mut()
237    .resource_table
238    .get::<WasmStreamingResource>(rid)?;
239
240  wasm_streaming.0.borrow_mut().on_bytes_received(bytes);
241
242  Ok(())
243}
244
245#[op2(fast)]
246pub fn op_wasm_streaming_set_url(
247  state: &mut OpState,
248  #[smi] rid: ResourceId,
249  #[string] url: &str,
250) -> Result<(), Error> {
251  let wasm_streaming =
252    state.resource_table.get::<WasmStreamingResource>(rid)?;
253
254  wasm_streaming.0.borrow_mut().set_url(url);
255
256  Ok(())
257}
258
259#[op2(async)]
260async fn op_read(
261  state: Rc<RefCell<OpState>>,
262  #[smi] rid: ResourceId,
263  #[buffer] buf: JsBuffer,
264) -> Result<u32, Error> {
265  let resource = state.borrow().resource_table.get_any(rid)?;
266  let view = BufMutView::from(buf);
267  resource.read_byob(view).await.map(|(n, _)| n as u32)
268}
269
270#[op2(async)]
271#[buffer]
272async fn op_read_all(
273  state: Rc<RefCell<OpState>>,
274  #[smi] rid: ResourceId,
275) -> Result<BytesMut, Error> {
276  let resource = state.borrow().resource_table.get_any(rid)?;
277
278  let (min, maybe_max) = resource.size_hint();
279  let mut buffer_strategy =
280    AdaptiveBufferStrategy::new_from_hint_u64(min, maybe_max);
281  let mut buf = BufMutView::new(buffer_strategy.buffer_size());
282
283  loop {
284    #[allow(deprecated)]
285    buf.maybe_grow(buffer_strategy.buffer_size()).unwrap();
286
287    let (n, new_buf) = resource.clone().read_byob(buf).await?;
288    buf = new_buf;
289    buf.advance_cursor(n);
290    if n == 0 {
291      break;
292    }
293
294    buffer_strategy.notify_read(n);
295  }
296
297  let nread = buf.reset_cursor();
298  // If the buffer is larger than the amount of data read, shrink it to the
299  // amount of data read.
300  buf.truncate(nread);
301
302  Ok(buf.maybe_unwrap_bytes().unwrap())
303}
304
305#[op2(async)]
306async fn op_write(
307  state: Rc<RefCell<OpState>>,
308  #[smi] rid: ResourceId,
309  #[buffer] buf: JsBuffer,
310) -> Result<u32, Error> {
311  let resource = state.borrow().resource_table.get_any(rid)?;
312  let view = BufView::from(buf);
313  let resp = resource.write(view).await?;
314  Ok(resp.nwritten() as u32)
315}
316
317#[op2(fast)]
318fn op_read_sync(
319  state: Rc<RefCell<OpState>>,
320  #[smi] rid: ResourceId,
321  #[buffer] data: &mut [u8],
322) -> Result<u32, Error> {
323  let resource = state.borrow_mut().resource_table.get_any(rid)?;
324  resource.read_byob_sync(data).map(|n| n as u32)
325}
326
327#[op2(fast)]
328fn op_write_sync(
329  state: Rc<RefCell<OpState>>,
330  #[smi] rid: ResourceId,
331  #[buffer] data: &[u8],
332) -> Result<u32, Error> {
333  let resource = state.borrow_mut().resource_table.get_any(rid)?;
334  let nwritten = resource.write_sync(data)?;
335  Ok(nwritten as u32)
336}
337
338#[op2(async)]
339async fn op_write_all(
340  state: Rc<RefCell<OpState>>,
341  #[smi] rid: ResourceId,
342  #[buffer] buf: JsBuffer,
343) -> Result<(), Error> {
344  let resource = state.borrow().resource_table.get_any(rid)?;
345  let view = BufView::from(buf);
346  resource.write_all(view).await?;
347  Ok(())
348}
349
350#[op2(async)]
351async fn op_write_type_error(
352  state: Rc<RefCell<OpState>>,
353  #[smi] rid: ResourceId,
354  #[string] error: String,
355) -> Result<(), Error> {
356  let resource = state.borrow().resource_table.get_any(rid)?;
357  resource.write_error(type_error(error)).await?;
358  Ok(())
359}
360
361#[op2(async)]
362async fn op_shutdown(
363  state: Rc<RefCell<OpState>>,
364  #[smi] rid: ResourceId,
365) -> Result<(), Error> {
366  let resource = state.borrow().resource_table.get_any(rid)?;
367  resource.shutdown().await
368}
369
370#[op2]
371#[string]
372fn op_format_file_name(#[string] file_name: &str) -> String {
373  format_file_name(file_name)
374}
375
376#[op2]
377fn op_str_byte_length(
378  scope: &mut v8::HandleScope,
379  value: v8::Local<v8::Value>,
380) -> u32 {
381  if let Ok(string) = v8::Local::<v8::String>::try_from(value) {
382    string.utf8_length(scope) as u32
383  } else {
384    0
385  }
386}
387
388/// Creates a [`CancelHandle`] resource that can be used to cancel invocations of certain ops.
389#[op2(fast)]
390#[smi]
391pub fn op_cancel_handle(state: &mut OpState) -> u32 {
392  state.resource_table.add(CancelHandle::new())
393}
394
395#[op2]
396#[serde]
397fn op_encode_binary_string(#[buffer] s: &[u8]) -> ByteString {
398  ByteString::from(s)
399}
400
401#[op2(fast)]
402fn op_is_terminal(
403  state: &mut OpState,
404  #[smi] rid: ResourceId,
405) -> Result<bool, Error> {
406  let handle = state.resource_table.get_handle(rid)?;
407  Ok(handle.is_terminal())
408}