Skip to content

Commit de53adb

Browse files
Gabriel "_|Nix|_" Schulhofzherczeg
Gabriel "_|Nix|_" Schulhof
authored andcommitted
Lazily create a linked list of context items (jerryscript-project#1833)
This approach has the benefit that it does not require any *a priori* initialization, and that each context pointer is identified by the way in which it was created. Additionally, retrieving the context pointer now requires that the entity responsible for creating/destroying it (the manager) be given. Since managers are stored in global static const structures, they should not normally be visible across source files, and thus there should be no danger that a context item will be retrieved by the wrong manager and thus cast into the wrong data type. Since the items are stored in a linked list, their number will be limited to exactly as many as are needed for a given context, with the caveat that storing too many on a context will cause slow retrieval. Thanks @mhdawson for the idea! Fixes jerryscript-project#1845 JerryScript-DCO-1.0-Signed-off-by: Gabriel Schulhof [email protected]
1 parent 29f57ec commit de53adb

File tree

6 files changed

+233
-185
lines changed

6 files changed

+233
-185
lines changed

docs/02.API-REFERENCE.md

+76-94
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,25 @@ created by API functions has the error flag set.
101101
typedef uint32_t jerry_value_t;
102102
```
103103

104+
## jerry_context_data_manager_t
105+
106+
**Summary**
107+
108+
Structure that defines how a context data item will be initialized and deinitialized. JerryScript zeroes out the memory
109+
for the item by default, and if the `init_cb` field is not NULL, it will be called with the pointer to the memory as
110+
an additional custom initializer.
111+
112+
**Prototype**
113+
114+
```c
115+
typedef struct
116+
{
117+
void (*init_cb) (void *); /**< callback responsible for initializing a context item, or NULL */
118+
void (*deinit_cb) (void *); /**< callback responsible for deinitializing a context item */
119+
size_t bytes_needed; /**< number of bytes to allocate for this manager */
120+
} jerry_context_data_manager_t;
121+
```
122+
104123
## jerry_property_descriptor_t
105124

106125
**Summary**
@@ -264,9 +283,7 @@ typedef jerry_value_t (*jerry_vm_exec_stop_callback_t) (void *user_p);
264283
**Summary**
265284

266285
Initializes the JerryScript engine, making it possible to run JavaScript code and perform operations
267-
on JavaScript values. See also [jerry_init_with_user_context](#jerry_init_with_user_context) if you
268-
wish to initialize the JerryScript engine in such a way that its context contains a custom pointer
269-
which you can later retrieve using [jerry_get_user_context](#jerry_get_user_context).
286+
on JavaScript values.
270287

271288
**Prototype**
272289

@@ -299,91 +316,13 @@ jerry_init (jerry_init_flag_t flags)
299316
**See also**
300317

301318
- [jerry_cleanup](#jerry_cleanup)
302-
- [jerry_init_with_user_context](#jerry_init_with_user_context)
303-
304-
305-
## jerry_init_with_user_context
306-
307-
**Summary**
308-
309-
Calls [jerry_init](#jerry_init) to initialize the JerryScript engine, thereby making it possible
310-
to run JavaScript code and perform operations on JavaScript values. In addition to the first
311-
parameter this function accepts two more parameters with which it allows the caller to store a
312-
`void *` pointer inside the context being initialized with `jerry_init ()`. The function calls the
313-
callback given in its `init_cb` parameter to allocate the memory for the pointer and it stores the
314-
function pointer given in the `deinit_cb` parameter along with the pointer so that it may be called
315-
to free the stored pointer when `jerry_cleanup ()` is later called to dispose of the context.
316-
317-
**Prototype**
318-
319-
```c
320-
void
321-
jerry_init_with_user_context (jerry_init_flag_t flags,
322-
jerry_user_context_init_cb init_cb,
323-
jerry_user_context_deinit_cb deinit_cb);
324-
```
325-
326-
`flags` - combination of various engine configuration flags:
327-
328-
- `JERRY_INIT_EMPTY` - no flags, just initialize in default configuration.
329-
- `JERRY_INIT_SHOW_OPCODES` - print compiled byte-code.
330-
- `JERRY_INIT_SHOW_REGEXP_OPCODES` - print compiled regexp byte-code.
331-
- `JERRY_INIT_MEM_STATS` - dump memory statistics.
332-
- `JERRY_INIT_MEM_STATS_SEPARATE` - dump memory statistics and reset peak values after parse.
333-
- `JERRY_INIT_DEBUGGER` - enable all features required by debugging.
334-
335-
`init_cb` - a function pointer that will be called to allocate the custom pointer.
336-
337-
`deinit_cb` - a function pointer that will be called when the custom pointer must be freed.
338-
339-
**Example**
340-
341-
```c
342-
void *
343-
init_user_context (void)
344-
{
345-
void *return_value;
346-
347-
/* allocate and initialize return_value */
348-
349-
return return_value;
350-
} /* init_user_context */
351-
352-
void
353-
free_user_context (void *context)
354-
{
355-
356-
/* free the value allocated above */
357-
358-
} /* free_user_context */
359-
360-
{
361-
/* init_user_context () will be called before the call below returns */
362-
jerry_init_with_user_context (JERRY_INIT_SHOW_OPCODES | JERRY_INIT_SHOW_REGEXP_OPCODES,
363-
init_user_context,
364-
free_user_context);
365-
366-
/* ... */
367-
368-
/* free_user_context () will be called before the call below returns */
369-
jerry_cleanup ();
370-
}
371-
```
372-
373-
**See also**
374-
375-
- [jerry_cleanup](#jerry_cleanup)
376-
- [jerry_get_user_context](#jerry_get_user_context)
377319

378320

379321
## jerry_cleanup
380322

381323
**Summary**
382324

383-
Finish JavaScript engine execution, freeing memory and JavaScript values. If the context was
384-
initialized with `jerry_init_with_user_context ()` and a `deinit_cb` was provided, then it will
385-
be called to free the memory at the custom pointer which was associated with the context being
386-
cleaned up.
325+
Finish JavaScript engine execution, freeing memory and JavaScript values.
387326

388327
*Note*: JavaScript values, received from engine, will be inaccessible after the cleanup.
389328

@@ -397,37 +336,80 @@ jerry_cleanup (void);
397336
**See also**
398337

399338
- [jerry_init](#jerry_init)
400-
- [jerry_init_with_user_context](#jerry_init_with_user_context)
401339

402340

403-
## jerry_get_user_context
341+
## jerry_get_context_data
404342

405343
**Summary**
406344

407-
Retrieve the pointer stored within the current context.
345+
Retrieve a pointer to the item stored within the current context by the given manager.
346+
347+
*Note*: Since internally the pointer to a manager's context data item is linked to the next such pointer in a linked
348+
list, it is inadvisable to invoke too many different managers, because doing so will increase the time it takes
349+
to retrieve a manager's context data item, degrading performance. For example, try to keep the number of
350+
managers below five.
408351

409352
**Prototype**
410353

411354
```c
412355
void *
413-
jerry_get_user_context (void);
356+
jerry_get_context_data (const jerry_context_data_manager *manager_p);
414357
```
415358

416-
- return value: the pointer that was assigned during `jerry_init_with_user_context ()`
359+
- `manager_p`: the manager of this context data item.
360+
- return value: the item created by `manager_p` when `jerry_get_context_data ()` was first called, or a new item created
361+
by `manager_p`, which will be stored for future identical calls to `jerry_get_context_data ()`, and which will be
362+
deinitialized using the `deinit_cb` callback provided by `manager_p` when the context will be destroyed.
417363

418364
**Example**
419365

420366
```c
367+
typedef struct
421368
{
422-
/* ... */
423-
my_context *custom_data = (my_context *) jerry_get_user_context ();
424-
/* ... */
369+
int my_data1;
370+
double my_data2;
371+
char *my_data3;
372+
} my_context_data_t;
373+
374+
/* Define how context items will be initialized. */
375+
static void
376+
my_context_data_new (void *user_data_p)
377+
{
378+
my_context_data_t *my_data_p = (my_context_data_t *) user_data_p;
379+
380+
/*
381+
* Initialize my_data_p. JerryScript will store it on the current context and return it whenever
382+
* jerry_get_context_data () is called with a pointer to my_manager as defined below.
383+
*/
425384
}
426-
```
427385

428-
**See also**
429-
- [jerry_init_with_user_context](#jerry_init_with_user_context)
430-
- [jerry_cleanup](#jerry_cleanup)
386+
/* Define how context items will be deinitialized */
387+
static void
388+
my_context_data_free (void *user_data_p)
389+
{
390+
my_context_data_t *my_data_p = ((my_context_data_t *) user_data_p);
391+
392+
/* Perform any necessary cleanup on my_data. JerryScript will free the pointer after this function completes. */
393+
}
394+
395+
/* Wrap the creation and destruction functions into a manager */
396+
static const jerry_context_data_manager_t my_manager =
397+
{
398+
.init_cb = my_context_data_new,
399+
.deinit_cb = my_context_data_free,
400+
.bytes_needed = sizeof (my_context_data_t)
401+
};
402+
403+
/*
404+
* Then, in some function in your code, you can retrieve an item of type my_context_data_t from the currently active
405+
* context such that JerryScript will create and store such an item if one was not previously created
406+
*/
407+
void someplace_in_the_code (void)
408+
{
409+
my_context_data_t *my_data = (my_context_data_t *) jerry_get_context_data (&my_manager);
410+
/* Perform useful things using the data found in my_data */
411+
}
412+
```
431413

432414

433415
## jerry_register_magic_strings

jerry-core/api/jerry.c

+40-26
Original file line numberDiff line numberDiff line change
@@ -166,22 +166,6 @@ jerry_init (jerry_init_flag_t flags) /**< combination of Jerry flags */
166166
#endif /* JERRY_DEBUGGER */
167167
} /* jerry_init */
168168

169-
/**
170-
* Initialize Jerry engine with custom user context.
171-
*/
172-
void
173-
jerry_init_with_user_context (jerry_init_flag_t flags, /**< combination of Jerry flags */
174-
jerry_user_context_init_t init_cb, /**< callback to call to create the user context or
175-
* NULL, in which case no user context will be
176-
* created */
177-
jerry_user_context_deinit_t deinit_cb) /**< callback to call to free the user context or
178-
* NULL if it does not need to be freed */
179-
{
180-
jerry_init (flags);
181-
JERRY_CONTEXT (user_context_p) = (init_cb ? init_cb () : NULL);
182-
JERRY_CONTEXT (user_context_deinit_cb) = deinit_cb;
183-
} /* jerry_init_with_user_context */
184-
185169
/**
186170
* Terminate Jerry engine
187171
*/
@@ -190,6 +174,15 @@ jerry_cleanup (void)
190174
{
191175
jerry_assert_api_available ();
192176

177+
for (jerry_context_data_header_t *this_p = JERRY_CONTEXT (context_data_p), *next_p = NULL;
178+
this_p != NULL;
179+
this_p = next_p)
180+
{
181+
next_p = this_p->next_p;
182+
this_p->manager_p->deinit_cb (JERRY_CONTEXT_DATA_HEADER_USER_DATA (this_p));
183+
jmem_heap_free_block (this_p, sizeof (jerry_context_data_header_t) + this_p->manager_p->bytes_needed);
184+
}
185+
193186
ecma_finalize ();
194187

195188
#ifdef JERRY_DEBUGGER
@@ -201,23 +194,44 @@ jerry_cleanup (void)
201194

202195
jmem_finalize ();
203196
jerry_make_api_unavailable ();
204-
205-
if (JERRY_CONTEXT (user_context_deinit_cb))
206-
{
207-
JERRY_CONTEXT (user_context_deinit_cb) (JERRY_CONTEXT (user_context_p));
208-
}
209197
} /* jerry_cleanup */
210198

211199
/**
212-
* Retrieve user context.
200+
* Retrieve a context data item, or create a new one.
213201
*
214-
* @return the user-provided context-specific pointer
202+
* @param manager_p pointer to the manager whose context data item should be returned.
203+
*
204+
* @return a pointer to the user-provided context-specific data item for the given manager, creating such a pointer if
205+
* none was found.
215206
*/
216207
void *
217-
jerry_get_user_context (void)
208+
jerry_get_context_data (const jerry_context_data_manager_t *manager_p)
218209
{
219-
return JERRY_CONTEXT (user_context_p);
220-
} /* jerry_get_user_context */
210+
void *ret = NULL;
211+
jerry_context_data_header_t *item_p;
212+
213+
for (item_p = JERRY_CONTEXT (context_data_p); item_p != NULL; item_p = item_p->next_p)
214+
{
215+
if (item_p->manager_p == manager_p)
216+
{
217+
return JERRY_CONTEXT_DATA_HEADER_USER_DATA (item_p);
218+
}
219+
}
220+
221+
item_p = jmem_heap_alloc_block (sizeof (jerry_context_data_header_t) + manager_p->bytes_needed);
222+
item_p->manager_p = manager_p;
223+
item_p->next_p = JERRY_CONTEXT (context_data_p);
224+
JERRY_CONTEXT (context_data_p) = item_p;
225+
ret = JERRY_CONTEXT_DATA_HEADER_USER_DATA (item_p);
226+
227+
memset (ret, 0, manager_p->bytes_needed);
228+
if (manager_p->init_cb)
229+
{
230+
manager_p->init_cb (ret);
231+
}
232+
233+
return ret;
234+
} /* jerry_get_context_data */
221235

222236
/**
223237
* Register external magic string array

jerry-core/include/jerryscript-core.h

+8-10
Original file line numberDiff line numberDiff line change
@@ -195,15 +195,16 @@ typedef jerry_value_t (*jerry_vm_exec_stop_callback_t) (void *user_p);
195195
typedef bool (*jerry_object_property_foreach_t) (const jerry_value_t property_name,
196196
const jerry_value_t property_value,
197197
void *user_data_p);
198-
/**
199-
* Function type for user context allocation
200-
*/
201-
typedef void *(*jerry_user_context_init_t) (void);
202198

203199
/**
204-
* Function type for user context deallocation
200+
* User context item manager
205201
*/
206-
typedef void (*jerry_user_context_deinit_t) (void *user_context_p);
202+
typedef struct
203+
{
204+
void (*init_cb) (void *); /**< callback responsible for initializing a context item, or NULL to zero out the memory */
205+
void (*deinit_cb) (void *); /**< callback responsible for deinitializing a context item */
206+
size_t bytes_needed; /**< number of bytes to allocate for this manager */
207+
} jerry_context_data_manager_t;
207208

208209
/**
209210
* Function type for allocating buffer for JerryScript instance.
@@ -227,15 +228,12 @@ typedef struct jerry_instance_t jerry_instance_t;
227228
* General engine functions.
228229
*/
229230
void jerry_init (jerry_init_flag_t flags);
230-
void jerry_init_with_user_context (jerry_init_flag_t flags,
231-
jerry_user_context_init_t init_cb,
232-
jerry_user_context_deinit_t deinit_cb);
233231
void jerry_cleanup (void);
234232
void jerry_register_magic_strings (const jerry_char_ptr_t *ex_str_items_p, uint32_t count,
235233
const jerry_length_t *str_lengths_p);
236234
void jerry_get_memory_limits (size_t *out_data_bss_brk_limit_p, size_t *out_stack_limit_p);
237235
void jerry_gc (void);
238-
void *jerry_get_user_context (void);
236+
void *jerry_get_context_data (const jerry_context_data_manager_t *manager_p);
239237

240238
/**
241239
* Parser and executor functions.

jerry-core/jcontext/jcontext.h

+14-2
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@
2626
#include "jmem.h"
2727
#include "re-bytecode.h"
2828
#include "vm-defines.h"
29+
#include "jerryscript.h"
2930

3031
/** \addtogroup context Context
3132
* @{
@@ -36,6 +37,18 @@
3637
*/
3738
#define JERRY_CONTEXT_FIRST_MEMBER ecma_builtin_objects
3839

40+
/**
41+
* User context item
42+
*/
43+
typedef struct jerry_context_data_header
44+
{
45+
struct jerry_context_data_header *next_p; /**< pointer to next context item */
46+
const jerry_context_data_manager_t *manager_p; /**< manager responsible for deleting this item */
47+
} jerry_context_data_header_t;
48+
49+
#define JERRY_CONTEXT_DATA_HEADER_USER_DATA(item_p) \
50+
((uint8_t *) (item_p + 1))
51+
3952
/**
4053
* JerryScript context
4154
*
@@ -63,8 +76,7 @@ typedef struct
6376
ecma_lit_storage_item_t *number_list_first_p; /**< first item of the literal number list */
6477
ecma_object_t *ecma_global_lex_env_p; /**< global lexical environment */
6578
vm_frame_ctx_t *vm_top_context_p; /**< top (current) interpreter context */
66-
void *user_context_p; /**< user-provided context-specific pointer */
67-
ecma_user_context_deinit_t user_context_deinit_cb; /**< user-provided deleter for context-specific pointer */
79+
jerry_context_data_header_t *context_data_p; /**< linked list of user-provided context-specific pointers */
6880
size_t ecma_gc_objects_number; /**< number of currently allocated objects */
6981
size_t ecma_gc_new_objects; /**< number of newly allocated objects since last GC session */
7082
size_t jmem_heap_allocated_size; /**< size of allocated regions */

0 commit comments

Comments
 (0)