From a9021cb55f98e28cfd318a36773f203842ddbf5c Mon Sep 17 00:00:00 2001 From: Peter M Date: Fri, 27 Dec 2024 21:58:07 +0100 Subject: [PATCH 01/10] Chore: Bump esp-idf Use 5.3.2 for releases. Signed-off-by: Peter M --- .github/workflows/esp32-build.yaml | 8 ++++---- .github/workflows/esp32-mkimage.yaml | 2 +- doc/release-notes.md.in | 6 +++--- 3 files changed, 8 insertions(+), 8 deletions(-) diff --git a/.github/workflows/esp32-build.yaml b/.github/workflows/esp32-build.yaml index 741ae9e4d..dd036776a 100644 --- a/.github/workflows/esp32-build.yaml +++ b/.github/workflows/esp32-build.yaml @@ -39,15 +39,15 @@ jobs: esp-idf-target: ["esp32", "esp32c3"] idf-version: - 'v5.0.7' - - 'v5.1.4' - - 'v5.2.2' - - 'v5.3.1' + - 'v5.1.5' + - 'v5.2.3' + - 'v5.3.2' exclude: - esp-idf-target: "esp32c3" idf-version: 'v5.0.7' - esp-idf-target: "esp32c3" - idf-version: 'v5.1.4' + idf-version: 'v5.1.5' steps: - name: Checkout repo uses: actions/checkout@v4 diff --git a/.github/workflows/esp32-mkimage.yaml b/.github/workflows/esp32-mkimage.yaml index 001a15d69..4e16fc1a7 100644 --- a/.github/workflows/esp32-mkimage.yaml +++ b/.github/workflows/esp32-mkimage.yaml @@ -37,7 +37,7 @@ jobs: strategy: matrix: - idf-version: ["5.3.1"] + idf-version: ["5.3.2"] cc: ["clang-14"] cxx: ["clang++-14"] cflags: ["-O3"] diff --git a/doc/release-notes.md.in b/doc/release-notes.md.in index f3f46540d..382c71f25 100644 --- a/doc/release-notes.md.in +++ b/doc/release-notes.md.in @@ -68,9 +68,9 @@ AtomVM currently supports the following versions of ESP-IDF: | IDF SDK supported versions | AtomVM support | |------------------------------|----------------| | ESP-IDF [v5.0](https://docs.espressif.com/projects/esp-idf/en/v5.0.7/esp32/get-started/index.html) | ✅ | -| ESP-IDF [v5.1](https://docs.espressif.com/projects/esp-idf/en/v5.1.4/esp32/get-started/index.html) | ✅ | -| ESP-IDF [v5.2](https://docs.espressif.com/projects/esp-idf/en/v5.2.2/esp32/get-started/index.html) | ✅ | -| ESP-IDF [v5.3](https://docs.espressif.com/projects/esp-idf/en/v5.3/esp32/get-started/index.html) | ✅ | +| ESP-IDF [v5.1](https://docs.espressif.com/projects/esp-idf/en/v5.1.5/esp32/get-started/index.html) | ✅ | +| ESP-IDF [v5.2](https://docs.espressif.com/projects/esp-idf/en/v5.2.3/esp32/get-started/index.html) | ✅ | +| ESP-IDF [v5.3](https://docs.espressif.com/projects/esp-idf/en/v5.3.2/esp32/get-started/index.html) | ✅ | Building the AtomVM virtual machine for ESP32 is optional. In most cases, you can simply download a release image from the AtomVM [release](https://github.com/atomvm/AtomVM/releases) repository. If you wish to work on development of the VM or use one on the additional drivers that are available in the [AtomVM repositories](https://github.com/atomvm) you will to build AtomVM from source. See the [Build Instructions](build-instructions.md) for information about how to build AtomVM from source code. We recommend you to use the latest subminor (patch) versions for source builds. You can check the current version used for testing in the [esp32-build.yaml](https://github.com/atomvm/AtomVM/actions/workflows/esp32-build.yaml) workflow. From 96d9cf5d92065f2cd05c459d9da4ae214919aa74 Mon Sep 17 00:00:00 2001 From: Davide Bettio Date: Sat, 28 Dec 2024 11:56:47 +0100 Subject: [PATCH 02/10] CI: Update CodeQL action before deprecation Use v3 before v2 is deprecated, also explicitly set ubuntu version to 24.04 so we can switch in advance and test it. Signed-off-by: Davide Bettio --- .github/workflows/codeql-analysis.yaml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/codeql-analysis.yaml b/.github/workflows/codeql-analysis.yaml index f8e9ae190..8dc9d48e4 100644 --- a/.github/workflows/codeql-analysis.yaml +++ b/.github/workflows/codeql-analysis.yaml @@ -39,7 +39,7 @@ concurrency: jobs: analyze: name: Analyze - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 permissions: actions: read contents: read @@ -61,7 +61,7 @@ jobs: uses: actions/checkout@v4 - name: "Initialize CodeQL" - uses: github/codeql-action/init@v2 + uses: github/codeql-action/init@v3 with: languages: ${{ matrix.language }} queries: +./code-queries/term-to-non-term-func.ql,./code-queries/non-term-to-term-func.ql @@ -74,4 +74,4 @@ jobs: ninja - name: "Perform CodeQL Analysis" - uses: github/codeql-action/analyze@v2 + uses: github/codeql-action/analyze@v3 From b9f79d760e99a0b677a4708f5aca1e2f2c70a3fd Mon Sep 17 00:00:00 2001 From: Davide Bettio Date: Sat, 28 Dec 2024 12:16:20 +0100 Subject: [PATCH 03/10] CI: use ubuntu-24.04 instead of ubuntu-latest Using ubuntu-24.04 allows us to start testing it in advance and to decide when making any future change. Signed-off-by: Davide Bettio --- .github/workflows/esp32-build.yaml | 2 +- .github/workflows/reuse-lint.yaml | 2 +- .github/workflows/wasm-build.yaml | 6 +++--- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/.github/workflows/esp32-build.yaml b/.github/workflows/esp32-build.yaml index dd036776a..86ded3b56 100644 --- a/.github/workflows/esp32-build.yaml +++ b/.github/workflows/esp32-build.yaml @@ -29,7 +29,7 @@ concurrency: jobs: esp-idf: - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 container: espressif/idf:${{ matrix.idf-version }} strategy: diff --git a/.github/workflows/reuse-lint.yaml b/.github/workflows/reuse-lint.yaml index ed63ea86a..cf5e72a31 100644 --- a/.github/workflows/reuse-lint.yaml +++ b/.github/workflows/reuse-lint.yaml @@ -12,7 +12,7 @@ concurrency: jobs: test: - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 steps: - uses: actions/checkout@v4 - name: REUSE Compliance Check diff --git a/.github/workflows/wasm-build.yaml b/.github/workflows/wasm-build.yaml index 8de03e7f1..4a67d5316 100644 --- a/.github/workflows/wasm-build.yaml +++ b/.github/workflows/wasm-build.yaml @@ -78,7 +78,7 @@ jobs: wasm_build_and_test_node: needs: compile_tests - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 container: emscripten/emsdk steps: - name: Checkout repo @@ -145,7 +145,7 @@ jobs: src/platforms/emscripten/build/src/AtomVM-node-${{ github.ref_name }}.wasm.sha256 wasm_build_web: - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 container: emscripten/emsdk steps: - name: Checkout repo @@ -175,7 +175,7 @@ jobs: wasm_test_web: needs: [compile_tests, wasm_build_web] - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 steps: - name: Checkout repo uses: actions/checkout@v4 From bb2f067265a1ed1131a5e1c5b17a8157b392cf01 Mon Sep 17 00:00:00 2001 From: Peter M Date: Sun, 29 Dec 2024 12:32:48 +0100 Subject: [PATCH 04/10] Test esp-idf v5.4-rc1 in sim_test Future proof logic around esp32p4/h2, so they are expanded for a full_sim_test. Signed-off-by: Peter M --- .github/workflows/esp32-simtest.yaml | 23 ++++++++++++++++++----- 1 file changed, 18 insertions(+), 5 deletions(-) diff --git a/.github/workflows/esp32-simtest.yaml b/.github/workflows/esp32-simtest.yaml index 0c04c3c6e..cb5d1f74a 100644 --- a/.github/workflows/esp32-simtest.yaml +++ b/.github/workflows/esp32-simtest.yaml @@ -59,13 +59,26 @@ jobs: fail-fast: false # focus on device diversity. matrix: - esp-idf-target: ["esp32", "esp32s2", "esp32s3", "esp32c3", "esp32c6"] - idf-version: ${{ ((contains(github.event.head_commit.message, 'full_sim_test')||contains(github.event.pull_request.title, 'full_sim_test')) && fromJSON('["v5.1.5", "v5.2.3", "v5.3.2", "v5.4-beta1"]')) || fromJSON('["v5.3.2"]') }} - include: + esp-idf-target: + [ + "esp32", + "esp32s2", + "esp32s3", + "esp32c3", + "esp32c6", + "esp32h2", + "esp32p4", + ] + idf-version: ${{ ((contains(github.event.head_commit.message, 'full_sim_test')||contains(github.event.pull_request.title, 'full_sim_test')) && fromJSON('["v5.1.5", "v5.2.3", "v5.3.2", "v5.4-rc1"]')) || fromJSON('["v5.3.2"]') }} + exclude: - esp-idf-target: "esp32p4" - idf-version: "v5.3.2" + idf-version: "v5.1.5" + - esp-idf-target: "esp32p4" + idf-version: "v5.2.3" + - esp-idf-target: "esp32h2" + idf-version: "v5.1.5" - esp-idf-target: "esp32h2" - idf-version: "v5.3.2" + idf-version: "v5.2.3" steps: - name: Checkout repo From f7592e23a1c8938dcc7cfc388275cb3a34d33c7d Mon Sep 17 00:00:00 2001 From: Paul Guyot Date: Fri, 3 Jan 2025 19:09:18 +0100 Subject: [PATCH 05/10] Add workaround for latest debian gcc-arm-none-eabi Fixes #1445 Signed-off-by: Paul Guyot --- .github/workflows/pico-build.yaml | 2 +- .github/workflows/stm32-build.yaml | 13 +------------ CHANGELOG.md | 1 + src/libAtomVM/term.h | 3 +++ 4 files changed, 6 insertions(+), 13 deletions(-) diff --git a/.github/workflows/pico-build.yaml b/.github/workflows/pico-build.yaml index 0d1fec96b..25c8d516c 100644 --- a/.github/workflows/pico-build.yaml +++ b/.github/workflows/pico-build.yaml @@ -33,7 +33,7 @@ concurrency: jobs: pico: - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 strategy: matrix: board: ["pico", "pico_w"] diff --git a/.github/workflows/stm32-build.yaml b/.github/workflows/stm32-build.yaml index b33fcccb0..feacf4985 100644 --- a/.github/workflows/stm32-build.yaml +++ b/.github/workflows/stm32-build.yaml @@ -44,17 +44,6 @@ jobs: https://repo.hex.pm https://cdn.jsdelivr.net/hex - - name: Install arm-embedded toolchain - if: ${{ steps.builddeps-cache.outputs.cache-hit != 'true' }} - working-directory: /home/runner - run: | - set -euo pipefail - cd /home/runner - wget https://developer.arm.com/-/media/Files/downloads/gnu/11.3.rel1/binrel/arm-gnu-toolchain-11.3.rel1-x86_64-arm-none-eabi.tar.xz \ - --output-document=$RUNNER_TEMP/arm-gnu-toolchain-11.3.rel1-x86_64-arm-none-eabi.tar.xz - tar xJf $RUNNER_TEMP/arm-gnu-toolchain-11.3.rel1-x86_64-arm-none-eabi.tar.xz - pwd && ls - - name: Checkout and build libopencm3 if: ${{ steps.builddeps-cache.outputs.cache-hit != 'true' }} working-directory: /home/runner @@ -71,7 +60,7 @@ jobs: run: sudo apt update - name: "Install deps" - run: sudo apt install -y cmake gperf + run: sudo apt install -y cmake gperf gcc-arm-none-eabi - name: Checkout repo uses: actions/checkout@v4 diff --git a/CHANGELOG.md b/CHANGELOG.md index 18b7d8d9c..55e71914a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -27,6 +27,7 @@ certain VM instructions are used. - Fixed an issue where a timeout would occur immediately in a race condition - Fixed SPI close command - Added missing lock on socket structure +- Fixed compilation with latest debian gcc-arm-none-eabi ## [0.6.5] - 2024-10-15 diff --git a/src/libAtomVM/term.h b/src/libAtomVM/term.h index f926ef6d5..cecad50c5 100644 --- a/src/libAtomVM/term.h +++ b/src/libAtomVM/term.h @@ -28,6 +28,9 @@ #ifndef _TERM_H_ #define _TERM_H_ +// gcc-arm-none-eabi 13.2.1 with newlib requires this first +#include + #include #include #include From af808361c5025e761d5dc635149e7e5f88c8ed25 Mon Sep 17 00:00:00 2001 From: Paul Guyot Date: Thu, 2 Jan 2025 20:24:06 +0100 Subject: [PATCH 06/10] Fix double free of esp32 uart driver on close - Also fix usage of memory_ensure_free in esp32 uart driver - Also remove unnecessary condition in context_destroy Signed-off-by: Paul Guyot --- CHANGELOG.md | 1 + src/libAtomVM/context.c | 4 +--- .../esp32/components/avm_builtins/uart_driver.c | 9 +++++---- 3 files changed, 7 insertions(+), 7 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 18b7d8d9c..091339644 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -27,6 +27,7 @@ certain VM instructions are used. - Fixed an issue where a timeout would occur immediately in a race condition - Fixed SPI close command - Added missing lock on socket structure +- Fixed a double free when esp32 uart driver was closed, yielding an assert abort ## [0.6.5] - 2024-10-15 diff --git a/src/libAtomVM/context.c b/src/libAtomVM/context.c index 46cff747e..1fd22be10 100644 --- a/src/libAtomVM/context.c +++ b/src/libAtomVM/context.c @@ -178,9 +178,7 @@ void context_destroy(Context *ctx) // globalcontext_get_process_lock before accessing platform_data. // Here, the context can no longer be acquired with // globalcontext_get_process_lock, so it's safe to free the pointer. - if (ctx->platform_data) { - free(ctx->platform_data); - } + free(ctx->platform_data); free(ctx); } diff --git a/src/platforms/esp32/components/avm_builtins/uart_driver.c b/src/platforms/esp32/components/avm_builtins/uart_driver.c index 22f9ed398..8f141d77b 100644 --- a/src/platforms/esp32/components/avm_builtins/uart_driver.c +++ b/src/platforms/esp32/components/avm_builtins/uart_driver.c @@ -309,7 +309,7 @@ static void uart_driver_do_read(Context *ctx, GenMessage gen_message) int local_pid = term_to_local_process_id(pid); if (uart_data->reader_process_pid != term_invalid_term()) { - if (UNLIKELY(memory_ensure_free(ctx, TUPLE_SIZE(2) * 2 + REF_SIZE) != MEMORY_GC_OK)) { + if (UNLIKELY(memory_ensure_free_with_roots(ctx, TUPLE_SIZE(2) * 2 , 1, &ref, MEMORY_CAN_SHRINK) != MEMORY_GC_OK)) { ESP_LOGE(TAG, "[uart_driver_do_read] Failed to allocate space for error tuple"); globalcontext_send_message(glb, local_pid, MEMORY_ATOM); return; @@ -326,7 +326,7 @@ static void uart_driver_do_read(Context *ctx, GenMessage gen_message) if (count > 0) { int bin_size = term_binary_heap_size(count); - if (UNLIKELY(memory_ensure_free(ctx, bin_size + TUPLE_SIZE(2) * 2 + REF_SIZE) != MEMORY_GC_OK)) { + if (UNLIKELY(memory_ensure_free_with_roots(ctx, bin_size + TUPLE_SIZE(2) * 2, 1, &ref, MEMORY_CAN_SHRINK) != MEMORY_GC_OK)) { ESP_LOGE(TAG, "[uart_driver_do_read] Failed to allocate space for return value"); globalcontext_send_message(glb, local_pid, MEMORY_ATOM); } @@ -387,7 +387,7 @@ static void uart_driver_do_write(Context *ctx, GenMessage gen_message) free(buffer); int local_pid = term_to_local_process_id(pid); - if (UNLIKELY(memory_ensure_free(ctx, TUPLE_SIZE(2) + REF_SIZE) != MEMORY_GC_OK)) { + if (UNLIKELY(memory_ensure_free_with_roots(ctx, TUPLE_SIZE(2), 1, &ref, MEMORY_CAN_SHRINK) != MEMORY_GC_OK)) { ESP_LOGE(TAG, "[uart_driver_do_write] Failed to allocate space for return value"); globalcontext_send_message(glb, local_pid, MEMORY_ATOM); } @@ -406,7 +406,7 @@ static void uart_driver_do_close(Context *ctx, GenMessage gen_message) sys_unregister_listener(glb, &uart_data->listener); - if (UNLIKELY(memory_ensure_free(ctx, TUPLE_SIZE(2) + REF_SIZE) != MEMORY_GC_OK)) { + if (UNLIKELY(memory_ensure_free_with_roots(ctx, TUPLE_SIZE(2), 1, &ref, MEMORY_CAN_SHRINK) != MEMORY_GC_OK)) { ESP_LOGE(TAG, "[uart_driver_do_close] Failed to allocate space for return value"); globalcontext_send_message(glb, local_pid, MEMORY_ATOM); } @@ -419,6 +419,7 @@ static void uart_driver_do_close(Context *ctx, GenMessage gen_message) } free(uart_data); + ctx->platform_data = NULL; } static NativeHandlerResult uart_driver_consume_mailbox(Context *ctx) From 46cde9838d284472a5585b99c63cd9fdacbadf7d Mon Sep 17 00:00:00 2001 From: Paul Guyot Date: Sun, 5 Jan 2025 11:28:15 +0100 Subject: [PATCH 07/10] Fix dubious logic revealed by esp idf 5.4 warning Signed-off-by: Paul Guyot --- src/platforms/esp32/components/avm_sys/sys.c | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/src/platforms/esp32/components/avm_sys/sys.c b/src/platforms/esp32/components/avm_sys/sys.c index acb6b173d..1581cb83d 100644 --- a/src/platforms/esp32/components/avm_sys/sys.c +++ b/src/platforms/esp32/components/avm_sys/sys.c @@ -677,12 +677,12 @@ static void *select_thread_loop(void *arg) { GlobalContext *glb = arg; struct ESP32PlatformData *platform = glb->platform_data; - struct pollfd *fds = malloc(0); + struct pollfd *fds = NULL; while (!platform->select_thread_exit) { int select_events_poll_count = platform->select_events_poll_count; int poll_count = 1; int fd_index; - if (select_events_poll_count < 0) { + if (fds == NULL || select_events_poll_count < 0) { // Means it is dirty and should be rebuilt. struct ListHead *select_events = synclist_wrlock(&glb->select_events); size_t select_events_new_count; @@ -692,7 +692,11 @@ static void *select_thread_loop(void *arg) select_events_new_count = select_events_poll_count; } - fds = realloc(fds, sizeof(struct pollfd) * (poll_count + select_events_new_count)); + if (fds) { + fds = realloc(fds, sizeof(struct pollfd) * (poll_count + select_events_new_count)); + } else { + fds = malloc(sizeof(struct pollfd) * (poll_count + select_events_new_count)); + } fds[0].fd = platform->signal_fd; fds[0].events = POLLIN; From e487ea71c97e1eabe6b47ffa3cb1bbfe303577da Mon Sep 17 00:00:00 2001 From: Winford Date: Sun, 5 Jan 2025 08:06:42 +0000 Subject: [PATCH 08/10] Add ESP-IDF v5.4 (official) to workflows and supported build environment in relase notes. Bumps version the esp32-build.yaml, and esp32-simtest.yaml for runs that include a full_sim_test to ESP-IDF v5.4. Adds ESP-IDF v5.4 to supported build environments in the release-0.6 release notes. Signed-off-by: Winford --- .github/workflows/esp32-build.yaml | 1 + .github/workflows/esp32-simtest.yaml | 2 +- doc/release-notes.md.in | 1 + 3 files changed, 3 insertions(+), 1 deletion(-) diff --git a/.github/workflows/esp32-build.yaml b/.github/workflows/esp32-build.yaml index 86ded3b56..1c2b3ab0d 100644 --- a/.github/workflows/esp32-build.yaml +++ b/.github/workflows/esp32-build.yaml @@ -42,6 +42,7 @@ jobs: - 'v5.1.5' - 'v5.2.3' - 'v5.3.2' + - 'v5.4' exclude: - esp-idf-target: "esp32c3" diff --git a/.github/workflows/esp32-simtest.yaml b/.github/workflows/esp32-simtest.yaml index cb5d1f74a..55177a47e 100644 --- a/.github/workflows/esp32-simtest.yaml +++ b/.github/workflows/esp32-simtest.yaml @@ -69,7 +69,7 @@ jobs: "esp32h2", "esp32p4", ] - idf-version: ${{ ((contains(github.event.head_commit.message, 'full_sim_test')||contains(github.event.pull_request.title, 'full_sim_test')) && fromJSON('["v5.1.5", "v5.2.3", "v5.3.2", "v5.4-rc1"]')) || fromJSON('["v5.3.2"]') }} + idf-version: ${{ ((contains(github.event.head_commit.message, 'full_sim_test')||contains(github.event.pull_request.title, 'full_sim_test')) && fromJSON('["v5.1.5", "v5.2.3", "v5.3.2", "v5.4"]')) || fromJSON('["v5.3.2"]') }} exclude: - esp-idf-target: "esp32p4" idf-version: "v5.1.5" diff --git a/doc/release-notes.md.in b/doc/release-notes.md.in index 382c71f25..1de96f072 100644 --- a/doc/release-notes.md.in +++ b/doc/release-notes.md.in @@ -71,6 +71,7 @@ AtomVM currently supports the following versions of ESP-IDF: | ESP-IDF [v5.1](https://docs.espressif.com/projects/esp-idf/en/v5.1.5/esp32/get-started/index.html) | ✅ | | ESP-IDF [v5.2](https://docs.espressif.com/projects/esp-idf/en/v5.2.3/esp32/get-started/index.html) | ✅ | | ESP-IDF [v5.3](https://docs.espressif.com/projects/esp-idf/en/v5.3.2/esp32/get-started/index.html) | ✅ | +| ESP-IDF [v5.4](https://docs.espressif.com/projects/esp-idf/en/v5.4/esp32/get-started/index.html) | ✅ | Building the AtomVM virtual machine for ESP32 is optional. In most cases, you can simply download a release image from the AtomVM [release](https://github.com/atomvm/AtomVM/releases) repository. If you wish to work on development of the VM or use one on the additional drivers that are available in the [AtomVM repositories](https://github.com/atomvm) you will to build AtomVM from source. See the [Build Instructions](build-instructions.md) for information about how to build AtomVM from source code. We recommend you to use the latest subminor (patch) versions for source builds. You can check the current version used for testing in the [esp32-build.yaml](https://github.com/atomvm/AtomVM/actions/workflows/esp32-build.yaml) workflow. From 50fca6d9f0d06175b759949bf970cac5958a633a Mon Sep 17 00:00:00 2001 From: Paul Guyot Date: Sun, 5 Jan 2025 22:06:33 +0100 Subject: [PATCH 09/10] Complete supervisor behavior - add missing `supervisor:terminate_child/2`, `supervisor:delete_child/2` and `supervisor:restart_child/2` - fix termination of children of supervisor - add more termination strategies Signed-off-by: Paul Guyot --- CHANGELOG.md | 1 + libs/estdlib/src/gen_server.erl | 40 ++++---- libs/estdlib/src/supervisor.erl | 121 ++++++++++++++++++++++++- src/libAtomVM/defaultatoms.def | 2 + src/libAtomVM/opcodesswitch.h | 4 +- tests/libs/estdlib/test_supervisor.erl | 62 ++++++++++++- 6 files changed, 205 insertions(+), 25 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 9e436e129..1e4b68ed5 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -14,6 +14,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added support for `registered_name` in `erlang:process_info/2` and `Process.info/2` - Added `net:gethostname/0` on platforms with gethostname(3). - Added `socket:getopt/2` +- Added `supervisor:terminate_child/2`, `supervisor:restart_child/2` and `supervisor:delete_child/2` ### Fixed - ESP32: improved sntp sync speed from a cold boot. diff --git a/libs/estdlib/src/gen_server.erl b/libs/estdlib/src/gen_server.erl index 7aa1f7640..12e16882d 100644 --- a/libs/estdlib/src/gen_server.erl +++ b/libs/estdlib/src/gen_server.erl @@ -211,8 +211,8 @@ init_it(Starter, Module, Args, Options) -> end, case StateT of undefined -> ok; - {State, {continue, Continue}} -> loop(State, {continue, Continue}); - {State, Timeout} -> loop(State, Timeout) + {State, {continue, Continue}} -> loop(Starter, State, {continue, Continue}); + {State, Timeout} -> loop(Starter, State, Timeout) end. init_ack(Parent, Return) -> @@ -499,34 +499,34 @@ reply({Pid, Ref}, Reply) -> %% %% @private -loop(#state{mod = Mod, mod_state = ModState} = State, {continue, Continue}) -> +loop(Parent, #state{mod = Mod, mod_state = ModState} = State, {continue, Continue}) -> case Mod:handle_continue(Continue, ModState) of {noreply, NewModState} -> - loop(State#state{mod_state = NewModState}, infinity); + loop(Parent, State#state{mod_state = NewModState}, infinity); {noreply, NewModState, {continue, NewContinue}} -> - loop(State#state{mod_state = NewModState}, {continue, NewContinue}); + loop(Parent, State#state{mod_state = NewModState}, {continue, NewContinue}); {stop, Reason, NewModState} -> do_terminate(State, Reason, NewModState) end; -loop(#state{mod = Mod, mod_state = ModState} = State, Timeout) -> +loop(Parent, #state{mod = Mod, mod_state = ModState} = State, Timeout) -> receive {'$call', {_Pid, _Ref} = From, Request} -> case Mod:handle_call(Request, From, ModState) of {reply, Reply, NewModState} -> ok = reply(From, Reply), - loop(State#state{mod_state = NewModState}, infinity); + loop(Parent, State#state{mod_state = NewModState}, infinity); {reply, Reply, NewModState, {continue, Continue}} -> ok = reply(From, Reply), - loop(State#state{mod_state = NewModState}, {continue, Continue}); + loop(Parent, State#state{mod_state = NewModState}, {continue, Continue}); {reply, Reply, NewModState, NewTimeout} -> ok = reply(From, Reply), - loop(State#state{mod_state = NewModState}, NewTimeout); + loop(Parent, State#state{mod_state = NewModState}, NewTimeout); {noreply, NewModState} -> - loop(State#state{mod_state = NewModState}, infinity); + loop(Parent, State#state{mod_state = NewModState}, infinity); {noreply, NewModState, {continue, Continue}} -> - loop(State#state{mod_state = NewModState}, {continue, Continue}); + loop(Parent, State#state{mod_state = NewModState}, {continue, Continue}); {noreply, NewModState, NewTimeout} -> - loop(State#state{mod_state = NewModState}, NewTimeout); + loop(Parent, State#state{mod_state = NewModState}, NewTimeout); {stop, Reason, Reply, NewModState} -> ok = reply(From, Reply), do_terminate(State, Reason, NewModState); @@ -538,11 +538,11 @@ loop(#state{mod = Mod, mod_state = ModState} = State, Timeout) -> {'$cast', Request} -> case Mod:handle_cast(Request, ModState) of {noreply, NewModState} -> - loop(State#state{mod_state = NewModState}, infinity); + loop(Parent, State#state{mod_state = NewModState}, infinity); {noreply, NewModState, {continue, Continue}} -> - loop(State#state{mod_state = NewModState}, {continue, Continue}); + loop(Parent, State#state{mod_state = NewModState}, {continue, Continue}); {noreply, NewModState, NewTimeout} -> - loop(State#state{mod_state = NewModState}, NewTimeout); + loop(Parent, State#state{mod_state = NewModState}, NewTimeout); {stop, Reason, NewModState} -> do_terminate(State, Reason, NewModState); _ -> @@ -550,12 +550,14 @@ loop(#state{mod = Mod, mod_state = ModState} = State, Timeout) -> end; {'$stop', Reason} -> do_terminate(State, Reason, ModState); + {'EXIT', Parent, Reason} -> + do_terminate(State, Reason, ModState); Info -> case Mod:handle_info(Info, ModState) of {noreply, NewModState} -> - loop(State#state{mod_state = NewModState}, infinity); + loop(Parent, State#state{mod_state = NewModState}, infinity); {noreply, NewModState, NewTimeout} -> - loop(State#state{mod_state = NewModState}, NewTimeout); + loop(Parent, State#state{mod_state = NewModState}, NewTimeout); {stop, Reason, NewModState} -> do_terminate(State, Reason, NewModState); _ -> @@ -564,9 +566,9 @@ loop(#state{mod = Mod, mod_state = ModState} = State, Timeout) -> after Timeout -> case Mod:handle_info(timeout, ModState) of {noreply, NewModState} -> - loop(State#state{mod_state = NewModState}, infinity); + loop(Parent, State#state{mod_state = NewModState}, infinity); {noreply, NewModState, NewTimeout} -> - loop(State#state{mod_state = NewModState}, NewTimeout); + loop(Parent, State#state{mod_state = NewModState}, NewTimeout); {stop, Reason, NewModState} -> do_terminate(State, Reason, NewModState); _ -> diff --git a/libs/estdlib/src/supervisor.erl b/libs/estdlib/src/supervisor.erl index 8239d55e9..1db075599 100644 --- a/libs/estdlib/src/supervisor.erl +++ b/libs/estdlib/src/supervisor.erl @@ -25,14 +25,18 @@ -export([ start_link/2, start_link/3, - start_child/2 + start_child/2, + terminate_child/2, + restart_child/2, + delete_child/2 ]). -export([ init/1, handle_call/3, handle_cast/2, - handle_info/2 + handle_info/2, + terminate/2 ]). -export_type([ @@ -41,7 +45,11 @@ sup_flags/0 ]). --type restart() :: permanent | transient | temporary. +-type restart() :: + permanent + | transient + | temporary + | {terminating, permanent | transient | temporary, gen_server:from()}. -type shutdown() :: brutal_kill | timeout(). -type child_type() :: worker | supervisor. @@ -90,6 +98,15 @@ start_link(SupName, Module, Args) -> start_child(Supervisor, ChildSpec) -> gen_server:call(Supervisor, {start_child, ChildSpec}). +terminate_child(Supervisor, ChildId) -> + gen_server:call(Supervisor, {terminate_child, ChildId}). + +restart_child(Supervisor, ChildId) -> + gen_server:call(Supervisor, {restart_child, ChildId}). + +delete_child(Supervisor, ChildId) -> + gen_server:call(Supervisor, {delete_child, ChildId}). + init({Mod, Args}) -> erlang:process_flag(trap_exit, true), case Mod:init(Args) of @@ -152,6 +169,16 @@ restart_child(Pid, Reason, State) -> case lists:keyfind(Pid, #child.pid, State#state.children) of false -> {ok, State}; + #child{restart = {terminating, temporary, From}} -> + gen_server:reply(From, ok), + NewChildren = lists:keydelete(Pid, #child.pid, State#state.children), + {ok, State#state{children = NewChildren}}; + #child{restart = {terminating, Restart, From}} = Child -> + gen_server:reply(From, ok), + NewChildren = lists:keyreplace(Pid, #child.pid, State#state.children, Child#child{ + pid = undefined, restart = Restart + }), + {ok, State#state{children = NewChildren}}; #child{} = Child -> case should_restart(Reason, Child#child.restart) of true -> @@ -195,6 +222,46 @@ handle_call({start_child, ChildSpec}, _From, #state{children = Children} = State {error, _Reason} = ErrorT -> {reply, ErrorT, State} end + end; +handle_call({terminate_child, ID}, From, #state{children = Children} = State) -> + case lists:keyfind(ID, #child.id, Children) of + #child{pid = undefined} -> + {reply, ok, State}; + #child{restart = Restart} = Child -> + do_terminate(Child), + NewChild = Child#child{restart = {terminating, Restart, From}}, + NewChildren = lists:keyreplace(ID, #child.id, Children, NewChild), + {noreply, State#state{children = NewChildren}}; + false -> + {reply, {error, not_found}, State} + end; +handle_call({restart_child, ID}, _From, #state{children = Children} = State) -> + case lists:keyfind(ID, #child.id, Children) of + #child{pid = undefined} = Child -> + case try_start(Child) of + {ok, NewPid, Result} -> + NewChild = Child#child{pid = NewPid}, + NewChildren = lists:keyreplace( + ID, #child.id, Children, NewChild + ), + {reply, Result, State#state{children = NewChildren}}; + {error, _Reason} = ErrorT -> + {reply, ErrorT, State} + end; + #child{} -> + {reply, {error, running}, State}; + false -> + {reply, {error, not_found}, State} + end; +handle_call({delete_child, ID}, _From, #state{children = Children} = State) -> + case lists:keyfind(ID, #child.id, Children) of + #child{pid = undefined} -> + NewChildren = lists:keydelete(ID, #child.id, Children), + {reply, ok, State#state{children = NewChildren}}; + #child{} -> + {reply, {error, running}, State}; + false -> + {reply, {error, not_found}, State} end. handle_cast(_Msg, State) -> @@ -207,10 +274,50 @@ handle_info({'EXIT', Pid, Reason}, State) -> {shutdown, State1} -> {stop, shutdown, State1} end; +handle_info({ensure_killed, Pid}, State) -> + case lists:keyfind(Pid, #child.pid, State#state.children) of + false -> + {noreply, State}; + #child{} -> + exit(Pid, kill), + {noreply, State} + end; handle_info(_Msg, State) -> %TODO: log unexpected message {noreply, State}. +%% @hidden +terminate(_Reason, #state{children = Children} = State) -> + RemainingChildren = loop_terminate(Children, []), + loop_wait_termination(RemainingChildren), + {ok, State}. + +loop_terminate([#child{pid = undefined} | Tail], AccRemaining) -> + loop_terminate(Tail, AccRemaining); +loop_terminate([#child{pid = Pid} = Child | Tail], AccRemaining) when is_pid(Pid) -> + do_terminate(Child), + loop_terminate(Tail, [Pid | AccRemaining]); +loop_terminate([], AccRemaining) -> + AccRemaining. + +loop_wait_termination([]) -> + ok; +loop_wait_termination(RemainingChildren0) -> + receive + {'EXIT', Pid, _Reason} -> + RemainingChildren1 = lists:delete(Pid, RemainingChildren0), + loop_wait_termination(RemainingChildren1); + {ensure_killed, Pid} -> + case lists:member(Pid, RemainingChildren0) of + true -> + exit(Pid, kill), + RemainingChildren1 = lists:delete(Pid, RemainingChildren0), + loop_wait_termination(RemainingChildren1); + false -> + loop_wait_termination(RemainingChildren0) + end + end. + try_start(#child{start = {M, F, Args}} = Record) -> try case apply(M, F, Args) of @@ -229,3 +336,11 @@ try_start(#child{start = {M, F, Args}} = Record) -> error:Error -> {error, {{'EXIT', Error}, Record}} end. + +do_terminate(#child{pid = Pid, shutdown = brutal_kill}) -> + exit(Pid, kill); +do_terminate(#child{pid = Pid, shutdown = infinity}) -> + exit(Pid, shutdown); +do_terminate(#child{pid = Pid, shutdown = Timeout}) when is_integer(Timeout) -> + exit(Pid, shutdown), + erlang:send_after(Timeout, self(), {ensure_killed, Pid}). diff --git a/src/libAtomVM/defaultatoms.def b/src/libAtomVM/defaultatoms.def index 2f1d2949d..9e768a355 100644 --- a/src/libAtomVM/defaultatoms.def +++ b/src/libAtomVM/defaultatoms.def @@ -165,3 +165,5 @@ X(EXTERNAL_ATOM, "\x8", "external") X(LOCAL_ATOM, "\x5", "local") X(REGISTERED_NAME_ATOM, "\xF", "registered_name") + +X(SHUTDOWN_ATOM, "\x8", "shutdown") diff --git a/src/libAtomVM/opcodesswitch.h b/src/libAtomVM/opcodesswitch.h index f83dd1f60..c0ed1fe30 100644 --- a/src/libAtomVM/opcodesswitch.h +++ b/src/libAtomVM/opcodesswitch.h @@ -7053,8 +7053,8 @@ HOT_FUNC int scheduler_entry_point(GlobalContext *glb) } } - // Do not print crash dump if reason is normal. - if (x_regs[0] != LOWERCASE_EXIT_ATOM || x_regs[1] != NORMAL_ATOM) { + // Do not print crash dump if reason is normal or shutdown. + if (x_regs[0] != LOWERCASE_EXIT_ATOM || (x_regs[1] != NORMAL_ATOM && x_regs[1] != SHUTDOWN_ATOM)) { dump(ctx); } diff --git a/tests/libs/estdlib/test_supervisor.erl b/tests/libs/estdlib/test_supervisor.erl index b18944113..cea4d9024 100644 --- a/tests/libs/estdlib/test_supervisor.erl +++ b/tests/libs/estdlib/test_supervisor.erl @@ -31,6 +31,8 @@ test() -> ok = test_start_child(), ok = test_start_child_ping_pong(), ok = test_supervisor_order(), + ok = test_terminate_delete_child(), + ok = test_terminate_timeout(), ok. test_basic_supervisor() -> @@ -85,6 +87,53 @@ test_start_child() -> exit(SupPid, shutdown), ok. +test_terminate_delete_child() -> + {ok, SupPid} = supervisor:start_link(?MODULE, {test_no_child, self()}), + {ok, Pid} = supervisor:start_child(SupPid, #{ + id => child_start, start => {?MODULE, child_start, [start]} + }), + {error, not_found} = supervisor:terminate_child(SupPid, Pid), + {error, running} = supervisor:delete_child(SupPid, child_start), + ok = supervisor:terminate_child(SupPid, child_start), + ok = supervisor:delete_child(SupPid, child_start), + {error, not_found} = supervisor:delete_child(SupPid, child_start), + unlink(SupPid), + exit(SupPid, shutdown), + ok. + +test_terminate_timeout() -> + {ok, SupPid} = supervisor:start_link(?MODULE, {test_no_child, self()}), + Self = self(), + {ok, Pid} = supervisor:start_child(SupPid, #{ + id => child_start, start => {?MODULE, child_start, [{trap_exit, Self}]}, shutdown => 500 + }), + ok = supervisor:terminate_child(SupPid, child_start), + ok = + receive + {Pid, {SupPid, shutdown}} -> ok + after 1000 -> timeout + end, + {ok, Pid2} = supervisor:restart_child(SupPid, child_start), + Pid2 ! ok, + ok = supervisor:terminate_child(SupPid, child_start), + ok = + receive + {Pid2, {SupPid, shutdown}} -> ok + after 1000 -> timeout + end, + ok = supervisor:delete_child(SupPid, child_start), + {ok, Pid3} = supervisor:start_child(SupPid, #{ + id => child_start, start => {?MODULE, child_start, [{trap_exit, Self}]}, shutdown => 500 + }), + unlink(SupPid), + exit(SupPid, shutdown), + ok = + receive + {Pid3, {SupPid, shutdown}} -> ok + after 1000 -> timeout + end, + ok. + child_start(ignore) -> ignore; child_start(start) -> @@ -104,7 +153,18 @@ child_start(info) -> child_start(error) -> {error, child_error}; child_start(fail) -> - fail. + fail; +child_start({trap_exit, Parent}) -> + Pid = spawn_link(fun() -> + process_flag(trap_exit, true), + receive + {'EXIT', From, Reason} -> Parent ! {self(), {From, Reason}} + end, + receive + ok -> ok + end + end), + {ok, Pid}. test_ping_pong(SupPid) -> Pid1 = get_and_test_server(), From 777a1ba98394dff291722fabd5ae5e31d8829d09 Mon Sep 17 00:00:00 2001 From: Paul Guyot Date: Mon, 6 Jan 2025 22:37:52 +0100 Subject: [PATCH 10/10] Fix race condition in timeout handling Signed-off-by: Paul Guyot --- CHANGELOG.md | 1 + src/libAtomVM/scheduler.c | 4 ++-- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 18b7d8d9c..af2dfba82 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -27,6 +27,7 @@ certain VM instructions are used. - Fixed an issue where a timeout would occur immediately in a race condition - Fixed SPI close command - Added missing lock on socket structure +- Fixed a race condition affecting multi-core MCUs where a timeout would not be properly cleared ## [0.6.5] - 2024-10-15 diff --git a/src/libAtomVM/scheduler.c b/src/libAtomVM/scheduler.c index 21f1d7d75..ce97bbb16 100644 --- a/src/libAtomVM/scheduler.c +++ b/src/libAtomVM/scheduler.c @@ -414,11 +414,11 @@ void scheduler_cancel_timeout(Context *ctx) { GlobalContext *glb = ctx->global; - context_update_flags(ctx, ~(WaitingTimeout | WaitingTimeoutExpired), NoFlags); - struct TimerList *tw = &glb->timer_list; SMP_SPINLOCK_LOCK(&glb->timer_spinlock); timer_list_remove(tw, &ctx->timer_list_head); SMP_SPINLOCK_UNLOCK(&glb->timer_spinlock); + + context_update_flags(ctx, ~(WaitingTimeout | WaitingTimeoutExpired), NoFlags); }