├── .github
└── workflows
│ └── rust.yaml
├── .gitignore
├── Cargo.lock
├── Cargo.toml
├── LICENSE-APACHE
├── LICENSE-MIT
├── README.md
├── examples
├── in_fn.rs
├── in_mod.rs
├── on_block.rs
├── on_fn.rs
├── on_mod.rs
└── on_statement.rs
├── macros
├── Cargo.toml
├── build.rs
└── lib.rs
├── rust-toolchain.toml
├── rustfmt.toml
├── src
├── borrow_unchecked.md
├── lib.rs
└── turn_off_the_borrow_checker.md
└── tests
├── compile_tests.rs
├── stable-fail
└── not_on_fn.rs
├── stable-pass
├── on_fn.rs
├── on_fn.stderr
├── on_fn_empty.rs
├── on_fn_empty.stderr
├── on_mod.rs
└── on_mod.stderr
├── unstable-fail
└── not_on_fn.rs
└── unstable-pass
├── in_fn.rs
├── in_fn.stderr
├── in_mod.rs
├── in_mod.stderr
├── on_block.rs
├── on_block.stderr
├── on_fn.rs
├── on_fn.stderr
├── on_fn_empty.rs
├── on_fn_empty.stderr
├── on_mod.rs
├── on_mod.stderr
├── on_statement.rs
└── on_statement.stderr
/.github/workflows/rust.yaml:
--------------------------------------------------------------------------------
1 | on: [push, workflow_dispatch, pull_request]
2 | name: rust
3 | jobs:
4 | rust:
5 | strategy:
6 | matrix:
7 | include:
8 | - rust_toolchain: stable
9 | name: stable
10 | tests: --tests
11 | - rust_toolchain: nightly
12 | name: unstable nightly
13 | tests: --all
14 | fail-fast: false
15 | runs-on: ubuntu-latest
16 | name: ${{ matrix.name }}
17 | steps:
18 | - uses: actions/checkout@v1
19 | - uses: actions-rs/toolchain@v1
20 | with:
21 | toolchain: ${{ matrix.rust_toolchain }}
22 | - run: cargo +${{ matrix.rust_toolchain }} build --all
23 | - run: cargo +${{ matrix.rust_toolchain }} clean
24 | - run: cargo +${{ matrix.rust_toolchain }} test ${{ matrix.tests }}
25 | - run: cargo +${{ matrix.rust_toolchain }} doc --all
26 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | /target
2 |
--------------------------------------------------------------------------------
/Cargo.lock:
--------------------------------------------------------------------------------
1 | # This file is automatically @generated by Cargo.
2 | # It is not intended for manual editing.
3 | version = 3
4 |
5 | [[package]]
6 | name = "aho-corasick"
7 | version = "0.7.18"
8 | source = "registry+https://github.com/rust-lang/crates.io-index"
9 | checksum = "1e37cfd5e7657ada45f742d6e99ca5788580b5c529dc78faf11ece6dc702656f"
10 | dependencies = [
11 | "memchr",
12 | ]
13 |
14 | [[package]]
15 | name = "anyhow"
16 | version = "1.0.61"
17 | source = "registry+https://github.com/rust-lang/crates.io-index"
18 | checksum = "508b352bb5c066aac251f6daf6b36eccd03e8a88e8081cd44959ea277a3af9a8"
19 |
20 | [[package]]
21 | name = "bitflags"
22 | version = "1.3.2"
23 | source = "registry+https://github.com/rust-lang/crates.io-index"
24 | checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
25 |
26 | [[package]]
27 | name = "cfg-if"
28 | version = "1.0.0"
29 | source = "registry+https://github.com/rust-lang/crates.io-index"
30 | checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
31 |
32 | [[package]]
33 | name = "compiletest_rs"
34 | version = "0.8.0"
35 | source = "registry+https://github.com/rust-lang/crates.io-index"
36 | checksum = "262134ef87408da1ddfe45e33daa0ca43b75286d6b1076446e602d264cf9847e"
37 | dependencies = [
38 | "diff",
39 | "filetime",
40 | "getopts",
41 | "lazy_static",
42 | "libc",
43 | "log",
44 | "miow",
45 | "regex",
46 | "rustfix",
47 | "serde",
48 | "serde_derive",
49 | "serde_json",
50 | "tempfile",
51 | "tester",
52 | "winapi",
53 | ]
54 |
55 | [[package]]
56 | name = "diff"
57 | version = "0.1.13"
58 | source = "registry+https://github.com/rust-lang/crates.io-index"
59 | checksum = "56254986775e3233ffa9c4d7d3faaf6d36a2c09d30b20687e9f88bc8bafc16c8"
60 |
61 | [[package]]
62 | name = "dirs-next"
63 | version = "2.0.0"
64 | source = "registry+https://github.com/rust-lang/crates.io-index"
65 | checksum = "b98cf8ebf19c3d1b223e151f99a4f9f0690dca41414773390fc824184ac833e1"
66 | dependencies = [
67 | "cfg-if",
68 | "dirs-sys-next",
69 | ]
70 |
71 | [[package]]
72 | name = "dirs-sys-next"
73 | version = "0.1.2"
74 | source = "registry+https://github.com/rust-lang/crates.io-index"
75 | checksum = "4ebda144c4fe02d1f7ea1a7d9641b6fc6b580adcfa024ae48797ecdeb6825b4d"
76 | dependencies = [
77 | "libc",
78 | "redox_users",
79 | "winapi",
80 | ]
81 |
82 | [[package]]
83 | name = "fastrand"
84 | version = "1.8.0"
85 | source = "registry+https://github.com/rust-lang/crates.io-index"
86 | checksum = "a7a407cfaa3385c4ae6b23e84623d48c2798d06e3e6a1878f7f59f17b3f86499"
87 | dependencies = [
88 | "instant",
89 | ]
90 |
91 | [[package]]
92 | name = "filetime"
93 | version = "0.2.17"
94 | source = "registry+https://github.com/rust-lang/crates.io-index"
95 | checksum = "e94a7bbaa59354bc20dd75b67f23e2797b4490e9d6928203fb105c79e448c86c"
96 | dependencies = [
97 | "cfg-if",
98 | "libc",
99 | "redox_syscall",
100 | "windows-sys",
101 | ]
102 |
103 | [[package]]
104 | name = "getopts"
105 | version = "0.2.21"
106 | source = "registry+https://github.com/rust-lang/crates.io-index"
107 | checksum = "14dbbfd5c71d70241ecf9e6f13737f7b5ce823821063188d7e46c41d371eebd5"
108 | dependencies = [
109 | "unicode-width",
110 | ]
111 |
112 | [[package]]
113 | name = "getrandom"
114 | version = "0.2.7"
115 | source = "registry+https://github.com/rust-lang/crates.io-index"
116 | checksum = "4eb1a864a501629691edf6c15a593b7a51eebaa1e8468e9ddc623de7c9b58ec6"
117 | dependencies = [
118 | "cfg-if",
119 | "libc",
120 | "wasi",
121 | ]
122 |
123 | [[package]]
124 | name = "hermit-abi"
125 | version = "0.1.19"
126 | source = "registry+https://github.com/rust-lang/crates.io-index"
127 | checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33"
128 | dependencies = [
129 | "libc",
130 | ]
131 |
132 | [[package]]
133 | name = "instant"
134 | version = "0.1.12"
135 | source = "registry+https://github.com/rust-lang/crates.io-index"
136 | checksum = "7a5bbe824c507c5da5956355e86a746d82e0e1464f65d862cc5e71da70e94b2c"
137 | dependencies = [
138 | "cfg-if",
139 | ]
140 |
141 | [[package]]
142 | name = "itoa"
143 | version = "1.0.3"
144 | source = "registry+https://github.com/rust-lang/crates.io-index"
145 | checksum = "6c8af84674fe1f223a982c933a0ee1086ac4d4052aa0fb8060c12c6ad838e754"
146 |
147 | [[package]]
148 | name = "lazy_static"
149 | version = "1.4.0"
150 | source = "registry+https://github.com/rust-lang/crates.io-index"
151 | checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646"
152 |
153 | [[package]]
154 | name = "libc"
155 | version = "0.2.131"
156 | source = "registry+https://github.com/rust-lang/crates.io-index"
157 | checksum = "04c3b4822ccebfa39c02fc03d1534441b22ead323fa0f48bb7ddd8e6ba076a40"
158 |
159 | [[package]]
160 | name = "log"
161 | version = "0.4.17"
162 | source = "registry+https://github.com/rust-lang/crates.io-index"
163 | checksum = "abb12e687cfb44aa40f41fc3978ef76448f9b6038cad6aef4259d3c095a2382e"
164 | dependencies = [
165 | "cfg-if",
166 | ]
167 |
168 | [[package]]
169 | name = "memchr"
170 | version = "2.5.0"
171 | source = "registry+https://github.com/rust-lang/crates.io-index"
172 | checksum = "2dffe52ecf27772e601905b7522cb4ef790d2cc203488bbd0e2fe85fcb74566d"
173 |
174 | [[package]]
175 | name = "miow"
176 | version = "0.3.7"
177 | source = "registry+https://github.com/rust-lang/crates.io-index"
178 | checksum = "b9f1c5b025cda876f66ef43a113f91ebc9f4ccef34843000e0adf6ebbab84e21"
179 | dependencies = [
180 | "winapi",
181 | ]
182 |
183 | [[package]]
184 | name = "num_cpus"
185 | version = "1.13.1"
186 | source = "registry+https://github.com/rust-lang/crates.io-index"
187 | checksum = "19e64526ebdee182341572e50e9ad03965aa510cd94427a4549448f285e957a1"
188 | dependencies = [
189 | "hermit-abi",
190 | "libc",
191 | ]
192 |
193 | [[package]]
194 | name = "proc-macro2"
195 | version = "1.0.43"
196 | source = "registry+https://github.com/rust-lang/crates.io-index"
197 | checksum = "0a2ca2c61bc9f3d74d2886294ab7b9853abd9c1ad903a3ac7815c58989bb7bab"
198 | dependencies = [
199 | "unicode-ident",
200 | ]
201 |
202 | [[package]]
203 | name = "quote"
204 | version = "1.0.21"
205 | source = "registry+https://github.com/rust-lang/crates.io-index"
206 | checksum = "bbe448f377a7d6961e30f5955f9b8d106c3f5e449d493ee1b125c1d43c2b5179"
207 | dependencies = [
208 | "proc-macro2",
209 | ]
210 |
211 | [[package]]
212 | name = "redox_syscall"
213 | version = "0.2.16"
214 | source = "registry+https://github.com/rust-lang/crates.io-index"
215 | checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a"
216 | dependencies = [
217 | "bitflags",
218 | ]
219 |
220 | [[package]]
221 | name = "redox_users"
222 | version = "0.4.3"
223 | source = "registry+https://github.com/rust-lang/crates.io-index"
224 | checksum = "b033d837a7cf162d7993aded9304e30a83213c648b6e389db233191f891e5c2b"
225 | dependencies = [
226 | "getrandom",
227 | "redox_syscall",
228 | "thiserror",
229 | ]
230 |
231 | [[package]]
232 | name = "regex"
233 | version = "1.6.0"
234 | source = "registry+https://github.com/rust-lang/crates.io-index"
235 | checksum = "4c4eb3267174b8c6c2f654116623910a0fef09c4753f8dd83db29c48a0df988b"
236 | dependencies = [
237 | "aho-corasick",
238 | "memchr",
239 | "regex-syntax",
240 | ]
241 |
242 | [[package]]
243 | name = "regex-syntax"
244 | version = "0.6.27"
245 | source = "registry+https://github.com/rust-lang/crates.io-index"
246 | checksum = "a3f87b73ce11b1619a3c6332f45341e0047173771e8b8b73f87bfeefb7b56244"
247 |
248 | [[package]]
249 | name = "remove_dir_all"
250 | version = "0.5.3"
251 | source = "registry+https://github.com/rust-lang/crates.io-index"
252 | checksum = "3acd125665422973a33ac9d3dd2df85edad0f4ae9b00dafb1a05e43a9f5ef8e7"
253 | dependencies = [
254 | "winapi",
255 | ]
256 |
257 | [[package]]
258 | name = "rustc_version"
259 | version = "0.2.3"
260 | source = "registry+https://github.com/rust-lang/crates.io-index"
261 | checksum = "138e3e0acb6c9fb258b19b67cb8abd63c00679d2851805ea151465464fe9030a"
262 | dependencies = [
263 | "semver 0.9.0",
264 | ]
265 |
266 | [[package]]
267 | name = "rustc_version"
268 | version = "0.4.0"
269 | source = "registry+https://github.com/rust-lang/crates.io-index"
270 | checksum = "bfa0f585226d2e68097d4f95d113b15b83a82e819ab25717ec0590d9584ef366"
271 | dependencies = [
272 | "semver 1.0.13",
273 | ]
274 |
275 | [[package]]
276 | name = "rustfix"
277 | version = "0.6.1"
278 | source = "registry+https://github.com/rust-lang/crates.io-index"
279 | checksum = "ecd2853d9e26988467753bd9912c3a126f642d05d229a4b53f5752ee36c56481"
280 | dependencies = [
281 | "anyhow",
282 | "log",
283 | "serde",
284 | "serde_json",
285 | ]
286 |
287 | [[package]]
288 | name = "rustversion"
289 | version = "1.0.9"
290 | source = "registry+https://github.com/rust-lang/crates.io-index"
291 | checksum = "97477e48b4cf8603ad5f7aaf897467cf42ab4218a38ef76fb14c2d6773a6d6a8"
292 |
293 | [[package]]
294 | name = "ryu"
295 | version = "1.0.11"
296 | source = "registry+https://github.com/rust-lang/crates.io-index"
297 | checksum = "4501abdff3ae82a1c1b477a17252eb69cee9e66eb915c1abaa4f44d873df9f09"
298 |
299 | [[package]]
300 | name = "semver"
301 | version = "0.9.0"
302 | source = "registry+https://github.com/rust-lang/crates.io-index"
303 | checksum = "1d7eb9ef2c18661902cc47e535f9bc51b78acd254da71d375c2f6720d9a40403"
304 | dependencies = [
305 | "semver-parser",
306 | ]
307 |
308 | [[package]]
309 | name = "semver"
310 | version = "1.0.13"
311 | source = "registry+https://github.com/rust-lang/crates.io-index"
312 | checksum = "93f6841e709003d68bb2deee8c343572bf446003ec20a583e76f7b15cebf3711"
313 |
314 | [[package]]
315 | name = "semver-parser"
316 | version = "0.7.0"
317 | source = "registry+https://github.com/rust-lang/crates.io-index"
318 | checksum = "388a1df253eca08550bef6c72392cfe7c30914bf41df5269b68cbd6ff8f570a3"
319 |
320 | [[package]]
321 | name = "serde"
322 | version = "1.0.143"
323 | source = "registry+https://github.com/rust-lang/crates.io-index"
324 | checksum = "53e8e5d5b70924f74ff5c6d64d9a5acd91422117c60f48c4e07855238a254553"
325 | dependencies = [
326 | "serde_derive",
327 | ]
328 |
329 | [[package]]
330 | name = "serde_derive"
331 | version = "1.0.143"
332 | source = "registry+https://github.com/rust-lang/crates.io-index"
333 | checksum = "d3d8e8de557aee63c26b85b947f5e59b690d0454c753f3adeb5cd7835ab88391"
334 | dependencies = [
335 | "proc-macro2",
336 | "quote",
337 | "syn",
338 | ]
339 |
340 | [[package]]
341 | name = "serde_json"
342 | version = "1.0.83"
343 | source = "registry+https://github.com/rust-lang/crates.io-index"
344 | checksum = "38dd04e3c8279e75b31ef29dbdceebfe5ad89f4d0937213c53f7d49d01b3d5a7"
345 | dependencies = [
346 | "itoa",
347 | "ryu",
348 | "serde",
349 | ]
350 |
351 | [[package]]
352 | name = "syn"
353 | version = "1.0.99"
354 | source = "registry+https://github.com/rust-lang/crates.io-index"
355 | checksum = "58dbef6ec655055e20b86b15a8cc6d439cca19b667537ac6a1369572d151ab13"
356 | dependencies = [
357 | "proc-macro2",
358 | "quote",
359 | "unicode-ident",
360 | ]
361 |
362 | [[package]]
363 | name = "tempfile"
364 | version = "3.3.0"
365 | source = "registry+https://github.com/rust-lang/crates.io-index"
366 | checksum = "5cdb1ef4eaeeaddc8fbd371e5017057064af0911902ef36b39801f67cc6d79e4"
367 | dependencies = [
368 | "cfg-if",
369 | "fastrand",
370 | "libc",
371 | "redox_syscall",
372 | "remove_dir_all",
373 | "winapi",
374 | ]
375 |
376 | [[package]]
377 | name = "term"
378 | version = "0.7.0"
379 | source = "registry+https://github.com/rust-lang/crates.io-index"
380 | checksum = "c59df8ac95d96ff9bede18eb7300b0fda5e5d8d90960e76f8e14ae765eedbf1f"
381 | dependencies = [
382 | "dirs-next",
383 | "rustversion",
384 | "winapi",
385 | ]
386 |
387 | [[package]]
388 | name = "tester"
389 | version = "0.9.0"
390 | source = "registry+https://github.com/rust-lang/crates.io-index"
391 | checksum = "0639d10d8f4615f223a57275cf40f9bdb7cfbb806bcb7f7cc56e3beb55a576eb"
392 | dependencies = [
393 | "cfg-if",
394 | "getopts",
395 | "libc",
396 | "num_cpus",
397 | "term",
398 | ]
399 |
400 | [[package]]
401 | name = "thiserror"
402 | version = "1.0.32"
403 | source = "registry+https://github.com/rust-lang/crates.io-index"
404 | checksum = "f5f6586b7f764adc0231f4c79be7b920e766bb2f3e51b3661cdb263828f19994"
405 | dependencies = [
406 | "thiserror-impl",
407 | ]
408 |
409 | [[package]]
410 | name = "thiserror-impl"
411 | version = "1.0.32"
412 | source = "registry+https://github.com/rust-lang/crates.io-index"
413 | checksum = "12bafc5b54507e0149cdf1b145a5d80ab80a90bcd9275df43d4fff68460f6c21"
414 | dependencies = [
415 | "proc-macro2",
416 | "quote",
417 | "syn",
418 | ]
419 |
420 | [[package]]
421 | name = "unicode-ident"
422 | version = "1.0.3"
423 | source = "registry+https://github.com/rust-lang/crates.io-index"
424 | checksum = "c4f5b37a154999a8f3f98cc23a628d850e154479cd94decf3414696e12e31aaf"
425 |
426 | [[package]]
427 | name = "unicode-width"
428 | version = "0.1.9"
429 | source = "registry+https://github.com/rust-lang/crates.io-index"
430 | checksum = "3ed742d4ea2bd1176e236172c8429aaf54486e7ac098db29ffe6529e0ce50973"
431 |
432 | [[package]]
433 | name = "wasi"
434 | version = "0.11.0+wasi-snapshot-preview1"
435 | source = "registry+https://github.com/rust-lang/crates.io-index"
436 | checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423"
437 |
438 | [[package]]
439 | name = "winapi"
440 | version = "0.3.9"
441 | source = "registry+https://github.com/rust-lang/crates.io-index"
442 | checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419"
443 | dependencies = [
444 | "winapi-i686-pc-windows-gnu",
445 | "winapi-x86_64-pc-windows-gnu",
446 | ]
447 |
448 | [[package]]
449 | name = "winapi-i686-pc-windows-gnu"
450 | version = "0.4.0"
451 | source = "registry+https://github.com/rust-lang/crates.io-index"
452 | checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
453 |
454 | [[package]]
455 | name = "winapi-x86_64-pc-windows-gnu"
456 | version = "0.4.0"
457 | source = "registry+https://github.com/rust-lang/crates.io-index"
458 | checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
459 |
460 | [[package]]
461 | name = "windows-sys"
462 | version = "0.36.1"
463 | source = "registry+https://github.com/rust-lang/crates.io-index"
464 | checksum = "ea04155a16a59f9eab786fe12a4a450e75cdb175f9e0d80da1e17db09f55b8d2"
465 | dependencies = [
466 | "windows_aarch64_msvc",
467 | "windows_i686_gnu",
468 | "windows_i686_msvc",
469 | "windows_x86_64_gnu",
470 | "windows_x86_64_msvc",
471 | ]
472 |
473 | [[package]]
474 | name = "windows_aarch64_msvc"
475 | version = "0.36.1"
476 | source = "registry+https://github.com/rust-lang/crates.io-index"
477 | checksum = "9bb8c3fd39ade2d67e9874ac4f3db21f0d710bee00fe7cab16949ec184eeaa47"
478 |
479 | [[package]]
480 | name = "windows_i686_gnu"
481 | version = "0.36.1"
482 | source = "registry+https://github.com/rust-lang/crates.io-index"
483 | checksum = "180e6ccf01daf4c426b846dfc66db1fc518f074baa793aa7d9b9aaeffad6a3b6"
484 |
485 | [[package]]
486 | name = "windows_i686_msvc"
487 | version = "0.36.1"
488 | source = "registry+https://github.com/rust-lang/crates.io-index"
489 | checksum = "e2e7917148b2812d1eeafaeb22a97e4813dfa60a3f8f78ebe204bcc88f12f024"
490 |
491 | [[package]]
492 | name = "windows_x86_64_gnu"
493 | version = "0.36.1"
494 | source = "registry+https://github.com/rust-lang/crates.io-index"
495 | checksum = "4dcd171b8776c41b97521e5da127a2d86ad280114807d0b2ab1e462bc764d9e1"
496 |
497 | [[package]]
498 | name = "windows_x86_64_msvc"
499 | version = "0.36.1"
500 | source = "registry+https://github.com/rust-lang/crates.io-index"
501 | checksum = "c811ca4a8c853ef420abd8592ba53ddbbac90410fab6903b3e79972a631f7680"
502 |
503 | [[package]]
504 | name = "you-can"
505 | version = "0.0.14"
506 | dependencies = [
507 | "compiletest_rs",
508 | "rustc_version 0.4.0",
509 | "you-can-build-macros",
510 | ]
511 |
512 | [[package]]
513 | name = "you-can-build-macros"
514 | version = "0.0.14"
515 | dependencies = [
516 | "quote",
517 | "rustc_version 0.2.3",
518 | "syn",
519 | ]
520 |
--------------------------------------------------------------------------------
/Cargo.toml:
--------------------------------------------------------------------------------
1 | [workspace]
2 | members = [".", "macros"]
3 |
4 | [package]
5 | authors = ["Jeremy Banks <_@jeremy.ca>"]
6 | description = "#[you_can::turn_off_the_borrow_checker]"
7 | edition = "2021"
8 | license = "MIT OR Apache-2.0"
9 | name = "you-can"
10 | repository = "https://github.com/jeremyBanks/you-can"
11 | version = "0.0.14"
12 |
13 | [dependencies]
14 | you-can-build-macros = { path = "./macros", version = "0.0.14" }
15 |
16 | [dev-dependencies]
17 | compiletest_rs = { version = "0.8.0", features=["tmp"] }
18 | rustc_version = "0.4.0"
19 |
20 | [workspace.metadata.release]
21 | allow-branch = ["trunk", "main", "master"]
22 | consolidate-commits = true
23 | consolidate-pushes = true
24 | dev-version = true
25 | dev-version-ext = "dev"
26 | post-release-commit-message = "v{{next_version}}"
27 | pre-release-commit-message = "v{{version}}"
28 | shared-version = true
29 | tag-message = "https://crates.io/crates/{{crate_name}}/{{version}}"
30 | tag-name = "{{version}}"
31 |
--------------------------------------------------------------------------------
/LICENSE-APACHE:
--------------------------------------------------------------------------------
1 | Copyright Jeremy Banks
2 |
3 | Apache License
4 | Version 2.0, January 2004
5 | http://www.apache.org/licenses/
6 |
7 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
8 |
9 | 1. Definitions.
10 |
11 | "License" shall mean the terms and conditions for use, reproduction,
12 | and distribution as defined by Sections 1 through 9 of this document.
13 |
14 | "Licensor" shall mean the copyright owner or entity authorized by
15 | the copyright owner that is granting the License.
16 |
17 | "Legal Entity" shall mean the union of the acting entity and all
18 | other entities that control, are controlled by, or are under common
19 | control with that entity. For the purposes of this definition,
20 | "control" means (i) the power, direct or indirect, to cause the
21 | direction or management of such entity, whether by contract or
22 | otherwise, or (ii) ownership of fifty percent (50%) or more of the
23 | outstanding shares, or (iii) beneficial ownership of such entity.
24 |
25 | "You" (or "Your") shall mean an individual or Legal Entity
26 | exercising permissions granted by this License.
27 |
28 | "Source" form shall mean the preferred form for making modifications,
29 | including but not limited to software source code, documentation
30 | source, and configuration files.
31 |
32 | "Object" form shall mean any form resulting from mechanical
33 | transformation or translation of a Source form, including but
34 | not limited to compiled object code, generated documentation,
35 | and conversions to other media types.
36 |
37 | "Work" shall mean the work of authorship, whether in Source or
38 | Object form, made available under the License, as indicated by a
39 | copyright notice that is included in or attached to the work
40 | (an example is provided in the Appendix below).
41 |
42 | "Derivative Works" shall mean any work, whether in Source or Object
43 | form, that is based on (or derived from) the Work and for which the
44 | editorial revisions, annotations, elaborations, or other modifications
45 | represent, as a whole, an original work of authorship. For the purposes
46 | of this License, Derivative Works shall not include works that remain
47 | separable from, or merely link (or bind by name) to the interfaces of,
48 | the Work and Derivative Works thereof.
49 |
50 | "Contribution" shall mean any work of authorship, including
51 | the original version of the Work and any modifications or additions
52 | to that Work or Derivative Works thereof, that is intentionally
53 | submitted to Licensor for inclusion in the Work by the copyright owner
54 | or by an individual or Legal Entity authorized to submit on behalf of
55 | the copyright owner. For the purposes of this definition, "submitted"
56 | means any form of electronic, verbal, or written communication sent
57 | to the Licensor or its representatives, including but not limited to
58 | communication on electronic mailing lists, source code control systems,
59 | and issue tracking systems that are managed by, or on behalf of, the
60 | Licensor for the purpose of discussing and improving the Work, but
61 | excluding communication that is conspicuously marked or otherwise
62 | designated in writing by the copyright owner as "Not a Contribution."
63 |
64 | "Contributor" shall mean Licensor and any individual or Legal Entity
65 | on behalf of whom a Contribution has been received by Licensor and
66 | subsequently incorporated within the Work.
67 |
68 | 2. Grant of Copyright License. Subject to the terms and conditions of
69 | this License, each Contributor hereby grants to You a perpetual,
70 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable
71 | copyright license to reproduce, prepare Derivative Works of,
72 | publicly display, publicly perform, sublicense, and distribute the
73 | Work and such Derivative Works in Source or Object form.
74 |
75 | 3. Grant of Patent License. Subject to the terms and conditions of
76 | this License, each Contributor hereby grants to You a perpetual,
77 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable
78 | (except as stated in this section) patent license to make, have made,
79 | use, offer to sell, sell, import, and otherwise transfer the Work,
80 | where such license applies only to those patent claims licensable
81 | by such Contributor that are necessarily infringed by their
82 | Contribution(s) alone or by combination of their Contribution(s)
83 | with the Work to which such Contribution(s) was submitted. If You
84 | institute patent litigation against any entity (including a
85 | cross-claim or counterclaim in a lawsuit) alleging that the Work
86 | or a Contribution incorporated within the Work constitutes direct
87 | or contributory patent infringement, then any patent licenses
88 | granted to You under this License for that Work shall terminate
89 | as of the date such litigation is filed.
90 |
91 | 4. Redistribution. You may reproduce and distribute copies of the
92 | Work or Derivative Works thereof in any medium, with or without
93 | modifications, and in Source or Object form, provided that You
94 | meet the following conditions:
95 |
96 | (a) You must give any other recipients of the Work or
97 | Derivative Works a copy of this License; and
98 |
99 | (b) You must cause any modified files to carry prominent notices
100 | stating that You changed the files; and
101 |
102 | (c) You must retain, in the Source form of any Derivative Works
103 | that You distribute, all copyright, patent, trademark, and
104 | attribution notices from the Source form of the Work,
105 | excluding those notices that do not pertain to any part of
106 | the Derivative Works; and
107 |
108 | (d) If the Work includes a "NOTICE" text file as part of its
109 | distribution, then any Derivative Works that You distribute must
110 | include a readable copy of the attribution notices contained
111 | within such NOTICE file, excluding those notices that do not
112 | pertain to any part of the Derivative Works, in at least one
113 | of the following places: within a NOTICE text file distributed
114 | as part of the Derivative Works; within the Source form or
115 | documentation, if provided along with the Derivative Works; or,
116 | within a display generated by the Derivative Works, if and
117 | wherever such third-party notices normally appear. The contents
118 | of the NOTICE file are for informational purposes only and
119 | do not modify the License. You may add Your own attribution
120 | notices within Derivative Works that You distribute, alongside
121 | or as an addendum to the NOTICE text from the Work, provided
122 | that such additional attribution notices cannot be construed
123 | as modifying the License.
124 |
125 | You may add Your own copyright statement to Your modifications and
126 | may provide additional or different license terms and conditions
127 | for use, reproduction, or distribution of Your modifications, or
128 | for any such Derivative Works as a whole, provided Your use,
129 | reproduction, and distribution of the Work otherwise complies with
130 | the conditions stated in this License.
131 |
132 | 5. Submission of Contributions. Unless You explicitly state otherwise,
133 | any Contribution intentionally submitted for inclusion in the Work
134 | by You to the Licensor shall be under the terms and conditions of
135 | this License, without any additional terms or conditions.
136 | Notwithstanding the above, nothing herein shall supersede or modify
137 | the terms of any separate license agreement you may have executed
138 | with Licensor regarding such Contributions.
139 |
140 | 6. Trademarks. This License does not grant permission to use the trade
141 | names, trademarks, service marks, or product names of the Licensor,
142 | except as required for reasonable and customary use in describing the
143 | origin of the Work and reproducing the content of the NOTICE file.
144 |
145 | 7. Disclaimer of Warranty. Unless required by applicable law or
146 | agreed to in writing, Licensor provides the Work (and each
147 | Contributor provides its Contributions) on an "AS IS" BASIS,
148 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
149 | implied, including, without limitation, any warranties or conditions
150 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
151 | PARTICULAR PURPOSE. You are solely responsible for determining the
152 | appropriateness of using or redistributing the Work and assume any
153 | risks associated with Your exercise of permissions under this License.
154 |
155 | 8. Limitation of Liability. In no event and under no legal theory,
156 | whether in tort (including negligence), contract, or otherwise,
157 | unless required by applicable law (such as deliberate and grossly
158 | negligent acts) or agreed to in writing, shall any Contributor be
159 | liable to You for damages, including any direct, indirect, special,
160 | incidental, or consequential damages of any character arising as a
161 | result of this License or out of the use or inability to use the
162 | Work (including but not limited to damages for loss of goodwill,
163 | work stoppage, computer failure or malfunction, or any and all
164 | other commercial damages or losses), even if such Contributor
165 | has been advised of the possibility of such damages.
166 |
167 | 9. Accepting Warranty or Additional Liability. While redistributing
168 | the Work or Derivative Works thereof, You may choose to offer,
169 | and charge a fee for, acceptance of support, warranty, indemnity,
170 | or other liability obligations and/or rights consistent with this
171 | License. However, in accepting such obligations, You may act only
172 | on Your own behalf and on Your sole responsibility, not on behalf
173 | of any other Contributor, and only if You agree to indemnify,
174 | defend, and hold each Contributor harmless for any liability
175 | incurred by, or claims asserted against, such Contributor by reason
176 | of your accepting any such warranty or additional liability.
177 |
178 | END OF TERMS AND CONDITIONS
179 |
--------------------------------------------------------------------------------
/LICENSE-MIT:
--------------------------------------------------------------------------------
1 | Copyright Jeremy Banks
2 |
3 | Permission is hereby granted, free of charge, to any
4 | person obtaining a copy of this software and associated
5 | documentation files (the "Software"), to deal in the
6 | Software without restriction, including without
7 | limitation the rights to use, copy, modify, merge,
8 | publish, distribute, sublicense, and/or sell copies of
9 | the Software, and to permit persons to whom the Software
10 | is furnished to do so, subject to the following
11 | conditions:
12 |
13 | The above copyright notice and this permission notice
14 | shall be included in all copies or substantial portions
15 | of the Software.
16 |
17 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF
18 | ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
19 | TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
20 | PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT
21 | SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
22 | CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
23 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR
24 | IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
25 | DEALINGS IN THE SOFTWARE.
26 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 |
2 |
--------------------------------------------------------------------------------
/examples/in_fn.rs:
--------------------------------------------------------------------------------
1 | #![feature(custom_inner_attributes)]
2 |
3 | pub fn main() {
4 | #![you_can::turn_off_the_borrow_checker]
5 |
6 | let mut owned = vec![1, 32];
7 |
8 | // unsound mutable aliasing
9 | let mut_1 = &mut owned[0];
10 | let mut_2 = &mut owned[1];
11 |
12 | // use after free
13 | drop(owned);
14 | let undefined = *mut_1 + *mut_2;
15 | println!("{undefined}");
16 | }
17 |
--------------------------------------------------------------------------------
/examples/in_mod.rs:
--------------------------------------------------------------------------------
1 | #![feature(custom_inner_attributes)]
2 |
3 | mod inner {
4 | #![you_can::turn_off_the_borrow_checker]
5 |
6 | pub fn main() {
7 | let mut owned = vec![1, 32];
8 |
9 | // unsound mutable aliasing
10 | let mut_1 = &mut owned[0];
11 | let mut_2 = &mut owned[1];
12 |
13 | // use after free
14 | drop(owned);
15 | let undefined = *mut_1 + *mut_2;
16 | println!("{undefined}");
17 | }
18 | }
19 |
20 | fn main() {
21 | inner::main();
22 | }
23 |
--------------------------------------------------------------------------------
/examples/on_block.rs:
--------------------------------------------------------------------------------
1 | #![feature(stmt_expr_attributes, proc_macro_hygiene)]
2 |
3 | pub fn main() {
4 | let mut owned = vec![1, 32];
5 |
6 | #[you_can::turn_off_the_borrow_checker]
7 | {
8 | // unsound mutable aliasing
9 | let mut_1 = &mut owned[0];
10 | let mut_2 = &mut owned[1];
11 |
12 | // use after free
13 | drop(owned);
14 | let undefined = *mut_1 + *mut_2;
15 | println!("{undefined}");
16 | }
17 | }
18 |
--------------------------------------------------------------------------------
/examples/on_fn.rs:
--------------------------------------------------------------------------------
1 | #[you_can::turn_off_the_borrow_checker]
2 | pub fn main() {
3 | let mut owned = vec![1, 32];
4 |
5 | // unsound mutable aliasing
6 | let mut_1 = &mut owned[0];
7 | let mut_2 = &mut owned[1];
8 |
9 | // use after free
10 | drop(owned);
11 | let undefined = *mut_1 + *mut_2;
12 | println!("{undefined}");
13 | }
14 |
--------------------------------------------------------------------------------
/examples/on_mod.rs:
--------------------------------------------------------------------------------
1 | #[you_can::turn_off_the_borrow_checker]
2 | mod inner {
3 | pub fn main() {
4 | let mut owned = vec![1, 32];
5 |
6 | // unsound mutable aliasing
7 | let mut_1 = &mut owned[0];
8 | let mut_2 = &mut owned[1];
9 |
10 | // use after free
11 | drop(owned);
12 | let undefined = *mut_1 + *mut_2;
13 | println!("{undefined}");
14 | }
15 | }
16 |
17 | fn main() {
18 | inner::main();
19 | }
20 |
--------------------------------------------------------------------------------
/examples/on_statement.rs:
--------------------------------------------------------------------------------
1 | #![feature(stmt_expr_attributes, proc_macro_hygiene)]
2 |
3 | pub fn main() {
4 | let mut owned = vec![1, 32];
5 | // unsound mutable aliasing
6 | #[you_can::turn_off_the_borrow_checker]
7 | let mut_1 = &mut owned[0];
8 | #[you_can::turn_off_the_borrow_checker]
9 | let mut_2 = &mut owned[1];
10 |
11 | // use after free
12 | drop(owned);
13 | let undefined = *mut_1 + *mut_2;
14 | println!("{undefined}");
15 | }
16 |
--------------------------------------------------------------------------------
/macros/Cargo.toml:
--------------------------------------------------------------------------------
1 | [package]
2 | authors = ["Jeremy Banks <_@jeremy.ca>"]
3 | description = "#[you_can::turn_off_the_borrow_checker]"
4 | documentation = "https://docs.rs/you-can"
5 | edition = "2021"
6 | license = "MIT OR Apache-2.0"
7 | name = "you-can-build-macros"
8 | repository = "https://github.com/jeremyBanks/you-can"
9 | version = "0.0.14"
10 |
11 | [lib]
12 | proc-macro = true
13 | path = "./lib.rs"
14 |
15 | [dependencies]
16 | quote = "1.0.14"
17 | syn = { version = "1.0.85", features = ["full", "visit", "fold"] }
18 |
19 | [build-dependencies]
20 | rustc_version = "0.2"
21 |
--------------------------------------------------------------------------------
/macros/build.rs:
--------------------------------------------------------------------------------
1 | fn main() {
2 | println!("cargo:rerun-if-env-changed=RUSTC_BOOTSTRAP");
3 | if rustc_is_unstable() {
4 | println!("cargo:rustc-cfg=rustc_is_unstable");
5 | }
6 | }
7 |
8 | fn rustc_is_unstable() -> bool {
9 | match rustc_version::version_meta().unwrap().channel {
10 | rustc_version::Channel::Nightly | rustc_version::Channel::Dev => true,
11 | _ => std::env::var("RUSTC_BOOTSTRAP").is_ok(),
12 | }
13 | }
14 |
--------------------------------------------------------------------------------
/macros/lib.rs:
--------------------------------------------------------------------------------
1 | //! internal proc macros for [`::you_can`]
2 | #![cfg_attr(rustc_is_unstable, feature(proc_macro_diagnostic, proc_macro_span))]
3 |
4 | use {
5 | proc_macro::{Span, TokenStream},
6 | quote::{quote, quote_spanned, ToTokens},
7 | syn::{fold::Fold, parse_quote_spanned, spanned::Spanned, visit::Visit},
8 | };
9 |
10 | /// Runs one of two branches depending on whether we're running on a stable
11 | /// version of the compiler (stable, beta), or an unstable version (nightly,
12 | /// dev, or anywhere that `RUSTC_BOOTSTRAP=1`).
13 | macro_rules! if_unstable {
14 | { then { $($then:tt)* } else { $($else:tt)* } } => {
15 | #[allow(unreachable_code)]
16 | if cfg!(rustc_is_unstable) {
17 | #[cfg(not(rustc_is_unstable))] {
18 | unreachable!();
19 | }
20 | #[cfg(rustc_is_unstable)] {
21 | $($then)*
22 | }
23 | } else {
24 | #[cfg(rustc_is_unstable)] {
25 | unreachable!();
26 | }
27 | $($else)*
28 | }
29 | }
30 | }
31 |
32 | #[proc_macro_attribute]
33 | pub fn turn_off_the_borrow_checker(_attribute: TokenStream, input: TokenStream) -> TokenStream {
34 | let mut suppressor = BorrowCheckerSuppressor {
35 | suppressed_references: vec![],
36 | };
37 |
38 | let output = if let Ok(as_file) = syn::parse(input.clone()) {
39 | suppressor.fold_file(as_file).to_token_stream()
40 | } else if let Ok(as_expr) = syn::parse(input.clone()) {
41 | suppressor.fold_expr(as_expr).to_token_stream()
42 | } else if let Ok(as_stmt) = syn::parse(input) {
43 | suppressor.fold_stmt(as_stmt).to_token_stream()
44 | } else {
45 | return quote! { compile_error!("unsupported use of #[turn_off_the_borrow_checker]") }
46 | .into();
47 | };
48 |
49 | if_unstable! {
50 | then {
51 | proc_macro::Diagnostic::spanned(
52 | vec![Span::call_site().parent().unwrap_or_else(Span::call_site)],
53 | proc_macro::Level::Warning,
54 | "this suppresses the borrow checker in an unsafe, unsound, and unstable way \
55 | that produces undefined behaviour. this is not suitable for any purpose beyond \
56 | educational experimentation.",
57 | ).emit();
58 |
59 | if suppressor.suppressed_references.len() > 1 {
60 | proc_macro::Diagnostic::spanned(
61 | suppressor.suppressed_references,
62 | proc_macro::Level::Warning,
63 | "the borrow checker is suppressed for these references.",
64 | ).emit();
65 | }
66 |
67 | output.into_token_stream().into()
68 | } else {
69 | static DANGER: std::sync::Once = std::sync::Once::new();
70 | DANGER.call_once(|| {
71 | eprintln!();
72 | eprintln!(" DANGER This project is using the the #[you_can::turn_off_the_borrow_checker]");
73 | eprintln!(" DANGER macro, which is inherently unsafe, unsound, and unstable. This is not");
74 | eprintln!(" DANGER suitable for any purpose beyond educational experimentation.");
75 | eprintln!();
76 | });
77 |
78 | quote_spanned! {
79 | Span::call_site().into() =>
80 | #[warn(unsafe_code)]
81 | #output
82 | }.into_token_stream().into()
83 | }
84 | }
85 | }
86 |
87 | /// Replaces all references (&T or &mut T) with unbounded references by wrapping
88 | /// them in calls to you_can::borrow_unchecked().
89 | #[derive(Debug, Default)]
90 | struct BorrowCheckerSuppressor {
91 | suppressed_references: Vec,
92 | }
93 |
94 | impl Fold for BorrowCheckerSuppressor {
95 | fn fold_expr(&mut self, node: syn::Expr) -> syn::Expr {
96 | match node {
97 | syn::Expr::Reference(node) => {
98 | let node = syn::fold::fold_expr_reference(self, node);
99 | self.suppressed_references.push(node.span().unwrap());
100 | syn::Expr::Block(parse_quote_spanned! { node.span() =>
101 | {
102 | let r#ref = #node;
103 | unsafe { ::you_can::borrow_unchecked(r#ref) }
104 | }
105 | })
106 | },
107 | _ => syn::fold::fold_expr(self, node),
108 | }
109 | }
110 |
111 | fn fold_expr_if(&mut self, mut node: syn::ExprIf) -> syn::ExprIf {
112 | if matches!(*node.cond, syn::Expr::Let(_)) {
113 | let mut ref_collector = RefCollector::default();
114 | ref_collector.visit_expr(&node.cond);
115 | let refs = ref_collector.refs;
116 | self.suppressed_references.extend(ref_collector.spans);
117 | let then_stmts = node.then_branch.stmts.clone();
118 | node.then_branch = parse_quote_spanned! { node.span() =>
119 | {
120 | #(let #refs = unsafe { ::you_can::borrow_unchecked(#refs) };)*
121 | #(#then_stmts)*
122 | }
123 | };
124 | }
125 | syn::fold::fold_expr_if(self, node)
126 | }
127 |
128 | fn fold_arm(&mut self, mut node: syn::Arm) -> syn::Arm {
129 | let mut ref_collector = RefCollector::default();
130 | ref_collector.visit_pat(&node.pat);
131 | let refs = ref_collector.refs;
132 | self.suppressed_references.extend(ref_collector.spans);
133 | let body = node.body.clone();
134 | node.body = parse_quote_spanned! { node.span() =>
135 | {
136 | #(let #refs = unsafe { ::you_can::borrow_unchecked(#refs) };)*
137 | #body
138 | }
139 | };
140 | syn::fold::fold_arm(self, node)
141 | }
142 | }
143 |
144 | #[derive(Debug, Default)]
145 | struct RefCollector {
146 | refs: Vec,
147 | spans: Vec,
148 | }
149 |
150 | impl<'ast> Visit<'ast> for RefCollector {
151 | fn visit_pat_ident(&mut self, node: &'ast syn::PatIdent) {
152 | if node.by_ref.is_some() {
153 | self.refs.push(node.ident.clone());
154 | self.spans.push(node.span().unwrap());
155 | }
156 | }
157 | }
158 |
--------------------------------------------------------------------------------
/rust-toolchain.toml:
--------------------------------------------------------------------------------
1 | [toolchain]
2 | channel = "nightly"
3 |
--------------------------------------------------------------------------------
/rustfmt.toml:
--------------------------------------------------------------------------------
1 | force_explicit_abi = false
2 | format_code_in_doc_comments = true
3 | format_macro_bodies = true
4 | format_macro_matchers = true
5 | format_strings = true
6 | group_imports = "StdExternalCrate"
7 | imports_granularity = "One"
8 | match_arm_blocks = false
9 | match_block_trailing_comma = true
10 | newline_style = "Unix"
11 | normalize_comments = true
12 | normalize_doc_attributes = true
13 | overflow_delimited_expr = true
14 | reorder_impl_items = true
15 | unstable_features = true
16 | use_field_init_shorthand = true
17 | wrap_comments = true
18 |
--------------------------------------------------------------------------------
/src/borrow_unchecked.md:
--------------------------------------------------------------------------------
1 | Unbinds the lifetime in a [reference][REF] (`&T` or `&mut T`). This disconnects
2 | the input lifetime from the output lifetime, so use of the output reference will
3 | no longer cause input reference to be kept "live" (as defined in
4 | [RFC 2094: non-lexical lifetimes][NLL]). The output reference is given an
5 | [unbounded lifetime][UBL]. This causes the borrow checker to effectively ignore
6 | it. The output lifetime may still be implicitly bound by any references within
7 | the referenced type `T`. You may need to unbind these as well, or replace them
8 | with `'static`.
9 |
10 | # Safety
11 |
12 | _The section intentionally left blank._
13 |
14 | # Example
15 |
16 | Here we create two (aliasing) unbounded mutable references at once, and then
17 | continue to use them even after the original value is dropped (and its true
18 | lifetime has ended).
19 |
20 | ```rust
21 | let mut x = 0;
22 |
23 | let mut_1 = unsafe { ::you_can::borrow_unchecked(&mut x) };
24 | let mut_2 = unsafe { ::you_can::borrow_unchecked(&mut x) };
25 |
26 | drop(x);
27 |
28 | *mut_1 = 1;
29 | *mut_2 = 2;
30 |
31 | assert_eq!(*mut_1, *mut_2, "I hope LLVM is in a good mood!");
32 | ```
33 |
34 | [REF]: https://doc.rust-lang.org/std/primitive.reference.html
35 | [NLL]: https://rust-lang.github.io/rfcs/2094-nll.html
36 | [UBL]: https://doc.rust-lang.org/nomicon/unbounded-lifetimes.html
37 |
--------------------------------------------------------------------------------
/src/lib.rs:
--------------------------------------------------------------------------------
1 | //! [`#[you_can::turn_off_the_borrow_checker]`][turn_off_the_borrow_checker]
2 | #![no_std]
3 | #![allow(clippy::needless_doctest_main)]
4 | #![warn(unused_extern_crates)]
5 | #![deny(unsafe_op_in_unsafe_fn, rustdoc::broken_intra_doc_links)]
6 | #![doc(
7 | html_favicon_url = "",
8 | html_logo_url = ""
9 | )]
10 |
11 | #[doc = include_str!("./turn_off_the_borrow_checker.md")]
12 | #[doc(inline)]
13 | pub use you_can_build_macros::turn_off_the_borrow_checker;
14 |
15 | #[doc = include_str!("./borrow_unchecked.md")]
16 | #[inline(always)]
17 | pub unsafe fn borrow_unchecked<
18 | 'original,
19 | 'unbounded,
20 | Ref: BorrowUnchecked<'original, 'unbounded>,
21 | >(
22 | reference: Ref,
23 | ) -> Ref::Unbounded {
24 | unsafe { BorrowUnchecked::borrow_unchecked(reference) }
25 | }
26 |
27 | #[doc(hidden)]
28 | pub unsafe trait BorrowUnchecked<'original, 'unbounded> {
29 | type Unbounded;
30 |
31 | unsafe fn borrow_unchecked(self) -> Self::Unbounded;
32 | }
33 |
34 | unsafe impl<'original, 'unbounded, T: 'unbounded> BorrowUnchecked<'original, 'unbounded>
35 | for &'original T
36 | {
37 | type Unbounded = &'unbounded T;
38 |
39 | #[inline(always)]
40 | unsafe fn borrow_unchecked(self) -> Self::Unbounded {
41 | unsafe { ::core::mem::transmute(self) }
42 | }
43 | }
44 |
45 | unsafe impl<'original, 'unbounded, T: 'unbounded> BorrowUnchecked<'original, 'unbounded>
46 | for &'original mut T
47 | {
48 | type Unbounded = &'unbounded mut T;
49 |
50 | #[inline(always)]
51 | unsafe fn borrow_unchecked(self) -> Self::Unbounded {
52 | unsafe { ::core::mem::transmute(self) }
53 | }
54 | }
55 |
--------------------------------------------------------------------------------
/src/turn_off_the_borrow_checker.md:
--------------------------------------------------------------------------------
1 | [You can't "turn off the borrow checker" in Rust][OFF], and you shouldn't want
2 | to. [Rust's references][REF] aren't pointers, and the compiler is free to
3 | decimate code that tries to use references as though they are. If you need raw
4 | pointer behaviour in Rust, don't use this, use
5 | [Rust's actual raw pointers][PTR], which don't make the same aliasing guarantees
6 | to the compiler. However, if you would like to pretend the borrow checker
7 | doesn't exist **for educational purposes and never in production code**, this
8 | macro that will suppress many (though not all) borrow checker errors in the code
9 | it's applied to.
10 |
11 | This shouldn't break any otherwise-valid code; the borrow checker doesn't affect
12 | compilation output, only verify input validity. However, it will allow unsound
13 | and unsafe nonsense that will fail unpredictably and dangerously. This **is not
14 | safe to use**.
15 |
16 | ### Example
17 |
18 | #### Without Macro
19 |
20 | ```compile_fail
21 | fn main() {
22 | let mut owned = vec![1, 32];
23 |
24 | let mut_1 = &mut owned[0];
25 | let mut_2 = &mut owned[1];
26 | //~^ ERROR cannot borrow `owned` as mutable more than once at a time
27 |
28 | drop(owned);
29 | //~^ ERROR cannot move out of `owned` because it is borrowed
30 | let undefined = *mut_1 + *mut_2;
31 | println!("{undefined}");
32 | }
33 | ```
34 |
35 | #### With Macro
36 |
37 | ```rust
38 | #[you_can::turn_off_the_borrow_checker]
39 | fn main() {
40 | let mut owned = vec![1, 32];
41 |
42 | let mut_1 = &mut owned[0];
43 | let mut_2 = &mut owned[1];
44 | //~^ WARNING the borrow checker is suppressed for these references.
45 |
46 | drop(owned);
47 | let undefined = *mut_1 + *mut_2;
48 | println!("{undefined}");
49 | }
50 | ```
51 |
52 | ## Explanation
53 |
54 | The macro looks for references created in the code by use of the `&` or `&mut`
55 | operators or the `ref` and `ref mut` bindings, and wraps them with our
56 | [`borrow_unchecked()`] function to [unbind their lifetimes][UBL], causing the
57 | borrow checker to effectively ignore them. If running on nightly, it adds new
58 | warning diagnostic messages for every reference it modifies.
59 |
60 | #### Expanded
61 |
62 | ```rust
63 | fn main() {
64 | let mut owned = vec![1, 32];
65 |
66 | let mut_1 = unsafe { ::you_can::borrow_unchecked(&mut owned[0]) };
67 | let mut_2 = unsafe { ::you_can::borrow_unchecked(&mut owned[1]) };
68 |
69 | drop(owned);
70 | let undefined = *mut_1 + *mut_2;
71 | println!("{undefined}");
72 | }
73 | ```
74 |
75 | This approached is limited. It can't suppress errors resulting from the code
76 | illegally composing lifetimes created elsewhere, or references created
77 | implicitly. As a workaround, prefixing `&*` can sometimes be used to force an
78 | explicit reference where one is needed, such as as in the example below.
79 |
80 | ### Example
81 |
82 | ```rust
83 | #[you_can::turn_off_the_borrow_checker]
84 | fn main() {
85 | let mut source = Some(1);
86 | let inner_mut = &*source.as_ref().unwrap();
87 | let mutable_alias = &mut source;
88 |
89 | source = None;
90 | *mutable_alias = Some(2);
91 |
92 | if let Some(ref mut inner_a) = source {
93 | match source {
94 | Some(ref mut inner_b) => {
95 | *inner_b = inner_mut + 1;
96 | *inner_a = inner_mut + 2;
97 | },
98 | None => {
99 | println!("none");
100 | },
101 | }
102 | }
103 |
104 | println!("{source:?}");
105 | }
106 | ```
107 |
108 | #### Expanded
109 |
110 | ```rust
111 | fn main() {
112 | let mut source = Some(1);
113 | let inner_mut = unsafe { ::you_can::borrow_unchecked(&*source.as_ref().unwrap()) };
114 | let mutable_alias = unsafe { ::you_can::borrow_unchecked(&mut source) };
115 |
116 | source = None;
117 | *mutable_alias = Some(2);
118 |
119 | if let Some(ref mut inner_a) = source {
120 | let inner_a = unsafe { ::you_can::borrow_unchecked(inner_a) };
121 |
122 | match source {
123 | Some(ref mut inner_b) => {
124 | let inner_b = unsafe { ::you_can::borrow_unchecked(inner_b) };
125 |
126 | *inner_b = inner_mut + 1;
127 | *inner_a = inner_mut + 2;
128 | },
129 | None => {
130 | println!("none");
131 | },
132 | }
133 | }
134 |
135 | println!("{source:?}");
136 | }
137 | ```
138 |
139 | ## Discussions
140 |
141 | Here are some related discussions, mostly about why you shouldn't do this:
142 |
143 | -
144 | -
145 | -
146 | -
147 | -
148 |
149 | [OFF]: https://steveklabnik.com/writing/you-can-t-turn-off-the-borrow-checker-in-rust
150 | [PTR]: https://doc.rust-lang.org/std/primitive.pointer.html
151 | [REF]: https://doc.rust-lang.org/std/primitive.reference.html
152 | [UBL]: https://doc.rust-lang.org/nomicon/unbounded-lifetimes.html
153 |
--------------------------------------------------------------------------------
/tests/compile_tests.rs:
--------------------------------------------------------------------------------
1 | extern crate compiletest_rs as compiletest;
2 |
3 | #[test]
4 | fn run_pass() {
5 | let mut config = compiletest::Config {
6 | mode: compiletest::common::Mode::Ui,
7 | src_base: std::path::PathBuf::from(if rustc_is_unstable() {
8 | "tests/unstable-pass"
9 | } else {
10 | "tests/stable-pass"
11 | }),
12 | bless: std::env::var_os("BLESS").is_some(),
13 | edition: Some("2021".into()),
14 | target_rustcflags: Some(
15 | [
16 | "--extern you_can",
17 | "--extern unbounded",
18 | "-L target/debug/deps",
19 | ]
20 | .join(" "),
21 | ),
22 | ..Default::default()
23 | };
24 |
25 | config.link_deps();
26 | config.clean_rmeta();
27 |
28 | compiletest::run_tests(&config);
29 | }
30 |
31 | #[test]
32 | fn compile_fail() {
33 | let mut config = compiletest::Config {
34 | mode: compiletest::common::Mode::CompileFail,
35 | src_base: std::path::PathBuf::from(if rustc_is_unstable() {
36 | "tests/unstable-fail"
37 | } else {
38 | "tests/stable-fail"
39 | }),
40 | bless: std::env::var_os("BLESS").is_some(),
41 | edition: Some("2021".into()),
42 | target_rustcflags: Some(
43 | [
44 | "--extern you_can",
45 | "--extern unbounded",
46 | "-L target/debug/deps",
47 | ]
48 | .join(" "),
49 | ),
50 | ..Default::default()
51 | };
52 |
53 | config.link_deps();
54 | config.clean_rmeta();
55 |
56 | compiletest::run_tests(&config);
57 | }
58 |
59 | fn rustc_is_unstable() -> bool {
60 | match rustc_version::version_meta().unwrap().channel {
61 | rustc_version::Channel::Nightly | rustc_version::Channel::Dev => true,
62 | _ => option_env!("RUSTC_BOOTSTRAP").is_some(),
63 | }
64 | }
65 |
--------------------------------------------------------------------------------
/tests/stable-fail/not_on_fn.rs:
--------------------------------------------------------------------------------
1 | fn main() {
2 | let mut owned = vec![1, 32];
3 |
4 | // unsound mutable aliasing
5 | let mut_1 = &mut owned[0];
6 | let mut_2 = &mut owned[1];
7 | //~^ ERROR cannot borrow `owned` as mutable more than once at a time
8 |
9 | // use after free
10 | drop(owned);
11 | //~^ ERROR cannot move out of `owned` because it is borrowed
12 | let undefined = *mut_1 + *mut_2;
13 | println!("{undefined}");
14 | }
15 |
--------------------------------------------------------------------------------
/tests/stable-pass/on_fn.rs:
--------------------------------------------------------------------------------
1 |
2 | #[you_can::turn_off_the_borrow_checker]
3 | fn main() {
4 | let mut owned = vec![1, 32];
5 |
6 | // unsound mutable aliasing
7 | let mut_1 = &mut owned[0];
8 | let mut_2 = &mut owned[1];
9 |
10 | // use after free
11 | drop(owned);
12 | let undefined = *mut_1 + *mut_2;
13 | println!("{undefined}");
14 | }
15 |
--------------------------------------------------------------------------------
/tests/stable-pass/on_fn.stderr:
--------------------------------------------------------------------------------
1 |
2 | DANGER This project is using the the #[you_can::turn_off_the_borrow_checker]
3 | DANGER macro, which is inherently unsafe, unsound, and unstable. This is not
4 | DANGER suitable for any purpose beyond educational experimentation.
5 |
6 | warning: usage of an `unsafe` block
7 | --> $DIR/on_fn.rs:7:17
8 | |
9 | 7 | let mut_1 = &mut owned[0];
10 | | ^
11 | |
12 | note: the lint level is defined here
13 | --> $DIR/on_fn.rs:2:1
14 | |
15 | 2 | #[you_can::turn_off_the_borrow_checker]
16 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
17 | = note: this warning originates in the attribute macro `you_can::turn_off_the_borrow_checker` (in Nightly builds, run with -Z macro-backtrace for more info)
18 |
19 | warning: usage of an `unsafe` block
20 | --> $DIR/on_fn.rs:8:17
21 | |
22 | 8 | let mut_2 = &mut owned[1];
23 | | ^
24 |
25 | warning: 2 warnings emitted
26 |
27 |
--------------------------------------------------------------------------------
/tests/stable-pass/on_fn_empty.rs:
--------------------------------------------------------------------------------
1 | #[you_can::turn_off_the_borrow_checker]
2 | fn main() {}
3 |
--------------------------------------------------------------------------------
/tests/stable-pass/on_fn_empty.stderr:
--------------------------------------------------------------------------------
1 |
2 | DANGER This project is using the the #[you_can::turn_off_the_borrow_checker]
3 | DANGER macro, which is inherently unsafe, unsound, and unstable. This is not
4 | DANGER suitable for any purpose beyond educational experimentation.
5 |
6 |
--------------------------------------------------------------------------------
/tests/stable-pass/on_mod.rs:
--------------------------------------------------------------------------------
1 | #[you_can::turn_off_the_borrow_checker]
2 | mod inner {
3 | pub fn main() {
4 | let mut owned = vec![1, 32];
5 |
6 | // unsound mutable aliasing
7 | let mut_1 = &mut owned[0];
8 | let mut_2 = &mut owned[1];
9 |
10 | // use after free
11 | drop(owned);
12 | let undefined = *mut_1 + *mut_2;
13 | println!("{undefined}");
14 | }
15 | }
16 |
17 | fn main() {
18 | inner::main();
19 | }
20 |
--------------------------------------------------------------------------------
/tests/stable-pass/on_mod.stderr:
--------------------------------------------------------------------------------
1 |
2 | DANGER This project is using the the #[you_can::turn_off_the_borrow_checker]
3 | DANGER macro, which is inherently unsafe, unsound, and unstable. This is not
4 | DANGER suitable for any purpose beyond educational experimentation.
5 |
6 | warning: usage of an `unsafe` block
7 | --> $DIR/on_mod.rs:7:21
8 | |
9 | 7 | let mut_1 = &mut owned[0];
10 | | ^
11 | |
12 | note: the lint level is defined here
13 | --> $DIR/on_mod.rs:1:1
14 | |
15 | 1 | #[you_can::turn_off_the_borrow_checker]
16 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
17 | = note: this warning originates in the attribute macro `you_can::turn_off_the_borrow_checker` (in Nightly builds, run with -Z macro-backtrace for more info)
18 |
19 | warning: usage of an `unsafe` block
20 | --> $DIR/on_mod.rs:8:21
21 | |
22 | 8 | let mut_2 = &mut owned[1];
23 | | ^
24 |
25 | warning: 2 warnings emitted
26 |
27 |
--------------------------------------------------------------------------------
/tests/unstable-fail/not_on_fn.rs:
--------------------------------------------------------------------------------
1 | fn main() {
2 | let mut owned = vec![1, 32];
3 |
4 | // unsound mutable aliasing
5 | let mut_1 = &mut owned[0];
6 | let mut_2 = &mut owned[1];
7 | //~^ ERROR cannot borrow `owned` as mutable more than once at a time
8 |
9 | // use after free
10 | drop(owned);
11 | //~^ ERROR cannot move out of `owned` because it is borrowed
12 | let undefined = *mut_1 + *mut_2;
13 | println!("{undefined}");
14 | }
15 |
--------------------------------------------------------------------------------
/tests/unstable-pass/in_fn.rs:
--------------------------------------------------------------------------------
1 | #![feature(custom_inner_attributes)]
2 |
3 | pub fn main() {
4 | #![you_can::turn_off_the_borrow_checker]
5 |
6 | let mut owned = vec![1, 32];
7 |
8 | // unsound mutable aliasing
9 | let mut_1 = &mut owned[0];
10 | let mut_2 = &mut owned[1];
11 |
12 | // use after free
13 | drop(owned);
14 | let undefined = *mut_1 + *mut_2;
15 | println!("{undefined}");
16 | }
17 |
--------------------------------------------------------------------------------
/tests/unstable-pass/in_fn.stderr:
--------------------------------------------------------------------------------
1 | warning: this suppresses the borrow checker in an unsafe, unsound, and unstable way that produces undefined behaviour. this is not suitable for any purpose beyond educational experimentation.
2 | --> $DIR/in_fn.rs:4:5
3 | |
4 | 4 | #![you_can::turn_off_the_borrow_checker]
5 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6 |
7 | warning: the borrow checker is suppressed for these references.
8 | --> $DIR/in_fn.rs:9:17
9 | |
10 | 9 | let mut_1 = &mut owned[0];
11 | | ^^^^^^^^^^^^^
12 | 10 | let mut_2 = &mut owned[1];
13 | | ^^^^^^^^^^^^^
14 |
15 | warning: 2 warnings emitted
16 |
17 |
--------------------------------------------------------------------------------
/tests/unstable-pass/in_mod.rs:
--------------------------------------------------------------------------------
1 | #![feature(custom_inner_attributes)]
2 |
3 | mod inner {
4 | #![you_can::turn_off_the_borrow_checker]
5 |
6 | pub fn main() {
7 | let mut owned = vec![1, 32];
8 |
9 | // unsound mutable aliasing
10 | let mut_1 = &mut owned[0];
11 | let mut_2 = &mut owned[1];
12 |
13 | // use after free
14 | drop(owned);
15 | let undefined = *mut_1 + *mut_2;
16 | println!("{undefined}");
17 | }
18 | }
19 |
20 | fn main() {
21 | inner::main();
22 | }
23 |
--------------------------------------------------------------------------------
/tests/unstable-pass/in_mod.stderr:
--------------------------------------------------------------------------------
1 | warning: this suppresses the borrow checker in an unsafe, unsound, and unstable way that produces undefined behaviour. this is not suitable for any purpose beyond educational experimentation.
2 | --> $DIR/in_mod.rs:4:5
3 | |
4 | 4 | #![you_can::turn_off_the_borrow_checker]
5 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6 |
7 | warning: the borrow checker is suppressed for these references.
8 | --> $DIR/in_mod.rs:10:21
9 | |
10 | 10 | let mut_1 = &mut owned[0];
11 | | ^^^^^^^^^^^^^
12 | 11 | let mut_2 = &mut owned[1];
13 | | ^^^^^^^^^^^^^
14 |
15 | warning: 2 warnings emitted
16 |
17 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_block.rs:
--------------------------------------------------------------------------------
1 | #![feature(stmt_expr_attributes, proc_macro_hygiene)]
2 |
3 | pub fn main() {
4 | let mut owned = vec![1, 32];
5 |
6 | #[you_can::turn_off_the_borrow_checker]
7 | {
8 | // unsound mutable aliasing
9 | let mut_1 = &mut owned[0];
10 | let mut_2 = &mut owned[1];
11 |
12 | // use after free
13 | drop(owned);
14 | let undefined = *mut_1 + *mut_2;
15 | println!("{undefined}");
16 | }
17 | }
18 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_block.stderr:
--------------------------------------------------------------------------------
1 | warning: this suppresses the borrow checker in an unsafe, unsound, and unstable way that produces undefined behaviour. this is not suitable for any purpose beyond educational experimentation.
2 | --> $DIR/on_block.rs:6:5
3 | |
4 | 6 | #[you_can::turn_off_the_borrow_checker]
5 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6 |
7 | warning: the borrow checker is suppressed for these references.
8 | --> $DIR/on_block.rs:9:21
9 | |
10 | 9 | let mut_1 = &mut owned[0];
11 | | ^^^^^^^^^^^^^
12 | 10 | let mut_2 = &mut owned[1];
13 | | ^^^^^^^^^^^^^
14 |
15 | warning: 2 warnings emitted
16 |
17 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_fn.rs:
--------------------------------------------------------------------------------
1 | #[you_can::turn_off_the_borrow_checker]
2 | fn main() {
3 | let mut owned = vec![1, 32];
4 |
5 | // unsound mutable aliasing
6 | let mut_1 = &mut owned[0];
7 | let mut_2 = &mut owned[1];
8 |
9 | // use after free
10 | drop(owned);
11 | let undefined = *mut_1 + *mut_2;
12 | println!("{undefined}");
13 | }
14 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_fn.stderr:
--------------------------------------------------------------------------------
1 | warning: this suppresses the borrow checker in an unsafe, unsound, and unstable way that produces undefined behaviour. this is not suitable for any purpose beyond educational experimentation.
2 | --> $DIR/on_fn.rs:1:1
3 | |
4 | 1 | #[you_can::turn_off_the_borrow_checker]
5 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6 |
7 | warning: the borrow checker is suppressed for these references.
8 | --> $DIR/on_fn.rs:6:17
9 | |
10 | 6 | let mut_1 = &mut owned[0];
11 | | ^^^^^^^^^^^^^
12 | 7 | let mut_2 = &mut owned[1];
13 | | ^^^^^^^^^^^^^
14 |
15 | warning: 2 warnings emitted
16 |
17 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_fn_empty.rs:
--------------------------------------------------------------------------------
1 | #[you_can::turn_off_the_borrow_checker]
2 | fn main() {}
3 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_fn_empty.stderr:
--------------------------------------------------------------------------------
1 | warning: this suppresses the borrow checker in an unsafe, unsound, and unstable way that produces undefined behaviour. this is not suitable for any purpose beyond educational experimentation.
2 | --> $DIR/on_fn_empty.rs:1:1
3 | |
4 | 1 | #[you_can::turn_off_the_borrow_checker]
5 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6 |
7 | warning: 1 warning emitted
8 |
9 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_mod.rs:
--------------------------------------------------------------------------------
1 | #[you_can::turn_off_the_borrow_checker]
2 | mod inner {
3 | pub fn main() {
4 | let mut owned = vec![1, 32];
5 |
6 | // unsound mutable aliasing
7 | let mut_1 = &mut owned[0];
8 | let mut_2 = &mut owned[1];
9 |
10 | // use after free
11 | drop(owned);
12 | let undefined = *mut_1 + *mut_2;
13 | println!("{undefined}");
14 | }
15 | }
16 |
17 | fn main() {
18 | inner::main();
19 | }
20 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_mod.stderr:
--------------------------------------------------------------------------------
1 | warning: this suppresses the borrow checker in an unsafe, unsound, and unstable way that produces undefined behaviour. this is not suitable for any purpose beyond educational experimentation.
2 | --> $DIR/on_mod.rs:1:1
3 | |
4 | 1 | #[you_can::turn_off_the_borrow_checker]
5 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6 |
7 | warning: the borrow checker is suppressed for these references.
8 | --> $DIR/on_mod.rs:7:21
9 | |
10 | 7 | let mut_1 = &mut owned[0];
11 | | ^^^^^^^^^^^^^
12 | 8 | let mut_2 = &mut owned[1];
13 | | ^^^^^^^^^^^^^
14 |
15 | warning: 2 warnings emitted
16 |
17 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_statement.rs:
--------------------------------------------------------------------------------
1 | #![feature(stmt_expr_attributes, proc_macro_hygiene)]
2 |
3 | pub fn main() {
4 | let mut owned = vec![1, 32];
5 | // unsound mutable aliasing
6 | #[you_can::turn_off_the_borrow_checker]
7 | let mut_1 = &mut owned[0];
8 | #[you_can::turn_off_the_borrow_checker]
9 | let mut_2 = &mut owned[1];
10 |
11 | // use after free
12 | drop(owned);
13 | let undefined = *mut_1 + *mut_2;
14 | println!("{undefined}");
15 | }
16 |
--------------------------------------------------------------------------------
/tests/unstable-pass/on_statement.stderr:
--------------------------------------------------------------------------------
1 | warning: this suppresses the borrow checker in an unsafe, unsound, and unstable way that produces undefined behaviour. this is not suitable for any purpose beyond educational experimentation.
2 | --> $DIR/on_statement.rs:6:5
3 | |
4 | 6 | #[you_can::turn_off_the_borrow_checker]
5 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6 |
7 | warning: this suppresses the borrow checker in an unsafe, unsound, and unstable way that produces undefined behaviour. this is not suitable for any purpose beyond educational experimentation.
8 | --> $DIR/on_statement.rs:8:5
9 | |
10 | 8 | #[you_can::turn_off_the_borrow_checker]
11 | | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
12 |
13 | warning: 2 warnings emitted
14 |
15 |
--------------------------------------------------------------------------------