Skip to content

Commit c890ba5

Browse files
committed
Add @min_optlevel to set a min-optlevel for a module
1 parent f91436e commit c890ba5

10 files changed

+66
-0
lines changed

NEWS.md

+3
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,9 @@ New language features
2323
- actual running time for the task (`Base.Experimental.task_running_time_ns`), and
2424
- wall-time for the task (`Base.Experimental.task_wall_time_ns`).
2525
- Support for Unicode 16 ([#56925]).
26+
* `Base.Experimental.@min_optlevel level` sets a minimum optlevel (`--min_optlevel`) for a
27+
module, ensuring that level of optimization even if the process is run at a lower optlevel
28+
(`-O`) ([#]).
2629

2730
Language changes
2831
----------------

base/experimental.jl

+21
Original file line numberDiff line numberDiff line change
@@ -124,6 +124,27 @@ macro optlevel(n::Int)
124124
return Expr(:meta, :optlevel, n)
125125
end
126126

127+
"""
128+
Experimental.@min_optlevel n::Int
129+
130+
Set the minimum optimization level (equivalent to the `--min-optlevel` command line
131+
argument) for code in the current module. Submodules inherit the setting of their
132+
parent module.
133+
134+
Supported values are 0, 1, 2, and 3.
135+
136+
This sets a lower-bound for optimization level, such that the effective optimization
137+
level, `o` is `@min_optlevel <= o <= @optlevel` and `--min-optlevel <= o <= -O`.
138+
139+
NOTE: This min optimization level will only be applied to LLVM functions that are compiled
140+
from this module. If a function defined in this module is *inlined* into a function defined
141+
outside the module, that function will not inherit this min optimization level. Consider
142+
whether you need to introduce `@noinline` if you require users to observe your min optlevel.
143+
"""
144+
macro min_optlevel(n::Int)
145+
return Expr(:meta, :min_optlevel, n)
146+
end
147+
127148
"""
128149
Experimental.@max_methods n::Int
129150

src/ast.c

+2
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,7 @@ JL_DLLEXPORT jl_sym_t *jl_escape_sym;
100100
JL_DLLEXPORT jl_sym_t *jl_aliasscope_sym;
101101
JL_DLLEXPORT jl_sym_t *jl_popaliasscope_sym;
102102
JL_DLLEXPORT jl_sym_t *jl_optlevel_sym;
103+
JL_DLLEXPORT jl_sym_t *jl_min_optlevel_sym;
103104
JL_DLLEXPORT jl_sym_t *jl_thismodule_sym;
104105
JL_DLLEXPORT jl_sym_t *jl_eval_sym;
105106
JL_DLLEXPORT jl_sym_t *jl_include_sym;
@@ -392,6 +393,7 @@ void jl_init_common_symbols(void)
392393
jl_specialize_sym = jl_symbol("specialize");
393394
jl_nospecializeinfer_sym = jl_symbol("nospecializeinfer");
394395
jl_optlevel_sym = jl_symbol("optlevel");
396+
jl_min_optlevel_sym = jl_symbol("min_optlevel");
395397
jl_compile_sym = jl_symbol("compile");
396398
jl_force_compile_sym = jl_symbol("force_compile");
397399
jl_infer_sym = jl_symbol("infer");

src/codegen.cpp

+5
Original file line numberDiff line numberDiff line change
@@ -8673,6 +8673,11 @@ static jl_llvm_functions_t
86738673
static const char* const optLevelStrings[] = { "0", "1", "2", "3" };
86748674
FnAttrs.addAttribute("julia-optimization-level", optLevelStrings[optlevel]);
86758675
}
8676+
int min_optlevel = jl_get_module_min_optlevel(ctx.module);
8677+
if (min_optlevel >= 0 && min_optlevel <= 3) {
8678+
static const char* const optLevelStrings[] = { "0", "1", "2", "3" };
8679+
FnAttrs.addAttribute("julia-min-optimization-level", optLevelStrings[min_optlevel]);
8680+
}
86768681

86778682
ctx.f = f;
86788683

src/interpreter.c

+6
Original file line numberDiff line numberDiff line change
@@ -659,6 +659,12 @@ static jl_value_t *eval_body(jl_array_t *stmts, interpreter_state *s, size_t ip,
659659
jl_set_module_optlevel(s->module, n);
660660
}
661661
}
662+
else if (jl_exprarg(stmt, 0) == (jl_value_t*)jl_min_optlevel_sym) {
663+
if (jl_is_long(jl_exprarg(stmt, 1))) {
664+
int n = jl_unbox_long(jl_exprarg(stmt, 1));
665+
jl_set_module_min_optlevel(s->module, n);
666+
}
667+
}
662668
else if (jl_exprarg(stmt, 0) == (jl_value_t*)jl_compile_sym) {
663669
if (jl_is_long(jl_exprarg(stmt, 1))) {
664670
jl_set_module_compile(s->module, jl_unbox_long(jl_exprarg(stmt, 1)));

src/jitlayers.cpp

+8
Original file line numberDiff line numberDiff line change
@@ -1049,6 +1049,14 @@ static orc::ThreadSafeModule selectOptLevel(orc::ThreadSafeModule TSM) JL_NOTSAF
10491049
if (ol < opt_level)
10501050
opt_level = ol;
10511051
}
1052+
attr = F.getFnAttribute("julia-min-optimization-level");
1053+
val = attr.getValueAsString();
1054+
if (val != "") {
1055+
size_t ol = (size_t)val[0] - '0';
1056+
if (ol > opt_level)
1057+
opt_level = ol;
1058+
jl_safe_printf("Function %s opt: %d\n", F.getName().str().c_str(), (int)opt_level);
1059+
}
10521060
}
10531061
}
10541062
if (opt_level < opt_level_min)

src/jl_exported_funcs.inc

+2
Original file line numberDiff line numberDiff line change
@@ -213,6 +213,7 @@
213213
XX(jl_get_module_infer) \
214214
XX(jl_get_module_of_binding) \
215215
XX(jl_get_module_optlevel) \
216+
XX(jl_get_module_min_optlevel) \
216217
XX(jl_get_next_task) \
217218
XX(jl_get_nth_field) \
218219
XX(jl_get_nth_field_checked) \
@@ -411,6 +412,7 @@
411412
XX(jl_set_module_infer) \
412413
XX(jl_set_module_nospecialize) \
413414
XX(jl_set_module_optlevel) \
415+
XX(jl_set_module_min_optlevel) \
414416
XX(jl_set_module_uuid) \
415417
XX(jl_set_next_task) \
416418
XX(jl_set_nth_field) \

src/julia.h

+3
Original file line numberDiff line numberDiff line change
@@ -724,6 +724,7 @@ typedef struct _jl_module_t {
724724
_Atomic(uint32_t) counter;
725725
int32_t nospecialize; // global bit flags: initialization for new methods
726726
int8_t optlevel;
727+
int8_t min_optlevel;
727728
int8_t compile;
728729
int8_t infer;
729730
uint8_t istopmod;
@@ -1989,7 +1990,9 @@ extern JL_DLLIMPORT jl_module_t *jl_libdl_module JL_GLOBALLY_ROOTED;
19891990
JL_DLLEXPORT jl_module_t *jl_new_module(jl_sym_t *name, jl_module_t *parent);
19901991
JL_DLLEXPORT void jl_set_module_nospecialize(jl_module_t *self, int on);
19911992
JL_DLLEXPORT void jl_set_module_optlevel(jl_module_t *self, int lvl);
1993+
JL_DLLEXPORT void jl_set_module_min_optlevel(jl_module_t *self, int lvl);
19921994
JL_DLLEXPORT int jl_get_module_optlevel(jl_module_t *m);
1995+
JL_DLLEXPORT int jl_get_module_min_optlevel(jl_module_t *m);
19931996
JL_DLLEXPORT void jl_set_module_compile(jl_module_t *self, int value);
19941997
JL_DLLEXPORT int jl_get_module_compile(jl_module_t *m);
19951998
JL_DLLEXPORT void jl_set_module_infer(jl_module_t *self, int value);

src/julia_internal.h

+1
Original file line numberDiff line numberDiff line change
@@ -1867,6 +1867,7 @@ extern JL_DLLEXPORT jl_sym_t *jl_escape_sym;
18671867
extern JL_DLLEXPORT jl_sym_t *jl_aliasscope_sym;
18681868
extern JL_DLLEXPORT jl_sym_t *jl_popaliasscope_sym;
18691869
extern JL_DLLEXPORT jl_sym_t *jl_optlevel_sym;
1870+
extern JL_DLLEXPORT jl_sym_t *jl_min_optlevel_sym;
18701871
extern JL_DLLEXPORT jl_sym_t *jl_thismodule_sym;
18711872
extern JL_DLLEXPORT jl_sym_t *jl_eval_sym;
18721873
extern JL_DLLEXPORT jl_sym_t *jl_include_sym;

src/module.c

+15
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,7 @@ JL_DLLEXPORT jl_module_t *jl_new_module_(jl_sym_t *name, jl_module_t *parent, ui
9191
jl_atomic_store_relaxed(&m->counter, 1);
9292
m->nospecialize = 0;
9393
m->optlevel = -1;
94+
m->min_optlevel = -1;
9495
m->compile = -1;
9596
m->infer = -1;
9697
m->max_methods = -1;
@@ -145,6 +146,10 @@ JL_DLLEXPORT void jl_set_module_optlevel(jl_module_t *self, int lvl)
145146
{
146147
self->optlevel = lvl;
147148
}
149+
JL_DLLEXPORT void jl_set_module_min_optlevel(jl_module_t *self, int lvl)
150+
{
151+
self->min_optlevel = lvl;
152+
}
148153

149154
JL_DLLEXPORT int jl_get_module_optlevel(jl_module_t *m)
150155
{
@@ -155,6 +160,16 @@ JL_DLLEXPORT int jl_get_module_optlevel(jl_module_t *m)
155160
}
156161
return lvl;
157162
}
163+
JL_DLLEXPORT int jl_get_module_min_optlevel(jl_module_t *m)
164+
{
165+
int lvl = m->min_optlevel;
166+
while (lvl == -1 && m->parent != m && m != jl_base_module) {
167+
m = m->parent;
168+
lvl = m->min_optlevel;
169+
}
170+
return lvl;
171+
}
172+
158173

159174
JL_DLLEXPORT void jl_set_module_compile(jl_module_t *self, int value)
160175
{

0 commit comments

Comments
 (0)