Skip to content

Commit 6c80e54

Browse files
authored
Merge pull request #33864 from JuliaLang/jb/softscope
bring back v0.6 scope rules in the REPL
2 parents b37d094 + 95bfb9a commit 6c80e54

File tree

12 files changed

+675
-247
lines changed

12 files changed

+675
-247
lines changed

NEWS.md

+13-1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,19 @@ New language features
88
Language changes
99
----------------
1010

11+
* The interactive REPL now uses "soft scope" for top-level expressions: an assignment inside a
12+
scope block such as a `for` loop automatically assigns to a global variable if one has been
13+
defined already. This matches the behavior of Julia versions 0.6 and prior, as well as
14+
[IJulia](https://github.com/JuliaLang/IJulia.jl).
15+
Note that this only affects expressions interactively typed or pasted directly into the
16+
default REPL ([#28789], [#33864]).
17+
18+
* Outside of the REPL (e.g. in a file), assigning to a variable within a top-level scope
19+
block is considered ambiguous if a global variable with the same name exists.
20+
A warning is given if that happens, to alert you that the code will work differently
21+
than in the REPL.
22+
A new command line option `--warn-scope` controls this warning ([#33864]).
23+
1124
* Converting arbitrary tuples to `NTuple`, e.g. `convert(NTuple, (1, ""))` now gives an error,
1225
where it used to be incorrectly allowed. This is because `NTuple` refers only to homogeneous
1326
tuples (this meaning has not changed) ([#34272]).
@@ -19,7 +32,6 @@ Language changes
1932
(in addition to the one that enters the help mode) to see the full
2033
docstring. ([#25930])
2134

22-
2335
Multi-threading changes
2436
-----------------------
2537

base/options.jl

+1
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ struct JLOptions
4141
outputji::Ptr{UInt8}
4242
output_code_coverage::Ptr{UInt8}
4343
incremental::Int8
44+
warn_scope::Int8
4445
end
4546

4647
# This runs early in the sysimage != is not defined yet

doc/src/manual/variables-and-scoping.md

+331-206
Large diffs are not rendered by default.

src/ast.c

+54-6
Original file line numberDiff line numberDiff line change
@@ -131,13 +131,34 @@ static jl_value_t *scm_to_julia(fl_context_t *fl_ctx, value_t e, jl_module_t *mo
131131
static value_t julia_to_scm(fl_context_t *fl_ctx, jl_value_t *v);
132132
static jl_value_t *jl_expand_macros(jl_value_t *expr, jl_module_t *inmodule, struct macroctx_stack *macroctx, int onelevel);
133133

134+
value_t fl_defined_julia_global(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
135+
{
136+
// tells whether a var is defined in and *by* the current module
137+
argcount(fl_ctx, "defined-julia-global", nargs, 1);
138+
(void)tosymbol(fl_ctx, args[0], "defined-julia-global");
139+
jl_ast_context_t *ctx = jl_ast_ctx(fl_ctx);
140+
jl_sym_t *var = jl_symbol(symbol_name(fl_ctx, args[0]));
141+
jl_binding_t *b = jl_get_module_binding(ctx->module, var);
142+
return (b != NULL && b->owner == ctx->module) ? fl_ctx->T : fl_ctx->F;
143+
}
144+
134145
value_t fl_current_module_counter(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
135146
{
136147
jl_ast_context_t *ctx = jl_ast_ctx(fl_ctx);
137148
assert(ctx->module);
138149
return fixnum(jl_module_next_counter(ctx->module));
139150
}
140151

152+
value_t fl_julia_current_file(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
153+
{
154+
return symbol(fl_ctx, jl_filename);
155+
}
156+
157+
value_t fl_julia_current_line(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
158+
{
159+
return fixnum(jl_lineno);
160+
}
161+
141162
// Check whether v is a scalar for purposes of inlining fused-broadcast
142163
// arguments when lowering; should agree with broadcast.jl on what is a
143164
// scalar. When in doubt, return false, since this is only an optimization.
@@ -197,9 +218,12 @@ value_t fl_julia_logmsg(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
197218
}
198219

199220
static const builtinspec_t julia_flisp_ast_ext[] = {
221+
{ "defined-julia-global", fl_defined_julia_global },
200222
{ "current-julia-module-counter", fl_current_module_counter },
201223
{ "julia-scalar?", fl_julia_scalar },
202224
{ "julia-logmsg", fl_julia_logmsg },
225+
{ "julia-current-file", fl_julia_current_file },
226+
{ "julia-current-line", fl_julia_current_line },
203227
{ NULL, NULL }
204228
};
205229

@@ -227,6 +251,7 @@ static void jl_init_ast_ctx(jl_ast_context_t *ast_ctx) JL_NOTSAFEPOINT
227251
ctx->task = NULL;
228252
ctx->module = NULL;
229253
set(symbol(fl_ctx, "*depwarn-opt*"), fixnum(jl_options.depwarn));
254+
set(symbol(fl_ctx, "*scopewarn-opt*"), fixnum(jl_options.warn_scope));
230255
}
231256

232257
// There should be no GC allocation while holding this lock
@@ -859,9 +884,10 @@ jl_value_t *jl_parse_eval_all(const char *fname,
859884
}
860885
// expand non-final expressions in statement position (value unused)
861886
expression =
862-
fl_applyn(fl_ctx, 3,
863-
symbol_value(symbol(fl_ctx, iscons(cdr_(ast)) ? "jl-expand-to-thunk-stmt" : "jl-expand-to-thunk")),
864-
expression, symbol(fl_ctx, jl_filename), fixnum(jl_lineno));
887+
fl_applyn(fl_ctx, 4,
888+
symbol_value(symbol(fl_ctx, "jl-expand-to-thunk-warn")),
889+
expression, symbol(fl_ctx, jl_filename), fixnum(jl_lineno),
890+
iscons(cdr_(ast)) ? fl_ctx->T : fl_ctx->F);
865891
}
866892
jl_get_ptls_states()->world_age = jl_world_counter;
867893
form = scm_to_julia(fl_ctx, expression, inmodule);
@@ -1171,6 +1197,13 @@ JL_DLLEXPORT jl_value_t *jl_macroexpand1(jl_value_t *expr, jl_module_t *inmodule
11711197
return expr;
11721198
}
11731199

1200+
// Lower an expression tree into Julia's intermediate-representation.
1201+
JL_DLLEXPORT jl_value_t *jl_expand(jl_value_t *expr, jl_module_t *inmodule)
1202+
{
1203+
return jl_expand_with_loc(expr, inmodule, "none", 0);
1204+
}
1205+
1206+
// Lowering, with starting program location specified
11741207
JL_DLLEXPORT jl_value_t *jl_expand_with_loc(jl_value_t *expr, jl_module_t *inmodule,
11751208
const char *file, int line)
11761209
{
@@ -1183,10 +1216,25 @@ JL_DLLEXPORT jl_value_t *jl_expand_with_loc(jl_value_t *expr, jl_module_t *inmod
11831216
return expr;
11841217
}
11851218

1186-
// Lower an expression tree into Julia's intermediate-representation.
1187-
JL_DLLEXPORT jl_value_t *jl_expand(jl_value_t *expr, jl_module_t *inmodule)
1219+
// Same as the above, but printing warnings when applicable
1220+
JL_DLLEXPORT jl_value_t *jl_expand_with_loc_warn(jl_value_t *expr, jl_module_t *inmodule,
1221+
const char *file, int line)
11881222
{
1189-
return jl_expand_with_loc(expr, inmodule, "none", 0);
1223+
JL_TIMING(LOWERING);
1224+
JL_GC_PUSH1(&expr);
1225+
expr = jl_copy_ast(expr);
1226+
expr = jl_expand_macros(expr, inmodule, NULL, 0);
1227+
jl_ast_context_t *ctx = jl_ast_ctx_enter();
1228+
fl_context_t *fl_ctx = &ctx->fl;
1229+
JL_AST_PRESERVE_PUSH(ctx, old_roots, inmodule);
1230+
value_t arg = julia_to_scm(fl_ctx, expr);
1231+
value_t e = fl_applyn(fl_ctx, 4, symbol_value(symbol(fl_ctx, "jl-expand-to-thunk-warn")), arg,
1232+
symbol(fl_ctx, file), fixnum(line), fl_ctx->F);
1233+
expr = scm_to_julia(fl_ctx, e, inmodule);
1234+
JL_AST_PRESERVE_POP(ctx, old_roots);
1235+
jl_ast_ctx_leave(ctx);
1236+
JL_GC_POP();
1237+
return expr;
11901238
}
11911239

11921240
// expand in a context where the expression value is unused

src/jlfrontend.scm

+35-6
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,11 @@
2929
'(error "malformed expression"))))
3030
thk))
3131

32+
;; this is overwritten when we run in actual julia
33+
(define (defined-julia-global v) #f)
34+
(define (julia-current-file) 'none)
35+
(define (julia-current-line) 0)
36+
3237
;; parser entry points
3338

3439
;; parse one expression (if greedy) or atom, returning end position
@@ -128,16 +133,38 @@
128133
(begin0 (expand-toplevel-expr-- e file line)
129134
(set! *in-expand* last))))))
130135

131-
; expand a piece of raw surface syntax to an executable thunk
132-
(define (jl-expand-to-thunk expr file line)
136+
;; used to collect warnings during lowering, which are usually discarded
137+
;; unless logging is requested
138+
(define lowering-warning (lambda lst (void)))
139+
140+
;; expand a piece of raw surface syntax to an executable thunk
141+
142+
(define (expand-to-thunk- expr file line)
133143
(error-wrap (lambda ()
134144
(expand-toplevel-expr expr file line))))
135145

146+
(define (expand-to-thunk-stmt- expr file line)
147+
(expand-to-thunk- (if (toplevel-only-expr? expr)
148+
expr
149+
`(block ,expr (null)))
150+
file line))
151+
152+
(define (jl-expand-to-thunk-warn expr file line stmt)
153+
(let ((warnings '()))
154+
(with-bindings
155+
((lowering-warning (lambda lst (set! warnings (cons lst warnings)))))
156+
(begin0
157+
(if stmt
158+
(expand-to-thunk-stmt- expr file line)
159+
(expand-to-thunk- expr file line))
160+
(for-each (lambda (args) (apply julia-logmsg args))
161+
(reverse warnings))))))
162+
163+
(define (jl-expand-to-thunk expr file line)
164+
(expand-to-thunk- expr file line))
165+
136166
(define (jl-expand-to-thunk-stmt expr file line)
137-
(jl-expand-to-thunk (if (toplevel-only-expr? expr)
138-
expr
139-
`(block ,expr (null)))
140-
file line))
167+
(expand-to-thunk-stmt- expr file line))
141168

142169
(define (jl-expand-macroscope expr)
143170
(error-wrap (lambda ()
@@ -215,6 +242,8 @@
215242
(if (equal? instead "") ""
216243
(string #\newline "Use `" instead "` instead."))))
217244

245+
(define *scopewarn-opt* 1)
246+
218247
; Corresponds to --depwarn 0="no", 1="yes", 2="error"
219248
(define *depwarn-opt* 1)
220249

src/jloptions.c

+15-3
Original file line numberDiff line numberDiff line change
@@ -71,7 +71,8 @@ jl_options_t jl_options = { 0, // quiet
7171
NULL, // output-ji
7272
NULL, // output-code_coverage
7373
0, // incremental
74-
0 // image_file_specified
74+
0, // image_file_specified
75+
JL_OPTIONS_WARN_SCOPE_ON // ambiguous scope warning
7576
};
7677

7778
static const char usage[] = "julia [switches] -- [programfile] [args...]\n";
@@ -109,7 +110,8 @@ static const char opts[] =
109110

110111
// error and warning options
111112
" --depwarn={yes|no|error} Enable or disable syntax and method deprecation warnings (\"error\" turns warnings into errors)\n"
112-
" --warn-overwrite={yes|no} Enable or disable method overwrite warnings\n\n"
113+
" --warn-overwrite={yes|no} Enable or disable method overwrite warnings\n"
114+
" --warn-scope={yes|no} Enable or disable warning for ambiguous top-level scope\n\n"
113115

114116
// code generation options
115117
" -C, --cpu-target <target> Limit usage of CPU features up to <target>; set to \"help\" to see the available options\n"
@@ -169,6 +171,7 @@ JL_DLLEXPORT void jl_parse_opts(int *argcp, char ***argvp)
169171
opt_output_bc,
170172
opt_depwarn,
171173
opt_warn_overwrite,
174+
opt_warn_scope,
172175
opt_inline,
173176
opt_polly,
174177
opt_trace_compile,
@@ -225,6 +228,7 @@ JL_DLLEXPORT void jl_parse_opts(int *argcp, char ***argvp)
225228
{ "output-incremental",required_argument, 0, opt_incremental },
226229
{ "depwarn", required_argument, 0, opt_depwarn },
227230
{ "warn-overwrite", required_argument, 0, opt_warn_overwrite },
231+
{ "warn-scope", required_argument, 0, opt_warn_scope },
228232
{ "inline", required_argument, 0, opt_inline },
229233
{ "polly", required_argument, 0, opt_polly },
230234
{ "trace-compile", required_argument, 0, opt_trace_compile },
@@ -554,7 +558,15 @@ JL_DLLEXPORT void jl_parse_opts(int *argcp, char ***argvp)
554558
else if (!strcmp(optarg,"no"))
555559
jl_options.warn_overwrite = JL_OPTIONS_WARN_OVERWRITE_OFF;
556560
else
557-
jl_errorf("julia: invalid argument to --warn-overwrite={yes|no|} (%s)", optarg);
561+
jl_errorf("julia: invalid argument to --warn-overwrite={yes|no} (%s)", optarg);
562+
break;
563+
case opt_warn_scope:
564+
if (!strcmp(optarg,"yes"))
565+
jl_options.warn_scope = JL_OPTIONS_WARN_SCOPE_ON;
566+
else if (!strcmp(optarg,"no"))
567+
jl_options.warn_scope = JL_OPTIONS_WARN_SCOPE_OFF;
568+
else
569+
jl_errorf("julia: invalid argument to --warn-scope={yes|no} (%s)", optarg);
558570
break;
559571
case opt_inline:
560572
if (!strcmp(optarg,"yes"))

0 commit comments

Comments
 (0)