all right let's kick it off thanks everybody for joining this is community meeting number 10 so let's start with a quick agenda we have an awesome talk from Evan on linear non-destructible types in Mojo and then we have a couple of Q&A questions that we're going to go through so G hand it over to Evan oh I think you're muted how about now there we go cool yeah give me sorry give me just a second I'm trying to figure out how to open speaker mode on this thing again there we go okay y'all see my screen yep we can see it cool okay so yeah I'm about to talk to you all about linear types and uh oh yeah this is my first time uh here in this community meeting so super happy to be here and I'm super glad to hear uh all your thoughts on what I'm about to talk about um so yeah we're going to talk about linear types uh I also kind of call them explicitly destroyed types or explicitly destroyable types kind of depending on what mood I'm in um but Evan do you wanna do you want to introduce yourself a bit for folks who don't know you um yeah yeah sure um so I'm Evan I'm a Mojo compiler engineer uh on the Mojo team um so I've been working a lot on the uh on the compiler internals and also a little bit on uh figuring out what features go into Mojo So yeah I'm pretty excited about that that um yeah that's that's pretty much me um so uh I'll explain what linear types are uh but before I go too deep into it I want to kind of show you all a riddle so can you spot the problem with these comments uh the first one is saying you have to remember to call join or detach uh on a thread before it goes out of scope uh sorry let me mess with the screen size there okay yeah the first one is saying if to remember to call join orach on the thread before c a scope uh the second one is saying hey caller I am returning to you a promise please remember to call set value on it uh the third one is saying remember to decelerate the car before you stop driving and the fourth one is saying remember to destroy or sorry before you destroy an entity remember to remove its ID from a location map so without knowing any of the context of any of these you can kind of tell the common problem is that they all rely on the programmer to remember to do something and what humans are like not great at remembering to do something especially this human uh and modern compilers don't really help us that much they're great at telling us what we can't ever do like with static typing but they don't really remind us to ever do something and we can kind of get halfway there with r aii but that really doesn't help with any of these examples so we're going to talk about what's a linear type how it prevents bugs like these and then uh the third thing I'm going to show you all the linear types hidden superpower uh that kind of blows my mind and I can't wait to get to that slide um and if you have any questions uh just remember the number in the low right there and I'll answer them at the end okay so what's linear uh a linear object must eventually be consumed exactly once that's the normal definition I usually see on the internet but it's kind of a terrible definition and so if you ever see a definition like this I recommend ignoring it because it really doesn't it doesn't capture the potential of linear types um so I recommend I recommend ignoring it and uh we're going to use my definition instead for now uh my definition is a linear object can't just go out of scope you must eventually explicitly destroy it in a specific way and I'll show you what I mean with this basic example that I think really shows how they shine um if any of you have used C+ plus STD thread before you've run into this and you you probably have Battle Scars From The Encounter um if you accidentally let this thread go out of scope like this code on the right uh without calling join or detach on it first it's Destructor does doesn't know what to do does it wait for the thread to join does it detach and let it run in the background the poort destructor doesn't know so it does the reasonable thing which is panic and crash your entire program by calling SD determinate uh so there is and there is for those of you who know C++ really well there is a half solution called stdj thread um but that's kind of a half solution and I'll show you I'll show you more about that after we show how linear types solves it in a way that in my opinion is a little bit better okay so it would be nice if we had something like this nojo code on the right we have a thread in a local variable named T and when we let it go out of scope the compiler tells us can't delete T call join or detach instead and it would be pretty nice to catch that at compile time like that okay so how do we make that happen well let's look at the definition of struct thread right here and first notice this explicit destroy annotation this means there's no Destructor an object of this type can't just go out of scope and right after that we specify a hint to show the user in the airor message when they accidentally do let it go out of scope but the user has to get rid of it somehow so let's have uh some methods that take ownership of the object using this owned keyword uh and this isn't new like for those of you who know Mojo really well already it's just the keyword that means it's a pointer to data that you now own uh and this method does some things and then it uses this new proposed destroy keyword to finally destroy it um now join and attach I want to I want to emphasize here join and detach are just regular methods it's the destroy keyword that would be speci special um and a types user like like that function Fu uh they wouldn't use destroyed directly it's really one of the type's own methods that uses destroy when it's finally ready to finally explicitly end the object um and you can kind of see how like okay if the user doesn't call destroy if they can't call destroy um and only a method can call destroy then that means that the user is forced to eventually call one of these methods okay so yeah some details on the destroy keyword uh it'll eventually well it'll end the object uh it'll call any destructors on any normal members inside it and if there's a linear type inside this linear type then it will show any error if you haven't handled that member linear type and that's that's kind of weird and then feel free to ask questions about that afterwards um so yeah we now have a linear type uh or in other words an explicitly destroyed type okay so how does the user fix their function now well they have some options uh they can either call a method that takes it as an owned argument and destroys it like you know a call to DET a call to detach um and and more on this uh see how it says yeah sorry I'm getting confused by my notes here um another option is to postpone destroying this thread and instead maybe uh add it to some longer live data structure like this threads list and whoever owns this list of thread the compiler helps them remember to not let that go out of scope to instead do something like drain the list uh and do something with all those threads before that list goes out of scope uh a takeaway from this slide is that thread is linear so a list of thread is linear too and the compiler will make sure that someone will handle those threads in that list and I'll talk a little bit about how it is that a list is linear if the member thread is linear uh in a bit okay so the last option is to return T karat uh which means of course uh move T into a return value for our caller uh and now the caller is responsible for figuring out what to do with this thread now they have to call either join or detach or move it to somewhere else so that's the first example of a linear type uh the compiler ensures that you eventually actually decide when and how this object should go away you could say that it ensures you eventually make a decision and that's a clue to uh that's a clue to the linear type hidden superpower that I'm going to get to in a bit uh it ensures you eventually make a decision okay so there are a lot of types like this where the where the compiler shouldn't assume when and shouldn't assume how you want to destroy your object um I mentioned jthread earlier J thread tries to solve this problem by just assuming that you want to join but this can be harmful like in this code here uh where we have a loop and we accidentally let the thread s out of scope too early and you can see by using J thread we've opened ourselves up to a bit of a risk where we accidentally forget to do something with it and we accidentally serialize this Loop that we were hoping would be parallel so as you can see even though RI was you know a huge step forward and it can solve some things that doesn't really help here linear types help in much more situations um they take it a step a little I like to say that it takes it a step further than ra AI uh you specify exactly when and exactly how your object goes away uh and also if some of you are wondering why C++ doesn't do linear types it's because of exceptions and stack unwinding um and stuff like that and I'll talk about that in a bit too um yeah okay so that was one example but there are a ton of ways uh that linear types help us and let's talk about that second one don't worry we won't go through all of these we don't have enough time but I'll just talk about the my favorite ones um yeah let's talk about this second one so this is another example which also coincidentally has something to do with threads uh if you want to give a value to to another thread you often use a promise and when you call set value on it the other thread can use the value and sorry can see the value and use it but if you forget like in this code then the other thread never gets the value so if only something could help us remember to call set value uh so here's the Mojo equivalent you can see how it shows an error if we don't call set value and let's see the Promise struct now uh here's the promise struct with the explicit destroy annotation to opt in to it being a linear type so it can't just go out of scope and there's us specifying that error message and of course we got to give the user some way to get rid of these things so we give them this set Value method which takes an owned self and also requires the value argument right and does some dot dot dot things and then destroys the promise uh so how do we fix that error message well once again we can move it into a method or we can postpone it by moving it somewhere else but let's say that we do call set value and oh wait we realize now we never actually calculate the result so let's do that now there we go okay so it's kind of like r i but the destructor has a name and can require arguments so and here's here's the second clue to the linear type uh hidden superpowers the compiler ensured that we eventually called set value and it kind of also ensured that we eventually calculated that result right there so that's the clue okay last example uh so continuing the theme of threading here's an example with a future that will contain an important linear thing perhaps it represents a pending request whose response they need to act on or perhaps it represents the airplane's current flight that they need to correctly land a user shouldn't just let this thing go out of scope fortunately they get an error about it and if you're curious this is how the future struct would be defined uh so how does the user fix this error well like the hint says they can call F Karat doget and hey they get the result out of that which is nice like you see the thing equals f.g uh it's almost like linear types ensured that we got that result and took ownership of it um it's kind of like our AI where the destructor returns a value too you could say uh so oh yeah this is the third clue this is the third clue to the hidden superpower it also ensured that we eventually took ownership of that result thing like I said but that's the clue okay so from these you can kind of see a pattern emerging and no it's it's not that they all have to do with threading that's just a coincidence in the examples I chose so these are the three things that linear types did for us uh they ensured we eventually made a decision like about those threads they ensured we eventually calculated and gave a value like to that promise and they ensured that we eventually took ownership of a value like from that future so these hint at something and if there's one thing I want everyone to take away from all this it's the linear type hidden superpower at long last which is that with linear types you literally control the future you can craft your linear types and their methods to ensure that specific things eventually happen in some order at some point in the future this is amazing if you're making an API for other people to use by the way um yeah and so like you don't know when these things will happen you don't know when they're going to call these methods that uh are blessed to destroy this object but as soon as you create that linear object you know that those decisions will be made that data will be calculated given or taken and destroyed in the correct way and you can combine these in interesting ways to help your user uh correctly use your types and I'll show you in a few slides just how powerful this is when a simple linear type single-handedly solved three cach inconsistency bugs uh for me uh yeah so but before I talk about that I want to nerd snipe everyone uh and I'm going to talk about all these ways that I think linear types can help us all these situations and for most of these I'll just quickly lay out the scenarios and I'll leave their Solutions as an exercise to the reader um but ask me in the Q&A after if uh if you want to know specifics okay so we saw already how they help with threads Futures and Promises they can also ensure that you correctly decelerate before ending your current driving session for a car for example uh they can also ensure that you stop a rocket booster correctly depending on the situation not too early not too late uh they can ensure that you handle a failed request rather than Dropping It On The Ground they can make sure that you actually update your database row to reflect the outcome of the calculation they can prevent node leak this one's this one's a little Arcane and I wish I could you know go into this a little more deeply but I'll I'll leave it vague for now uh they can prevent node leaks when you accidentally drop your owning handle so to speak and cause orphan nodes in a uh a vector backed collection uh they can ensure that a message is correctly handled and not forgotten in an orphaned channel so to speak this is a common problem in goang for example so I'm pretty excited that we can actually solve this with linear types uh and this one blew my mind so spectacular spectacularly uh that it gets its own slide so this is something from Veil uh if if uh for those of you that don't know me uh I also work on uh my own personal project called Veil uh and it's a it's where I implemented linear types uh in the first place about three four years ago and that's where that's where I actually saw this happen and so that's why I'm excited to bring it to Mojo potentially um so yeah here's an example from Veil and I'll show it in Mojo syntax just to be merciful to you all uh so here's a couple stru that I was using for a game and I'll explain what they do this is the central list of entities in the level uh and we wrap it in this live entity list struct right here and we have an add method and a remove method nothing surprising so far but this is interesting the add method returns two live entity handles that refer to the entity we just added now both of these are instances of this live entity handle struct that has its index in the list and explicit destroy on it so that can't just let it go out of scope it looks like the only way to get rid of them is to eventually give them back to this remove method uh yeah so these handles only exist while the entity is still in the list so where did they go in the meantime well in my case they were in some maps they were in a location to entity handle map uh to tell who's at what location and also a map of team to what entities are in that team and here's some comments just to emphasize that uh so what were the implications well the first one is that if you know like if you have a live entity handle or if you have a borrow reference to it you know that it is still in the live entity list uh it's kind of like weak pointer but it's it's more General like uh weak pointer you can know whether something still exists somewhere but with this this shows that if we have a handle we can assume an arbitrary property of a distant object such as its membership in our source of truth list over here it's kind of like spooky knowledge at a distance or linear entanglement but we still have to come up with a cool name for that if this happens also uh thank goodness dangling live entity handles are impossible and that's because you must take the handles out of the maps before you can use them uh to remove them to remove the entity from the list uh so the maps can never get out of sync with the list because of this in other words we just solved inconsistency bugs for these collections by crafting our linear types in a certain way and making use of these linear type superpowers and like maybe I'm just a language nerd I mean we know I'm a language nerd but that's ridiculously cool to me uh I someone said that the hardest two problems in computer science were naming and cash invalidation and I like to think that now there's only one problem now naming uh so this blew my mind when I discovered this uh I thought I was just making a stct linear on a whim to see what happened uh but I followed the ripple effect seeing what Veil uh was trying to show me and suddenly three places in the code kind of lit up and it showed exactly where i' accidentally removed from only some of the collections and not all of them and I sat back and I was like holy crap like a compiler just did that that's crazy um and I should say like this doesn't solve every bug and this isn't a Panacea um like for example you can still hand into mismatched handles for example but in my opinion that's a much better problem to have like that's a much easier problem to see and it's a much easier problem to solve rather than uh you know State inconsistency problems which are the bane of my existence okay so one last thing to freak people out and I I don't expect anyone to understand this like this is like deep dark Arcane magic and I'm not explaining it very well um but if you think of these two candles as references then this is kind of like some sort of non-scoped borrowing or compile time reference counting or something because like these are references that just Escape into the wild right but then you kind of put a constraint on the universe that like people have to do things like other functions have to orchestrate it somehow so that these two handles make it back to me so that I can be sure to destroy them at the same time I'm removing it from the list uh and that kind of that kind of Rhymes that kind of Echoes with like reference counting in a way but this has no runtime overheads this is like compile time reference counting I know I'm not making any sense about that so like feel free to feel free to ping me on Discord uh if you want more more insights into that one um okay so back uh back down to earth um um yeah so fitting this better into the context of Mojo we already know that Mojo kind of has like I don't want to say Mojo has R AI right because a lot of people think that you have to like it's like C++ a rust style RI where an object uh is destroyed at the end of the scope but it it kind of has our AI in a way but in the form of ASAP destruction um and uh for for those of you who don't know what asep destruction is um yeah it means that Mojo in inserts destructors not at the end of the scope it inserts them as soon as it can in this case it sees that the last use of ship is right here so it inserts the the Dell Destructor call right here um and this is awesome uh but uh it fits in with linear types in a certain way and so this yeah in this slide I want to talk about how linear types fit in with that aspect of Mojo and also with statements and uh and asyn away so ASAP destruction is in my opinion really good for freeing things right or like releasing some buffers on a GPU or doing things that don't have any non-local effects uh yeah that that have any like complex side effects in the system uh I think ASAP destruction is good for that and with statements are really good for when you want to make something explicit we know that with statements will delay something's destruction until the end of the scope that's great but it only works in the current scope and that's kind of the same thing with asyn a we but in a concurrent way there's no way to ensure that something happens past the end of your current scope and that's where linear types fit in and I also kind of think that linear types would be good for when we want to ensure destruction happens uh in in a way that has complex side effects because it's explicit right so if something's Destruction has side effects in the rest of the system I think it should be explicit but that's my own personal opinion and I kind of want to hear what you all think about that um also I don't think I have enough time to talk about the container problem because I think I'm running long no you have you have plenty of time you can talk oh okay cool I'm how long do I have however long you want okay awesome yeah okay I'm Gonna Keep geeking out about the container problem then um so okay so some of you are probably wondering uh how do we make this work for like like list right uh or I guess okay I'll I'll stick to the slide okay box for now I'll talk about list in a bit okay so how do we make it work with generic containers like list optional dictionary box and so on they have their destructors but their elements might not so what happens in other words just to clarify the problem we have a box of T which has a d Destructor method which usually calls T's D Destructor method but if T doesn't have one like if it's a linear type what happens we get that error right so the answer is to use conditional conformance where our box only has a Dell method if T hasit Del method that we can call something like this and uh this I implement it in this Veil but I don't have it implemented in vojo and I don't vojo I don't have it implemented in Mojo yet um but this is how it worked in Veil and I'll keep showing it in in Mojo syntax um so our delete only exists when T has a delete Sorry Destructor I shouldn't say delete and uh and this approach works for linear types uh like we apply it here to the Dell method um and that makes it work for containers like this so uh I want to restate something to make it super clear what's going on here there's two ways to make a linear type in this kind of proposal here um there's there's using the explicit destroy annotation and there's using a condition Destructor like this if we have a conditional Destructor that means that when the destructor doesn't exist the type is a linear type and that's what we see going on here the other way to make a linear type is to use the explicit destroy annotation um so yeah Mojo will likely be a bit different in the syntax than this uh because you know that's something we all have to discuss um and it'll be using a trait instead of asking directly for whether this function exists uh but it's close enough to how I did it in Veil a long time ago that I'm pretty confident it will work um yeah so that's pretty much the end of the content I had uh I've got a few notes here um yeah one of the big open questions is whether the linear types uh whether the benefits of linear types are worth this extra thing we have to keep in mind for our Dru uh for our containers like this um and it's it's something I want to get all of your thoughts on um because like it's really hard to see how like what kind of an effect this feature could have on a language like Mojo uh without actually trying it and like you know since I'm on the compiler team I managed to like you know hack up a little prototype of it and on a weekend but I haven't yet figured out how it could affect the architectures of our program um so I'm also going to be relying on on the crystal balling skills of of all of you out here to to kind of think about like how could this affect Mojo programs um yeah okay the other note I have the other note I have here is um that there is something there's something I didn't get to talk about in this talk uh which is super interesting to me because I'm a language nerd um but some of you might know what I'm talking about here and that's that linear types have an interesting interaction with borrowing uh like like Mojo's like Mojo's going for and oh man can I yeah I'm going to go back to that slide uh oh yeah it's this slide yeah okay so here on this slide I showed uh an add and remove method for the live entity list struct if you think about implementing a get method now normally the get would take an integer Index right and it might return an optional uh reference to an entity uh and it's optional because if the user hands in an index like what if they handed in something out of bounds right so that's why it would normally have to return an optional of a reference to that entity but hold my beer here's what I'm thinking if we also make a get method that takes in not an index but a reference to one of these live entity handles it's kind of like we can use that as proof that that entity is still in the list so we no longer have to return an optional reference to the entity we can just return the reference to the entity so it's like we've kind of solved the look up after remove problem right the user can't accidentally hand in an index to an entity that no longer exists which means that our get method can return not an optional but an actual reference to the entity so it's it it's something that I think could like have a few like run time you know speed implications uh a lot of reliability implications we've narrowed out an entire class of bugs there um but yeah if anyone wants any more details on where I think I'm going with that just uh ask ask me in the Q&A um okay I'm I'm losing myself in the slides here yeah so I think that's I'll stop for I'll stop here for now um but yeah if anyone has any questions feel free to ask them here's some good unanswered ones that I didn't answer in the talk uh if you uh if you want to talk about these so yeah that's all I got for now awesome thank you so much um any questions for Evan yeah awesome talk kin thanks so much man that's that's really cool um I was just wondering about like say for unsafe pointer are you able to use a linear type on that to make sure that it gets Freed at some point yes yes and I really like that because like when you're dealing with unsafe pointers you're often dealing with things that don't fit well into the you know safe type system in general and often those things have you know explicit effects on the system um so yeah like in in principle you can make unsafe pointer into a linear type like you you could change the language to do that make all unsafe pointers into linear types I don't know if I would make all unsafe pointers into linear types though um I think the right approach in that case might be to make a a linear struct that's a wrapper around that particular unsafe pointer um that's like crafted to your specific used case like like so for example if you have a unsafe pointer to oh I don't know I'll fall back on my normal example a spaceship right if you have an unsafe pointer to a spaceship right um You might want to make a linear struct wrapper around that unsafe pointer called uh unsafe spaceship pointer and that will enforce that you correctly destroy that spaceship in the particular way that makes sense for that situation in that particular unsafe pointer does that make sense yeah yeah we like for for circumstances where you might not want to free something like uh just let the program free it at the end of like once it finishes executing yes yes exactly and that that's one of the strengths of linear types is that like they make it so you don't have to just free something into Destructor like because destructors and like the Dell method are mainly for cleaning up and I think they should stay that way because of ASAP destruction the language moves the destructor as soon as it can and sometimes that's surprising like I I think Destructor in mode should only be for cleanup anything more advanced than that I really I really think should be in a either withd statement or a linear type but that's my own personal preference I'd love to hear what the rest of you think that's really cool it just be interesting to put it like on on safe point and see all the places in the code base where we're we're not freeing properly yeah yeah I have a question um how should we think about like the main downsides of of linear types obviously you made a lot of you know points about why they're awesome and and the hidden superpowers but you know what what would you say is like the main reason that more languages don't have them yes I'm so glad you asked that because I totally forgot about that I was going to talk about that and I didn't um so the main reason that for example C++ and rust don't do this is because they made in my opinion well I can't really call it a mistake cuz they had really good reasons and do give them benefits but I think they made a decision that Mojo could improve on um their decision was to conflate exception handling with normal control flow uh so um in C++ and rust right if you know function a calls B calls B calls C calls D calls e um and E like at the very Lea function e throws an exception that exception is going to like bust it way through all the stack frames until it gets back up to main or a or a tri catch block or a catch unwind or whatever it is um and because of that because this stack unwinding needs to clean up any objects in its path it needs to clean up any locals that exist um it's going to need a way to clean them up it's going to need a function to call to free all these buffers that are in local variables uh that are pointing to the Heap and so on and so it needs a zero Arc Destructor to call um and that's why C++ and rust they saw that situation and so they decided to use the destructor which they were also already using as a way to just clean up things when they go out of scope normally I think that they should I think what we can do differently it's complicated cuz like mojo as far as I know Mojo doesn't have panics someone can correct me if I'm wrong because I'm I'm new here right um but since Mojo doesn't have panics it doesn't really have that problem it doesn't have exceptions it has errors which are much more explicit uh we don't have to have a zero argument Destructor so we don't have that problem but in in going back to the case of C++ um if they for example DEC conflated if they decoupled exception handling from normal going out of scope they could have had a separate on Panic Handler or I guess for C++ it'd be called on exception Handler and that would be called when an exception is blasting its way through the stack frames um same with rust it could be called on Panic there uh in Veil this was what I was planning to do with Veil is have an on Panic Handler called specifically for stack unwinding and then the object is free to choose for normal control flow um whether it wants to do something like I set destruction or linear types so in a way we decouple exception handling and panic handling and stack and finding from normal control flow handling which can be accomplished really nicely with ASAP destruction or linear types um yeah I hope I hope that makes sense yeah that does thank you uh couple questions in the chat uh linear types have been around in the programming language world for over 30 years is there any example of a non-toy linear language that would be a helpful case study yeah yeah thank thanks to whoever answered Veil to that I appreciate that um Veil is is uh depending on on how you define it Veil could be called a toy language like I don't call it a toy language you know it's it's working on for a decade and it's 140,000 lines but uh it doesn't it doesn't have that like widespread adoption and uh that's important here because uh that means that we don't know how linear types will do at scale right um like for example there's something uh I noticed architectural benefits and costs to using them in Veil uh one of the costs was that it made our generic functions a little bit trickier uh for these uh generic containers like box list dictionary and so on um another potential drawback that I don't know how it's going to do at scale is um a type being linear seems partially viral like you saw how a if a if a thread is linear then a list of thread becomes linear when well sometimes whoever it is that owns the list of thread uh they're just like a class or a struct somewhere and they don't know what to do with this list of linear uh with this list of threads and so they make themselves linear so they leave it to their owner to figure it out um in other words what I sometimes saw in the little prototypes I did in Veil is that this linearity would kind of virally trickle up to the root container of the entire program this didn't always happen um but this sometimes happened and in all the cases that I saw this was actually a benefit but this can have downsides um every viral constraint has downsides asyn a weight linear types uh SN and send uh in Rust is an example of this um so I so I really like your example of like have there ever been any non-toy linear languages no I think I think for us the answer is no and so we don't know how this would turn out uh and so that's why I really want us to like consider carefully uh what this feature could be like what it could be like specifically in Mojo and what it could be like in the real world in Mojo programs uh and for various different architectures that might be in the wild um yeah I hope that answers the question one other question in the chat uh is there a point at which the type system gives up and will there be an escape hatch for instance if I have a bunch of very messy rules around when objects are removed from a collection and freed such as when I get a message from another server that says to free uo idx do you attempt to prove that the list is empty before I free the list oh that's a super good question I love that one yes this is a case I ran into Veil uh ran into unveil and I it took me like I swear a week of bashing my head against the wall before I figured this one out um yes the answer is yes the list uh we do have to prove that the list is before we free the list and most of the time it is uh the user would call a function called drain I think I called it drain or drain all um and that is a method that consumes the list and also calls a user specified function for every element in the list um and that wasn't the only option the other option was a function called uh like assert empty uh or it was called assume empty and that consumed the list and also didn't assert uh to make sure that the list was indeed empty so I I expect that these kind of like Escape hatches like assume empty will kind of naturally pop up in the apis for these linear types like they did for uh list in veil's case but I think even past that like like that's an escape hatch that the definer of list can Define if they are looking forward enough to see that the user will need this escape hatch I see some Merit in the argument of maybe we should go a little bit further than that like maybe we should just allow users to call the destroy keyword on any type I and it it sucks because like saying that means that I'm suggesting we offer a feature which undermines the guarantees uh that linear types give us but I think in real world codebases situations often arise where like it's extremely difficult to work within a very strict system in fact we we saw this in Rust all the time uh this is why rust has ref cell uh and RC and uh Arc and mutex and RW lock uh these kinds of like constraints are very helpful until the last 5% of use cases where it's like oh but the real world doesn't work as well as you know math and linear types too so I personally am leaning towards recommending we have the user be able to called destroy from anywhere um maybe we could have some like extra rules of thumb like hey if you're using destroy keyword uh outside of the class that def find it like please put a comment explaining why and like we've seen uh a similar recommendation for example in Rust with the unsafe uh with unsafe blocks like if you use an unsafe block and you want to break out of the the strong type system guarantees of borrow checking put a comment over it and that's like I think there's linters uh that enforce that in Rust too um so yeah that that that would be the escape hatch I recommend I'm just going to reread the question real quick um yeah yeah I think that answers it but let me know if that didn't answer it there's actually one other question that was uh submitted ahead of time and I was going to share it in the Q&A section but we can just do it now so I read your blog post about the seven Arcane uses of linear types I'm just wondering what the differences are between the approach to linear types in Veil compared to linear types in Mojo would we be able to use linear types to prevent leaks for example by the way I love your blog and that's from uh Richard I know you covered some of it but yeah oh my God I love that um yeah so the differences between Vil and Mojo I I think at a basic level the approaches would be the same [Music] um there's there's there's one big difference that I found so far um yeah there it is so so right here this this is Veil thinking right here this is not how I think we would do it in Mojo and the veil thinking here is that it says where exists T colon colon Dell now Veil can do this thing called concept functions um where you can enable a function if a specific you can enable a function like this Destructor Dell if another function exists for a given type uh and that's because in Veil like there's no there's no difference between methods and functions they're all just functions right so instead of a a moja style thinking of asking if a type has a method you know Veil asks if a function exists anywhere in the visible scope um and and on top of that Mojo doesn't even think in terms of functions like this it thinks in terms of traits this line in Mojo would probably look something like where T implements trait implicitly destructible for example and implicitly destructible might be a trait that just defines the Dell function um that that's how that's how Mojo works right now and I don't think we should change that I don't I don't that was a bit of an adventure in Veil that was an experiment uh doing this concept functions thing um I think in this case Mojo will want to put a constraint on this to say where T implements some sort of trait that defines you know a delete method so that's that's the only big difference that pops out um between Vil and what I think we could do in Mojo um right uh mojo as far as I know doesn't have stack unwinding so it doesn't face the whole you know do we need a zero argument Destructor problem Veil would have a difference here in my mind veil was going to do some crazy things with like software transactional memory and I had all these like you know crazy you know plans on the wall uh and so Veil would have an on Panic Handler for that case but luckily moo doesn't have that problem so I guess that would be another difference between Mojo and Veil uh someone in the chat says Mojo has conditional methods yes yes it does uh it doesn't have them for destructors yet uh and it does think uh it does think in terms of implementing traits uh for these other methods already which is really nice um so yeah that'll fit in really well with moa's existing conditional methods yeah okay so there was another question how do you handle something no sorry there was a second half to that question you said Caroline can you remind me what that second half was and I think you kind of you did kind of touch on this before but would we be able to use linear types to prevent leaks for example yes yes and and it gets even better actually um so like regular ASAP destruction and regular RI they they kind of solve leaks right but but I'm going to blow people's minds here they actually how do I say this there's actually a class of hidden leaks that R aaii has not yet solved and an ASAP destruction doesn't yet solve and I'm going to try and tldr this as best I can if you imagine a doubly linked list implemented in something like C++ you would have a unique pointer an owning pointer from each node to the next one and all of the back pointers would be regular non-owning just raw pointers um but we see a certain phenomenon when we try and translate that C++ doubly link list into something that uses a borrowing system uh like Ruster Mojo and what we see is we end up putting all these nodes in a vector uh an array list and then we refer to and then a node will'll refer to other nodes via IND indices indexes integers um and there's a risk in there in that we lost our ai's power of ensuring that every node is eventually deleted right there's certain ways you can mess up a vector backed doubly length list that you can't mess up in C++ um and that's that we can accidentally mishandle the owning index so to speak of a given node linear types actually help that um with linear types you can have all of your dou link list nodes in a vector and what you would do is you would have every node would have a um an an optional linear wrapper around an index that refers to the next node if it has one and because that linear wrapper around the index is linear uh the optional uh becomes linear and so you're forced to handle in various places in the code which eventually means that you can't actually accidentally orphan a node in this Vector backed doubly link list um if you've ever heard someone claim that Java can have memory leaks this is also what they're referring to because this can happen in Java too um you will often have a central canonical source of truth list for all of your Java objects right and someone else is responsible for removing things from this list but they forget right because they forget to call dispose or they forget to do something um and that's a case where even Java can have something like a memory leak because someone didn't correctly handle uh their responsibility for removing something from a list uh and linear types are great at tracking responsibility heck they even they they enforce that you eventually call methods that's that's exactly tracking responsibility so yes linear types handles memory leaks and other kinds of leaks in a way that goes even past our Ai and an asep destruction which is pretty cool uh yeah I'm so glad I got to I got to ramble about that because that's one of my favorite things about linear types okay so there was a question question um in the chat there's one from Gabriel uh can the Readiness of being destroyed be encoded in the type of object as a parameter oh man you're going into some deep dark arts here yes yes it can um so uh if so yes and the way this can happen is if we have conditional conformance where a struct can can conditionally Implement a certain trait um okay actually my brain's kind of falling apart on this one um but I think what you're kind of getting at is some sort of type State programming I just want to say that at least um type State programming uh is where you get closer and closer and closer to making your type system really represent all the possible types sorry all the possible states of a certain State machine um and uh one one thing that I think is going to happen uh if we introduce linear types is that people will have a for example they will have a um a struct representing a rocket that is currently in flight right and like maybe this rocket is delivering you know snacks up to the International Space Station for these astronauts right and so you'll have a struct that represents this rocket that is currently in Flight right and then at some point um oh and you'd want to make that linear right because you don't want to just drop this rocket on the floor right you don't want to accidentally forget to go through its whole shutdown sequence so that would be linear and the only way to get rid of that linear type will be to initiate its its uh booster shutdown sequence and once you initiate that booster shutdown sequence it'll consume that inflight rocket and produce a currently shutting down rocket that's literally what the struct to be called the currently shutting down rocket um and then eventually you can only destroy that by calling land right once it actually gets to the the surface of the Moon or the ISS or whatever um now these are differences I I just describe different entire structs but there are ways that you can represent uh these differences not as entirely different structs but instead as a uh input generic parameter and I think that's kind of what you're getting at Gabriel um is encoding uh things in the type of the object that could influence whether or not we can destroy things and I think we can do that I don't know if Mojo has the machinery for this yet Gabriel you'd know better than I do um but I think if we can make it if we can take in an input generic parameter of a of a type and then make our spaceship struct in implicitly or explicitly extend that type that we're given in I I think we could do something like that I I'm 90% sure we can do what you're talking about but my brain's kind of falling apart um so I'll leave it at that for now okay we have uh two final questions hopefully your brain hasn't fully Fallen apart uh how do you handle something which is created on one device but destroyed on another for instance if I need to pass a linear lock guard or a ghost sell like proof token into a GPU and also since arm has brought it back to life how do we deal with Hardware transactional memory and linear types oh man okay I'm going to answer that that second one first because that's I had no idea that Hardware transactional memory is even a thing like I I think I saw some papers on that but I didn't know that was a thing yet um so in in Vil I I did explore kind of a software transactional memory kind of thing um cuz cuz Veil used regions under the hood and so there's this really cool thing where you could say um that within this scope this block uh there's kind of an inside world right which is the code inside the block and the data created inside the block and there's kind of all the data created outside the block that is reachable from the Block but it existed before the block and um one of the things that was cool about Veil which I don't know if Mojo can do and that'd be really cool if it could is that the type system would track which data came before and which data uh was created within this block and by the way I didn't implement this in Veil this is this is just like Theory crafting like this this could fall apart for a bunch of different reasons but since the type system knows the difference between data created inside the block and data created before the block then it could track any modifications that happen to data outside the block and then it could in theory record those changes in a what I call a mutation list and this mutation list uh would just be an Ever growing thing right an Ever growing uh not a linked list but some sort of like uh deck or a vector or something that would record all the changes to all the data that happened in this uh outside data and then if an if a panic happens inside that scope inside the block it would would just destroy all the data created inside the block and it would go through this list in reverse order and undo every single thing that this one uh scope did to the outside world's data um and how this interacts with linear types this yeah this is another one of the things that I was like bashing my head against the wall for a week on I'm like oh wait but what about linear types if a if a linear type enters this block and then like a panic happens and we destroy all the data created in the block block what happens turns out it's actually not a problem it's it's not a problem because as we go through all of these mutations that happened and you know do the kind of undos um it'll naturally just happen that the linear type that was brought into uh this um this scope would naturally be just ejected back out of the scope just as a natural course of doing all these undo operations so linear types actually isn't uh isn't isn't incomp with uh transactional memory at all I think I think um but of course I'm assuming that Hardware transactional memory works like veils region-based software transaction memory design worked and like I don't I don't even know if that's an an apt metaphor I don't even know um okay so you asked yeah your first question though was how do you handle something which is created on one device but destroyed on another um so this one yeah a similar question to this one is how we handle linear types that start on one thread and end on another thread and I love talking about that but I only have 5 minutes I think so I I'll keep this one as a tldr before I answer your actual question if we make a linear type on one thread and we want to send it to another thread we need to make sure that it actually gets to the other thread right and it's not just dropped as you know in an orphaned Channel which often happens in goang um so what we do in that case is we make it so uh instead of having a single Channel class we have uh one channel is represented by two uh objects a sender half and a receiver half and the sender half uh when we destroy it it will send a linear doesn't have to be linear it will send a close message right the receiving half is the interesting one the receiving half is the linear type that can only be destroyed if someone hands it the uh the closing message uh that was sent by the sending half and that in a way forces it to wait and read all the messages out of the channel until it gets to the final closing message okay but it just read all of those you know messages that were in the channel and so one of those will be the linear message that we originally wanted to send from The Source thread so yeah that's how we can kind of enforce that something happens over the thread boundaries and I think something could kind of be done in that way over two machines uh in other words one machine would have a linear type that can only be destroyed once the sending half has correctly closed down their connection um and and that that's kind of the the rough idea of how it would work the big flaw in this is that this is the internet and we're talking about TCP and TCP is unreliable as heck right and it's like there's there's no way to guarantee that we're going to get any sort of close message right so we're going to have up here it is an escape hatch uh for when the connection is just lost um and and designing that would be actually really interesting because how do you design the escape hatch um but also trying to uphold a lot of the guarantees that we get from linear types um I think it's possible I don't know every time I think something's impossible with linear types I end up smashing my head against the wall for a week and finding out that it is actually possible so I think there's hope for that one um hopefully I answered that well okay it looks like there's one last question um I think that was all of them um oh yeah just the last comment thank you so much Evan for the presentation and for answering so many questions let's all give Evan a big virtual Round of Applause if you want to come off meute won't Force you um all right so uh do do you mind stopping your screen share for a second yeah yeah thank you so real quick um we also have have a little bit of Q&A today we already answered the first question or we already answered uh this one for Evan but the first question uh is I believe Chris has mentioned the team was working on a road map I was wondering if there was a road map for the road map which is a very fair question uh the short answer is that we are aiming to share something in December the longer answer is is that uh we're hoping to have more specifics in terms of the timeline for you soon uh and we want to do this as soon as possible and really make sure everyone is on the same page um regarding the road map within our community so the other questions um Chris had a last minute Conflict for today so I want to give him the chance to answer the other questions there were really great questions submitted um those questions will be added to the agenda doc um I'll drop the link to the agenda doc in the chat as well but I think most of you have seen it and Chris will answer those um async in the doc but thank you everyone for coming and for the awesome questions and comments um our next meeting will be in three weeks so if you have any interest in presenting or sharing something in that meeting please reach out on Discord and thanks so much for joining bye for now Back To Top