aboutsummaryrefslogtreecommitdiff
path: root/test/link/wasm/bss/build.zig
blob: faf8202cd96664d959f8083cc96b5a1f27589d8c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
const std = @import("std");

pub const requires_stage2 = true;

pub fn build(b: *std.Build) void {
    const test_step = b.step("test", "Test");
    b.default_step = test_step;

    add(b, test_step, .Debug, true);
    add(b, test_step, .ReleaseFast, false);
    add(b, test_step, .ReleaseSmall, false);
    add(b, test_step, .ReleaseSafe, true);
}

fn add(b: *std.Build, test_step: *std.Build.Step, optimize_mode: std.builtin.OptimizeMode, is_safe: bool) void {
    {
        const lib = b.addExecutable(.{
            .name = "lib",
            .root_source_file = .{ .path = "lib.zig" },
            .target = .{ .cpu_arch = .wasm32, .os_tag = .freestanding },
            .optimize = optimize_mode,
        });
        lib.entry = .disabled;
        lib.use_llvm = false;
        lib.use_lld = false;
        lib.strip = false;
        // to make sure the bss segment is emitted, we must import memory
        lib.import_memory = true;
        lib.link_gc_sections = false;

        const check_lib = lib.checkObject();

        // since we import memory, make sure it exists with the correct naming
        check_lib.checkStart();
        check_lib.checkExact("Section import");
        check_lib.checkExact("entries 1");
        check_lib.checkExact("module env"); // default module name is "env"
        check_lib.checkExact("name memory"); // as per linker specification

        // since we are importing memory, ensure it's not exported
        check_lib.checkStart();
        check_lib.checkNotPresent("Section export");

        // validate the name of the stack pointer
        check_lib.checkStart();
        check_lib.checkExact("Section custom");
        check_lib.checkExact("type data_segment");
        check_lib.checkExact("names 2");
        check_lib.checkExact("index 0");
        check_lib.checkExact("name .rodata");
        // for safe optimization modes `undefined` is stored in data instead of bss.
        if (is_safe) {
            check_lib.checkExact("index 1");
            check_lib.checkExact("name .data");
            check_lib.checkNotPresent("name .bss");
        } else {
            check_lib.checkExact("index 1"); // bss section always last
            check_lib.checkExact("name .bss");
        }
        test_step.dependOn(&check_lib.step);
    }

    // verify zero'd declaration is stored in bss for all optimization modes.
    {
        const lib = b.addExecutable(.{
            .name = "lib",
            .root_source_file = .{ .path = "lib2.zig" },
            .target = .{ .cpu_arch = .wasm32, .os_tag = .freestanding },
            .optimize = optimize_mode,
        });
        lib.entry = .disabled;
        lib.use_llvm = false;
        lib.use_lld = false;
        lib.strip = false;
        // to make sure the bss segment is emitted, we must import memory
        lib.import_memory = true;
        lib.link_gc_sections = false;

        const check_lib = lib.checkObject();
        check_lib.checkStart();
        check_lib.checkExact("Section custom");
        check_lib.checkExact("type data_segment");
        check_lib.checkExact("names 2");
        check_lib.checkExact("index 0");
        check_lib.checkExact("name .rodata");
        check_lib.checkExact("index 1");
        check_lib.checkExact("name .bss");

        test_step.dependOn(&check_lib.step);
    }
}