ArduinoOSC
OSC subscriber / publisher for Arduino
NOTE (>= v0.3.x) : BREAKING API CHANGES
- almost all apis has have changed and got much simpler
- dropped support for
OscSerial(recommend to use MsgPacketizer for much smaller packet size)
NOTE (>= v0.4.0) : DEPENDENT LIBRARIES REMOVED
If you have already installed this library, please follow:
- Cloned from GitHub (manually): Please install dependent libraries manually
- Installed from library manager: re-install this library from library manager
- Dependent libraries will be installed automatically
Feature
- simple usage
- flexible callback registration with lambda
- directly binding osc packet to values
- osc packet sending in one-line
- publishing osc packet in one-line
- support basic OSC types based on oscpkt
- TF (
bool: true, false) - i (
int32_t) - h (
int64_t) - f (
float) - d (
double) - s (
string) - b (
bundle)
- TF (
- support pattern-matching (wildcards)
Usage
Include ArduinoOSC
Please include #include "ArduinoOSC.h first.
If you use the board which has both WiFi and Ethernet, you can't use #include <ArduinoOSC.h>. Please replace it with #include <ArduinoOSCWiFi.h> or #include <ArduinoOSCEther.h> depending on the interface you want to use.
// For the boards which can use ether WiFi or Ethernet #include <ArduinoOSC.h> // OR use WiFi on the board which can use both WiFi and Ethernet #include <ArduinoOSCWiFi.h> // OR use Ethenet on the board which can use both WiFi and Ethernet #include <ArduinoOSCEther.h>
Following examples use OscWiFi.
To use with Ethernet, please change OscWiFi to OscEther.
One-Line Subscriber / Publisher
#include <ArduinoOSCWiFi.h> // #include <ArduinoOSC.h> // you can use this if your borad supports only WiFi or Ethernet int i; float f; String s; void setup() { // WiFi stuff WiFi.begin(ssid, pwd); WiFi.config(ip, gateway, subnet); // subscribe osc packet and directly bind to variable OscWiFi.subscribe(bind_port, "/bind/values", i, f, s); // publish osc packet in 30 times/sec (default) OscWiFi.publish(host, publish_port, "/publish/value", i, f, s); // function can also be published OscWiFi.publish(host, publish_port, "/publish/func", &millis, µs) ->setFrameRate(1); // and publish it once per second } void loop() { OscWiFi.update(); // should be called to subscribe + publish osc }
Bind OSC to Lambda Arguments and One-Line Send
void setup() { // WiFi stuff // ... OscWiFi.subscribe(bind_port, "/lambda/bind/args", [&](int& i, float& f, String& s) { Serial.print("/lambda/bind/args "); Serial.print(i); Serial.print(" "); Serial.print(f); Serial.print(" "); Serial.print(s); Serial.println(); // One-Line Send Back OscWiFi.send(host, send_port, "/reply", i, f, s); } ); } void loop() { OscWiFi.update(); // should be called }
Other Way to Subscribe
// OscMessage as lambda argument OscWiFi.subscribe(recv_port, "/lambda/msg", [](const OscMessage& m) { Serial.print(m.remoteIP()); Serial.print(" "); Serial.print(m.remotePort()); Serial.print(" "); Serial.print(m.size()); Serial.print(" "); Serial.print(m.address()); Serial.print(" "); Serial.print(m.arg<int>(0)); Serial.print(" "); Serial.print(m.arg<float>(1)); Serial.print(" "); Serial.print(m.arg<String>(2)); Serial.println(); } ); // wildcard address pattern matching OscWiFi.subscribe(recv_port, "/wildcard/*/test", [](const OscMessage& m) { Serial.print(m.remoteIP()); Serial.print(" "); Serial.print(m.remotePort()); Serial.print(" "); Serial.print(m.size()); Serial.print(" "); Serial.print(m.address()); Serial.print(" "); Serial.print(m.arg<int>(0)); Serial.println(); } ); // no arguments OscWiFi.subscribe(recv_port, "/need/reply", []() { OscWiFi.send(host, send_port, "/reply", i, f, s); }); // pre-defined callback OscWiFi.subscribe(recv_port, "/callback", onOscReceived);
Supported Platform
This library currently supports following platforms and interfaces. Please feel free to send PR or request for more board support!
WiFi
- ESP32
- ESP8266
- Raspberry Pi Pico W
- Arduino Uno R4 WiFi
- Arduino Uno WiFi Rev2
- Arduino MKR VIDOR 4000
- Arduino MKR WiFi 1010
- Arduino MKR WiFi 1000
- Arduino Nano 33 IoT
Ethernet
- Almost all platforms which has
Ethernet(andETH) library
Limitation and Options for NO-STL Boards
STL is used to handle packet data by default, but for following boards/architectures, ArxContainer is used to store the packet data because STL can not be used for such boards. The storage size of such boards for packets, queue of packets, max packet binary size, callbacks are limited.
- AVR
- megaAVR
Usage Recommendation for Arduino Uno (and other boards with tiny memory size)
For the boards which has tiny memory size (e.g. Arduino Uno), I reccomend not to use publisher and subscriber.
Though you can use them on such boards, such rich functions requires more memory.
The reccomended way is to use send and parse manually.
The example is shown in examples/arduino/OscEtherUno, so please consider to use it.
#include <ArduinoOSCEther.h> // #include <ArduinoOSC.h> // you can use this because Uno supports only Ethernet // required to use manual packet parsing OscEtherServer server(recv_port); OscEtherClient client; // OscEtherClient client(local_port); // set the local port of client manually (default: 9) void setup() { Ethernet.begin(mac, ip); } void loop() { // manual sending instead of publishers static uint32_t prev_func_ms = millis(); if (millis() > prev_func_ms + 500) { client.send(host, publish_port, "/publish/func", millis(), micros()); prev_func_ms = millis(); } // manual parsing instead of subscribers if (server.parse()) { const OscMessage* msg = server.message(); if (msg->address() == "/need/reply") { Serial.println("/need/reply"); int i = millis(); float f = (float)micros() / 1000.f; String s = F("hello"); client.send(host, send_port, "/reply", i, f, s); } } }
Memory Management (only for NO-STL Boards)
As mentioned above, for such boards like Arduino Uno, the storage sizes are limited. And of course you can manage them by defining following macros. But these default values are optimized for such boards, please be careful not to excess your boards storage/memory.
#define ARDUINOOSC_MAX_MSG_ARGUMENT_SIZE 8 #define ARDUINOOSC_MAX_MSG_BYTE_SIZE 128 #define ARDUINOOSC_MAX_MSG_QUEUE_SIZE 1 #define ARDUINOOSC_MAX_PUBLISH_DESTINATION 4 #define ARDUINOOSC_MAX_SUBSCRIBE_ADDRESS_PER_PORT 4 #define ARDUINOOSC_MAX_SUBSCRIBE_PORTS 2
Enable Bundle for NO-STL Boards
OSC bundle option is disabled for such boards.
If you want to use that, please use this macro and handle packets manually.
ArduinoOSC does not use bundle by default.
#define ARDUINOOSC_ENABLE_BUNDLE #define ARDUINOOSC_MAX_MSG_BUNDLE_SIZE 128
Enable Debug Logger
You can see the debug log when you insert following line before include ArduinoOSC.
#define ARDUINOOSC_DEBUGLOG_ENABLE #include <ArduinoOSC.h>
APIs
Main Class (OscWiFi / OscEther)
Subscribing to OSC Messages
// Subscribe a value to an OSC message OscWiFi.subscribe(const uint16_t port, const String& addr, T& value); // Subscribe multiple values to an OSC message OscWiFi.subscribe(const uint16_t port, const String& addr, T1& v1, T2& v2, ...); // Subscribe a lambda to an OSC message with arguments OscWiFi.subscribe(const uint16_t port, const String& addr, [](T1 arg1, T2 arg2, ...) { ... }); // Subscribe a lambda to an OSC message with OscMessage argument OscWiFi.subscribe(const uint16_t port, const String& addr, [](const OscMessage& msg) { ... }); // Subscribe a function to an OSC message OscWiFi.subscribe(const uint16_t port, const String& addr, onOscReceived);
Unsubscribing from OSC Messages
// Unsubscribe from a specific address on a port OscWiFi.unsubscribe(const uint16_t port, const String& addr); // Unsubscribe from all addresses on a port OscWiFi.unsubscribe(const uint16_t port); // Unsubscribe from all addresses on all ports OscWiFi.unsubscribe();
Sending OSC Messages
// Send an OSC message with arguments OscWiFi.send(const String& ip, const uint16_t port, const String& addr, T1 arg1, T2 arg2, ...);
Publishing OSC Messages
// Publish a value periodically OscWiFi.publish(const String& ip, const uint16_t port, const String& addr, T& value) ->setFrameRate(float fps); // Publish multiple values periodically OscWiFi.publish(const String& ip, const uint16_t port, const String& addr, T1& v1, T2& v2, ...) ->setIntervalMsec(float ms); // Publish function results periodically OscWiFi.publish(const String& ip, const uint16_t port, const String& addr, &func1, &func2) ->setIntervalSec(float sec);
OSC Bundle Support
// Create and send OSC bundles OscWiFi.begin_bundle(const TimeTag& tt = TimeTag::immediate()); OscWiFi.add_bundle(const String& addr, T1 arg1, T2 arg2, ...); OscWiFi.end_bundle(); OscWiFi.send_bundle(const String& ip, const uint16_t port);
Update Functions
// Parse incoming OSC messages (server) OscWiFi.parse(); // Parse incoming OSC messages and publish outgoing messages (server + client) OscWiFi.update(); // Send published OSC messages (client) OscWiFi.post();
OscMessage
Argument Getters
msg.arg<T>(const uint8_t index); // Get argument as type T msg.getArgAsInt32(const size_t i); msg.getArgAsInt64(const size_t i); msg.getArgAsFloat(const size_t i); msg.getArgAsDouble(const size_t i); msg.getArgAsString(const size_t i); msg.getArgAsBlob(const size_t i); msg.getArgAsBool(const size_t i);
Type Checkers
msg.isBool(const size_t i); msg.isInt32(const size_t i); msg.isInt64(const size_t i); msg.isFloat(const size_t i); msg.isDouble(const size_t i); msg.isStr(const size_t i); msg.isBlob(const size_t i);
Message Information
msg.address(); // Get OSC address msg.size(); // Get number of arguments msg.typeTags(); // Get type tag string msg.remoteIP(); // Get sender's IP address msg.remotePort(); // Get sender's port msg.match(const String& pattern); // Check if address matches pattern
Manual Packet Handling (for boards with limited memory)
// Server for receiving OscEtherServer server(recv_port); if (server.parse()) { const OscMessage* msg = server.message(); // Process message... } // Client for sending OscEtherClient client; client.send(host, send_port, "/addr", arg1, arg2);
Dependent Libraries
Embedded Libraries
Special Thanks
License
MIT