Self-referential structs (in Rust)

  Рет қаралды 55,181

fasterthanlime

fasterthanlime

Күн бұрын

Пікірлер: 147
@Kazyek
@Kazyek 2 жыл бұрын
Scoped threads were added in Rust 1.63.0, which could be used to cleanly solve your specific simple example without unsafe
@m.sierra5258
@m.sierra5258 2 жыл бұрын
Just wanted to let you know that scoped threads now made it to Rust's Std library in 1.63.
@yardez5990
@yardez5990 2 жыл бұрын
1.63*
@dexio85
@dexio85 2 жыл бұрын
Still, another patchup on the misguided idea that compiler will "catch all problems".
@yardez5990
@yardez5990 2 жыл бұрын
​@@dexio85 why is it misguided? every language's compiler tries to find as many problems in your code as it can, and if it doesnt devs try to make some additional tools for that
@m.sierra5258
@m.sierra5258 2 жыл бұрын
@@yardez5990 Exactly. And Rust does guarantee no undefined behavior without the unsafe keyword. This is not misguided, it is a guarantee. (and a compiler bug if not held)
@m.sierra5258
@m.sierra5258 2 жыл бұрын
@@dexio85 If I learned one thing during my years of working as a programmer, then it is that developers are absolutely never reliable, so every bit that the compiler enforces is valuable. Look at it as built-in sanity checks. Why is this misguided?
@Iifesteal
@Iifesteal Жыл бұрын
Thanks for starting with a normal for loop and working your way up to idiomatic rust. Really appreciated by a starter like myself.
@gomesroney
@gomesroney 2 жыл бұрын
Very nice video. It's hard to find examples online of memory corruption in Rust, so it was nice that you demonstrated how that could be possible. Also, as you iterated over the example I was getting high hopes that you would explore pinning but then the video ended, haha. I was ready for more 30 minutes just on Pin 🙂. I hope you can explore that in the future as your examples oriented approach makes everything much more comprehensible.
@m.sierra5258
@m.sierra5258 2 жыл бұрын
Pin then definitely no longer counts as "simple example" :P
@flyingsquirrel3271
@flyingsquirrel3271 3 жыл бұрын
Awesome! Thanks for the video! Although I knew most of this before, watching videos like this feels like casually "strengthening" my inner understanding of rust which is always appreciated :)
@kajacx
@kajacx Жыл бұрын
18:00 The reason that move doesn't work is because rust compiler doesn't understand the difference between shallow and deep borrow. Since the "good_lines" vec borrows (points to) the heap where "input" points, it's actually safe to move input while good_lines exist. If good_lines pointed to the input's length, for example, then you couldn't move the input, because then good_lines would point to invalid memory. But the Rust compiler has no way of knowing that. There are no type annotations for it. The same problem is with shallow/deep mutability. Sometimes you have to declare a variable binding as mut even when you don't need shallow mutability.
@alagaika8515
@alagaika8515 26 күн бұрын
Actually, it is mare subtle. If you move input into the thread and the thread assumes in has sole owner, it might mutate the data and cause a reallocation, at which point the borrow will became invalid. Moving immutably is not passible because if you move, you explicitly want the new owner to mutate the data - at the very least once it is finished with it and drops it.
@daniellambert6207
@daniellambert6207 3 жыл бұрын
14:30 the worker thread will outlive main() {} if you decided to add a conditional panic!() before the join. 16:00 aah, so crossbeam's scope will be dropped at that diabolical panic, ensuring the threads join
@neopalm2050
@neopalm2050 Жыл бұрын
I guess this illustrates why it's important that things are dropped in the event of a panic!()
@jotch_7627
@jotch_7627 Жыл бұрын
​@@neopalm2050no, actually. rust does not allow memory unsafety just because a value wasnt dropped. thats why std::mem::forget is safe to call. the way it actually works is that the callback is run in a way that panics are caught and cleanup is performed before continuing the panic
@Verrisin
@Verrisin Жыл бұрын
The more I learn about Rust (and what it makes explicit about how hard it used to be in C++) .... The more I appreciate simplicity of programming with GC.
@squelchedotter
@squelchedotter 3 жыл бұрын
Where is Cool Bear though 🐻 😞
@asdfghyter
@asdfghyter Жыл бұрын
4:19 was there any specific reason to wrap the filter function in a lambda instead of just writing .filter(starts_with_capital_letter) directly?
@calisti9308
@calisti9308 Жыл бұрын
23:03 Swapping the pointer, length and capacity of the two strings (input2 and s.input) does nothing to the references stored in S::good_lines. And I would imagine it also does nothing at all to each heap-stored string slice (pointed to by the pointer field of input2 and s.input - so the pointers on the stack are swapped, but the bytes in heap aren't.) But why does it then start failing when enclosed in a scope (curly braces)?!?
@naitikmundra8511
@naitikmundra8511 2 жыл бұрын
The feeling when you finally understand ownership! Great video.
@linkernick5379
@linkernick5379 3 жыл бұрын
I love your blog, learned from it so much.
@wololo1657
@wololo1657 3 жыл бұрын
I did not understand *all* of the parts of this video, but I still managed to learn a lot by watching it. I will be coming back to this video, certainly.
@sardobi
@sardobi 2 жыл бұрын
Love your work! this video was pitched at the perfect level for me
@gauravtatke4793
@gauravtatke4793 Жыл бұрын
Hi, At 22:00 to 22:15 time, why does return S not complain about returning a reference to local variable input_slice? good_lines hold a reference to input_slice which is a local variable. Can someone please explain?
@calisti9308
@calisti9308 Жыл бұрын
input_slice is a reference to memory on the heap, not a binding of a local stack value. good_lines reference sub-slices of the same heap string slice that input_slice points to. The reference's lifetime is made 'static by the unsafe block, even though the heap value is not actually 'static - which is why this is unsafe. (If the heap value is dropped, then both input_slice and the vec entries of good_lines point at garbage memory.
@mattdavies6820
@mattdavies6820 3 жыл бұрын
Great video - what is the drawing tool you're using for doing the diagrams?
@samuelhurel6402
@samuelhurel6402 2 жыл бұрын
Have you found out ? I'd like to know too
@cthutu
@cthutu 2 жыл бұрын
@@samuelhurel6402 No, but it's the same tools that 3Blue1Brown uses in his Maths videos. Coincidentally, I literally asked the same question on one of his videos. It's blowing my mind that you should reply to my original ask from 6 months ago!
@walterkrieg5396
@walterkrieg5396 9 ай бұрын
It’s a python extension called manim
@1000percent1000
@1000percent1000 Жыл бұрын
i have been wondering for months why joining threads doesn't join lifetimes, you are a lifesaver for demonstrating crossbeam's scope. so useful, thank you so much :DDD
@d3line
@d3line Жыл бұрын
Scoped threads are now in standard library btw
@SophieJMore
@SophieJMore Жыл бұрын
Joining threads doesn't join lifetimes because join() can be called at any time (including never), or it can be called in some conditions and not others. Also, the compiler is unable to reason about any complex control flow, it just relies on the lifetime system, and the lifetime of the closure you pass when you spawn the thread is 'static, which was chosen for the abovementioned reason. That 'static is literally everything the compiler sees, and it will reject any attempt to give it a closure that reference anything non-static
@antonionatilla9848
@antonionatilla9848 3 жыл бұрын
Very interesting demonstration and explanation! I dealt with the same situation a couple of weeks ago, but I didn't delve into the "unsafe territory" since I'm "still learning" Rust. Finding out about this video was a pleasant surprise, in the past days!
@m.sierra5258
@m.sierra5258 2 жыл бұрын
It's hard to believe to me that many people still strongly advocate C/C++ where unsafe ist just normal
@Hwyadylaw
@Hwyadylaw 2 жыл бұрын
I think it's quite nice that writing unsafe code is so explicit that it makes you think "maybe I shouldn't", whereas in other languages you might write rust-unsafe code without even realising the potential issues
@smidimir
@smidimir Жыл бұрын
I'm currently new to rust but, I know C++ very well. In rust I tried to solve exactly this problem for about 3 hours using "safe" rust without success. Having my background, I had some assumptions about how certain things work in rust. - Unlike C++, rust "moves" are just memcpy's. - Moving the container (String or Vec) shoud not mean moving its data from one location to another. It's just a handle. I even checked if String in rust has short string optimization. And it doesn't. So, what's the problem, rust? Why don't you let me do this clearly safe thing? In your video you asked exactly the questions I had in my head while trying to make it work. I also started to think, that "unsafe" rust is the solution to the problem. Thank you for the thought process and the solution. Great video!
@SviatoslavAlekseev
@SviatoslavAlekseev 2 жыл бұрын
Great video. How about Pin and Unpin. They should help with these problems (almost) without unsafe. Are there a any good guides on how to use it in self-referential structs?
@abhishekshah11
@abhishekshah11 3 жыл бұрын
Yeah this was really insightful into subtle workings of rust safe code. Would like to see more of it. I'll try to write a "safe" fn woops this afternoon hopefully, now that I'm curious to make it work :P
@connormcmk
@connormcmk Жыл бұрын
i don't think anyone has mentioned it but scoped threads were added rust's std 1.63.0, wish someone would have commented this earlier
@SianaGearz
@SianaGearz Жыл бұрын
Is it deep sarcasm or genuine? I can't tell.
@connormcmk
@connormcmk Жыл бұрын
@@SianaGearz i saw it mentioned 3 times so wanted to make sure it got mentioned at least once
@nikandfor
@nikandfor Жыл бұрын
What a good simple language!
@luckystrike91
@luckystrike91 4 ай бұрын
you can use Pin to prevent move
@MetalManiacBoy
@MetalManiacBoy Жыл бұрын
Insanely well explained, fasterthanlime ! Keep up the good work :D
@CottidaeSEA
@CottidaeSEA Жыл бұрын
Once you mentioned the second thread, all I could think was "you could just do the entire thing in the new thread".
@AbelShields
@AbelShields Жыл бұрын
Hmmm... I wonder if this problem could be solved with a little language help - can we assume it's fine to hold a reference to a heap value, such as a T in `Box` while the box itself is moved?
@d3line
@d3line Жыл бұрын
It's not really about the move, it's more about the drop. Box could be moved to a function that immediately drops it, so box is deallocated and references become invalid. So no.
@AbelShields
@AbelShields Жыл бұрын
@@d3line huh, that makes a lot of sense. Thanks :)
@MasterHigure
@MasterHigure 2 жыл бұрын
4:30 filter(|x| f(x)) is the same as just filter(f).
@qm3ster
@qm3ster Жыл бұрын
Nope. `f` is `Fn(&str)` `|x| f(x)` is `Fn(&&str)` There is a dereferencing happening. The closure is unavoidable.
@asdfghyter
@asdfghyter Жыл бұрын
I wonder if it would be possible for Rust to be aware of the distinct lifetimes of the String struct and the string data it points to? when moving something, you invalidate anything that points to it directly, but it shouldn't invalidate pointers to things it points to. it doesn't seem like a super difficult rule to get right and it should be a fairly common case: if i borrow a pointer from a struct, you can move the struct, but you can't drop it nor mutate the pointer part of the struct. the last part is similar to rules that already exist, since you can partially borrow a struct and then the rest of the struct is still available *edit:* there seems to be a crate that attempts to solve this, called owning_ref. unfortunately it (and the example in the video) seems to be unsound/UB for some technical reasons, which is proposed to be fixed by the "maybe_dangling" rfc. *edit2:* another crate called _self_cell_ claims to be sound, so maybe that's the way to go. it uses a macro to define the struct and the allowed operations
@zperk13
@zperk13 Жыл бұрын
7:00 How in the world did you get rust analyzer to highlight line 17 too?
@zperk13
@zperk13 Жыл бұрын
Figured it out. You need to set Error Lens to also highlight hints
@ゾカリクゾ
@ゾカリクゾ Жыл бұрын
So... what approach would you use if you were to avoid unsafe?
Жыл бұрын
Hi bro, thanks for all of your hard work. I would like you know, which plug-in are using, to show the errors on the line in Vs Code
@tuesss
@tuesss Жыл бұрын
I have only used VSCode very briefly, but I think it's Error Lens.
@syncpoint
@syncpoint 3 жыл бұрын
Hopefully there are more in que regarding rust.
@alexzander__6334
@alexzander__6334 2 жыл бұрын
loved that part with the diagram
@jeffg4686
@jeffg4686 2 жыл бұрын
The comment about a move being a memcpy, is that only true in the context of moving across a thread boundary? I always think of it as more of "compile-time rules enforcement" (playing the game of the language so to speak) than an actual memory operation at runtime, but I don't really know ... Was just the way I had envisioned. Maybe it's a runtime thing for thread boundary, but compile time thing otherwise. Something to ponder. I would think for a thread boundary, it would have to be an actual runtime consideration (move to stack of other thread), but otherwise I envision compiler enforcement only, and no memory op. I guess the bigger point is that a move may or may not be a memory op (i think...)
@AyoDamilareMichael-g3y
@AyoDamilareMichael-g3y 7 ай бұрын
What font are you using in the video ?
@thepuzzlemaker2159
@thepuzzlemaker2159 2 жыл бұрын
Nice video! Just curious, what's that program around 8:18?
@fasterthanlime
@fasterthanlime 2 жыл бұрын
That's draw.io, now called diagrams.net
@kajacx
@kajacx Жыл бұрын
This is about deep references. Still interesting, but I was hoping it would be about shallow self-referencing. Much more fun, since you cannot even move the struct then. Does "Pin" help there?
@Laura-hl3hg
@Laura-hl3hg 2 жыл бұрын
Hey, I love this video, you're amazing at explaining! Could you tell me what software you're using to make those diagrams? I feel like it could be insanely helpful for sketching out the logic flow of programs.
@fasterthanlime
@fasterthanlime 2 жыл бұрын
I'm using draw.io, now named diagrams.net
@lucasemanuelgenova9179
@lucasemanuelgenova9179 Жыл бұрын
Which drawing program is that?
@miniappletheapple
@miniappletheapple 2 жыл бұрын
Which application are you using?
@jx6773
@jx6773 2 жыл бұрын
what vscodes extensions are you using???? those seems useful
@dj-maxus
@dj-maxus Жыл бұрын
Currently, `std::thread` also have `scope()` which is pretty nice
@newclarity4228
@newclarity4228 9 ай бұрын
What software do you use for your diagrams?
@fasterthanlime
@fasterthanlime 9 ай бұрын
In this video, draw.io / diagrams.net
@fasterthanlime
@fasterthanlime 9 ай бұрын
In this video, draw.io / diagrams.net
@JimmyZeng
@JimmyZeng Жыл бұрын
I think there's some serious misunderstanding about "move" in rust, "move" here does not mean the memory address is _moved_, it's just the ownership of a value been moved.
@micycle8778
@micycle8778 Жыл бұрын
but when a closure in rust has the move keyword, would it not copy stack allocated memory into that closure's stack, therefore changing the memory address?
@JimmyZeng
@JimmyZeng Жыл бұрын
@@micycle8778 ​ I think the point is: when we are talking about move, we are talking about ownership, not address.
@calisti9308
@calisti9308 Жыл бұрын
@@JimmyZeng Ownership definitely moves, while the address *might* move, or be optimized away.
@baggern
@baggern 3 жыл бұрын
"load bearing keyword" lol
@joshuahab
@joshuahab 3 жыл бұрын
What if you created an ImmutableString type that can be built from String? In that case it seems like self-reference to the ImmutableString could be made safe.
@DeviRuto
@DeviRuto 2 жыл бұрын
You can always leak() the string. Makes anything 'static
@SophieJMore
@SophieJMore Жыл бұрын
I think the main problem here is that you can't move values that are being borrowed, even immutably. Generally, it's unsound to do so, because it might invalidate the data that's being pointed to. Here it's not unsound, since the data that's being pointed to is actually on the heap. However, the compiler can't really meaningfully differentiate between the two cases. To it, &String and &str both borrow from String, so it would prevent the String from being moved. Even if you wrap a String to make it immutable (psst, better use a Box instead in this case), you still don't be and to make a self referential struct with it (without unsafe). So, if by "could be made safe", you mean not using unsafe blocks, then I don't think it's possible. Otherwise, you can always make a safe API around unsafe code, just be careful.
@gabrielnuetzi
@gabrielnuetzi Жыл бұрын
In 2023 is there an safe way of achieving self referencial structs? ❤??
@zerker2000
@zerker2000 Жыл бұрын
OwningRef seems like it would be useful here, if generally limited.
@buzdxkysguge2334
@buzdxkysguge2334 3 жыл бұрын
Good work! May I ask what's the software you used to draw those diagrams.
@meain_
@meain_ 3 жыл бұрын
Kinda looks like app.diagrams.net/
@numtostr
@numtostr 3 жыл бұрын
This was indeed fun. Thanks!
@kh0kh0
@kh0kh0 3 жыл бұрын
More please!
@maxwellclarke1862
@maxwellclarke1862 Жыл бұрын
I think RcString is the best solution here
@manawa3832
@manawa3832 Жыл бұрын
Great video also keep in mind aliasing could still screw things up.
@contactron
@contactron 3 жыл бұрын
So you are saying std::thread::spawn never drops any resource moved into it? That seems like a major source of memory leaks. Good to know.
@fasterthanlime
@fasterthanlime 3 жыл бұрын
It definitely drops resources at the end of the thread, but the compiler cannot statically tell *when* unless you use something like crossbeam's scoped threads (which is not always what you want either). But it's easy to check that it does in fact drop stuff when they fall out of scope: play.rust-lang.org/?version=stable&mode=debug&edition=2018&gist=2968a70cb054657ff7134dce01679c49
@contactron
@contactron 3 жыл бұрын
@@fasterthanlime OK, good, that's what I would expect. I originally misunderstood what you were saying at 14:00, I thought you were asking why it wasn't safe to drop input, which seemed odd to me because you had moved input into the new thread. But really you were asking *why* you had to move it into the thread. Anyways nice video.
@timo7964
@timo7964 3 жыл бұрын
Amazing video! thank you!
@joaoalves1359
@joaoalves1359 3 жыл бұрын
Excellent video
@bozhidaratanasov7800
@bozhidaratanasov7800 Жыл бұрын
Can anyone point me to good material on the topic of building your own LinkedList/ Graph type structures in Rust? This is what I understand by "self-referential".
@bozhidaratanasov7800
@bozhidaratanasov7800 Жыл бұрын
To answer my own question, there is a reading online called "Learning Rust with entirely too many linked lists". It's dence with useful Rust examples.
@JihChiLee
@JihChiLee 3 жыл бұрын
Love you videos! keep it up!
@ibrozdemir
@ibrozdemir Жыл бұрын
relatable intro xD even tho i never coded in rust, i understand the pain you are having.. why would anyone make another language as fast as c++ but puts tonns of restriction so no one can use it as free as they want
@TheMrKeksLp
@TheMrKeksLp Жыл бұрын
Because having the compiler check the code for you is better than you checking the code and failing
@ibrozdemir
@ibrozdemir Жыл бұрын
@@TheMrKeksLp thats not checking thats restricting.. and yes you are right, with c++ you will get more crashes and bugs, however thats (in time) fixable.. since %99.9 of programmers not writing codes for spaceships, its allright
@thirdreplicator
@thirdreplicator Жыл бұрын
Very nice!
@pepoviola
@pepoviola 3 жыл бұрын
Awesome video!! thanks!! :)
@kibar9155
@kibar9155 3 жыл бұрын
awesome! thanks
@Hector-bj3ls
@Hector-bj3ls Жыл бұрын
Why did we get Pin instead of relative pointers? The issue is that references and pointers have absolute addresses. So &a returns the absolute address of a. If we had some way to do relative pointers/references we could have a pointer that is more of a "take my address and sub/add x to get to the thing I point to" instead of "I contain the address of the thing I point to".
@LeandroCoutinho
@LeandroCoutinho 3 жыл бұрын
Amazing!!!
@mattdavies6820
@mattdavies6820 3 жыл бұрын
`scoped_threadpool` may be lighter weight than crossbeam
@WolvericCatkin
@WolvericCatkin 2 жыл бұрын
_`starts_with_capital_letter`..._ 🤦‍♀️ Or just `.starts_with(|&x| x.is_uppercase())`...
@Dorumin
@Dorumin 2 жыл бұрын
Yeah that function was really awkwardly written, but it did the trick I guess. Is &x necessary? Shouldn't the dot operator auto deref?
@TheMrKeksLp
@TheMrKeksLp Жыл бұрын
Many roads lead to rome 🤷‍♂
@MykolaTheVaultDweller
@MykolaTheVaultDweller 11 ай бұрын
good video. but where is self referential structs? looks like clickbait title
@theLowestPointInMyLife
@theLowestPointInMyLife Жыл бұрын
GUI in rust lol
@qaqkirby9781
@qaqkirby9781 Жыл бұрын
😂😂😂
@alekxcrafter
@alekxcrafter 2 жыл бұрын
rust is cringe
@happygofishing
@happygofishing Жыл бұрын
Language for transsexuals that failed to use makefiles and pointers
@jonathanmoore5619
@jonathanmoore5619 3 жыл бұрын
Contrived. "Main" is the program. Join the thread and end, else send the changed input to another file. You're creating an unnecessary problem.
@Iogoslavia
@Iogoslavia 2 жыл бұрын
Oh man, I thought there was going to be some neat trick to create self referential structures without the unsafe keyword 🥲
Causing problems with Rust traits (then fixing them)
28:06
fasterthanlime
Рет қаралды 20 М.
Choose the Right Option
18:14
Logan Smith
Рет қаралды 75 М.
Counter-Strike 2 - Новый кс. Cтарый я
13:10
Marmok
Рет қаралды 2,8 МЛН
OCCUPIED #shortssprintbrasil
0:37
Natan por Aí
Рет қаралды 131 МЛН
Как Ходили родители в ШКОЛУ!
0:49
Family Box
Рет қаралды 2,3 МЛН
Catching up with async Rust
17:55
fasterthanlime
Рет қаралды 22 М.
Rust's Most Important Containers 📦 10 Useful Patterns
17:11
Code to the Moon
Рет қаралды 131 М.
strace feels like magic - let’s fix that (with Rust)
16:21
fasterthanlime
Рет қаралды 50 М.
Rust Functions Are Weird (But Be Glad)
19:52
Logan Smith
Рет қаралды 144 М.
5 traits your Rust types must implement
10:39
Let's Get Rusty
Рет қаралды 63 М.
Let's make an htop-like in your browser (with Rust)
51:25
fasterthanlime
Рет қаралды 84 М.
The Dome Paradox: A Loophole in Newton's Laws
22:59
Up and Atom
Рет қаралды 1,1 МЛН
Improve your Rust APIs with the type state pattern
14:45
Let's Get Rusty
Рет қаралды 91 М.
POV: I'm on my third coffee and you just asked me how the internet works
21:20
Rust Demystified 🪄 Simplifying The Toughest Parts
14:05
Code to the Moon
Рет қаралды 192 М.
Counter-Strike 2 - Новый кс. Cтарый я
13:10
Marmok
Рет қаралды 2,8 МЛН