test memory limit
Some checks failed
memory-limit / Build judge (push) Successful in 18s
memory-limit / Linux / gcc (push) Failing after 14s
memory-limit / Linux / clang (push) Failing after 15s
memory-limit / Windows / clang (push) Failing after 16s
memory-limit / Windows / msvc (push) Failing after 17s

This commit is contained in:
2026-04-10 23:41:37 +03:00
parent a977d4d9f5
commit e9f07dc47b
3 changed files with 188 additions and 0 deletions

126
.gitea/workflows/memory.yml Normal file
View File

@@ -0,0 +1,126 @@
name: memory-limit
run-name: "memory_limit enforcement check"
on:
push:
branches: [main]
pull_request:
workflow_dispatch:
env:
SUITE_FILE: mem.jdg
EXAMPLE_DIR: example/mem-limit
jobs:
build_judge:
name: Build judge
runs-on: Linux-Runner
timeout-minutes: 10
steps:
- uses: actions/checkout@v4
- name: Set up Go
uses: actions/setup-go@v5
with:
go-version: '1.22'
- name: Cross-compile judge
shell: bash
run: |
mkdir -p dist
CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build -o dist/judge-linux-amd64 ./cmd/cli
CGO_ENABLED=0 GOOS=windows GOARCH=amd64 go build -o dist/judge-windows-amd64.exe ./cmd/cli
- name: Upload judge binaries
uses: https://github.com/christopherHX/gitea-upload-artifact@v4
with:
name: judge-bin-mem
path: dist/
retention-days: 1
check:
needs: build_judge
name: "${{ matrix.toolchain.system }} / ${{ matrix.toolchain.use_compiler }}"
strategy:
fail-fast: false
matrix:
toolchain:
- { system: Linux, use_compiler: gcc }
- { system: Linux, use_compiler: clang }
- { system: Windows, use_compiler: clang }
- { system: Windows, use_compiler: msvc }
runs-on: ${{ matrix.toolchain.system }}-Runner
timeout-minutes: 10
env:
CC: ${{ matrix.toolchain.use_compiler }}
steps:
- name: Checkout judge harness
uses: actions/checkout@v4
- name: Set up MSVC environment
if: matrix.toolchain.use_compiler == 'msvc'
uses: ilammy/msvc-dev-cmd@v1
- name: Download judge binary
uses: https://github.com/christopherHX/gitea-download-artifact@v4
with:
name: judge-bin-mem
path: judge-bin
- name: Install judge on PATH
shell: bash
run: |
mkdir -p bin
if [ "${{ matrix.toolchain.system }}" = "Windows" ]; then
cp judge-bin/judge-windows-amd64.exe bin/judge.exe
else
cp judge-bin/judge-linux-amd64 bin/judge
chmod +x bin/judge
fi
echo "$PWD/bin" >> "$GITHUB_PATH"
- name: Install jq (Linux)
if: matrix.toolchain.system == 'Linux'
run: sudo apt-get update && sudo apt-get install -y jq
- name: Run judge and capture JSON
shell: bash
working-directory: ${{ env.EXAMPLE_DIR }}
run: |
# `|| true` — the "exceeds_limit" group is *expected* to have a
# failing test, so judge will exit non-zero.
judge "$SUITE_FILE" . --json > report.json || true
cat report.json
- name: Assert enforcement
shell: bash
working-directory: ${{ env.EXAMPLE_DIR }}
run: |
set -euo pipefail
pass_status=$(jq -r '.groups[] | select(.name=="within_limit") | .tests[] | select(.name=="allocate_16mb") | .status' report.json)
fail_status=$(jq -r '.groups[] | select(.name=="exceeds_limit") | .tests[] | select(.name=="allocate_256mb") | .status' report.json)
pass_peak=$(jq -r '.groups[] | select(.name=="within_limit") | .tests[] | select(.name=="allocate_16mb") | .peak_memory_kb // 0' report.json)
fail_peak=$(jq -r '.groups[] | select(.name=="exceeds_limit") | .tests[] | select(.name=="allocate_256mb") | .peak_memory_kb // 0' report.json)
echo "within_limit/allocate_16mb: status=$pass_status peak=${pass_peak} KiB"
echo "exceeds_limit/allocate_256mb: status=$fail_status peak=${fail_peak} KiB"
rc=0
if [ "$pass_status" != "PASS" ]; then
echo "FAIL: within_limit test should PASS, got $pass_status"
rc=1
fi
if [ "$fail_status" != "MLE" ]; then
echo "FAIL: exceeds_limit test should be MLE, got $fail_status"
rc=1
fi
if [ "${pass_peak:-0}" -le 0 ]; then
echo "FAIL: within_limit peak memory not reported (got $pass_peak)"
rc=1
fi
exit $rc

35
example/mem-limit/mem.jdg Normal file
View File

@@ -0,0 +1,35 @@
// End-to-end check that memory_limit is actually enforced by the runner.
// The solution allocates N MiB (argv[1]) and touches every page.
//
// - "within_limit" allocates 16 MiB under a 256 MiB cap → expected PASS
// - "exceeds_limit" allocates 256 MiB under a 64 MiB cap → expected MLE
//
// The workflow runs judge --json and asserts these statuses via jq.
build "$CC -O2 -std=c11 -Wall -Wextra solution.c -o solution"
build_windows "if /I \"%CC%\"==\"msvc\" (cl /nologo /O2 /W3 solution.c /Fe:solution.exe) else (%CC% -O2 -std=c11 -Wall -Wextra solution.c -o solution.exe)"
binary = "solution"
timeout 10s
normalize_crlf = true
group("within_limit") {
weight = 1.0
memory_limit = 256M
test("allocate_16mb") {
args = ["16"]
stdout = "ok 16\n"
}
}
group("exceeds_limit") {
weight = 0.0 // score doesn't matter — we assert the status in CI
memory_limit = 64M
test("allocate_256mb") {
args = ["256"]
// no stdout matcher: process is expected to be killed before it prints
}
}

View File

@@ -0,0 +1,27 @@
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char **argv) {
if (argc < 2) {
fprintf(stderr, "usage: %s <megabytes>\n", argv[0]);
return 1;
}
long mb = strtol(argv[1], NULL, 10);
if (mb <= 0) return 1;
size_t bytes = (size_t)mb * 1024 * 1024;
char *p = (char *)malloc(bytes);
if (!p) {
fprintf(stderr, "malloc failed\n");
return 2;
}
for (size_t i = 0; i < bytes; i += 4096) {
p[i] = (char)(i & 0xff);
}
volatile char sink = p[bytes - 1];
(void)sink;
printf("ok %ld\n", mb);
return 0;
}