-
Notifications
You must be signed in to change notification settings - Fork 85
Make extension callbacks part of the public MPS #213
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Merged
Merged
Changes from all commits
Commits
Show all changes
22 commits
Select commit
Hold shift + click to select a range
9919050
Part of making extension callbacks part of the public MPS.
rptb1 fd040ed
Add test for arena extend and contract callbacks
thejayps d252cbd
Add extcon.c to testci target so it runs during CI
thejayps b4d5485
Fix test to ensure that cold pointer exists in colder stack frame.
thejayps 7e45b2e
Adding extcon extension/contraction test to Posix builds. Fixing war…
rptb1 32e21e3
Detabifying extcon.c.
rptb1 24e4c86
Adding an assertion that forces gcc not to reorder locals, working ar…
rptb1 627ded3
Adding an attribute to test_main to prevent clang 14 from inlining it…
rptb1 921a639
Removing unnecessary arena_park.
rptb1 3f7be39
Adding detailed output to extcon about memory reservation and GC cycl…
rptb1 a4190fa
add output to aid cold end of stack debugging
thejayps f737421
Moving the root of objects into a static to avoid problems with the c…
rptb1 874cb1b
Disabling the Insist comparing the testobj array to the cold end of t…
rptb1 b4e5256
Disable extcon test except on Windows to workaround our inability to …
rptb1 90d54f1
remove homebrew format from extcon testbench and replace with dylan f…
thejayps 612c1a3
add dylan test object to comm.gmk for extcon, missing from last commit
thejayps d9087b1
consistently use die() in extcon.c
thejayps 74d77c8
executing proc.review.edit
thejayps b8ac0d1
correction to last commit
thejayps fb93d86
Adding arena contraction callback to all chunk deallocations, so that…
rptb1 2833d45
Improving comments in response to review <https://github.com/Ravenbro…
rptb1 a20116a
Clarifying comments in response to review <https://github.com/Ravenbr…
rptb1 File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,306 @@ | ||
| /* extcon.c: ARENA EXTENDED AND CONTRACTED CALLBACK TEST | ||
| * | ||
| * $Id$ | ||
| * Copyright (c) 2022-2023 Ravenbrook Limited. See end of file for license. | ||
| * | ||
| * .overview: This test case allocates a bunch of large objects, of a size | ||
| * similar to the size of the arena, to force the arena to extend. It then | ||
| * discards the base pointers to those objects, and forces a collection. | ||
| * | ||
| * .limitations: This test checks that the EXTENDED and CONTRACTED | ||
| * callbacks were called at least once, and that they are called the | ||
| * same number of times. It does not check that the extensions and | ||
| * contractions themselves were performed correctly, nor does it check | ||
| * that an appropriate number of extensions and contractions took | ||
| * place, nor does it check that they took place at sensible times. | ||
| * | ||
| * .dylan: This test uses Dylan format objects in common with most | ||
| * other tests for convenience and brevity. | ||
| */ | ||
|
|
||
| #include "mps.h" | ||
| #include "testlib.h" | ||
| #include "fmtdy.h" | ||
| #include "fmtdytst.h" | ||
| #include "mpsavm.h" | ||
| #include "mpscamc.h" | ||
| #include <stdio.h> | ||
| #include <stdlib.h> | ||
|
|
||
| /* Number of test objects to allocate */ | ||
| #define N_TESTOBJ 100 | ||
| /* The number of slots determines the size of each object */ | ||
| #define N_SLOT_TESTOBJ 10000 | ||
| /* The initial arena size is requested to be bigger the test object by | ||
| this many bytes */ | ||
| #define SIZEDIFF 10 | ||
|
|
||
| /* Set alignment to mps_word_ts */ | ||
| #define ALIGNMENT sizeof(mps_word_t) | ||
|
|
||
| /* Align size upwards to the next multiple of the word size. */ | ||
| #define ALIGN_WORD(size) \ | ||
| (((size) + ALIGNMENT - 1) & ~(ALIGNMENT - 1)) | ||
|
|
||
| /* Global objects*/ | ||
| static mps_arena_t arena; /* the arena */ | ||
| static mps_pool_t obj_pool; /* pool for test objects */ | ||
| static mps_ap_t obj_ap; /* allocation point used to allocate objects */ | ||
|
|
||
| /* Count of number of arena contractions and extensions */ | ||
| static int n_contract = 0; | ||
| static int n_extend = 0; | ||
|
|
||
| /* Callback functions for arena extension and contraction */ | ||
| static void arena_extended_cb(mps_arena_t arena_in, mps_addr_t addr, size_t size) | ||
| { | ||
| testlib_unused(arena_in); | ||
| testlib_unused(addr); | ||
| testlib_unused(size); | ||
| printf("Arena extended by %"PRIuLONGEST" bytes\n", (ulongest_t)size); | ||
| n_extend++; | ||
| } | ||
|
|
||
| static void arena_contracted_cb(mps_arena_t arena_in, mps_addr_t addr, size_t size) | ||
| { | ||
| testlib_unused(arena_in); | ||
| testlib_unused(addr); | ||
| testlib_unused(size); | ||
| printf("Arena contracted by %"PRIuLONGEST" bytes\n", (ulongest_t)size); | ||
| n_contract++; | ||
| } | ||
|
|
||
| /* Messages for testbench debugging */ | ||
| static void print_messages(void) | ||
| { | ||
| mps_message_type_t type; | ||
|
|
||
| while (mps_message_queue_type(&type, arena)) { | ||
| mps_message_t message; | ||
|
|
||
| cdie(mps_message_get(&message, arena, type), | ||
| "get"); | ||
|
|
||
| switch(type) { | ||
| case mps_message_type_gc_start(): | ||
| printf("GC start at %"PRIuLONGEST": %s\n", | ||
| (ulongest_t)mps_message_clock(arena, message), | ||
| mps_message_gc_start_why(arena, message)); | ||
| break; | ||
|
|
||
| case mps_message_type_gc(): | ||
| printf("GC end at %"PRIuLONGEST" " | ||
| "condemned %"PRIuLONGEST" " | ||
| "not condemned %"PRIuLONGEST" " | ||
| "live %"PRIuLONGEST"\n", | ||
| (ulongest_t)mps_message_clock(arena, message), | ||
| (ulongest_t)mps_message_gc_condemned_size(arena, message), | ||
| (ulongest_t)mps_message_gc_not_condemned_size(arena, message), | ||
| (ulongest_t)mps_message_gc_live_size(arena, message)); | ||
| break; | ||
|
|
||
| default: | ||
| cdie(0, "message type"); | ||
| break; | ||
| } | ||
|
|
||
| mps_message_discard(arena, message); | ||
| } | ||
| } | ||
|
|
||
| /* Disabling inlining is necessary (but perhaps not sufficient) if using stack roots. | ||
| See comment below with link to GitHub issue*/ | ||
| ATTRIBUTE_NOINLINE | ||
| static void test_main(void *cold_stack_end) | ||
| { | ||
| mps_fmt_t obj_fmt; | ||
| mps_thr_t thread; | ||
| mps_root_t stack_root, testobj_root; | ||
| size_t arena_size, obj_size; | ||
| int i; | ||
| /* In the original version of extcon this was a stack root, but we | ||
| observed unreliable failures to do with registering the cold end | ||
| of the stack. See GitHub issue #210 | ||
| <https://github.com/Ravenbrook/mps/issues/210>. For now, we | ||
| declare this as a separate root. */ | ||
| static mps_word_t testobj[N_TESTOBJ]; | ||
|
|
||
| /* The testobj array must be below (on all current Posix platforms) | ||
| the cold end of the stack in order for the MPS to scan it. We | ||
| have observed a Heisenbug where GCC will inline test_main into | ||
| main and lose this condition if the expression below is removed. | ||
| This is a problem we are analysing in GitHub issue #210 | ||
| <https://github.com/Ravenbrook/mps/issues/210>. For now, we | ||
| disable this Insist to allow the test to run with a static | ||
| testobj array. */ | ||
| #if 0 | ||
| Insist((void *)&testobj[N_TESTOBJ] <= cold_stack_end); | ||
| if ((void *)&testobj[N_TESTOBJ] > cold_stack_end) | ||
| printf("Cold stack marker invalid!\n"); | ||
| else | ||
| printf("Cold stack marker probably valid.\n"); | ||
| #endif | ||
|
|
||
| /* Make initial arena size slightly bigger than the test object size to force an extension as early as possible */ | ||
| /* See definition of make_dylan_vector() in fmtdytst.c for calculation of vector size */ | ||
| obj_size = ALIGN_WORD((N_SLOT_TESTOBJ + 2) * sizeof(mps_word_t)); | ||
| arena_size = ALIGN_WORD(obj_size + SIZEDIFF); | ||
|
|
||
| /* Create arena and register callbacks */ | ||
| MPS_ARGS_BEGIN(args) { | ||
| MPS_ARGS_ADD(args, MPS_KEY_ARENA_SIZE, arena_size); | ||
| MPS_ARGS_ADD(args, MPS_KEY_ARENA_EXTENDED, (mps_fun_t)&arena_extended_cb); | ||
| MPS_ARGS_ADD(args, MPS_KEY_ARENA_CONTRACTED, (mps_fun_t)&arena_contracted_cb); | ||
| die(mps_arena_create_k(&arena, mps_arena_class_vm(), args), "mps_arena_create_k"); | ||
| } MPS_ARGS_END(args); | ||
|
|
||
| printf("Initial reservation %"PRIuLONGEST".\n", (ulongest_t)mps_arena_reserved(arena)); | ||
|
|
||
| die(dylan_fmt(&obj_fmt, arena), "dylan_fmt()"); | ||
|
|
||
| /* Create new pool */ | ||
| MPS_ARGS_BEGIN(args) { | ||
| MPS_ARGS_ADD(args, MPS_KEY_FORMAT, obj_fmt); | ||
| die(mps_pool_create_k(&obj_pool, arena, mps_class_amcz(), args), | ||
| "mps_pool_create_k"); | ||
| } MPS_ARGS_END(args); | ||
|
|
||
| /* Register thread */ | ||
| die(mps_thread_reg(&thread, arena), "Thread reg"); | ||
|
|
||
| /* Register stack roots */ | ||
| /* Since this testbench is currently not using a stack root, #IF 0 this out */ | ||
| testlib_unused(cold_stack_end); | ||
| testlib_unused(stack_root); | ||
| #if 0 | ||
| die(mps_root_create_thread(&stack_root, arena, thread, cold_stack_end), "Create Stack root"); | ||
| #endif | ||
|
|
||
| /* Register ambiguous array of object roots. */ | ||
| die(mps_root_create_area(&testobj_root, arena, | ||
| mps_rank_ambig(), (mps_rm_t)0, | ||
| &testobj[0], &testobj[N_TESTOBJ], | ||
| mps_scan_area, NULL), | ||
| "root_create_area(testobj)"); | ||
|
|
||
| /* Create allocation point */ | ||
| die(mps_ap_create_k(&obj_ap, obj_pool, mps_args_none), "Create Allocation point"); | ||
|
|
||
| mps_message_type_enable(arena, mps_message_type_gc_start()); | ||
| mps_message_type_enable(arena, mps_message_type_gc()); | ||
|
|
||
| /* Allocate objects and force arena extension */ | ||
| for (i = 0; i < N_TESTOBJ; i++) { | ||
|
|
||
| die(make_dylan_vector(&testobj[i], obj_ap, N_SLOT_TESTOBJ), "make_dylan_vector"); | ||
|
|
||
| printf("Object %d committed. " | ||
| "Arena reserved: %"PRIuLONGEST".\n", | ||
| i, | ||
| (ulongest_t)mps_arena_reserved(arena)); | ||
|
|
||
| print_messages(); | ||
| } | ||
|
|
||
| /* overwrite all the references to the objects*/ | ||
| for (i = 0; i < N_TESTOBJ; i++) { | ||
|
|
||
| /* bonus test of mps_addr_object */ | ||
| #if 0 /* Comment this out until mps_addr_object becomes available. */ | ||
| mps_addr_t out; | ||
| Insist(N_TESTOBJ <= N_INT_TESTOBJ); | ||
|
|
||
| /* use "i" to as a convenient way to generate different interior pointers | ||
| To guarantee the i index will give us an interior pointer the number of test | ||
| objects must be <= the number of integers in each object */ | ||
| Insist(N_TESTOBJ <= N_INT_TESTOBJ); | ||
| die(mps_addr_object(&out, arena, &(testobj[i])->int_array[i]), "Address object"); | ||
|
|
||
| Insist(out == testobj[i]); | ||
|
|
||
| /* end piggy back testbench */ | ||
| #endif | ||
|
|
||
| /* now overwrite the ref */ | ||
| testobj[i] = (mps_word_t)NULL; | ||
|
|
||
| print_messages(); | ||
| } | ||
|
|
||
| /* Collect */ | ||
| mps_arena_collect(arena); | ||
|
|
||
| print_messages(); | ||
|
|
||
| /* Clean up */ | ||
| mps_root_destroy(testobj_root); | ||
| /* mps_root_destroy(stack_root);*/ /*commented out while not using stack root */ | ||
| mps_thread_dereg(thread); | ||
| mps_ap_destroy(obj_ap); | ||
| mps_pool_destroy(obj_pool); | ||
| mps_fmt_destroy(obj_fmt); | ||
| mps_arena_destroy(arena); | ||
|
|
||
| /* Destroying the arena should cause contraction callbacks on all | ||
| remaining chunks, even if they had contents. */ | ||
| Insist(n_extend == n_contract); | ||
|
|
||
| printf("Arena extended %d times\n", n_extend); | ||
| printf("Arena contracted %d times\n", n_contract); | ||
|
|
||
| /* comment out some diagnostics for investigating issue #210 mentioned above */ | ||
| #if 0 | ||
| printf("&testobj[N_TESTOBJ] = %p\n", (void *)&testobj[N_TESTOBJ]); | ||
| printf("cold_stack_end = %p\n", cold_stack_end); | ||
| #endif | ||
| if (n_extend == 0) | ||
| printf("No callbacks received upon arena extended!\n"); | ||
| if (n_contract == 0) | ||
| printf("No callbacks received upon arena contracted!\n"); | ||
|
|
||
| if (n_contract == 0 || n_extend == 0) | ||
| exit(EXIT_FAILURE); | ||
| } | ||
|
|
||
| int main(int argc, char* argv[]) | ||
| { | ||
| void *stack_marker = &stack_marker; | ||
|
|
||
| testlib_init(argc, argv); | ||
|
|
||
| test_main(stack_marker); | ||
|
|
||
| printf("%s: Conclusion: Failed to find any defects.\n", argv[0]); | ||
|
|
||
| return 0; | ||
| } | ||
|
|
||
|
|
||
| /* C. COPYRIGHT AND LICENSE | ||
| * | ||
| * Copyright (C) 2022-2023 Ravenbrook Limited <https://www.ravenbrook.com/>. | ||
| * | ||
| * Redistribution and use in source and binary forms, with or without | ||
| * modification, are permitted provided that the following conditions are | ||
| * met: | ||
| * | ||
| * 1. Redistributions of source code must retain the above copyright | ||
| * notice, this list of conditions and the following disclaimer. | ||
| * | ||
| * 2. Redistributions in binary form must reproduce the above copyright | ||
| * notice, this list of conditions and the following disclaimer in the | ||
| * documentation and/or other materials provided with the | ||
| * distribution. | ||
| * | ||
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS | ||
| * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED | ||
| * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A | ||
| * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | ||
| * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | ||
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | ||
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | ||
| * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | ||
| * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
| */ | ||
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Uh oh!
There was an error while loading. Please reload this page.