1 /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
2 * vim: set ts=8 sts=4 et sw=4 tw=99:
3 * This Source Code Form is subject to the terms of the Mozilla Public
4 * License, v. 2.0. If a copy of the MPL was not distributed with this
5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
7 #ifndef js_StructuredClone_h
8 #define js_StructuredClone_h
14 #include "js/RootingAPI.h"
15 #include "js/TypeDecls.h"
19 struct JSStructuredCloneReader
;
20 struct JSStructuredCloneWriter
;
22 // API for the HTML5 internal structured cloning algorithm.
25 enum TransferableOwnership
{
26 // Transferable data has not been filled in yet
27 SCTAG_TMO_UNFILLED
= 0,
29 // Structured clone buffer does not yet own the data
30 SCTAG_TMO_UNOWNED
= 1,
32 // All values at least this large are owned by the clone buffer
33 SCTAG_TMO_FIRST_OWNED
= 2,
35 // Data is a pointer that can be freed
36 SCTAG_TMO_ALLOC_DATA
= 2,
38 // Data is a SharedArrayBufferObject's buffer
39 SCTAG_TMO_SHARED_BUFFER
= 3,
41 // Data is a memory mapped pointer
42 SCTAG_TMO_MAPPED_DATA
= 4,
44 // Data is embedding-specific. The engine can free it by calling the
45 // freeTransfer op. The embedding can also use SCTAG_TMO_USER_MIN and
46 // greater, up to 32 bits, to distinguish specific ownership variants.
53 // Read structured data from the reader r. This hook is used to read a value
54 // previously serialized by a call to the WriteStructuredCloneOp hook.
56 // tag and data are the pair of uint32_t values from the header. The callback
57 // may use the JS_Read* APIs to read any other relevant parts of the object
58 // from the reader r. closure is any value passed to the JS_ReadStructuredClone
59 // function. Return the new object on success, nullptr on error/exception.
60 typedef JSObject
* (*ReadStructuredCloneOp
)(JSContext
* cx
, JSStructuredCloneReader
* r
,
61 uint32_t tag
, uint32_t data
, void* closure
);
63 // Structured data serialization hook. The engine can write primitive values,
64 // Objects, Arrays, Dates, RegExps, TypedArrays, ArrayBuffers, Sets, Maps,
65 // and SharedTypedArrays. Any other type of object requires application support.
66 // This callback must first use the JS_WriteUint32Pair API to write an object
67 // header, passing a value greater than JS_SCTAG_USER to the tag parameter.
68 // Then it can use the JS_Write* APIs to write any other relevant parts of
69 // the value v to the writer w. closure is any value passed to the
70 // JS_WriteStructuredClone function.
72 // Return true on success, false on error/exception.
73 typedef bool (*WriteStructuredCloneOp
)(JSContext
* cx
, JSStructuredCloneWriter
* w
,
74 JS::HandleObject obj
, void* closure
);
76 // This is called when JS_WriteStructuredClone is given an invalid transferable.
77 // To follow HTML5, the application must throw a DATA_CLONE_ERR DOMException
78 // with error set to one of the JS_SCERR_* values.
79 typedef void (*StructuredCloneErrorOp
)(JSContext
* cx
, uint32_t errorid
);
81 // This is called when JS_ReadStructuredClone receives a transferable object
82 // not known to the engine. If this hook does not exist or returns false, the
83 // JS engine calls the reportError op if set, otherwise it throws a
84 // DATA_CLONE_ERR DOM Exception. This method is called before any other
85 // callback and must return a non-null object in returnObject on success.
86 typedef bool (*ReadTransferStructuredCloneOp
)(JSContext
* cx
, JSStructuredCloneReader
* r
,
87 uint32_t tag
, void* content
, uint64_t extraData
,
89 JS::MutableHandleObject returnObject
);
91 // Called when JS_WriteStructuredClone receives a transferable object not
92 // handled by the engine. If this hook does not exist or returns false, the JS
93 // engine will call the reportError hook or fall back to throwing a
94 // DATA_CLONE_ERR DOM Exception. This method is called before any other
97 // tag: indicates what type of transferable this is. Must be greater than
98 // 0xFFFF0201 (value of the internal SCTAG_TRANSFER_MAP_PENDING_ENTRY)
100 // ownership: see TransferableOwnership, above. Used to communicate any needed
101 // ownership info to the FreeTransferStructuredCloneOp.
103 // content, extraData: what the ReadTransferStructuredCloneOp will receive
105 typedef bool (*TransferStructuredCloneOp
)(JSContext
* cx
,
106 JS::Handle
<JSObject
*> obj
,
110 JS::TransferableOwnership
* ownership
,
112 uint64_t* extraData
);
114 // Called when JS_ClearStructuredClone has to free an unknown transferable
115 // object. Note that it should never trigger a garbage collection (and will
116 // assert in a debug build if it does.)
117 typedef void (*FreeTransferStructuredCloneOp
)(uint32_t tag
, JS::TransferableOwnership ownership
,
118 void* content
, uint64_t extraData
, void* closure
);
120 // The maximum supported structured-clone serialization format version.
121 // Increment this when anything at all changes in the serialization format.
122 // (Note that this does not need to be bumped for Transferable-only changes,
123 // since they are never saved to persistent storage.)
124 #define JS_STRUCTURED_CLONE_VERSION 5
126 struct JSStructuredCloneCallbacks
{
127 ReadStructuredCloneOp read
;
128 WriteStructuredCloneOp write
;
129 StructuredCloneErrorOp reportError
;
130 ReadTransferStructuredCloneOp readTransfer
;
131 TransferStructuredCloneOp writeTransfer
;
132 FreeTransferStructuredCloneOp freeTransfer
;
135 // Note: if the *data contains transferable objects, it can be read only once.
137 JS_ReadStructuredClone(JSContext
* cx
, uint64_t* data
, size_t nbytes
, uint32_t version
,
138 JS::MutableHandleValue vp
,
139 const JSStructuredCloneCallbacks
* optionalCallbacks
, void* closure
);
141 // Note: On success, the caller is responsible for calling
142 // JS_ClearStructuredClone(*datap, nbytes, optionalCallbacks, closure).
144 JS_WriteStructuredClone(JSContext
* cx
, JS::HandleValue v
, uint64_t** datap
, size_t* nbytesp
,
145 const JSStructuredCloneCallbacks
* optionalCallbacks
,
146 void* closure
, JS::HandleValue transferable
);
149 JS_ClearStructuredClone(uint64_t* data
, size_t nbytes
,
150 const JSStructuredCloneCallbacks
* optionalCallbacks
,
154 JS_StructuredCloneHasTransferables(const uint64_t* data
, size_t nbytes
, bool* hasTransferable
);
157 JS_StructuredClone(JSContext
* cx
, JS::HandleValue v
, JS::MutableHandleValue vp
,
158 const JSStructuredCloneCallbacks
* optionalCallbacks
, void* closure
);
160 // RAII sugar for JS_WriteStructuredClone.
161 class JS_PUBLIC_API(JSAutoStructuredCloneBuffer
) {
165 const JSStructuredCloneCallbacks
* callbacks_
;
169 JSAutoStructuredCloneBuffer()
170 : data_(nullptr), nbytes_(0), version_(JS_STRUCTURED_CLONE_VERSION
),
171 callbacks_(nullptr), closure_(nullptr)
174 JSAutoStructuredCloneBuffer(const JSStructuredCloneCallbacks
* callbacks
, void* closure
)
175 : data_(nullptr), nbytes_(0), version_(JS_STRUCTURED_CLONE_VERSION
),
176 callbacks_(callbacks
), closure_(closure
)
179 JSAutoStructuredCloneBuffer(JSAutoStructuredCloneBuffer
&& other
);
180 JSAutoStructuredCloneBuffer
& operator=(JSAutoStructuredCloneBuffer
&& other
);
182 ~JSAutoStructuredCloneBuffer() { clear(); }
184 uint64_t* data() const { return data_
; }
185 size_t nbytes() const { return nbytes_
; }
189 // Copy some memory. It will be automatically freed by the destructor.
190 bool copy(const uint64_t* data
, size_t nbytes
, uint32_t version
=JS_STRUCTURED_CLONE_VERSION
);
192 // Adopt some memory. It will be automatically freed by the destructor.
193 // data must have been allocated by the JS engine (e.g., extracted via
194 // JSAutoStructuredCloneBuffer::steal).
195 void adopt(uint64_t* data
, size_t nbytes
, uint32_t version
=JS_STRUCTURED_CLONE_VERSION
);
197 // Remove the buffer so that it will not be automatically freed.
198 // After this, the caller is responsible for feeding the memory back to
199 // JSAutoStructuredCloneBuffer::adopt.
200 void steal(uint64_t** datap
, size_t* nbytesp
, uint32_t* versionp
=nullptr);
202 bool read(JSContext
* cx
, JS::MutableHandleValue vp
,
203 const JSStructuredCloneCallbacks
* optionalCallbacks
=nullptr, void* closure
=nullptr);
205 bool write(JSContext
* cx
, JS::HandleValue v
,
206 const JSStructuredCloneCallbacks
* optionalCallbacks
=nullptr, void* closure
=nullptr);
208 bool write(JSContext
* cx
, JS::HandleValue v
, JS::HandleValue transferable
,
209 const JSStructuredCloneCallbacks
* optionalCallbacks
=nullptr, void* closure
=nullptr);
212 // Copy and assignment are not supported.
213 JSAutoStructuredCloneBuffer(const JSAutoStructuredCloneBuffer
& other
) = delete;
214 JSAutoStructuredCloneBuffer
& operator=(const JSAutoStructuredCloneBuffer
& other
) = delete;
217 // The range of tag values the application may use for its own custom object types.
218 #define JS_SCTAG_USER_MIN ((uint32_t) 0xFFFF8000)
219 #define JS_SCTAG_USER_MAX ((uint32_t) 0xFFFFFFFF)
221 #define JS_SCERR_RECURSION 0
222 #define JS_SCERR_TRANSFERABLE 1
225 JS_SetStructuredCloneCallbacks(JSRuntime
* rt
, const JSStructuredCloneCallbacks
* callbacks
);
228 JS_ReadUint32Pair(JSStructuredCloneReader
* r
, uint32_t* p1
, uint32_t* p2
);
231 JS_ReadBytes(JSStructuredCloneReader
* r
, void* p
, size_t len
);
234 JS_ReadTypedArray(JSStructuredCloneReader
* r
, JS::MutableHandleValue vp
);
237 JS_WriteUint32Pair(JSStructuredCloneWriter
* w
, uint32_t tag
, uint32_t data
);
240 JS_WriteBytes(JSStructuredCloneWriter
* w
, const void* p
, size_t len
);
243 JS_WriteTypedArray(JSStructuredCloneWriter
* w
, JS::HandleValue v
);
245 #endif /* js_StructuredClone_h */