COME HANG OUT ON TWITCH KEKW www.twitch.tv/lowlevellearning
@FoxDr2 жыл бұрын
This is now much more explicit, but not quite Rust-y yet. One of Rust's great things is that it helps prevent logic bugs (such as writing to a non-output pin) using the typesystem (technically you can do some of that in C, but in Rust it's the idiomatic way). The GPIO could return an OutPin for example, and that OutPin would have the set method instead, guaranteeing that you never write to a non-output and letting you not have to use the pin number again. The GPIO could even return an Option, which would be None if it's already used, by you or an other program (since it's DMA, other programs that use the GPIO would also need to set the same registers)
@edgeeffect2 жыл бұрын
Make invalid states unrepresentable.
@paulsimon14942 жыл бұрын
Read-modify-write operations for set/clear registers are useless. The whole purpose of set/clear registers is to avoid RMW, even datasheet states this: > Separating the set and clear functions removes the need for read-modify-write operations. Also there are 6 GPIO Function Select registers, not just 3. If you really want a proper way of implementing a PAC, see svd2rust, it generates code which is much closer to idiomatic Rust.
@jan.tichavsky2 жыл бұрын
Agree, embedded processors have dedicated instructions and registers so you can toggle or set specific bits in one go, using read modify write slows it down significantly.
@RayBellis2 жыл бұрын
I came here to say this - as you say it's right there in the data sheet shown in the video - you don't need to mask the bits that you're _not_ setting, specifically to avoid a RMW operation.
@newlinjohn8 ай бұрын
I also came here to say this, I see I’m a year too late though 😂
@DanForbesAgain2 жыл бұрын
Love your videos, but an error message of "Something has gone terribly wrong" is questionable at best when it comes to advice aimed at improving code 😅
@LowLevelTV2 жыл бұрын
Lol... concur. You should never get to that part of the code, but agree that maybe the error message isn't great XD
@DanForbesAgain2 жыл бұрын
It would totally be possible to enter that code path since the function accepts a u32 and not like an enum or something, but yeah I totally hear what you're saying. But you know what they say...one good error message is worth 1000 GitHub Issues 😅
@emilien.breton2 жыл бұрын
In my opinion this error shouldn't even exist. The function type signature indicates it takes in a `u32`, but it really doesn't. It takes in a `Pin`. In most languages you'd blindly take in a u32 and throw an exception when it's out of range (which is conceptually what we have with that `panic!`). That's terrible practice. In any language that has a decent type system, there is no reason not to use total functions everywhere. It was incredibly frustrating to watch this video because the code that was written isn't Rust; at its core, it's a C program with Rust syntax slapped on.
@altairbueno56372 жыл бұрын
@@emilien.breton all these languages but you choose to speak with the truth lmaoo
@tagged5life2 жыл бұрын
@@emilien.breton I agree, I’d probably pass in a enum variant of Pin.
@carljacobs12872 жыл бұрын
I've not used this chip, but don't think you need to read-modify-write GPIO_SET0 and GPIO_CLR0. Just set-and-forget - it allows for very fast pin toggling and is inherently thread safe in multi-threaded environments.
@KitsuneAlex2 жыл бұрын
Unsafe blocks are also expressions, you can do let value = unsafe { ... }. Any expression inside of the unsafe block will be hoisted out by the compiler, as in, the type of the expression inside the unsafe block == the expression type of the unsafe block itself. Nice video :)
@zactron19972 жыл бұрын
Obviously all personal preference, but I would not use a u32 as the input type for a pin number, and instead use an enum of all the possible valid pins. It's pedantry, but an obvious bug in your current code is I could call set(1000000) to access a completely non-existent pin. You could use an if/match statement to ensure only valid u32's are passed in, but then you're potentially sacrificing performance at runtime which could instead be done at compile time. Otherwise, love the video, this is exactly how I imagine the unsafe block should be used in Rust; creating islands of danger that can be thoroughly scrutinized.
@darkceptor44 Жыл бұрын
"bug" this is so funny, i dont use rust but it looks like you're creating problems that don't exist, it falls onto the developer to put a existent pin number and the way i see it would be a constant anyway that can then be used in some kind of assertion test, its not like it will use pin numbers given by the end-user or something.
@wChris_2 жыл бұрын
At 6:08 the '~' is the right operation since we want a bitwise not while the '!' is a logical not and since mask is a shifted 3 which is a truthy value it will be turned into a 0 clearing the entire register!
@LowLevelTV2 жыл бұрын
Crap I totally missed that. Thank you!
@ericedin83622 жыл бұрын
Thanks, I was thinking "Why did Rust make that change from C?"
@JeffHanke2 жыл бұрын
In rust '!' *is* the bitwise not when used with integers. It's only logical not when used with booleans.
@xphreakyphilx2 жыл бұрын
I do wonder if Rust will ever use ~ as bitwise not as I believe the original reason it wasn't is that in the early days of rust ~ was to box objects.
@wChris_2 жыл бұрын
@@JeffHanke what?? why would rust do this? i guess it isnt C but still..
@chair5472 жыл бұрын
I would go a step further and have pins themselves be raii objects so you can only write to a pin if it's already been set to an output
@edgeeffect2 жыл бұрын
In the STM32 crates I've seen, they do exactly that one type for an uninitialized pin, one for an input pin, one for each kind of output pin..... "make invalid states unrepresentable".
@sohn77672 жыл бұрын
What’s the advantage of using a zero sized type over a module in this case?
@peter94772 жыл бұрын
In the current case probably none, but note that some chips have multiple GPIO peripherals at different addresses and you'll probably want something like the struct to help model that.
@BambeH2 жыл бұрын
Later on, the functions shown in the video could be moved to a trait, which the zero sized struct could implement.
@sohn77672 жыл бұрын
@@BambeH this makes sense with what @Peter Hansen said
@raffimolero642 жыл бұрын
I would suggest creating a gpio *module* rather than an empty struct to hold all of your functions and constants if you're not going to store any state in the GPIO struct itself. That way, users may simply `use gpio::set_output;` and do `set_output(21);` directly in their code. You could then also remove the prefixes from all of your consts. you'd just have this: mod gpio { /// function select zero const FSEL0: u32 = // value; // more private constants... pub fn set_output(pin: u32) { // code } // more public functions... } // example thing you can now do use gpio::set_output;
@amiwatchesyt2 жыл бұрын
Hi, I'm quite new to electronics and I have a doubt. If you are setting how to interpret a pin could you use that function to return a pin struct that holds all the relevant information to put it on and off? That way you can't possibly act on a pin you haven't set previously
@Maykeye2 жыл бұрын
Some general thoguhts: * If you are going for read-modify(reading comments, it might be not the best approach, don't know), it'd be cleaner to make something like `unsafe fn read_write_volatile(address: *mut u32, f:F) where F:FnOnce(u32)->u32` So for example `set` would become `unsafe {read_write_volatile(GPIO_SET0, |x| {x | (1
@nicolaslopeza22512 жыл бұрын
Why not using a HAL crate to use the GPIO enums, etc defined for that specific board (quite new to rust, and to embedded programming in it, it might be a silly question)
@Knirin2 жыл бұрын
That is the idiomatic way to do it.
@redumptious254411 ай бұрын
I'm not him but I feel like those videos are specifically to do stuff "from scratch". He even says at the beginning that he's going to build a PAC in the video which would be the basis for a HAL.
@IvesvanderFlaas2 жыл бұрын
How about concurrency? Should you have lock/mutex/disable interrupt around the read, change, write operations to avoid skipping a bit clear/set? Or will the above layer always run from the same thread?
@chair5472 жыл бұрын
So you would have to worry about concurrency if there was any actual multithreading code in this. There is not
@d3line2 жыл бұрын
This is kernel code, so it is in full control of what runs when. Threads are not a concern until you write the threads into existence!)
@peternrdstrm2 жыл бұрын
I REALLY love Rust content. You inspired me to start low level development myself, and I only use Rust. Rust gang for the win!!
@OnlyHerculean2 жыл бұрын
These two videos on the Raspberry Pi cleared up so many things for me it's insane, thank you very much and keep on with this high quality content.
@thatluminary2 жыл бұрын
I am fairly new at programming, I understand nothing at your videos (except the beginner tutorial ones about C) but I still enjoy watching all of them. The smallest amount of knowledge I might get is precious to me. Thank you for the great videos and keep them coming :))
@LowLevelTV2 жыл бұрын
It takes time! You'll get there.
@thatluminary2 жыл бұрын
@@LowLevelTV Thanks for the encouragement :))
@QckSGaming2 жыл бұрын
I'm at a bit of a loss here. What did this refactoring achieve apart from giving the pin addresses a name? It seems like the same unsafe code as it was, with some syntactic sugar, in the end being much more complicated than it has to be. Is it some Rust thing? Does it do magic in the background?
@boggarak-roblox2 жыл бұрын
Based on how I understand it, the refactoring *Low Level Learning* has done here is meant to the reduce the amount of code inside of an unsafe block, because, when you put code in the unsafe block, you are telling the Rust compiler that you know what you are doing and thus it *doesn't need to check for violations of the **_memory management rules_* typically imposed on Rust code. Because Rust's memory management rules that the compiler forces you to follow are there for a reason (and is a selling point for Rust in comparison to other Low-Level programming languages like C or C++), you typically don't want to use unsafe blocks unless if you _really_ have to (like reading or writing volatile data, which is typically a memory violation under the memory management rules but is crucial in Embedded Systems Programming). If there is too much code inside the unsafe block, especially if it is doing more than just simply reading or writing data to a volatile memory location, then there is a chance that there might be problems regarding memory that Rust will simply ignore - *defeating the entire purpose of using Rust of all languages.* Thus, it is important to only use unsafe blocks _just_ for reading or writing volatile data and nothing else. Otherwise, you are creating problems you could easily have avoided. (Not to say that the refactoring could've been improved to make the code more readable or easier to debug). (If I am wrong (_and I might be, since I am new to Rust_) about something I said, let me know so I can correct it.)
@williamdrum98992 жыл бұрын
Does the compiler inline functions if it can or is every function you write going to require a CALL and RET (I guess this is ARM so really it would be BL and BX LR?)
@begga96822 жыл бұрын
It's optimized by LLVM which probably has inlining conditions documented somewhere
@d3line2 жыл бұрын
In my experience rust almost always inlines small functions, especially if they are used only once. This would be controlled generally by the optimization level (O3 for fastest code, more inlines, Oz for smallest code, less inlines) and by #inline[always/never] attribute on a per-function basis.
@OnlyHerculean2 жыл бұрын
I think implementing the Drop trait for the GPIO struct might make the code even better.
@OnlyHerculean2 жыл бұрын
Haha shit fixing my typo removed the heart from the comment 😂 But whatever now it says correctly struct instead of strict
@markdlp2 жыл бұрын
Is it possible to replace the delay with a mills function? just to not stuff the memory with nops.
@sledgex92 жыл бұрын
Does rust have the equivalent of C++'s "enum class"? I would make enum classes for the pin/register values so that I would enforce type safety in the public API (or, in this case, the GPIO struct's functions)
@finnmonstar2 жыл бұрын
uhhhh maybe look into rust enums??
@sledgex92 жыл бұрын
@@finnmonstar Sure. If I wanted to learn about Rust. However, I only wanted to point out a possible direction based on the language I know, in case there is something equivalent in Rust.
@finnmonstar2 жыл бұрын
@@sledgex9 I do not know that much C++. Maybe I can help you with C features, but I know nothing about the former. The language is too bloated and complicated for my taste. Anyway, tell me the features "enum classes" have in C++. Maybe I can help you with feature parity across Rust and that language you mentioned.
@sledgex92 жыл бұрын
@@finnmonstar In C++ "enum" doesn't create a strong type (for a lack of a better word). Enum values are implicitly converted to/from ints when passed around. You can even accidentally pass the value of the wrong enum type into a function that expects another enum type and the compiler will happily accept it. However "enum class" (or strongly typed enums) don't allow for implicit int conversion nor do they allow you to pass the wrong enum type to the function. The compiler catches those errors. Since Rust is more new I assume that its "enum" will behave more like "enum class". It wouldn't make sense for Rust to adopt the old C/C++ style of enums.
@finnmonstar2 жыл бұрын
@@sledgex9 You are right about that. In Rust you need to convert everything to the right type, even 32-bit integers to 64-bit ones.
@saeidakbari63032 жыл бұрын
Again, watching a perfect example of quality stuff How about creating some longer and structured courses for rust on embedded?
@LowLevelTV2 жыл бұрын
In the works!
@Ma1ne22 жыл бұрын
@@LowLevelTV I'd love that! Great follow up video, but I am also very happy you made these "mistakes" in the first one! Great way to build up and show what is the idea behind Rust, to abstract these unsafe interactions into a safe(r) interface. Can't wait to grab my little RaspberryPi once I finished moving and follow these videos again hands on :)
@himbary Жыл бұрын
@@LowLevelTV ETA? I am a rust beginner and I want to get into embedded. This would be awesome.
@redcrafterlppa3032 жыл бұрын
You should replace the two "set" and "clear" functions with one with an additional parameter taking an enum with variants "clear" and "set" instead of copying the code twice.
@tagged5life2 жыл бұрын
You could have two wrapper functions and then call a private function with your enum variant passed in.
@redcrafterlppa3032 жыл бұрын
@@tagged5life yes thats possible but I don't really see the point. It's basically like a setter method for an enum field. Why would you make an extra function for each variant? You could also use a bool.
@herbertwestiron2 жыл бұрын
Thanks for using large font in this video. Really appreciate it.
@LowLevelTV2 жыл бұрын
No problem!
@RayanMADAO10 ай бұрын
I dont understand why we need to do the bit manipulations instead of directly setting the value
@xZise2 жыл бұрын
Hi is it even necessary to read the set register, setting the bit and then setting the register? Shouldn't also work if you remove line 68 and only set one bit and write it to the register so that the register "or"-s it itself?
@adrianbool45682 жыл бұрын
Processors don't allow you to write just one bit. If want to ensure that you're only changing the one but you care about then you need to read in the whole word; make just the specific change you want to that value and then write that "surgically" modified word back.
@animowany1112 жыл бұрын
@@adrianbool4568 The GPIO_SET and GPIO_CLEAR registers are special write-only hardware registers, this is not actually a memory address that would persist. By writing a full u32 value with just one bit set, you are effectively just setting or clearing that one pin without interfering with other pins.
@adrianbool45682 жыл бұрын
@@animowany111 Oh, OK, thanks for letting me know! Sure makes the code easier & safer like that!
@linkernick53792 жыл бұрын
Any working code is infinitely better than the crashing one 🙄
@code-dredd2 жыл бұрын
_Leaves no spaces between math operators_ Me: The code is terrible. It must be rewritten.
@jacksonbourne2 жыл бұрын
since all blocks return a value, you can just use: let mut val = unsafe { core::ptr::read_volatile(...) }
@alexloktionoff6833 Жыл бұрын
Can you make more videos about RPiPico programing in rust? Especially about proper flexible inline assembler with input, output and clobber registers.
@guillermomoreira56012 жыл бұрын
Have you tried zig?
@rallokkcaz2 жыл бұрын
I have, and I helped write the code from the stream where we work shopped the example code. Zig is fine, but the DX is trash and the documentation is balls. Rust wins over zig for me but I'll still give it a chance in the near future as an exercise in futility hahahah.
@bartpelle34602 жыл бұрын
zigma balls lmaoo
@DegradationDomain_stuff8 ай бұрын
What is the point if you wrapped both read and write into an unsafe block? Will it really become safe if you make operations on it in rust?
@datawolk2 жыл бұрын
Good tutorial, next a proper sleep/delay function?
@LowLevelTV2 жыл бұрын
Good idea!
@Little-bird-told-me2 жыл бұрын
which window manager are you using. I3 ?
@LowLevelTV2 жыл бұрын
Yea
@Little-bird-told-me2 жыл бұрын
@@LowLevelTV Noice
@Little-bird-told-me2 жыл бұрын
someone who is starting out now and wants to learn a low level language, which one would you recommend _C_ or _Rust_ ?
@LowLevelTV2 жыл бұрын
Start with C in my opinion. Rust is awesome but you need to know a decent amount about programming for it to make sense. C has far less abstractions
@Little-bird-told-me2 жыл бұрын
@@LowLevelTV Cool, thanks mate.
@KCrouch-t2o2 жыл бұрын
the 50000 "nop" s cause me mental pain
@Turalcar10 ай бұрын
49999, because 1..50000 it includes neither 0, nor 50000
@kayakMike10002 жыл бұрын
How dare you make this so much simpler! You must maintain the dark arts of firmware!!!
@OlivierEble7 ай бұрын
I don't know what is good "very low-level" rust. But Having a "safe" function that is not thread-safe doesn't feel right. (concurrent calls to set or clear are potential race conditions)
@thirtysixnanoseconds1086 Жыл бұрын
the unsafe is merely refactored out. i really dont get why you wouldnt stick with the tried and tested c
@HIRVIism9 ай бұрын
You concentrate the unsafe code in a few, marked locations. The rest of the program can then enjoy the benefits of rust, and you know where to look if you encounter weird behavior.
@jamespeterson79792 жыл бұрын
you could add debug attributes to remove the "not-used" warnings :)
@Rodrigo-lv8if Жыл бұрын
Could someone tell me if this tutorial works on Windows. If so, could you reply to this comment with a link to your project on github, because I tried the tutorial and it doesn't work for me on Windows?
@richardorourke45012 жыл бұрын
Just re-write it in Ada, do it right the from the start.
@edgeeffect2 жыл бұрын
Rust is Ada for the 21st century. ;)
@clubby78932 жыл бұрын
Might std::hint::spin_loop be a better alternative than a tight loop of nops?
@LowLevelTV2 жыл бұрын
no_std :(
@clubby78932 жыл бұрын
@@LowLevelTV Ah of course 😄Maybe the core::arch intrinsics used to implement it for your specific platform? I'm not familiar enough with Arm to tell if it's supported though
@animowany1112 жыл бұрын
@@LowLevelTV core::hint::spin_loop exists
@abdulrahman1s2 жыл бұрын
This code is unsafe, let's fix it with more unsafe blocks 😁
@LC-hd5dc2 жыл бұрын
more smaller blocks with limited scope > one big one that could be doing anything also after the cleanup it's clearer what the code is doing
@MrMooin2 жыл бұрын
This is Awesome! Is there a IDE like Eclipse where you can debug the Code and see the Content of Valuables by stepping slowly through the Code? I'm happy that i found your Channel!
@devdev30522 жыл бұрын
Was anyone able to compile for rp4 64 bit? I think the target triplet should look like this armv8-none-eabihf However rust says it cannot add a target of armv8-none
@paulsimon14942 жыл бұрын
Correct target name is aarch64-unknown-none
@devdev30522 жыл бұрын
@@paulsimon1494 yup. thanks
@paulwratt2 жыл бұрын
nice - cheers for showing _both_ ways
@LowLevelTV2 жыл бұрын
You bet
@kayakMike10002 жыл бұрын
Magic numbers kinda suck too, but that's a mistake in all languages. Also... What's up with that noop? Are there not proper timers? Also ... Screw make. Get Meson and Ninja. Much much better
@oxey_2 жыл бұрын
oh, you're overestimating me by a lot. I will definitely have to
@Dygear2 жыл бұрын
Thanks!
@Dygear2 жыл бұрын
*Chefs kiss*
@LowLevelTV2 жыл бұрын
:O Mark THANK YOU AGAIN my guy holy crap. Seriously this is so generous
@Dygear2 жыл бұрын
@@LowLevelTV This directly helps me. I'm working on a Time Clock for one of my customers. I've done it with a Raspberry Pi (In C) with Raspberry Pi OS and an RFID reader. Having to buy RFID keys has been come a pain so I'm going to do it with a PICO and a finger print reader next. At this point I might as well write it in Rust and I'd love to write it from the ground up. Plan is to have the code MIT licensed once I'm done with it as it's not a competitive advantage for me to keep the code a secret.
@wtfdoiputhere2 жыл бұрын
now do this to all the other videos my child
@LowLevelTV2 жыл бұрын
Yes Jesus 🙏
@vladlu63622 жыл бұрын
Couldn't you just have XOR'd the Val with itself to make sure it is 0?
@LowLevelTV2 жыл бұрын
Sure but then you’re destroying any state that was already there, which would suck if you want to use multiple pins on the same port
@davidgomez792 жыл бұрын
Rust is like putting training wheels on a ninja bike. C being the ninja bike. (bait)
@absalomdraconis2 жыл бұрын
No, but it is like putting a roll cage on one. Useful, but only an incremental change.
@HansBezemer2 жыл бұрын
Gee, no bitfields in Rust?
@DegradationDomain_stuff8 ай бұрын
RustberryPi when?
@mychromebook99352 жыл бұрын
the forth programming language is way easier than Rust for embedded stuff. Rust just seems overly complicated for simple stuff.
@memcpy2 жыл бұрын
Amen, have you tried flashforth for embedded ?
@LowLevelTV2 жыл бұрын
I'll have to give it a try!
@liam.3d2652 жыл бұрын
C will live forever
@mychromebook99352 жыл бұрын
@@memcpy yeah, and mecrisp forth for ARM make it easy to turn off and on gpio pins
@CallousCoder2 жыл бұрын
It’s been ages that I’ve used it. And it was quite hard to learn for a C/assembly programmer. But when you grasp the concept of “everything goes over the stack” it clicks.
@Ma_X64 Жыл бұрын
Oh! You just need a new language! More sAfE! 🤣
@glennmiller3942 жыл бұрын
Good stuff. Thanks,
@minneelyyyy2 жыл бұрын
... by rewriting it in C!
@davidjohnston42402 жыл бұрын
It magically became less readable as it became more rusty.
@adrianbool45682 жыл бұрын
I think that's mainly down to the added support for any pin number rather than original's operation only on PIN 21. The arithmetic does get a little messy. I guess that is what encapsulation is for.. :-/
@DanielRodriguez-ff5cs2 жыл бұрын
so in rust you must replace an unsafe code with another unsafe code, it's like the language was not meant for embedded systems
@LowLevelTV2 жыл бұрын
🤔
@finnmonstar2 жыл бұрын
well atleast the whole logic is not unsafe, and you can find memory bugs easier i think
@DanielRodriguez-ff5cs2 жыл бұрын
@@finnmonstar that's fair enough!
@rallokkcaz2 жыл бұрын
LLG.
@LowLevelTV2 жыл бұрын
LLG 5Life
@indefinitingdefinition Жыл бұрын
Very nice explanations about low level rust! What is this desktop environment? I heard a windows sound in one of your videos, is that a WSL? What is this windows manager? Do you have a video explaining your desktop setup? Looks really curious.
@edgeeffect2 жыл бұрын
That's wonderful... it takes a "real programmer" to point at their own code and say "this stinks".
@dries14542 жыл бұрын
Ironically, you ended up with the same unsafe code you had in the beginning. The only difference is that they are wrapped in small functions so you don't see a big unsafe block in the main function.
@raffimolero642 жыл бұрын
which then makes it easier to debug individually rather than as one whole swathe of danger
@ChungusTheLarge Жыл бұрын
Isn't that the definition of abstraction?
@der.Schtefan2 жыл бұрын
Rust: when you like C, but thought it is too readable.
@piotrek32822 жыл бұрын
still bad tho
@LowLevelTV2 жыл бұрын
no u ha gottem
@0xbinarylol10 ай бұрын
Syntax of rust is horrible. Why not they just take c grammar and added safety 😂
@metamud86862 жыл бұрын
The audio is horribly out of sync with your video / mouth movement.
@iyamroshan2 жыл бұрын
You are so handsome 😘
@BetaTester7042 жыл бұрын
Yeah rust is a no-go for me. Rather not melt my brain
@akaikangaroo2 жыл бұрын
Watching all these videos I understand that I could never be a good programmer for at least two reasons: 1) I still can't type on my keyboard while watching to the screen; 2) I am absolutely unfamiliar with Linux and never even tried to learn it😑
@virdvird2 жыл бұрын
And you just made code with worse performance for embedded. Same as arduino library do UPD. My statement is wrong see godbolt link below
@rallokkcaz2 жыл бұрын
Rust's zero size struct semantics are not the same as an abstraction like C/C++ with AVR dude.
@softwarelivre23892 жыл бұрын
@@rallokkcaz but now he has to calculate a bunch of masks instead of pasting at the memory location directly. As this code seems to only target the 21 pin, it ends up wasting more CPU cycles (unless the compiler already precalculates those values before the function call, which I doubt it does).
@saadisave2 жыл бұрын
@@softwarelivre2389 LLVM can easily inline simple calculations when all inputs are literals. It can even compute the result of a function at compile time and inline the result.
@softwarelivre23892 жыл бұрын
@@saadisave yeah, that I know, but as this is not a pure function, and it does memory allocations in its body, I don't know if that will be optimized beforehand or not in this context. If it is, then disregard my comment.
@virdvird2 жыл бұрын
@@rallokkcaz It's not about struct. Is `register = match reg ` zero cost abstraction?
@nevokrien95 Жыл бұрын
U wrote this code lol😂
@LordHog2 жыл бұрын
To make this Rust program perfect would be re-writing it in C!
@carljacobs12872 жыл бұрын
C++. The simple stuff compiles just as small, but then it gets cool!
@lilhaxxor Жыл бұрын
This is misleading. You just moved the unsafe code around, but it's still present. "No unsafe code" when you still have unsafe blocks. Call it an interface or encapsulation. What a joke. 😂
@AlFredo-sx2yy2 жыл бұрын
ooooor use C and stop wasting your time!
@anon_y_mousse2 жыл бұрын
Since hundreds of libraries exist already for C it would've been even easier. Instead he's writing it from scratch in a language which treats him like a baby and loses most of its proclaimed advantages at that level.
@AlFredo-sx2yy2 жыл бұрын
@@anon_y_mousse not only that, even if you're in a system where no libraries exist or you cant use any libraries for whatever reason, C gives you no limitations and does not treat you like a baby, so as you said, all advantages that rustaceans claim are lost at that level.
@d3line2 жыл бұрын
@@AlFredo-sx2yy i'll bite. What advantages are lost at this level? The only difference from C that I can see is needing to actually say "unsafe" when the thing that you're doing is unsafe. Also he doesn't actually use the rust features that would be used by a library author writing a hardware abstraction layer in rust. For example rust can prevent using nonexistent pin numbers or reading output pins/writing input pins at compile time with 0 runtime checks, haven't seen that in C. He also gets a big chunk of libraries if he needs them, and all others if he writes a heap allocator.
@AlFredo-sx2yy2 жыл бұрын
@@d3line the "advantages" provided by rust are lost when you try to go this low level lmao..... i thought you had watched the video? You havent seen C detect nonexistent pin numbers because you havent seen proper C code then. That's something that YOU have to program, the fact that Rust provides this out of the box for certain architectures doesnt mean shit, because good luck finding a build target for all of the architectures you can find lol.... Christ the shit i have to read i swear.
@d3line2 жыл бұрын
@@AlFredo-sx2yy 1) *which* advantages are lost? You keep exasperatedly saying that some nebulous advantages are lost without defining what, in fact, is lost. 2) I haven't seen stuff like compile time const functions in C because they don't exist in C, yes. C has no way to prevent you just using pin number 12345 except runtime "if" check. Rust can do this at compile time (not shown in video). 3) I'm not talking about using some magic from rust's platform support, you can code it yourself by using rust's type system, and provide your users with un-misuse-able API at no runtime cost. That's a rust advantage, and I don't see how it is in any way lost when programming for embedded devices. Also please chill...
@cozyGalvinism2 жыл бұрын
You may be able to move the linker stuff into a build.rs so it gets built automatically
@necauqua2 жыл бұрын
Nothing prevents you from doing GPIO::set_output(any random u32), this is not that Rusty still. Make. Invalid. State. Unrepresentable!!! (that is, something wrong _logically_ should not even compile! Or be hard to make it compile such that you'd have to deliberately try to) This was (with exclamation points hah) the greatest statement that Rust introduced me to and I replicate it in everything I write ever since. So your 'safe interface' is actually not very safe What you can do instead is this (or there are a few ways to do that, simplest that came to mind is a enum) #[repr(u32)] // you can do enum_value as u32 where you need it now, totally safe pub enum Pin { PIN21 = 0x1234 // whatever it is //.. and so on from the datasheet } impl Pin { pub fn from_raw(raw: u32) -> Option {..} pub unsafe fn from_raw_unchecked(raw: u32) -> Pin {..} } also unsafe can be an expression so you do `let mut val = unsafe { read_volatile }` - a bit shorter :) also², please install rust-analyzer lsp extension in vscode please, watching you do cargo build in the terminal every time is painful :)
@bartpelle34602 жыл бұрын
you make yourself sound so incredibly uneducated with the last paragraph that your (entirely valid) former paragraphs are out of the window.
@minimalhostage2 жыл бұрын
@necauqua You posted some really helpful information. I have a lot of experience in Rust and C, have been lucky enough to work on very difficult problems throughout my career and have led a lot of engineering teams. I say this mainly because its given me exposure to engineers with all types of communication styles. Its obvious to me that you were trying to be helpful by posting a rusty approach and a clean interface that takes the reader into account as much as the runtime. Your approach was solid and I think you wrote it in good faith. Because of that I want to post some constructive criticism for you, also in good faith. 1. It appears you don't know how to use "lol". As it stands, your comments come off as extremely self-important and patronizing. If you remove the "lol", then your statements are just blunt but not necessarily malicious. You seemed to use "lol" to punctuate important points but, unless you're trying to alienate your reader, I recommend you abandon "lol" entirely. You can also try replacing them with exclamation points or clarifying questions. 2. Your classification of embedded is limiting and elitist. Also, you're categorically wrong about "proper embedded". Embedded systems are simply units, within larger electrical systems, that have a processor, memory, and Input/Output capabilities for peripheral devices/components. In this video and the last, he replaced the OS with some kernel code and fed it to a bootloader. All he used on the physical board was... the processor, memory, and I/O pins. Maybe you'd rather he use an STM32 or some other MMU-less MCUs and spend half the video walking us through a circuit diagram. But there are already thousands of videos covering that for people who know where to look and what to look for. Maybe a big reason this video focuses on raspberry pi's is because the boards are versatile and ubiquitous. You can have a desktop machine or just a processor, memory, and I/O. An embedded approach to raspberry pi is much more inviting than the same video with an STM32 or something. Engineering, as a practice and discipline, is about trying to accomplish what you want with what you have. That's it. Everything about "proper" and "the right way" is just pedantic and frankly signals that the person has spent a lot of time in a narrow field, academia, or has little experience outside of their chosen discipline. If an engineer insists on a solution without bringing up trade-offs, alternatives, and cost benefit analysis, then 99% of the time they're just reciting dogma they read/heard or used to solve a problem in the past. Like I said, it is obvious to me that you were trying to be helpful. I don't mean to be rude, only clear. You mentioning "Make. Invalid. State. Unrepresentable!!!!" up front made it obvious to me that you have experience in this field; that you care about attention to detail, correctness, and stability. All of these are great qualities that are really hard to find. If I were hiring an engineer, you'd be a desirable candidate. If I were looking to assign a mentor, you'd be on my list. But your tone is very condescending and abrasive. You know so much that you assume the way you know how to do something is THE way to do it; You forget that much of what you've learned is just another engineer's opinion or ideology. An opinion that another engineer taught THEM and is either outdated or will likely become outdated. I could be wrong, but you sound like a student or a junior engineer that's very passionate about the subject but hasn't really been exposed to a variety of systems and problems. Regardless, here's my parting advice: The way you communicate just negates the value of the information you try to communicate. And people will opt for getting that information elsewhere. The hardest thing about engineering is having to collaborate with other humans and navigating the social nuance that emerges when solving difficult problems. Every tool you've mastered, or want to master, was built by a group of people that worked well together. I believe you have a lot to offer as a teacher or collaborator. It seems like you want to help and share or at the very least contribute. If that matters to you at all, consider asking people more questions and making less assumptions before proposing a solution. Technological systems are built in the context of social systems. Communication skills will serve you well in social systems. I think practicing them will go a long way and you'll even enjoy it as you get better. BUT If you're just a troll and like to feel superior to others or self-important then... carry on and ignore my comment; it was meant for somebody else haha
@bartpelle34602 жыл бұрын
@@minimalhostage Nothing but respect for the respectful way you phrased that; I don't have that in me but I commend it nonetheless Charles💯
@necauqua2 жыл бұрын
@@minimalhostage wow this was very detailed, thanks I mean I'm just used to be kind of rude/patronizing as you've said - because this is the internet and way more commonly you encounter obtuse people than good helpful people like you - and your approach also sounds kind of more 'formal' than I would've liked it to be - this was meant to be kind of a rant and I was annoyed (why? idk really actually, maybe it was elitism indeed, 'oh they learned some rust and sharing their incomplete knowledge') by some things in the video. Also non-native english I will edit some of my comment, everything you said is absolutely valid It is funny how one can hate something in others yet express it themselves to such an extent, thanks for opening my eyes - I really don't like elitism and static unchanging ideologies and when someone thinks some one thing is 'proper', yet I said exactly that huh Still hate that something that runs an OS is called embedded, but looks like my notion of 'embedded' was just invalid :)
@necauqua2 жыл бұрын
@@bartpelle3460 while I removed the paragraph as it was really poorly worded anyway indeed - I did think (oh well maybe this was wrong the entire time) that embedded is when you run something on a 'bare metal', w/o an OS - and raspberry pi's being a full-blown arm computers with linux installed were out of this definition - yet every youtuber seems to do 'embedded' with raspis. Although after the explanation from Charles it seems that if you run some code _in place_ of the OS, as it's done in this video - it is indeed embedded - I was under impression of some other youtubers doing like a blinking gpio led with like a python script running on that arm linux on raspi.