#151

Поделиться
HTML-код
  • Опубликовано: 31 дек 2024

Комментарии • 317

  • @OsoPolarClone
    @OsoPolarClone 5 лет назад +9

    Ralph: I have seen lots of ESP32 videos. Yours is the FIRST to talk about the queue and how to use it. Until yours I did not know it existed. The same goes for semaphores. WELL DONE (AS ALWAYS). Thanks for another GREAT VIDEO!

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      Thanks for that, Bruce, nice to hear from you.

  • @rjinnh3933
    @rjinnh3933 5 лет назад +3

    As always, outstanding video production quality.
    I simply marvel at the quality of each of your vids. Absolutely the best on YT!
    The content was pretty good also.
    Actually, the content was excellent begining with your 13 minute explanation of param passing and then diving into your code.
    Brings back memories. As a Hardware Designer, In 1978, I was writing my own diags and using semaphores in a Z80/Z8 design and having random crashes. A really sharp Software Engineer took pitty on me and showed me what you just showed us all. Saved my BACON. HAHA....
    It's amazing how much I've forgotten during the last 45+ yrs, but you've again re-awakened those little grey cells with this vid. A really big TankYou!
    Russ in NH

    • @RalphBacon
      @RalphBacon  5 лет назад

      Well, the good news, Russ, is that although you might think you have forgotten things they will quickly resurface once prompted. So you are ahead of the game!

  • @michaelmcnaughton1535
    @michaelmcnaughton1535 Год назад +1

    I like your videos. 1. Minimum blabby content. 2. Clear explanation with just the right degree of complexity. 3. Excellent downloadable examples. Your work is much appreciated, on a par with Andreas Spiess "the guy with the Swiss Accent".

    • @RalphBacon
      @RalphBacon  Год назад

      Awesome, thank you! I can't believe Andreas even knows I exist but I do occasionally watch his excellent content too.

  • @gpTeacher
    @gpTeacher 5 лет назад +2

    You hit it out of the park yet again Ralph! This is a FANTASTIC demo of queues for my Grade 12 CompSci course! Very approachable. I’ll be having the students work through it. Thanks!

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      Thanks Gord, if your students understand this then my work is complete!

  • @maoku7435
    @maoku7435 Год назад +1

    I just came here looking for copper and found gold. I learned a lot from this. you explain really well! Thanks for your video.

  • @TheRooster276
    @TheRooster276 Год назад +1

    Great job in explaining details like what "volatile" etc. does. I have purchased books on C++ and they often don't explain well what is going on in the code. Your videos are making all the difference. Thanks a bunch.

  • @davidharms3562
    @davidharms3562 Год назад +1

    Thanks Ralph! I thoroughly enjoy the detailed explanation in your videos. Having developed with arduino for a while now, the magic of the ESP32 is truly fascinating. 👍🏻

    • @RalphBacon
      @RalphBacon  Год назад

      Glad it was helpful! I tend to use the ESP32 for most projects unless it's trivial and doesn't ever need Wi-Fi connectivity.

  • @TheBionicbone
    @TheBionicbone 5 месяцев назад +1

    Great video and excellent info, a good discussion at the end too about speed vs power consumption that I wasn't expecting from the title. It is defo a video to watch in full - Thank you, subscribed !!

  • @secraiskille
    @secraiskille 4 года назад +4

    Because of the COVID, my teacher simply sent us here. Great video!

    • @RalphBacon
      @RalphBacon  4 года назад

      Awesome! Thank you! And thank your teacher for pointing you in this direction!

  • @annacermed4468
    @annacermed4468 4 года назад +1

    What a beautiful diving into the deep blue of multitasking. That's simply awesome. Thanks a lot (from Algeria).

    • @RalphBacon
      @RalphBacon  4 года назад

      Glad you liked it Annacer Med, great to hear from someone in Algeria, warmer than the UK, that's for sure.

  • @stompreaper
    @stompreaper 5 лет назад +1

    I really like how you are exploring these topics. Thanks for sharing!

    • @RalphBacon
      @RalphBacon  5 лет назад

      Thank you for your feedback, Stephen Thompson, noted and appreciated!

  • @dnarobo
    @dnarobo 3 года назад

    Your videos are an essential "library", I never needed Tasking before...but when I ran into a blocking conundrum today...RSB was my first and last stop to learn exactly what I needed.

    • @RalphBacon
      @RalphBacon  3 года назад

      Excellent! I am so glad this video helped you. I can never cover all aspects of a topic but enough to get you going in the right direction. Thanks for the feedback, David.

  • @robinharris4706
    @robinharris4706 5 лет назад

    Another excellent video, thanks Ralph. I get a lot of help and value from your videos and this one is a particularly good example - it goes into a deep technical topic in a clear and informative way. I like your style and that there are no distractions (poor audio, annoying habits etc.!). It is much more useful to me to watch this style of video than a 5 minute overview of something basic or well covered by others. Please keep producing the deep dive style video and I'll keep on watching and supporting! Thanks.

    • @RalphBacon
      @RalphBacon  5 лет назад

      Glad you liked this one, Robin, that ESP32 is a most powerful device indeed and it makes sense to know how to program it properly. Thanks for posting.

  • @iandrake4683
    @iandrake4683 3 года назад

    I'm a c# dev and just wanted to say I really appreciate your content. It's perfect for me, even if I already understand most concepts like queues and mutexes. Having them explained in the context of rtos is super helpful.
    Also, this is the first I've heard that delay on an esp32 releases the CPU! I'm used to the arduino uno. That tidbit helped a bunch.

    • @RalphBacon
      @RalphBacon  3 года назад

      Great to hear! From one C# developer to another it's all quite different in the embedded C++ world without the .NET safety net!

  • @george12121979
    @george12121979 5 лет назад +1

    thank you for continuing the videos about ESP32

    • @RalphBacon
      @RalphBacon  5 лет назад

      You are most welcome george kon, I'm glad you like the video. Nice to hear from you.

  • @bipolarchemist
    @bipolarchemist 5 лет назад +1

    Instead of Serial.println("..." + String("...")); at least on the ESP32 and maybe the ESP8266, you can use Serial.printf("%d %f %0.2f...
    ", varInt, varFloat, varFormattedFloat); to save yourself some typing. You can also get printf to work on the SAMD line as well, but it requires a few lines of code to 'turn on', but much easier than achieving the same thing for the AVR platform. Benefit, you can toss in as many variables as you want, format them to your hearts content and you don't have to use String.
    And thanks for another great addition to your ESP32 videos.

    • @RalphBacon
      @RalphBacon  5 лет назад

      Yes, you can indeed, hmx, and that is indeed the way to do it, but I knew if I did that I would confuse my audience. I tend to use sprintf on my ESP8266 for my HomeAlone project as it's easy to use and is "proper" C! I mentioned in the previous video that users can continue to use Arduino-friendly speak or slowly migrate to the underlying language constructs; maybe I need to expand on this?

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      Oh, and I _never_ use String (capital S) as it can fragment the Heap space, so std::string it is, but then we have to convert that back... the list goes on!

    • @bipolarchemist
      @bipolarchemist 5 лет назад

      Glad to hear that you are aware of sprintf. I go crosseyed every time I see example code and there are 5+ Serial.print() statements in a row just to output a single line of text. I am enjoying that you are encouraging people to use more standard functions. Can't wait to see what you have for us next.

  • @anispinner
    @anispinner 5 лет назад +2

    Thanks for the huge work on this board, again!

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      You are most welcome NLab ™, I'm glad you like the video. Nice to hear from you.

  • @jamescullins2709
    @jamescullins2709 4 года назад +1

    Well done Ralph. I Love your explanations.

    • @RalphBacon
      @RalphBacon  4 года назад

      Glad you liked it James!

  • @mikelopez9893
    @mikelopez9893 5 лет назад

    Great video Ralph and a good explanation of queues as a sync mechanism.

    • @RalphBacon
      @RalphBacon  5 лет назад

      Thanks for that, Mike Lopez, nice to hear from you.

  • @jackflash6377
    @jackflash6377 2 года назад +3

    The first computer I bought had 64K of RAM and a 1mhz CPU (Apple IIe bought new with 2 disk drives and a DOT matrix printer.. $2500 in 1979ish)
    The first computer I built was a 386 30mhz, 4mb RAM with a 40mb HD running DOS or Windows3.11 depending on what game I was playing at the time.
    The ESP32 is more powerful and has more onboard RAM than my full size PC had in that time.

    • @RalphBacon
      @RalphBacon  2 года назад

      $2,500 in 1979 is worth about $11,356 today! Wow! Expensive computer.
      See: www.in2013dollars.com/us/inflation/1978?amount=2500
      Yes, the ESP32 is a powerful beast for sure. One reason why I like it!

  • @vince_martyn
    @vince_martyn 4 года назад

    Hi Ralph, Just discovered your channel and have subscribed. The quality of your explanations and presentation style are excellent and I will be watching more as I plan to port an audio project from the ATMEGA328 to ESP32.

    • @RalphBacon
      @RalphBacon  4 года назад

      Welcome aboard! I'm looking at doing more ESP32 stuff, Vince, as it happens. Such a powerful chip. Keep tuned!

  • @readmoreon2390
    @readmoreon2390 3 года назад +1

    Just found your channel, absolutely love it thanks for great content as well as being very informational. Well deserved sub and like!

    • @RalphBacon
      @RalphBacon  3 года назад +1

      Welcome aboard! You only have around 233 more videos to watch, then! Oh, and the quiz to take (latest video). Good luck!

    • @readmoreon2390
      @readmoreon2390 3 года назад +1

      @@RalphBacon Ahahhahahah will do

  • @jackyli6716
    @jackyli6716 2 года назад

    best tutorial that i have never seen before! well done ! good job! i love it!

  • @andymouse
    @andymouse 5 лет назад +1

    Interesting stuff Ralph, but I'm definitely gonna need to watch a few times!

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      That's fine, Andy, repeat views are apparently a Good Thing as far as RUclips go! But it is worth understanding this correctly to avoid problems down the line.

    • @andymouse
      @andymouse 5 лет назад +1

      @@RalphBacon ​ Nice one Ralph and Happy Birthday! may you have many more and keep your tutorials coming ...I love 'em!

  • @dafydds
    @dafydds 3 года назад

    Another excellent video... Passing variables, and the bain of everyone's lives, global variables... I can't lie, I've been hellishly lazy in my recent project; it has global stuff all over the place along with threads.
    But as per usual, I've forgotten more than I remember from programming, so this video is great at just showing "here's a start, an idea, think and play, the don't cry over your cereal if you're using global variables" 😁🤣
    Thanks for a great video 👍

    • @RalphBacon
      @RalphBacon  3 года назад +1

      Well said! I'm using a task and a global variable in my ESP32 Web Radio project and it works very well.

  • @VictorianMaid99
    @VictorianMaid99 Год назад

    This is so useful, I am playing a project with multiple Arduinos talking back and forth.

    • @RalphBacon
      @RalphBacon  Год назад +1

      Sounds like you have an interesting project ahead.

  • @johnhoeppner1569
    @johnhoeppner1569 4 года назад +1

    Good video on the enigmatic ESP32. FYI: I encountered the issue with running a task on core 0 and getting a 'panic' message saying that the watchdog timer had timed out. Following this the CPU reset itself. This occurs if the new task loops forever. When a task is started using the xTaskCreatePinnedToCore, the watchdog for core 0 is started and needs to be reset periodically by the task running in core 0. One solution is to place a call to vTaskDelay(1) inside the infinite loop which restarts the watchdog. There are probably other methods but this worked for me.

    • @RalphBacon
      @RalphBacon  4 года назад +2

      Yes, putting small delay in the loop is one way to prevent the panic message, but what I do is delete the unwanted task as a first step, then the OS doesn't expect it to run. I demo this somewhere in my code, search my GitHubs for all things ESP32 and you will doubtless find it.

    • @johnhoeppner1569
      @johnhoeppner1569 4 года назад +2

      @@RalphBacon Hi Ralph, in my particular app I needed core 0 to continuously monitor time sensitive sensors while core 1 handled other asynchronous stuff. There is no issue with using either core in an infinite loop as long as the watchdog is refreshed periodically or disabled. The vTaskDelay() refreshes the watchdog kinda like time.sleep() does in Python but there may be more elegant ways to do this.

    • @rafaelhenrique7998
      @rafaelhenrique7998 4 года назад

      ​@@johnhoeppner1569 Hey john, i have the same error as you, can you please help me fix it?

  • @tomforrester377
    @tomforrester377 5 лет назад +1

    Thanks Ralph. I find the format of your videos really easy to follow.
    One question - what causes loop0 and loop1 to be started? Something special about that naming convention?

    • @RalphBacon
      @RalphBacon  5 лет назад

      Nothing special about loop0 and loop1 at all; these are arbitrary names and, as I mentioned, I would call them something very different in a real program. They are started the instant the setup() creates them. Which means that if you have several tasks to start the first one will be running long before the last one gets scheduled.
      Thanks for posting Tom, always good to hear from you.

    • @tomforrester377
      @tomforrester377 5 лет назад +3

      Ah okay, so xTaskCreatePinnedToCore() doesn’t only define the properties of the task, it also runs it.
      Thanks again Ralph!

  • @Tony770jr
    @Tony770jr 2 года назад

    Hi Ralph, excellent video. Keep it up my friend!

  • @antipainK
    @antipainK 4 года назад +2

    But queues that are used by multiple threads also need a concurrency lock (ec semaphores). Any operation on shared resource can create an error if accessed at the same time.

    • @RalphBacon
      @RalphBacon  4 года назад +1

      Indeed. I guess it depends on how atomic each operation is, but semaphores is what I would go with anyway if I decide to use a task to read the circular buffer.

  • @ArjanvanVught
    @ArjanvanVught 5 лет назад +1

    Yeah! Using a proper developement environment -> Eclipse Thank you!

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      Ha Ha! You're being cruel to the Arduino IDE now, Arjan! But, would you believe it, I'm also running *Visual Studio Code* and *Platformio* at the moment, mainly for ESP32 development but it also works for the Arduino and supports hundreds of other boards too. So maybe, that might become my IDE of choice in the future, you never know. Thanks for your enthusiasm for a 'proper' development IDE, keep tuned.

    • @W--ko9ms
      @W--ko9ms 3 года назад

      @@RalphBacon I've also found myself using vs code with platformio for the esp32. Had been a wonderful experience thus far. Thank for this explanation, cleared some things up for me.

  • @TheSimRacers
    @TheSimRacers 8 месяцев назад +1

    FYI: WROOM: WROOM stands for "Wireless Room" and refers to a compact module that integrates Espressif's Wi-Fi and Bluetooth SoC. These modules are designed to provide wireless connectivity capabilities to various IoT devices. They typically come in small form factors, making them suitable for applications where space is limited.
    WROVER: WROVER stands for "Wireless Room with RAM Overlay." Similar to WROOM, it includes Wi-Fi and Bluetooth functionality. However, WROVER modules also feature additional external RAM, which provides expanded memory capabilities compared to WROOM modules. This extra RAM allows for more complex applications or larger data storage requirements.

    • @RalphBacon
      @RalphBacon  8 месяцев назад

      That's very interesting, and plausible; if you can give me a citation where this info can be found, it would be even better. I searched high and low and could not find it). But for now, I'll use your explanation!

  • @davidlambert8368
    @davidlambert8368 5 лет назад

    Wonderfully succinct and so appreciated! Thanks Ralph.

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      Thank you for your feedback, David Lambert, noted and appreciated!

  • @sirrcharles1869
    @sirrcharles1869 2 года назад

    Thank you for.this content! Fantastic explanation! Was using global variabeles now I will think twice :)

  • @Enigma758
    @Enigma758 3 месяца назад

    If the shared global variable is an integer type (native to the CPU), then assigning it a value should be safe across tasks and cores without using a semaphore. The reason for that is that assignment to an int can be performed with a single instruction so it can't be interrupted, it is "atomic". However, a read modify write (e.g. x = x +1) to that same int is not safe and a semaphore is required. Granted, the former is a limited case but might be all you need in some cases.

    • @RalphBacon
      @RalphBacon  3 месяца назад +1

      Is it truly atomic? Updating both bytes of the 2-byte integer? If so, then great! If not, then a semaphore is still needed, and, as you rightly point out, during any kind of update on that variable.

  • @BerndFelsche
    @BerndFelsche 5 лет назад

    My recollection could be wrong but writing and reading to/from FreeRTOS queues is atomic. If several tasks are waiting for a queue, then only the first found with highest priority is woken when something is sent into the queue. Similarly at writing.
    One uses queue-driven tasks for e.g. ensuring that data written to a serial port doesn't get scrambled, which could happen if two or more tasks are writing to the same port "at the same time".

    • @RalphBacon
      @RalphBacon  5 лет назад

      I haven't even covered task priority yet (except in passing) but you are correct that the serial port most certainly does get corrupted (well, interleaved) with data from more than one task if the data is written in different statements. Perhaps the Serial library uses secret mutexes to ensure atomic operation?

  • @taobot8
    @taobot8 3 года назад

    This was incredibly useful! I'd been thinking about how to break away from the Arduino's delay() function to achieve a 250Hz sampling of an IMU and also how to move to a more easily tested codebase. This video really crystalized things for me and helped me appreciate that it's time for me to ditch the Arduino framework and just jump straight to tasks and queues in RTOS. Thankyou so much!

    • @RalphBacon
      @RalphBacon  3 года назад +1

      I'm glad it's opened your eyes to new possibilities, Stuart! But when you say "ditch the Arduino framework", just hold your horses!
      When you program for the ESP32, you still use what is effectively an Arduino-like framework (ported from Espressif's IDF development system). Whilst there will be new syntax to learn for the ESP32, you can still use just _most_ of the Arduino-speak you currently use which should speed up development.
      If you really want to embrace a better way of working (and cut the apron strings of Arduino) think about using PlatformIO (built on top of Visual Studio Code). It is far less hand-holding but ultimately leads to a better programming experience. I now use it all the time unless I'm doing a video! It still uses an Arduino-like framework (supports Arduino-speak C++) but gives you much greater freedom and control when developing. Just a thought!

    • @taobot8
      @taobot8 3 года назад

      @@RalphBacon I've been using PlatformIO since I started down this path back in November last year. I normally develop on non-embedded systems using TDD, so the opportunity to use an IDE that integrated with the Unity test framework swung it for me to use PlatformIO over the Arduino IDE. My understanding was that the ESP32 port of the Arduino framework was a wrapper layer on top of the Espressif IDF, am I correct in that? I'm wondering if that's why the String implementation you'd mentioned wasn't stable hadn't given you any problems on your ESP32.

    • @RalphBacon
      @RalphBacon  3 года назад +1

      The PlatformIO (and Arduino) implementation of the Espressif ESP32 platform are not wrappers for the IDF (which we would only use if we were developing software professionally, I guess). Espressif release these platforms (port them) regularly and although Arduino and PlatformIO might be at different versions I think they are basically the same thing.
      That allows the use of Arduino-speak C++ for all the millions of users out there.
      I haven't had any issues with string handling in the PlatformIO environment but maybe Espressif's implementation of String (capital S) doesn't turn your heap memory into Swiss cheese. That said, I tend to use std::string as a more standard way of using strings in C++.

  • @robertmurton7373
    @robertmurton7373 3 года назад

    A interesting video, I am new to the ESP32 so this help me understand it better.

  • @timblack4123
    @timblack4123 2 года назад

    Hi Ralph, about the static keyword. Not sure if it’s different in the Arduino world but in C a static “modifier” would make the variable or function local to the module, eg the compiler doesn’t export it and make it available to other modules. This could be useful if you want to make sure the mutex is protecting the variable in all cases.
    In C++ the static modifier WITHIN A CLASS makes it a class level variable, eg the variable is available to all instances of the class. The variable isn’t instantiated. So what you could do is instantiate an object of type x in each task and either one could access the variable. Of course the mutex would also need to be static. To reduce code and complexity you could simply have a function or method that would increment the variable with the mutex protection and return the new value. This is kind of implementing a design pattern called a Singleton. Of course for this simple example you could just just use a completely static class that wouldn’t need to be instantiated.

    • @RalphBacon
      @RalphBacon  2 года назад +1

      I use 'static' variables mainly to reduce the scope; that is, ensure they are not global. Within a function they are removed from the function by the compiler (so they are not re-initialised on each loop) but are not visible to other functions.
      Even so-called global variables (within a sketch) would benefit from being static to prevent the compiler exporting the name and potentially clashing with a name in another library (happened to me, only once or twice).
      I'd be very wary of declaring variables as static within a class as they are, as you say, shared between instances of the class. Might be what you want but probably not! I don't expect they are thread-safe on an ESP32 either, but I'm just guessing. Yes, we could protect the variable with semaphores, but that's beyond most beginner coding ability (although I did do a video on that, which everyone seemed to understand) 😜
      Thanks for the feedback, Tim, some good points you raise there 👍

  • @stephenborntrager6542
    @stephenborntrager6542 4 года назад +1

    Threading is an incredibly complicated topic, and you are brave for even touching on it! (And you have done a remarkable job covering what to do, without scaring people away.)
    I assume the ESP32 does not do instruction re-ordering like an x86 would, so we hopefully do not need to worry about memory access semantics at a hardware level.
    Does the platform not support the std::atomic library? There are many constructs far more appropriate than "volatile" in general.
    The trouble with volatile is that it has inconsistent meaning on different compilers and different hardware! (This is actually a HUGE problem, from a standards perspective!)
    Typically, it is a "full fence" barrier, which prevents memory access re-ordering both ways across all occurrences of that variable. (Typically the entire cache line, not just that variable.)
    This can really hurt optimization to the point where a single core might be better. (Supposedly, this impacts RISC chips like ARM too, not just intel or amd!)
    What we really need is acquire/release semantics, like what a proper implementation of the standard atomics library should provide.
    Basically, acquire/release fences are a way of getting the same anti-corruption guarantee, but while also allowing the compiler to optimize around the variable.
    Most of this is implementations specific, and I don't know how good the compiler for this chip is able to optimize in the first place, so it's hard to say how big of an issue volatile is.

    • @RalphBacon
      @RalphBacon  4 года назад

      Certainly having two tasks update the same variable, Stephen, is just asking for trouble - unless the developer has ensured the entire read/update operation is atomic, which you most certainly can do. Volatile alone may not do this (it's usually used to ensure register copies of memory values are not used). The question is why you might want the headache of doing this! Certainly not for beginners (which you show you are not, so you probably appreciate the coding complexity this would entail).
      You could use semaphores or queues, of course, but you always risk a deadlock. When using tasks I'd try and ensure the entire task was doing something independently (single responsibility principle) to avoid huge amounts of testing and debugging.

    • @stephenborntrager6542
      @stephenborntrager6542 4 года назад

      @@RalphBacon I may have gotten a little carried away! You're right, it's definitely better to keep tasks independent when possible. After looking around, It appears that std::atomic is available, if we need it.
      I haven't even gotten one of these chips yet, so maybe it's too soon for me to be trying to push the things limits ;)

    • @RalphBacon
      @RalphBacon  4 года назад

      Indeed, start slow and build on what you learn. Great chip the ESP32 and the RTOS is very good too, pity it's not a FULL implementation (ie certain functions are just not available).

  • @chrisandersson3428
    @chrisandersson3428 2 года назад

    Hi. Thank you very much for taking the time to explain concurrent computing by examples. For me, a beginner, to follow an example saves a lot of time in learning.
    I'm using an Arduino Due and could not see same behaviour as on your video. Then I read the "FreeRTOS Reference Manual" and realized that " vTaskStartScheduler()" has to be added to get the RTOS "started". One differens is that your board has the RTOS "built in"? whatever that means. is that it? Can you say a few words about a set of FreeRTOS functions to master to make advanced projects.

    • @RalphBacon
      @RalphBacon  2 года назад

      To really master FreeRTOS (especially in ESP32 terms and perhaps Due too) you need to work through the book "Free-RTOS for ESP32 Arduino" by Warren Gay. If you learn 1/10th of what's in there you will be head and shoulders above everyone else. Well, except me, of course 😉

  • @treefrogclassics
    @treefrogclassics 3 года назад

    Thanks for producing this and #150. It greatly demystified multi-core programming for me. I have a project That I think might benefit from this.
    I have a ESP32 based controller that centralizes control of lights, thermostat, and air compressor in my shop. The thermostat is controlled via wifi and the lights and compressor are controlled via relays. The user interface is via a touchscreen and a webserver. Occasionally, the touch screen is unresponsive. I suspect this is because the thermostat is being slow to respond over wifi and the request is blocking while waiting for an response. It appears to me that I could avoid this by using the second core.
    Core 0:
    - thermostat handling
    - web server
    Core 1:
    - touchscreen service
    - timer maintenance
    - relay control
    How does this approach sound? My thinking is that the Core 1 functions would work normally while Core 0 tasks were blocked while waiting on the thermostat to respond.
    The timer values are accessed by the web server, touchscreen, and timer maintenance functions. I don't see how a queue would serve this. So I guess I would use global variables with semaphores.

    • @RalphBacon
      @RalphBacon  3 года назад

      I'm glad you found the video interesting, Leon. However...
      However, although I might have used Core 0 and Core 1 to make a point about moving values between tasks, you really should not use Core 0 at all; it's for the RTOS and Wi-Fi and BT... all the stuff Espressif have put in there. Core 1 is for us developers.
      So, the best approach is not to use Core 0, just to create a new task, at the same priority as all other tasks (=1) pinned to Core 1. That's what I do with my ESP32 Web Radio: a separate task to handle the transfer of data from an audio buffer to an audio module. Before I did this it was very jittery due to other functions in Core 1 blocking (mostly waiting for) some Wi-Fi procedure to return. Now it runs like a dream.
      Tasks get a time-slice of about 1mS unless they "yield" their time slot. If the task has run out of work to do, or is itself blocked, you yield the remainder of its 1mS time-slice to other tasks of the same priority (which, counter-intuitively then only can use the remaining time left in that 1mS slice) but it all works amazingly well. I do this with the above-mentioned Web Radio task; if it runs out of data I just "yield" - no point in hogging a time-slice to do nothing in!
      You've listed several functions that you feel need a dedicated time-slot; set up new tasks for each, but do this slowly. Keep everything in the loop initially, which, of course, must be non-blocking in any way (no delay(x) calls) and move each function to a new task, one by one, ensuring the whole thing functions.
      Ensure you have enough debugging (monitoring) in each task but remove that (or switch it off) as soon as you feel it's running, as Serial prints are slow.
      The other thing I do, is to call each of the functions from the main loop() every few milliseconds, rather than doing all tasks all the time. So, for example, I update the screen every X milliseconds (given that it's mostly static). This then gives more time to other functions in the main loop( ). You may find you need fewer tasks than you think!
      Anyway, this is not the work of a few hours, it will take some design and experimentation - good luck and go slow!

  • @kenasuea3617
    @kenasuea3617 5 лет назад

    I like your enthusiasm. Good Job !

    • @RalphBacon
      @RalphBacon  5 лет назад

      I probably had too many coffees before this video, but inside I'm always enthusiastic! Good to hear from you, thanks for posting.

  • @jedandecko5585
    @jedandecko5585 5 лет назад

    Glad you continue with esp32 series, Arduino is dying slowly even them are planing a lot of new boards. Still love nano for offline stuff.
    Thank you on your effort.

    • @RalphBacon
      @RalphBacon  5 лет назад +2

      Arduino boards (perhaps, as you say, Jedan, in Nano format) are useful in simple projects but even Arduino.cc have just released a whole new set of Nanos to get the interest back. But when you look at this chip, which is cheaper than the Nano (official version) there is no comparison in capabilities, it is true.

  • @asagk
    @asagk 4 года назад

    Hey Ralph!
    "loop(){...}" might also work very well for asynchronous stuff like writing aggregated log files (error messages, etc) and else from time to time. So not using loop() at all might not be the case for most projects, but instead be used for low priority asynchronous stuff to do, like waiting for logging buffers to fill up, to be written somewhere when ever a buffer is filled to some amount, like page/block size in a flash device (e.g. some 64/128mbit spi flash).

    • @RalphBacon
      @RalphBacon  4 года назад

      Great tip! Sounds like it would be a good use for the loop.

  • @edgeeffect
    @edgeeffect 5 лет назад +1

    I've said this before but I'm still impressed... your little talking head at the bottom of the screen is a huge improvement on the popular format of "stuff on a green cutting mat".
    And you VHS-fast forward fade is great too... well done on bringing a bit of life to what's otherwise quite a boring video format.
    Having multiple cores and an RTOS would be great stuff... it's a pity I'm not that interested in WiFi and Bluetooth or I'd be going crazy for these here ESP32s.
    The kids who smoke cigs behind the bikesheds say that you could ditch that FOR( ; ; ) and replace it with a label and a GOTO.
    Wow... string concatenation with a "+" operator instead of strncat().... Blimey! At this rate, they might even pursuade me to start actually liking "C".

    • @RalphBacon
      @RalphBacon  5 лет назад

      Glad you like the format of the videos; and the content too! That "for{;;}" construct is just what has been always used. I don't think anyone would ever use a GOTO in C... would they? But the compiler probably compiles it to that anyway. Yes, C can be OK but C++ (object oriented) is better, of course.

  • @rayztech7384
    @rayztech7384 4 года назад

    Great video full of info!
    Would it be possible to a video like this where you can control several servos?

    • @RalphBacon
      @RalphBacon  4 года назад +1

      If you write one function to control one servo you could run that function many times (once per servo) in different tasks. Just pass in the parameter which servo it is that you are wishing to control from your main sketch and Bob's your uncle.

  • @jwcolby54
    @jwcolby54 5 лет назад +1

    The architecture is 32 bit which means 4 bytes of ram written and read at once. Even more importantly though is that higher math functions such as multiply and divide are software routines in an 8 bit processor whereas they are native instructions in a 32 bit processor. Often times hundreds of instructions on an 8 bit cpu.

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      Often true, John, but it's lucky for us the AVR 8-bit chips do have hardware multiplication. According to the datasheet, it allows 8- and 16-bit implementations, signed and unsigned routines, and fractional signed and unsigned multiply.
      Here's some bedtime reading: ww1.microchip.com/downloads/en/AppNotes/Atmel-1631-Using-the-AVR-Hardware-Multiplier_ApplicationNote_AVR201.pdf

  • @henrikjensen3278
    @henrikjensen3278 5 лет назад

    Very interesting video.
    My favorite multitasking synchronizing is Javas synchronize and queues, with these two I can do a lot of safe multitasking. With semaphores you have to be careful, at least when using multiple levels. Deadlocks are a PITA to locate! This happens very easily when you need a couple of resources and each has it own semaphore and each task have to reserve stuff at different parts in the code.
    Reading and writing variable asynchronous is usual safe when using the native word size, but with anything else semaphores are needed.
    My ESP32 boards runs at 80MHz, but I am going to check if I can change frequency. I use some 30pins boards, they looks like nice ESP32 hardware, the 36 pins board add a couple of useless pins (The ones needed for internal program access).

    • @RalphBacon
      @RalphBacon  5 лет назад

      I showed you how to change the CPU frequency, Henrik, (assuming you watched to the end) so a quick 5-line sketch should prove it one way or the other.
      As for deadlocks, I sort of mentioned that if you cannot get a semaphore in the allotted time it will return a response indicating this. It's something I must expand upon as they are, as you say, somewhat problematic.

    • @henrikjensen3278
      @henrikjensen3278 5 лет назад

      I have tried that now, but I do not have the functions to get/set CPU frequency.
      Searching the library code I found this:
      uint8_t getCpuFreqMHz(){ return CONFIG_ESP32_DEFAULT_CPU_FREQ_MHZ; }
      In the ESP class and it return 240
      The timeout on semaphores are not always available and it is mostly meaningless in some content anyway.

    • @RalphBacon
      @RalphBacon  5 лет назад

      Yes, 240Mhz is the default speed (and that used by the Arduino IDE). But if you have installed the recent boards from Espressif then you should have access to those function I show, to set and get the speed of the chip.

    • @henrikjensen3278
      @henrikjensen3278 5 лет назад

      I have not really worried about updates, but hoped that the Arduino IDE handled that, but that may not be true. I use this link for ESP32 in the board manager:
      dl.espressif.com/dl/package_esp32_index.json
      I do not know how to force a update. Maybe remove the link and add it again after a restart.

    • @RalphBacon
      @RalphBacon  5 лет назад

      Now, Henrik, I did a detailed, step-by-step demo on how to add _or update_ the board definitions so you obviously skipped ahead! Go to Tools, Boards, Boards Manager then type in ESP32 in the search box and you will see ESP32 version 1.02 (or even later). Upgrade to the latest if you haven't already. That should give you all the (new) commands you seem to be missing. Let me know how it goes!

  • @Axidhead
    @Axidhead 13 дней назад

    Hi, I'm working on an project, with a HX-711 and a stepper motor controlled by a driver, my problem is that everytime the value is read from the scale the my code is blocked and the motor isn't running smoothly even with the millis() function, is there a way to get rid of that problem, I thought of working with two cores on an arduino UNO is it possible?

    • @RalphBacon
      @RalphBacon  13 дней назад

      If only the humble Arduino UNO had two cores, it could have been the answer. 😢
      Unfortunately, as this is not the case we must think of another solution. If you run the scale-reading function using a timer-controlled function (eg every 100ms) it will be able to read the value, store it in a global value and then exit. Don't do any other processing in the scale-reading function, as it will slow down the main loop to the extent you have already noticed.
      What the main loop must do is service the stepper motor and check the global value to see if it has changed or reached a desired value. Only take action if this condition is true, at which point you probably want the stepper motor to stop anyway so problem with any blocking.
      However, you might be able to do this without a timer-controlled function (just use the millis-based method) but, as I say, do the absolute minimum in that function (basically, read a scale value). I can't remember off-hand but if the HX711 requires some time to do the reading; if so, then you must request the reading to start on the first invocation, then continually check to see if the reading is available, and only when it is then you can read the value and place it in a global variable. Several other sensors I have do take some time to do a conversion and this method works very well.
      That way the scale-reading function will not block the main loop and your stepper motor can run smoothly.

  • @avejst
    @avejst 5 лет назад

    Impressive multi processor programming design
    Thanks for sharing👍😀

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      Thanks for your post, Asger Vestbjerg, good to hear from you.

  • @SameerKhanna-zh7iq
    @SameerKhanna-zh7iq 2 года назад +1

    Thanks for this great toturial.

  • @GoatZilla
    @GoatZilla Год назад

    This seems to apply to interrupt context as well, although I've rarely seen this mentioned in Arduino land. Even with a single core, your "main" loop seemingly can get interrupted wherever unless they run it with interrupts off...

    • @RalphBacon
      @RalphBacon  Год назад +1

      Yes, the main flow of the code (the loop) will be interrupted by, well, an interrupt! In an ESP32 environment, each task gets a 1mS slice of time to do its thing. Very fair!

  • @PavolFilek
    @PavolFilek 2 года назад +1

    Hi Ralph, I can not change freq. of ESP32 under 80 MHz, 240 OK, 160 OK, 80 OK, but if I set 40 MHz, ESP32 crashes.Why ?

    • @RalphBacon
      @RalphBacon  2 года назад +1

      I suspect that the Wi-Fi, Bluetooth and other protocols/services are too delayed with a slow frequency. The question I'm asking myself is why you would want to reduce the frequency of an ESP32 _at all_ let along to below 80MHz? Let me know!

  • @MrAshrot
    @MrAshrot 11 месяцев назад

    Brilliant vid, loaded with info! I have a question! A simple setup. task on one core reads temperature off a sensor, and the task on the other core sends that temperature to an IoT cloud. It seems its safe to use a global variable, unless the sending task tries to read the variable at the same time reading task tries to update it. For such a setup, would a queue be more suitable or a semaphore?

    • @RalphBacon
      @RalphBacon  11 месяцев назад

      Using shared (volatile) global variables is always tricky; using a semaphore before reading a shared variable will be much safer.
      BTW I would not put the two tasks on different cores; run them at the same priority on Core 1 and it will be safer for all the Espressif stuff (BT/WiFI etc) running on Core 0.

  • @LimbaZero
    @LimbaZero 5 лет назад

    One note: If you are planning to do custom board for application that uses ESP32 then recommend to check directly these modules. Just noticed they were from 2e to 5e without vat (in Mouser). Only downside is shipping cost but I usually order that much that I get free shipping.

    • @RalphBacon
      @RalphBacon  5 лет назад

      I'll have a look to see how the prices compare to Chinese outlets, thanks for letting me know.

  • @noweare1
    @noweare1 5 лет назад +1

    Are these boards with the esp32 documented well. I know the chip is but are there schematics or pin out table. Also, a huge advantage of an Arduino is the number of libraries already written. Can these libs be used with the ESP32 ? Thanks Ralph, great video. Glad your getting into this chip.

    • @RalphBacon
      @RalphBacon  5 лет назад

      The ESP32 community is vibrant, Joey, as is the documentation and support (including the forums) from Espressif themselves, IMHO. You can use the Arduino libraries although I've only used a few so far. After all, if you're writing commands to the I2C bus then it makes no difference what hardware platform you're running on. For the sake of a few dollars it's well worthwhile just getting one and playing with it (also IMHO).

    • @noweare1
      @noweare1 5 лет назад +1

      @@RalphBacon Thanks Ralph

  • @scroogemcduckenjoyer
    @scroogemcduckenjoyer 3 года назад

    Whew, ESP-32 and 8266 has been my first and go-to MCU until now (well alongside arduino Nano). I never touched the RTOS part insofar i dabbled with prototyping devices, only with the Arduino-world. Now with multitasking like these, ESP-32 seems more OP than i've imagined. How could i missed such unlimited power & possibilities lol

    • @RalphBacon
      @RalphBacon  3 года назад

      RTOS is pretty well hidden from users of the ESP32; but now you have discovered a whole new world of possibilities! Keep away from Core 0, that is for WiFi, BT and RTOS. Use tasks on Core 1.

  • @thomasmcneil9598
    @thomasmcneil9598 2 года назад

    Thank you for this great tutorial, I am learning so much from the creators on RUclips it is fantastic. With som small exceptions, I am beginning to believe you could get a really great technical education just from what you can read/ follow along with on RUclips and the web. A couple of questions about the Semaphore. You define the volatile long count0, and that is what each task increments inside the code. Prior to incrementing, the task will take the semaphore, increment count0 and then give the semaphore back. Between the take and give it accesses the global variable count0.My question is: What associates this global variable count0 with the semaphore? In the semaphore declaration, the global variable is not mentioned. Does the compiler associate whatever variables are between TAKE-GIVE with the semaphore? Can the task access multiple global variables between the TAKE-GIVE and those variables are "protected " from the other task? Thanks Again

    • @RalphBacon
      @RalphBacon  2 года назад

      In a word (or two) there is _no relationship_ between a semaphore and what you are then doing.
      The fact that we are updating a variable is just coincidence. The semaphore prevents _concurrent_ updating by two tasks that are otherwise unaware that another task is in the process of updating the variable.
      By using the semaphore, we are just saying to any other task, "Sorry, you will have to wait a while to get that semaphore and continue whatever it is you were doing." That's why we need to be quick about handing back semaphore when we have carried out the critical process.

  • @victoriasilva-qo8lz
    @victoriasilva-qo8lz 3 месяца назад

    Hello, I'm working on a project that is quite bigger than anything I've done before and I'm just learning FreeRTO. In my project I will need to read some values from a few sensors and later move some steppers a number of degrees; thing is that in order to get those degrees from the data of the sensors, I first need to run quite long calculations with float numbers, I made a library to manage the functions for my maths, but now I'm wondering, how can I know If I could handle all those math procedures on core 0 without starving the bluetooth and wifi functions (I also need to implement bluetooth as a form of comunication)? and how can I tell how many bytes should I give for an especific task to run well? I've been told that float operations can be quite taxing on speed and memory.
    I'm just starting to learn these topics and any help or resource that would help me better understand these concepts is much apperciated.

    • @RalphBacon
      @RalphBacon  3 месяца назад

      Wow! Complicated project! But let's see...
      First of all, I do not recommend running anything on Core0 which is effectively for Espressif system code. But the Espressif police will not turn up at your door if you do, just not recommended. 🫨
      Run everything on Core 1, ideally in self contained tasks. That way, each task get its time-slice and you don't have to worry that parts of your code will hog all the available CPU time whilst starving other processes.
      If you design your system with separate, self contained elements (eg getting sensor readings, calculating some numbers when those sensor readings are available, then your tasks will be easier to visualise and manage).
      As an example, these days I always run my Wi-Fi connectivity (and monitoring) in a separate task. That way it does its own thing in the background (so to speak), doesn't interfere with my main loop code, and just does work with some fairly simple code that is easy to write and maintain.
      Communicate between tasks either with simple global variables, or much better with semaphores that will ensure that it really is time for another task to begin its work in a timely fashion (you GIVE and TAKE a semaphore, so your task will sit there waiting to be given a semaphore and then run with it before handing it back over).
      This might sound a bit involved but it works much better than trying to do everything a huge piece of loop code.
      Watch a couple of (my) videos on tasks but there are others out there (surprise!). Eg flash an LED only when you push a button - but the flashing LED code runs in a task, as does the button-push detection code in a separate task. Your loop, in this example, does nothing much at all!

  • @tomwatzek3500
    @tomwatzek3500 5 лет назад

    Nice Video, well explained, I'm happy with that! Go on that way, thanks!

    • @RalphBacon
      @RalphBacon  5 лет назад

      You are most welcome Tom Watzek, I'm glad you like the video. Nice to hear from you.

  • @TristanGomez
    @TristanGomez 5 лет назад +4

    I have one of those ESP32 with the Arduino UNO form factor and they're surprisingly compatible with many of the shields, with some caveats: those that require 5v for gpio won't work as the ESP32 only has 3v3 pins, and the older shields that have SDA and SCL on pins A4 and A5 (like the older non-R3 compliant data logger clones) won't work either, as those pins are input only for the ESP32.
    I'm maintaining my own wiki while doing research; some people might be interested so I'll link it here if you don't mind :) github.com/parasquid/mcu-iot-sensors/wiki/ESP32#espduino-32

    • @TristanGomez
      @TristanGomez 5 лет назад

      One other advantage of having the ESP32 in the same form factor as the Arduino UNO is ability to simply transfer the shields from one to the other. Just a couple of days ago I was doing a presentation to a javascript group about Espruino and part of it was a comparison of how code would look like if written for the Arduino. It really helped hammer home the fact that the hardware being controlled is the same, but the language being used to program is different.
      Just wanted to mention that one quickly :) The queueing discussion was really interesting and it's something I'd like to experiment on in the future.

    • @jimb032
      @jimb032 5 лет назад +1

      Normally it can work, but it depends on the shield. 3.3v is still seen as a logic high to a 5v input. And if you feed the shield only 3.3v it will return only 3.3v logic levels. You have to look at them case by case.

    • @RalphBacon
      @RalphBacon  5 лет назад

      So what we need, Tristan, is an ESP32-compatible shield format! Someone has probably created one already!

    • @RalphBacon
      @RalphBacon  5 лет назад

      Glad you like the queuing discussion; it can be a bit long-winded but once understood it's pretty straightforward.

    • @TristanGomez
      @TristanGomez 5 лет назад

      I think the most successful shields were the ones by Wemos (which you've covered quite thoroughly in the past) but that was for the ESP8266; and there were some that tried to ride on the form factor, but I haven't seen anything that really took off.
      One would think that it would be easy to design a pcb which would take the usual devkit-c boards and adapt it to the UNO form factor (much like the nano adapter boards you've also shown in the past) but again, I haven't seen anyone build that (yet).

  • @jorgerive7335
    @jorgerive7335 Год назад

    Excellent video…much appreciated. I just found you while I was looking for tutorials on the ESP32 (WROOM) and FreeRTos- I’m new to these processors and in programming with OS’s ( I’m an embedded HW guy, knowing enough programming to be dangerous, ;-) and your videos are very helpful. What IDE do you use?

    • @RalphBacon
      @RalphBacon  Год назад

      I currently use Visual Studio Code (free) with the PlatformIO add-in. After an initial learning curve (plenty of helpful videos out there, but not by me!) it was bliss (as I have used Visual Studio for decades in my day job!)

  • @KSITREVS
    @KSITREVS 4 года назад +2

    Second question, hello again, what ide do you use? Looks a lot better for developing with! :)

    • @RalphBacon
      @RalphBacon  4 года назад +2

      This: eclipse.baeyens.it/ but don't forget Arduino is developing a new IDE based on the Eclipse Thea framework, probably for later this year, Trevor. But you can use that Sloeber version in the meantime to get familiar with the way it works (expect to devote several dozen hours to getting your head around it).

    • @efwilson
      @efwilson 4 года назад

      @@RalphBacon Great video as always, thanks! How do you find the Arduino Pro IDE compares to it?

    • @stephenborntrager6542
      @stephenborntrager6542 4 года назад +1

      @@RalphBacon That's the first I've heard of an alternative IDE from Arduino! That's really exiting!
      Another option I'd just put out there is visual studio code (free, lightweight, multi-OS), which has an unofficial but decently supported Arduino plugin. It relies on Arduino IDE for part of the toolchain though, so it has the same crippling problems. (Weirdness involving header files and multiple source files in a project.) It CAN parse the source though, and provide code completion and suggestions, etc. Overall it's a very nice editor, but it can't overcome the issues inherent in the underlying build process.
      I've never had a good time with Eclipse, though I know it has come a long way, and literally anything is better than the plain Arduino IDE! So both this Sloeber thing and that new Arduino IDE are hugely good news, to me! Thanks for sharing!

  • @GBRiLFLRNTiN
    @GBRiLFLRNTiN 8 месяцев назад

    29:00 you set task0 with priority 10 and task1 with priority 1 ( is that why task0 runs many more times than task1? and, besides that, isnt task priorities limited to 1-5?)

    • @RalphBacon
      @RalphBacon  7 месяцев назад +2

      That looks like a typo, edited from a priority 0 (idle) to a standard priority 1. The thing is, if a higher priority task is ready to run (or continue to run) then it _will_ continue to run, starving any lower priority tasks from any cpu time at all.
      I'm sort of surprised this worked at all.
      TL;DR they should both be at priority 1 to give them an equal share of the processing time.

    • @GBRiLFLRNTiN
      @GBRiLFLRNTiN 7 месяцев назад +1

      @@RalphBacon thanks, man. Your video helped me a lot

  • @emmanuelsheshi1553
    @emmanuelsheshi1553 2 года назад

    the LOCK method does the same thing in python Threading module. awesome tutorials sir.

    • @RalphBacon
      @RalphBacon  2 года назад

      Thank you Emmanuel, glad you found it interesting.

  • @dalewolver8739
    @dalewolver8739 4 года назад +1

    Great explanation.

    • @RalphBacon
      @RalphBacon  4 года назад

      I'm very glad you think so, Dale, you're very welcome.

  • @unglaubichuberlieber8048
    @unglaubichuberlieber8048 5 лет назад

    you have explained this rtos, multitasking very well, what i like to know, is the ram on the board you have used, or tell me where to find that info, thanks in advance

    • @RalphBacon
      @RalphBacon  5 лет назад

      I'm glad you found this useful. Regarding your question, I'm not totally clear what you mean by "is the ram on the board you have used".
      The ESP32 has 4Mb of SRAM built into the chip, shared by both processors and is used just like any other memory would be. You know this. so what was your question, exactly?

  • @TheDiverJim
    @TheDiverJim 3 года назад

    Great vid. I know it’s a nit, but probably shouldn’t have held the the mutex while printing. Longer block than required.

    • @RalphBacon
      @RalphBacon  3 года назад +1

      Great point! Ok for this demo but in general you are spot on.

  • @thisnicklldo
    @thisnicklldo 5 лет назад

    Interesting - but I'm a bit perplexed. Taking the example using a global, it seems the ++count is not atomic - I see that ++count must take a few machine code instructions, and if you say so then I can see that the RTOS might choose the wrong moment to suspend the task doing ++count and start the other task that also wants to do ++count. But surely the underlying semaphore implementation must take several machine code instructions, so that is presumably treated as atomic, else it won't work. Is the position that the compiler only guarantees to create semaphores as atomic instructions and nothing else?

    • @RalphBacon
      @RalphBacon  5 лет назад

      Quite so. If the count is not atomic then the use of a semaphore is _essential_ but we can't be sure at which point the task handler switches contexts. But the semaphore ensures consistency.

  • @mumbaiverve2307
    @mumbaiverve2307 4 года назад

    Hi Ralph, your comment about the String() function vis-a-vis the ESP32 was a tad inaudible. Is the String function safer in ESP32, I mean is there an error catch , if heap allocation is unsuccessful ? Thanks for your amazing videos !

    • @RalphBacon
      @RalphBacon  4 года назад

      The use of the Arduino "String" with a capital S is known to cause heap fragmentation (and eventually a crash).
      However, heap management is allegedly better on the ESPxxxx processors and your heap space doesn't turn into Swiss cheese.
      That said, I've used std::string in my ESP8266 work and that runs 24x7 for months on end with no issues.
      Regarding a "catch" for heap fragmentation problems, I'm not sure you _can_ correct it - I mean what is your code going to do if your request for heap space fails?

  • @ardespmaker
    @ardespmaker 4 года назад

    Just came across this - I was really confused before. What a great presentation, thanks. Could you tell me - in the xQueueCreate would it be possible to send a structure with different data types. Also could you send an array?
    One thing I found - - (I only use Arduino IDE) using the serial print code within the tasks in the format with String caused the fatal error situation. Once I split it into 3 statements no problems at all. Interesting that the String format style does not cause problem when used in the standard loop section.

    • @RalphBacon
      @RalphBacon  4 года назад

      The _vparameters_ parameter type is just a void pointer (to a variable), so I guess you could point it to a struct or even a class instance (object)! I have some faint bells ringing but no definitive answer, this time. Strange about Serial.print but maybe a native C++ call might work better. Or output a char array rather than a String (with capital S).

    • @ardespmaker
      @ardespmaker 4 года назад +1

      @@RalphBacon Thanks Ralph. I've been able to pass both arrays and structs with no issues. Interesting was that the number of arrays or structs put on the queue works just like a normal variable in the xQueueCreate statement, as the sizeof () is just set to the array or struct variable. If you set to 1 then task0 just waits until it can put the next whole set of array data to the queue.

  • @peut
    @peut 5 лет назад

    Hi, for some reason, your example worked well, but in a bigger project I got a kernel panic (.../freertos/queue.c:1442 (xQueueGenericReceive)- assert failed! ) when taking or giving the semaphore.
    I had to initialize the semaphoers with Semaphore = xSemaphoreCreateMutex(). after that ot worked fine.
    Ideas?

    • @RalphBacon
      @RalphBacon  5 лет назад

      Indeed you have to create the semaphore, as I show in my setup() routine. Did you miss it or I have misunderstood your question, Jose?

    • @peut
      @peut 5 лет назад +1

      @@RalphBacon Yes, I missed it :-).
      Sorry.

  • @bharadwajtke
    @bharadwajtke 4 года назад

    thank u for the video..done a great work...I appreciate ur genorous nature

    • @bharadwajtke
      @bharadwajtke 4 года назад +1

      I wasn't subscribed now i subscribed...Thank u

    • @RalphBacon
      @RalphBacon  4 года назад

      And thank you back, welcome to my channel!

  • @FaizanAhmad-mf5wq
    @FaizanAhmad-mf5wq 5 месяцев назад

    Esp32 CAM
    Hii guys, i am stuck at designing a system.
    I want one task to continually check for touch sensor signal, if it comes, then save image and insert its name into queue.
    For second task i simply need to poll queue if there is a file name, send that file to email, pop the name and continue polling.
    I am unable to run both tasks concurrently on both cores, esp32cam resets itself when i run the program.
    Can someone help me how should i design my tasks so they can work concurrently?
    Please pardon me i am new to these things.

    • @RalphBacon
      @RalphBacon  3 месяца назад

      All tasks will start running the microsecond you start each one; have you given enough memory to each task, which is a common failure?

  • @akasickform
    @akasickform 5 лет назад

    This is great!
    I was refreshing my memory as to how queue and semaphores work etc.
    I forgot that you can change the processing speed.
    Does this mean you can do this per core?
    Also, would be interesting to see how much energy per cycle is used when the clock rate is lowered.
    IE. If using a battery system for a non-time critical application, is it more energy efficient to lower the clock cycle, and if so, how much benefit is this?
    (I'd imagine the results would not be linear. Also, it would generate less heat due to lower clock speeds, but how much does this actually save?)

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      It's pretty much a fallacy, IMHO, to think that you will save power by reducing the clock rate.
      If your microController is mostly asleep then that is where your power savings come from. Then, upon waking, you want to get things done pretty much as quickly as possible so you can get back to sleep as quickly as possible.
      Whilst slightly less power is consumed whilst on a lower clock frequency it does not make up for the fact that you will be awake for longer. That's my take on it, anyway.
      Of course, if you are dependent on a reliable clock (eg serial transmissions) then a slower internal one _may_ be better regulated but then you should really use a crystal anyway!

  • @chrisgibbons2304
    @chrisgibbons2304 3 года назад

    Great video, just started using the ESP32 and I'm finding it difficult to find these sorts of indepth explanations, keep up the good work. Regarding core 0, if you're are not using bluetooth/wireless would it be safe to use it?

    • @RalphBacon
      @RalphBacon  3 года назад

      Thanks Chris for your kind words. Now, if you're not using Wi-Fi or BT there is still the OS running on Core 0 but I would use it, with caution. If you upset it you will soon find out with a PANIC or similar on Core 0. Don't forget to yield tasks when there is no more work to do and to keep the priority of _your_ tasks to the default of 1 so that others in Core 0 continue to run.

  • @chriskosik663
    @chriskosik663 4 года назад

    Very solid video. Thank you so much!

  • @marianofranceschetti7847
    @marianofranceschetti7847 3 года назад

    Hello Ralph, thanks a lot for your videos. Do you have one of an introduction to ESP32 telling your prefered IDE and what to know about freeRTOS? Something like the correct way to work with ESP32.
    I'm currently using Arduino and should upgrade.

    • @RalphBacon
      @RalphBacon  3 года назад +2

      Espressif, the designers of the ESP32, have taken note of the popularity of the Arduino IDE, Mariano, and you can continue programming right there if you wish. Just install the ESP32 boards from the Tools, Boards Manager, ESP32 once you have added in the following entry into your Tools, Preferences, Additional Boards URL Manager:
      dl.espressif.com/dl/package_esp32_index.json
      That said, I would take the plunge and either use the new Arduino PRO IDE that is still in alpha (based on Eclipse) or use the open source and well-respected PlatformIO with Visual Studio Code (which I am currently using).
      Anything new will require a degree of familiarisation (aka a steep initial learning curve) but the PlatformIO community is mature with a wealth of documentation and videos.
      I decided to build my ESP32 Web Radio project using Platformio & VS Code (so I became familiar with it) and whilst at times it was quite frustrating, it eventually became a joy to use. No turning back for me now although I will keep trying the Arduino PRO IDE (it didn't compile my ESP32 project though!)

    • @marianofranceschetti7847
      @marianofranceschetti7847 3 года назад +1

      @@RalphBacon Great, thanks a lot. I'll study PlatformIO. So far Arduino has shown to be sufficient, but sometimes I fill it's not enough, specielly when debuging.
      Have a nice week-end : )

  • @JeanDAVID
    @JeanDAVID 3 года назад

    in maxprime process, there is a double loops computing float TEMP and long TEMP2 ? is this used to slow the process more ?

    • @RalphBacon
      @RalphBacon  3 года назад

      Yes, I put a number of things in there that I new would be computationally hard work for all the processors (although the ESP32, being 32-bit with a floating point unit, found it no trouble at all).

  • @AndrewJones-tj6et
    @AndrewJones-tj6et 5 лет назад

    When setting CPU frequency below the default of 240MHz are there any implications on the WiFi/Bluetooth tasks and can the CPU frequency be changed on the fly within the loop not just setup?

    • @RalphBacon
      @RalphBacon  5 лет назад +2

      Others have asked the same, Andrew, and Espressif indicate that 80Mhz should be considered a safe, minimum frequency to use. But, yes, you can adjust the frequency on the fly, dynamically as my demo shows. Have a go!

  • @TamPham-oj4os
    @TamPham-oj4os 3 года назад

    I want to save the raw array into non-volatile memory. Can I save the
    IRremtoe raw array by spiffs ? Which one is the best NVS, SPIFFS, EEPROM
    ? Thank you

    • @RalphBacon
      @RalphBacon  3 года назад +1

      The best way (and now advocated by Espressif, the makers of the ESP32) is to use LittleFS instead of SPIFFS.
      I did an entire video on this, and used it in my ESP32 Web Radio project.
      ruclips.net/video/4r6YZlLfKfw/видео.html

  • @4lnstudios833
    @4lnstudios833 3 года назад

    I want to use 2 tasks, one for a keypad and lcd interface and the 2nd to drive a stepper motor. Is Semaphore a good way to pass the input of how much to move the motor? Or perhaps to interrupt it if it is mid movement? Will calling the Semaphore code in a loop that controls a stepper motor delay the movement because it has to keep getting the value?

    • @RalphBacon
      @RalphBacon  3 года назад +1

      You sure you want to use two tasks for this? You could certainly use a separate, self-contained task to get input from the keyboard and display that on an LCD, but how about using the standard loop() to move the stepper motor?
      Why? I can hear you asking.
      Well, if the keyboard task updates a volatile, global variable then the main loop just reads that (you might get away with no semaphores if you ensure the read is atomic, that is, it reads the entire two-byte integer) and drives the motor the required amount. If you can use a single byte for the value then it will work without any semaphores or worrying about atomicity. And the stepper motor is only moved if that global variable changes value.
      Incidentally, the processor is so quick there will be no "delay" in the stepper motor just because you have to read a variable, whichever way you design your code.

    • @4lnstudios833
      @4lnstudios833 3 года назад +1

      @@RalphBacon Thank you so much for your time in replying. I know you didn't have to but that is what makes you a great person. I appreciate the input and I would love to avoid having to do multiple tasks. I just didn't want having to take input from a keypad, update a display and run a stepper motor at the same time to stop the smooth flow of the stepper motor. Based on what you have said, maybe I should try and get it working the basic way first and then look at improving my code if it doesn't work rather than giving myself hurdles I haven't even encountered yet. Thanks again, you rock!

  • @normcaissie5598
    @normcaissie5598 5 лет назад

    Keep up the good videos

    • @RalphBacon
      @RalphBacon  5 лет назад

      Thanks for your post, Norm Caissie, good to hear from you.

  • @michaelhyde9971
    @michaelhyde9971 5 лет назад +2

    Hi Ralph dose the clock speed upset the wifi .

    • @jimb032
      @jimb032 5 лет назад +1

      It's not supposed to...it's managed in the background by the RTOS. ESP8266 yes- you can seriously piss it off.

    • @michaelhyde9971
      @michaelhyde9971 5 лет назад

      @@jimb032 thanks

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      Espressif seem to say that 80Mhz is the slowest, reliable speed for any ESP32 variant (when running wifi). But the only way to know is to try it at lower speeds and see what happens.

    • @michaelhyde9971
      @michaelhyde9971 5 лет назад

      @@RalphBacon thanks for letting me know. I try when I have fixed my usb port after a little mishap. Lol.
      Great videos as usual.

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      "A little mishap" = "totally destroyed the motherboard"? Yes, I've had a few like that! I've heard anecdotally that the wifi signal gets "corrupted" at too low CPU Speed, but an identical ESP32 with the same speed can communicate with it no problem. Secret network!

  • @chevere3424
    @chevere3424 Год назад

    I am trying to show a message on an LED screen that comes from a telegram message and at the same time if this message is "led1" the led fades in and fades out (using pwm), but the telegram bot needs 1 second to wait for a new message, then I thought "I could use a core to update the message on the screen and another core to manage the class that makes the LED fade" however I can't do it, I see that you use two different tasks in the same core, if you use a delay(1000) in one task and delay(50) in the other, do they add up or are they still independent?

    • @RalphBacon
      @RalphBacon  Год назад +1

      If you are (still) using delays you need to learn how not to do that. Blocking code is wasteful in microcontroller cycles.
      At the very least you could YIELD that part of the timeslice it has been given, but better still is suspending the task for a particular period (either relative or absolute). Or just suspend the task until it is un-suspended (resumed) by another bit of code. Options, options!
      OK, enough wrist slapping! If you delay one task you will not affect the operation of other tasks (they all get an equal share of the time, usually 1mS each). (But if you do YIELD a task, the next task gets the remainder of its timeslice).
      BTW I've covered how not to use delay statements in more than video, good hunting (a searchable and linked pdf of all my videos in each video description).

    • @chevere3424
      @chevere3424 Год назад +1

      @@RalphBacon I really appreciate your help, since I learned to use millis on class objects, I no longer use delay. I hope to show you the progress of my project. Thank you!!!

  • @paulyorke1437
    @paulyorke1437 5 лет назад

    So very very clear as always, thanks Ralph. Just a question about cores, I am planning to run three tasks and was thinking of task0 on core0 and the other two tasks on core1. If I'm not using the wifi, will this be faster / more efficient than running all three tasks on core1?

    • @RalphBacon
      @RalphBacon  5 лет назад +1

      Short answer: yes, Paul. Longer answer: Yes. If you use two independent cores then each core can support X tasks before you see a noticeable slow-down. But even if you do not use WiFi or BT you might have to take further steps to kill (suspend?) the tasks that _are_ running on Core 0 as otherwise you may get a Core Panic and the ESP32 will reboot. That said, if you code well, that is don't hog all the CPU cycles in a tight loop, for example, it will all work anyway (as my demo from the previous video showed). Good luck!

    • @paulyorke1437
      @paulyorke1437 5 лет назад

      @@RalphBacon Thanks Ralph. Just a supplementary (and probably unnecessary) question, your excellent demo showed tasks sharing data via queues but on one core only. Can I assume that this technique will work between tasks running on different cores? All the best.

  • @KSITREVS
    @KSITREVS 4 года назад

    Why doesn’t serial.println need semaphores? Surely two tasks accessing this at the same time would cause a problem? Thank you very informative!!!

    • @RalphBacon
      @RalphBacon  4 года назад +1

      Yes, it can cause issues with one message mixed up with another!

    • @KSITREVS
      @KSITREVS 4 года назад

      @@RalphBacon interesting, ill remember this when i come to making something. Thank you!

  • @mohamedgendia4274
    @mohamedgendia4274 2 года назад

    Can the queue take more than one variable? Is it possible to pass variables like the functions ? If i want to give it the address to time struct or any thing complicated is that possible?

    • @RalphBacon
      @RalphBacon  2 года назад

      Yes, the queue can take any number of variables (structs, objects etc).

  • @bobp5776
    @bobp5776 9 месяцев назад

    Is it possible to run a i2c device in one of those Tasks? So far i haven't seen it anywhere. I.e. A simple AS5048 i2c Position Sensor running in one core , Making its Angle data available Globally to the other core . I cant get it to work.

    • @RalphBacon
      @RalphBacon  9 месяцев назад

      I2C (and, indeed, any other hardware device) can be used in any task. You just have to be careful not to conflict their use from multiple tasks. However, back to your question.
      If you have initialised the I2C device, either in the "setup" of your sketch, or the task itself, you should be able to determine it is present (perhaps using the "standard" ESP32 GPIO pins 21/22 but you can actually assign most GPIO pins as you see fit). Look at the ubiquitous I2C "scanner" sketch to ensure you have discovered the device, it's only about 6 lines long!
      Then, in your task's infinite loop (tasks must never end, except by deleting them) just use the I2C functionality as you need. It all works just as you would expect.
      Tip: Keep all your task on Core 1, leaving Core 0 for the ESP32's system processes (such as Wi-Fi, BT and so on). You can expose the data the task reads on the I2C bus by updating a shared variable (if you want to consume the data in another task, you will probably need to use semaphores so each task doesn't fight over - and corrupt - the data values).

  • @YordanYanakiev
    @YordanYanakiev 3 года назад

    Hello Ralph, I have the follow issue - I need to share a record of a few Strings, and integers between 3 tasks on ESP32. What's the best way ?

    • @RalphBacon
      @RalphBacon  3 года назад

      Firstly, don't share strings they take far too much space up. Use a value that means the same as a string. Eg 1=error, 2=too hot, 3= too cold... You get the idea. Push the value onto a queue as I described in this and other esp32 videos and the other task will detect the presence and continue.

  • @sevm7792
    @sevm7792 2 года назад

    "Bletooth" is the version the "based esp32" uses?

    • @RalphBacon
      @RalphBacon  2 года назад

      ESP32 does indeed have Bluetooth (the more recent chips support Low Energy BT, BLE). So the ESP32 can act as server or client for BLE.

  • @willofirony
    @willofirony 5 лет назад +1

    Static keyword: defines a variable that "lives" in the static data area i.e. not on the heap (dynamically created variable that lives until the memory is reallocated [deleted]) and not a temporary variable that lives on the stack and only lives until the end of the current scope. Now any variable that is declared outside a a function scope (including main()) is also designated space in the static data area. So, the static keyword is NOT essential. However it does make the code and intentions of the programmer easier to read. This is particularly the case when the static variable is declared outside but amongst a bunch of functions (either because the programmer just remembered it was needed by the following function or to indicate the close relationship between the variable and the following function). No matter were the variable is declared it lives until the program exits (when all the variables go to the variable cemetery - very sad).

    • @RalphBacon
      @RalphBacon  5 лет назад

      Indeed, Michael. So, given that the variable in question is declared within the scope of main() but outside of setup() and loop() it should exist for as long as main() runs (ie until the program exits). So with or without the static keyword the variable remains a Global Variable. I wonder why Espressif want it declared as static? Perhaps the SDK they use works a bit differently to the traditional Arduino setup?
      Thanks for the detailed response, very interesting, always good to hear from you.

  • @sudedemmanuel2975
    @sudedemmanuel2975 2 года назад

    Im trying to pass a dynamically changing variable to Task 1 (one task only) that is running like infinite loop, the variable has been defined in .h file and used by the software and dynamically changing, I want to keep track of its value
    I could pass a constant and it works, can you help please?

    • @RalphBacon
      @RalphBacon  2 года назад +1

      You need to pass it as a (void *) &reference (pointer) and then de-reference it in the task back to the correct variable type. I'll post some very simple code tomorrow for you; I do this in my project. What are you passing to the task?

    • @sudedemmanuel2975
      @sudedemmanuel2975 2 года назад

      @@RalphBacon can i send you my code so to show how i can modify please

    • @sudedemmanuel2975
      @sudedemmanuel2975 2 года назад

      @@RalphBacon thanks Ralph much appreciated im passing a float variable that is being used elsewhere and changing i need to track its value it is declared in a structure in .h file and used by .c files i think 2 i passed a constant float variable and that worked but the varying one is not giving me the right values

    • @RalphBacon
      @RalphBacon  2 года назад +2

      This is how I do it:
      1. When you create the (pinned) task, I pass the parameter like this:
      (void *)&myVariable, /* Task input parameter */
      2. In the task I dereference it (to get to the actual value) likes this:
      int myVariable = *((int *)parameter);
      You need to do this action whenever you want to get to the new (updated) values.
      I was wondering whether there was another way I could do this but as I am using a struct in an array it's the best way for me.
      One word of warning: floats are not a good variable type to pass around as they get rounding errors very easily. Better to use an integer (2 bytes) or a long integer (4 bytes) and then divide down to the float when you need to _display_ it. It's what I do.
      Example: float value of 10.45 received from some sensor.
      Multiply by 100 to get 1045 as the _integer_ value that you can then use without worries of rounding errors.
      When you need to display the value just do
      (float) myInteger / 100.00

    • @sudedemmanuel2975
      @sudedemmanuel2975 2 года назад

      @@RalphBacon Hi Ralph thanks for your reply this is my code in the task its running outside void loop()
      void task1(void *parameters){
      //contl.LutOut=34.0f;
      //volatile control_output_t contl;
      digitalWrite(2,HIGH);
      delay(5000);
      digitalWrite(2,LOW);
      delay(5000);
      for(;;){
      //volatile control_output_t contl;
      if(contl.LutOut>33.0f) {digitalWrite(2,HIGH);
      }
      else if (contl.LutOut

  • @MPElectronique
    @MPElectronique 3 года назад

    Hello Ralph, Why vTaskDelete (NULL); in the main loop?
    THanks.
    Marc.

    • @RalphBacon
      @RalphBacon  3 года назад

      Because we have our own task(s) running we don't need the "loop" any more so we can delete it in this manner; keeps things tidy and reduces resources.

  • @niekbeijloos8355
    @niekbeijloos8355 4 года назад +1

    Very nice thank you!

    • @RalphBacon
      @RalphBacon  4 года назад

      I'm glad you found it useful, thanks for posting.

  • @AKKJ420
    @AKKJ420 3 года назад

    I am not sure what I am doing wrong but I am getting the following error. I think when I comment out vTaskDelete(Null) it stops rebooting but nothing appears on the serial monitor.
    rst:0x1 (POWERON_RESET),boot:0x13 (SPI_FAST_FLASH_BOOT)
    configsip: 0, SPIWP:0xee
    clk_drv:0x00,q_drv:0x00,d_drv:0x00,cs0_drv:0x00,hd_drv:0x00,wp_drv:0x00
    mode:DIO, clock div:1
    load:0x3fff0030,len:1240
    load:0x40078000,len:13012
    load:0x40080400,len:3648
    entry 0x400805f8

    • @RalphBacon
      @RalphBacon  3 года назад +1

      Why did you delete that task? It is the loop( ) task which must either contain a single "delay(1)" or it must be removed by the line you commented out. Reinstate the line (or put in the delay) and see whether your device boots again. Don't forget to set the Debug Level to VERBOSE (level 5) as part of the Tools / Settings.

  • @chriskosik663
    @chriskosik663 4 года назад +2

    wroom vs wrover has to do with sram differences!

    • @RalphBacon
      @RalphBacon  4 года назад +1

      Good to know, Chris, pity Espressif didn't make it plain.

  • @gavinsmalley1513
    @gavinsmalley1513 5 лет назад

    I'm not sure of the origins of the names "WROOM" and "WROVER" but, as far as I understand it, essentially the two major differences are that the "WROOM" is a 3.3v core with no PSRAM and the "WROVER" uses a 1.8v core and has a PSRAM chip hiding under the can.

    • @RalphBacon
      @RalphBacon  5 лет назад

      Yes, that PSRAM caused them no end of problems, from I hear (and read)! You raise an important note there, Gavin, that these modules are NOT 5v tolerant (although some have experimented) but I would always use a level shifter). Thanks for posting, good to hear from you.

  • @W122ard1
    @W122ard1 2 года назад

    If i use volatile global variable will it be safe to use between threads?

    • @RalphBacon
      @RalphBacon  2 года назад +1

      No, Alexander, volatile variables are not thread-safe. You would have to use semaphores or use another signalling system to ensure you did not stomp all over a variable as it was being updated by another thread.
      Remember that *volatile variables* are simply those flagged to the compiler that they are unsafe to be referenced by a register where the compiler might have put a _copy_ of the variable's value; it will _always_ be retrieved from its memory location. But that doesn't help across threads at all!

    • @W122ard1
      @W122ard1 2 года назад

      @@RalphBacon Hm actually after some research and reading, looks like in some case i can use that. Depending on controller "bits" so in arduino it is safe to use byte size variables, and it case of ESP i could safely use up to 32bits variable. But if some structure to read is more than 32bits in this case could be problems

    • @RalphBacon
      @RalphBacon  2 года назад +1

      Well, OK, a single byte is OK, because the operation is _atomic_ - that is, it completes in its entirety or not at all. But updating an integer (2-bytes) can go very wrong, as one task updates the low-byte then before it has a chance to update the high-byte the other task stomps all over the low-byte value (or reads a now corrupt value).
      Personally, I think it would be foolhardy to _hope_ it will work when semaphores are very easy to use and are designed to avoid this exact situation of clashes. Just sayin'! 😜