diff --git a/src/core/specials.c b/src/core/specials.c index 9ccf8879..55ae14b0 100644 --- a/src/core/specials.c +++ b/src/core/specials.c @@ -154,6 +154,45 @@ static int destructure(JanetCompiler *c, for (int32_t i = 0; i < len; i++) { JanetSlot nextright = janetc_farslot(c); Janet subval = values[i]; + + if (!janet_cstrcmp(janet_unwrap_symbol(subval), "&")) { + JanetSlot argi = janetc_farslot(c); + JanetSlot arg = janetc_farslot(c); + JanetSlot len = janetc_farslot(c); + + janetc_emit_si(c, JOP_LOAD_INTEGER, argi, i, 0); + janetc_emit_ss(c, JOP_LENGTH, len, right, 0); + + // loop condition + // reuse arg slot for the condition result + int32_t label_loop_start = janetc_emit_sss(c, JOP_EQUALS, arg, argi, len, 0); + int32_t label_loop_cond_jump = janetc_emit_si(c, JOP_JUMP_IF, arg, 0, 0); + + // loop body + janetc_emit_sss(c, JOP_GET, arg, right, argi, 0); + janetc_emit_s(c, JOP_PUSH, arg, 0); + janetc_emit_ssi(c, JOP_ADD_IMMEDIATE, argi, argi, 1, 0); + + // loop + // jump back to the start of the loop + int32_t label_loop_loop = janet_v_count(c->buffer); + janetc_emit(c, JOP_JUMP); + int32_t label_loop_exit = janet_v_count(c->buffer); + + c->buffer[label_loop_cond_jump] |= (label_loop_exit - label_loop_cond_jump) << 16; + c->buffer[label_loop_loop] |= (label_loop_start - label_loop_loop) << 8; + + janetc_freeslot(c, argi); + janetc_freeslot(c, arg); + janetc_freeslot(c, len); + + janetc_emit_s(c, JOP_MAKE_TUPLE, nextright, 1); + + leaf(c, janet_unwrap_symbol(values[i + 1]), nextright, attr); + janetc_freeslot(c, nextright); + break; + } + if (i < 0x100) { janetc_emit_ssu(c, JOP_GET_INDEX, nextright, right, (uint8_t) i, 1); } else { diff --git a/test/suite0001.janet b/test/suite0001.janet index 958449ce..3fc0888f 100644 --- a/test/suite0001.janet +++ b/test/suite0001.janet @@ -137,6 +137,21 @@ (assert (= a 1) "dictionary destructuring 3") (assert (= b 2) "dictionary destructuring 4") (assert (= c 4) "dictionary destructuring 5 - expression as key")) +(let [test-tuple [:a :b 1 2]] + (def [a b one two] test-tuple) + (assert (= a :a) "tuple destructuring 1") + (assert (= b :b) "tuple destructuring 2") + (assert (= two 2) "tuple destructuring 3")) +(let [test-tuple [:a :b 1 2]] + (def [a & rest] test-tuple) + (assert (= a :a) "tuple destructuring 4 - rest") + (assert (= rest [:b 1 2]) "tuple destructuring 5 - rest")) +(do + (def [a b & rest] [:a :b nil :d]) + (assert (= a :a) "tuple destructuring 6 - rest") + (assert (= b :b) "tuple destructuring 7 - rest") + (pp rest) + (assert (= rest [nil :d]) "tuple destructuring 8 - rest")) # Marshal