Skip to content

Commit dd85daa

Browse files
committed
Merge pull request #7430 from JuliaLang/mn/nestedforbreak
fix #5154 (nested for loop syntax & break behavior)
2 parents a40bc88 + 1dc075b commit dd85daa

File tree

5 files changed

+53
-23
lines changed

5 files changed

+53
-23
lines changed

NEWS.md

+2
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,8 @@ New language features
5555

5656
* Improved reporting of syntax errors ([#6179])
5757

58+
* `break` inside a `for` loop with multiple ranges now exits the entire loop nest ([#5154])
59+
5860
REPL improvements
5961
-----------------
6062

doc/manual/control-flow.rst

+3
Original file line numberDiff line numberDiff line change
@@ -514,6 +514,9 @@ forming the cartesian product of its iterables:
514514
(2,3)
515515
(2,4)
516516

517+
A ``break`` statement inside such a loop exits the entire nest of loops,
518+
not just the inner one.
519+
517520
.. _man-exception-handling:
518521

519522
Exception Handling

src/julia-parser.scm

+3-4
Original file line numberDiff line numberDiff line change
@@ -1016,10 +1016,9 @@
10161016
(let* ((ranges (parse-comma-separated-iters s))
10171017
(body (parse-block s)))
10181018
(expect-end s)
1019-
(let nest ((r ranges))
1020-
(if (null? r)
1021-
body
1022-
`(for ,(car r) ,(nest (cdr r)))))))
1019+
`(for ,(if (length= ranges 1) (car ranges) (cons 'block ranges))
1020+
,body)))
1021+
10231022
((if)
10241023
(let* ((test (parse-cond s))
10251024
(then (if (memq (require-token s) '(else elseif))

src/julia-syntax.scm

+35-19
Original file line numberDiff line numberDiff line change
@@ -1494,6 +1494,24 @@
14941494
(cadr e)
14951495
(caddr e))))
14961496

1497+
(define (expand-for while lhs X body)
1498+
;; (for (= lhs X) body)
1499+
(let ((coll (gensy))
1500+
(state (gensy)))
1501+
`(scope-block
1502+
(block (= ,coll ,(expand-forms X))
1503+
(= ,state (call (top start) ,coll))
1504+
,(expand-forms
1505+
`(,while
1506+
(call (top !) (call (top done) ,coll ,state))
1507+
(scope-block
1508+
(block
1509+
;; NOTE: enable this to force loop-local var
1510+
#;,@(map (lambda (v) `(local ,v)) (lhs-vars lhs))
1511+
,(lower-tuple-assignment (list lhs state)
1512+
`(call (top next) ,coll ,state))
1513+
,body))))))))
1514+
14971515
(define (map-expand-forms e) (map expand-forms e))
14981516

14991517
(define (expand-forms e)
@@ -1793,6 +1811,13 @@
17931811
(break-block loop-cont
17941812
,(expand-forms (caddr e)))))))
17951813

1814+
'inner-while
1815+
(lambda (e)
1816+
`(scope-block
1817+
(_while ,(expand-forms (cadr e))
1818+
(break-block loop-cont
1819+
,(expand-forms (caddr e))))))
1820+
17961821
'break
17971822
(lambda (e)
17981823
(if (pair? (cdr e))
@@ -1803,25 +1828,16 @@
18031828

18041829
'for
18051830
(lambda (e)
1806-
(let ((X (caddr (cadr e)))
1807-
(lhs (cadr (cadr e)))
1808-
(body (caddr e)))
1809-
;; (for (= lhs X) body)
1810-
(let ((coll (gensy))
1811-
(state (gensy)))
1812-
`(scope-block
1813-
(block (= ,coll ,(expand-forms X))
1814-
(= ,state (call (top start) ,coll))
1815-
,(expand-forms
1816-
`(while
1817-
(call (top !) (call (top done) ,coll ,state))
1818-
(scope-block
1819-
(block
1820-
;; NOTE: enable this to force loop-local var
1821-
#;,@(map (lambda (v) `(local ,v)) (lhs-vars lhs))
1822-
,(lower-tuple-assignment (list lhs state)
1823-
`(call (top next) ,coll ,state))
1824-
,body)))))))))
1831+
(let nest ((ranges (if (eq? (car (cadr e)) 'block)
1832+
(cdr (cadr e))
1833+
(list (cadr e))))
1834+
(first #t))
1835+
(expand-for (if first 'while 'inner-while)
1836+
(cadr (car ranges))
1837+
(caddr (car ranges))
1838+
(if (null? (cdr ranges))
1839+
(caddr e) ;; body
1840+
(nest (cdr ranges) #f)))))
18251841

18261842
'+= lower-update-op
18271843
'-= lower-update-op

test/core.jl

+10
Original file line numberDiff line numberDiff line change
@@ -1772,3 +1772,13 @@ macro let_with_uninit()
17721772
end
17731773

17741774
@test @let_with_uninit() == 2
1775+
1776+
# issue #5154
1777+
let
1778+
v = {}
1779+
for i=1:3, j=1:3
1780+
push!(v, (i, j))
1781+
i == 1 && j == 2 && break
1782+
end
1783+
@test v == {(1,1), (1,2)}
1784+
end

0 commit comments

Comments
 (0)