going back to go

My mind appears unable to fully embrace how Rust is meant to be idiomatically used. I guess I’m too old. I’ve gone back to the Go language and discovered, once again, that it’s excellent at easily expressing complex ideas in code and having that code execute the way I expect. I see myself going back to Go over Rust and C++, and using Python in its MicroPython variant for embedded development. For software I write for others, I want a language that wraps everything up into a single binary executable that doesn’t require special support libraries be installed along side it. I can produce these types of binaries with C++, Go, and Rust. Python is complete on embedded devices, which is why I now do heavy development using MicroPython.

For this post I’ve written my devices listing application in Go, which I’ve previously written in Python, C++, and Rust. Here’s the Go listing.

package mainimport (  "errors"  "fmt"  "log"  "os"  "path/filepath"  "strings"  "encoding/json")func main() {  executable, error := os.Executable()  if error != nil {log.Fatal(error)  }  var annotations = make(map[string]string)  jsonPath := filepath.Dir(executable) + "/devices.json"  rawJSON, error := os.ReadFile(jsonPath)  if errors.Is(error, os.ErrNotExist) {fmt.Println("Annotations JSON file not found at:", jsonPath)  } else {json.Unmarshal(rawJSON, &annotations)  }  deviceById := "/dev/serial/by-id/"  files, error := os.ReadDir(deviceById)  if error != nil {log.Fatal(error)  }  for _, file := range files {symlink, _ := filepath.EvalSymlinks(deviceById + file.Name())device := filepath.Base(symlink)suffix := file.Name()[strings.LastIndex(file.Name(), "_")+1:]hexid := strings.Split(suffix, "-")[0]annotation, found := annotations[hexid]if found == true {  fmt.Println(device, "  ", annotation)} else {  fmt.Println(device, "  ", hexid)}  }}

Here’s a typical output. In this example I have three boards plugged into my Linux system, an ESP32-S3, an ESP32-C3, and a Raspberry Pi Pico 2:

ttyACM0Raspberry Pi Pico 2 - MicroPython 1.24.0 - RP2-EC93ttyUSB1ESP32-C3-DevKitC-1-N4 - ESP-IDF 5.3.1 DisplaysttyUSB0ESP32-S3-DevKitC-1.1-N32R8 - MicroPython-1.25.0-preview-xtensa-IDFv5.2.2 2024-11-08 - ESP32S3-7814

You can see the application’s found them all and their associated TTY port. This is quite useful when using command-line tools to communicated with them individually.

And finally, here’s a section of the JSON file the application reads in in order to match annotations with devices I have plugged in:

{  ...  "7a1c74c68740ec93": "Raspberry Pi Pico 2 - MicroPython 1.24.0 - RP2-EC93",  "7eab1642dbfaeb1198773ca4c6d924ec": "ESP32-C3-DevKitC-1-N4 - ESP-IDF 5.3.1 Displays",  "cc2ba0cbe2cfec11973d262686bdcd52": "ESP32-S3-DevKitC-1.1-N32R8 - MicroPython-1.25.0-preview-xtensa-IDFv5.2.2 2024-11-08 - ESP32S3-7814",  ...}

Quick Operational Description.

Here’s a bulleted list of what I’m trying to accomplish with the Go code.

  1. When started, determine where in the file system we were started (lines 14 and 18).
  2. With that file path, append our annotations file name to it so we can open it (lines 20 through 28). Note that the annotations file is in JSON.
  3. If the annotations file exists, read it in and unmarshal it as a map. If the file doesn’t exist, emit a message saying it couldn’t be found and where it is supposed to be located.
  4. Look for any devices plugged into the Linux system (line 31).
  5. Assuming you find any devices, extract the hex ID from each device name found (lines 38 through 42). Those hex IDs are unique.
  6. Try to look up the hex ID in the dictionary you loaded in step 3. If you find an entry, that’s your annotation. Print it out with the device you found in line 41. If there is no entry then print the device with the raw hex ID.

That’s basically it.

Final Thoughts

Life is way too short for me these days. I still like to write software, but now I want a language and supporting tools that make my efforts far more efficient. Go gives me all of that, and more. I started with Go over ten years ago (you can search for my entries in this blog) on very early Raspberry Pis, specifically the 3 and 4. Then I just sort of let it fade away while chasing other new and shiny languages like Rust. Now I’ve come back to Go for good, and will continue to use it for my purposes until I can no longer think in code, let alone write it.

fatal problem with esp-idf 5.2.3 building micropython 1.24.0 preview

Espressif released ESP-IDF version 5.2.3 two days ago from the date of this post. I installed a copy and then compiled the latest version of MicroPython I have on my system. When I then flashed the build onto one of my Espressif boards, an ESP32-S3 N32R8 DevKit C board, the MicroPython software crashed, performing an infinite boot loop. This is one cycle of the boot loop I recorded via Thonny plugged into the board’s micro USB port.

MPY: soft rebootMemory: 8,308,816 MB Flash: 8,388,608 MB  Platform: MicroPython 1.24.0 preview xtensa IDFv5.2.3 Unique ID: 68B6B33D7814  SSID: ESP32S3-7814 CPU Clock: 160,000,000 Hz   I2C: SoftI2C(scl=2, sda=1, freq=500000)   I2C: Devices found: ['0x3d']   I2C: SSD1306 OLED FoundA fatal error occurred. The crash dump printed below may be used to helpdetermine what caused it. If you are not already running the most recentversion of MicroPython, consider upgrading. New versions often fix bugs.To learn more about how to debug and/or report this crash visit the wikipage at: https://github.com/micropython/micropython/wiki/ESP32-debuggingMPY version : v1.19.1-2669.g17d823445.dirty on 2024-09-27IDF version : v5.2.3Machine : Generic ESP32S3 module with Octal-SPIRAM with ESP32S3Guru Meditation Error: Core  1 panic'ed (LoadProhibited). Exception was unhandled.Core  1 register dump:PC  : 0x4005544b  PS  : 0x00060830  A0  : 0x82007cc1  A1  : 0x3fcbb590  A2  : 0x00000000  A3  : 0x3c131a50  A4  : 0x00000000  A5  : 0x3fcbb5e4  A6  : 0x00000000  A7  : 0x00000001  A8  : 0x82008a70  A9  : 0x3fcbb540  A10 : 0x00000000  A11 : 0x3fca4184  A12 : 0x00000001  A13 : 0xffffffff  A14 : 0x00000000  A15 : 0x3fca41ac  SAR : 0x00000016  EXCCAUSE: 0x0000001c  EXCVADDR: 0x00000000  LBEG: 0x400570e8  LEND: 0x400570f3  LCOUNT  : 0x00000000  Backtrace: 0x40055448:0x3fcbb590 0x42007cbe:0x3fcbb5a0 0x4200837e:0x3fcbb5e0 0x4200dcbe:0x3fcbb610 0x42015279:0x3fcbb630 0x42015341:0x3fcbb650 0x40378e76:0x3fcbb670 0x4200ddc7:0x3fcbb710 0x42015279:0x3fcbb780 0x42015341:0x3fcbb7a0 0x40378e76:0x3fcbb7c0 0x4200ddc7:0x3fcbb860 0x42015279:0x3fcbb8b0 0x4201528e:0x3fcbb8d0 0x42022fef:0x3fcbb8f0 0x42023347:0x3fcbb980 0x42006922:0x3fcbb9b0

Lines 3 through 11 are output by my software. Line 14 is where the crash dump starts. All the output text my software prints comes from a single file, boot.py, which is the first file that is always executed. The next file to be executed is main.py, which is where the NeoPixel is exercised (it displays six colors then turns off). The last thing main.py does is instantiate an instance of my web server, then kick it off. I suspect that the crash has something to do with the built-in WiFi radio.

At this point I’ve simply dropped back from 5.2.3 to 5.2.2, recompiled MicroPython, re-flashed my board, and get it running happily again, just like it was doing before I tried to step up to the latest ESP-IDF release.

I have no idea why it’s crashing, and I have no desire to go looking. Since there’s nothing special in version 5.2.3 that I just have to have, I’ll stick with version 5.2.2 for building MicroPython, and version 5.3.1 for writing C++ for the Espressif chips. Be advised that ESP-IDF 5.2.3, at least for the ESP32-S3 in this circumstance, has a critical internal problem.