The MQTT PubSub client callback passes the payload
parameter as a pointer to a byte array and not a pointer to a char array - this is due to the content of the publish payload being application specific. In other words, from the point of view of the protocol, the payload can be a string, a file, and image, etc. By contrast, the topic
parameter is indeed expected to be a string, hence the buffer pointer char *topic
among the function parameters.
The result is that just casting the payload
byte parameter to a const char *
and converting to a String
type will not work because the ending null character will be missing. Given the published payload:
{"action":"subscribed"}
handled by the PubSub callback:
void mqttCallback(char *topic, byte *payload, unsigned int length) { String msgTopic = String(topic); String msgPayload = (const char *) payload; //...
will result in the msgPayload
string containing garbage:
{"action":"subscribed"}d}"
Perhaps a simpler way of dealing with the payload
parameter, in case the expected content is a string, would be to create a null-terminated character array:
void mqttCallback(char *topic, byte *payload, unsigned int length) { String msgTopic = String(topic); // payload is not null terminated and casting will not work char msgPayload[length + 1]; snprintf(msgPayload, length + 1, "%s", payload); Serial.println("Message received on topic: " + String(topic) + " with payload: " + String(msgPayload)); // ...
Now String(msgPayload)
will display the string properly. Unfortunately, a lot of available examples incorrectly convert the payload
parameter resulting in small yet ever-increasing memory corruption over time which leads to stability issues with the Arduino device.
For WeMoS ESP8266 the following trick can be used to force-flash an Arduino sketch. First, connect the flash pin to ground (ie: D3
on WeMoS mini), plug in the USB cable to flash a new sketch and then reset the board.
Similarly WeMoS ESP32 devices, along the lines of LOLIN that do not have a flash pin can be force-flashed by grounding the EN
pin (typically wired to the reset button on the board), starting a flash operation from Arduino, and when the "Connecting" message pops up with the dots (and sometimes dashes), releasing the EN
pin.
These methods are very useful in case the ESP pins that correspond to serial communication, or ESP pins that mess with the booting sequence of the ESP, are used up such that the only way to flash the device is to force-flash the device.
The following code is a translation of the lolcode random number generator to Arduino/C++.
/////////////////////////////////////////////////////////////////////////// // Copyright (C) Wizardry and Steamworks 2024 - License: GNU MIT // /////////////////////////////////////////////////////////////////////////// // from https://grimore.org/fuss/lolcode#random_number_generator // // the parameter "m" represents the upper bound of the random number // /////////////////////////////////////////////////////////////////////////// int lrng = 65536; int wasSpectrumLehmer(int m) { int g = 75; int n = 65537; int c = 0; do { lrng = (g * (lrng + c)) % n; } while(++c < m); return lrng % m; }
The code is somewhat useful given that generating random numbers, even PRNG numbers, seems to be tricky on Arduino and with various caveats depending on the methods being used.
Calling the wasSpectrumLehmer
function in order to generate random numbers can be performed as such:
int i = spectrumLehmer(100); Serial.println(i); i = spectrumLehmer(100); Serial.println(i); delay(1000);
and two random numbers should be printed to the serial console.
In order to convert a MAC address to a colon-spaced string, the following function can be used:
/////////////////////////////////////////////////////////////////////////// // Copyright (C) Wizardry and Steamworks 2015 - License: GNU MIT // /////////////////////////////////////////////////////////////////////////// const char* macToString(byte* mac) { char* buff = (char *)malloc(18 * sizeof(char)); sprintf(buff, "%02x:%02x:%02x:%02x:%02x:%02x", bootMAC[0], bootMAC[1], bootMAC[2], bootMAC[3], bootMAC[4], bootMAC[5] ); return (const char*) buff; }
For example, in order to obtain a colon-separated string of the WiFi MAC address, call the function like this:
byte mac[6]; WiFi.macAddress(mac); String wifiMAC = macToString(mac);
One of the issues with the default Arduino WiFi library is that it fails to remember the last allocated address such that when it requests an IP address from the DHCP server, it does so by asking for the any address 0.0.0.0
and the DHCP server gets confused due to the request:
2025-01-11T20:54:19.324705+00:00 gate dhcpd[3640098]: DHCPREQUEST for 0.0.0.0 from 57:e6:3f:d9:0b:a8 via br0: unknown lease 0.0.0.0. 2025-01-11T20:54:29.325581+00:00 gate dhcpd[3640098]: DHCPREQUEST for 0.0.0.0 from 57:e6:3f:d9:0b:a8 via br0: unknown lease 0.0.0.0. 2025-01-11T20:54:39.325186+00:00 gate dhcpd[3640098]: DHCPREQUEST for 0.0.0.0 from 57:e6:3f:d9:0b:a8 via br0: unknown lease 0.0.0.0. 2025-01-11T20:54:49.327665+00:00 gate dhcpd[3640098]: DHCPREQUEST for 0.0.0.0 from 57:e6:3f:d9:0b:a8 via br0: unknown lease 0.0.0.0.
In the end the Arduino WiFi stack gives up and maybe the device restarts such that the attempt is made again to retrieve the any address 0.0.0.0
ending up in a cycle.
One of the workarounds is to flush any previously known configuration just before starting the Arduino WiFi stack with WiFi.begin()
:
WiFi.config(INADDR_NONE, INADDR_NONE, INADDR_NONE); WiFi.begin(Ssid, password);
This will ensure that the Arduino WiFi stack will go directly to discovery and issue a DHCPDISCOVER
packet without even bothering about restoring its previous IP address. Considering that DHCP would imply dynamic allocation and that the option to set static addresses on the DHCP server itself exists, there does not seem to be too large of a downside to skip retrieving the last IP address.