Sharer is both a .NET and an Arduino Library. It allows a desktop application to read/write variables and remote call functions on Arduino, using the Sharer protocole accross a serial communication. Sharer has initially been developped for the Ballcuber project (https://ballcuber.github.io), but it is now a standalone library ;).
// C#
var connection = new SharerConnection("COM3", 115200);
connection.Connect();
Remark : for some boards, like Micro and Leonardo, it is necessary to set the RTS and DTR signals with the RtsEnable
and DtrEnable
properties.
// C# - My Arduino code has a function : int Sum(int a, byte b);
var result = connection.Call("Sum", 10, 12);
// result.Status : OK
// result.Type : int
// result.Value : 22
// C# - Read all digital pins : digitalRead(0) to digitalRead(13)
for(int i=0; i <= 13; i++){
var digitalValue = connection.Call("digitalRead", i);
// digitalValue.Status : OK
// digitalValue.Type : bool
// digitalValue.Value : true or false
}
// C# - Definition of all functions
var functions = connection.Functions;
// functions[0].Name : function name
// functions[0].ReturnType : enum void, int, long, ...
// functions[0].Arguments[0].Name // Name of first argument
// functions[0].Arguments[0].Type // enum void, int, long, ...
// C#
connection.WriteVariable("myVar", 15); // returns true if writting OK
// C# - Write simultaneously several variables by passing a List<WriteVariable>()
connection.WriteVariables(listOfVariablesToWrite);
// C# - Definition of all variables
var variables = connection.Variables;
// variables[0].Name : variable name
// variables[0].Type : enum int, long, ...
// C#
var value = connection.ReadVariable("myVar");
// value.Status : OK
// value.Value : 12
// C# - Read simultaneously several variables
var values = connection.ReadVariables(new string[] {"myVar", "anotherVar", "yetAnother"});
// C#
var info = connection.GetInfos();
// info.Board : Arduino UNO
// info.CPUFrequency : 16000000 (Hz)
// a lot more : info.CPlusPlusVersion, info.FunctionMaxCount, info.VariableMaxCount, ...
You can send and receive classic messages on the serial port with the WriteUserData
functions. Also, the UserDataReceived
event is raised when data is sent by the Arduino.
Warning : it is not possible to read or write variables and call functions in your UserDataReceived
event handler.
// C#
connection.WriteUserData("Hello!");
connection.WriteUserData(12.2);
connection.WriteUserData(new byte[] {0x12, 0x25, 0xFF});
// Event raised when new user data sent by Arduino
connection.UserDataReceived += UserDataReceived;
// C++ Arduino
#include <Sharer.h>
// A simple function that sums an integer and a byte and return an integer
int Sum(int a, byte b) {
return a + b;
}
// a simple integer
int myVar = 25;
void setup() {
Sharer.init(115200); // Init Serial communication with 115200 bauds
// Expose this function to Sharer : int Sum(int a, byte b)
Sharer_ShareFunction(int, Sum, int, a, byte, b);
// Share system functions
Sharer_ShareFunction(bool, digitalRead, uint8_t, pin);
Sharer_ShareVoid(pinMode, uint8_t, pin);
// Expose this variable to Sharer so that the desktop application can read/write it
Sharer_ShareVariable(int, myVar);
}
// Run Sharer engine in the main Loop
void loop() {
Sharer.run();
}
Sharer is divided into 2 repositories, one for the Arduino sources and the other for .NET sources
Sharer is available in the library manager available in the menu Tools/Manage Libraries...
. Just look for Sharer and install the latest version.
Please download the Sharer library archive : https://github.com/Rufus31415/Sharer/releases/latest/download/Sharer.zip
Sharer has been tested with Arduino UNO, NANO, MEGA and DUE. It may work with other boards.
Extract so that you get a Sharer directory in your Arduino "libraries" directory : C:\Program Files (x86)\Arduino\libraries\Sharer
You can then enjoy the examples by restarting your Arduino IDE and go to menu File / examples / Sharer
.
Sharer.NET is available on nuget : https://www.nuget.org/packages/Sharer/
Use this command line to install it with you package manager :
Install-Package Sharer
Sharer.dll assembly can be downloaded here : https://github.com/Rufus31415/Sharer.net/releases/latest/download/SharerAsssemblies.zip
This archive contains the nuget package Sharer.nupkg and Sharer.dll compiled in AnyCPU Release for the following targets:
- NET Framework 3.5, 4.0, 4.5.1, 4.5.2, 4.6, 4.6.1, 4.6.2, 4.7, 4.7.1, 4.7.2, 4.8
- NET Core 2.0, 2.1, 2.2, 3.0, 3.1
- NET Standard 2.1
Windows Forms example requires .NET Framework 3.5. It can be downloaded here : https://github.com/Rufus31415/Sharer.net/releases/latest/download/SharerWindowsFormsExample.zip
The console example run with .NET Core 3.0. But you don't need any runtime to execute it. The standalone console examples are available here :
- Windows 64 bits : https://github.com/Rufus31415/Sharer.net/releases/latest/download/SharerConsoleExample_win-x64.zip
- Windows 32 bits : https://github.com/Rufus31415/Sharer.net/releases/latest/download/SharerConsoleExample_win-x86.zip
- Windows ARM (for example Windows IOT for Raspberry PI) : https://github.com/Rufus31415/Sharer.net/releases/latest/download/SharerConsoleExample_win-arm.zip
- Linux ARM (for example Raspbian for Raspberry PI) : https://github.com/Rufus31415/Sharer.net/releases/latest/download/SharerConsoleExample_linux-arm.zip
- Linux 64 bits : https://github.com/Rufus31415/Sharer.net/releases/latest/download/SharerConsoleExample_linux-x64.zip
- Mac OS X : https://github.com/Rufus31415/Sharer.net/releases/latest/download/SharerConsoleExample_osx-x64.zip
Β© Rufus31415 See the license file for details.
The header file <Sharer.h> should be included.
In function setup()
, Sharer is initialized by passing a baudrate to function Sharer.init(...)
. It internally call Serial.init()
with the same baudrate.
In function loop()
, Sharer.run()
should be called. It runs the internal kernel of Sharer that decodes commands received.
#include <Sharer.h>
void setup() {
Sharer.init(115200);
}
void loop() {
Sharer.run();
}
You also can initialize Sharer with another stream, such as the Serial2 of Arduino DUE if you use Serial2 to communication with the desktop application :
void setup() {
// Initialize with another Serial interface
Serial2.begin(9600);
Sharer.init(Serial2);
}
To add a variable in the shared variable list, you should call the macro Sharer_ShareVariable
. Its first argument is the type of the variable, and the second is its name. This macro allows Sharer to have a pointer to the variable, its name, its type and its memory size.
byte myByteVar;
long myLongVar;
int myIntArray[2];
void setup() {
Sharer.init(115200);
Sharer_ShareVariable(byte, myByteVar);
Sharer_ShareVariable(long, myLongVar);
Sharer_ShareVariable(int, myIntArray[0]);
Sharer_ShareVariable(int, myIntArray[1]);
}
To add a variable in the shared function list, you should call the macro Sharer_ShareFunction
. Its first argument is the returned type, and the second is its name. Following arguments of the macro describe the argument of the shared function by its type and its name. There is no limit in the number of argument you can share, but all arguments should be shared.
Class method sharing is not supported and shared functions must be free functions (non-member static functions).
You can share your own functions, but all system functions like analogRead, digitalRead, digitalWrite, millis, ...
int Sum(int a, byte b) {
return a + b;
}
void setup() {
Sharer.init(115200);
// Share your function to Sharer : int Sum(int a, byte b)
Sharer_ShareFunction(int, Sum, int, a, byte, b);
// Sharer system functions
Sharer_ShareFunction(int, analogRead, uint8_t, pin);
Sharer_ShareFunction(bool, digitalRead, uint8_t, pin);
}
Void functions are functions without returned type. You should call the macro Sharer_ShareVoid
to share a void function. The first argument is its name, followed by type and name of each arguments.
void TurnLEDOn(void) {
pinMode(13, OUTPUT);
digitalWrite(13, true);
}
void SetLEDState(bool state) {
pinMode(13, OUTPUT);
digitalWrite(13, state);
}
void setup() {
Sharer.init(115200);
// Share your void fuctions
Sharer_ShareVoid(TurnLEDOn);
Sharer_ShareVoid(SetLEDState, bool, state);
// Sharer system void functions
Sharer_ShareVoid(digitalWrite, uint8_t, pin);
}
Sharer class inherits from Stream, so you can use the following functions. Please, never use Sharer.print(), Sharer.println() or Sharer.write() inside a shared function, writting woud be ignored.
void loop() {
Sharer.run();
//gets the number of bytes available in the stream
int available = Sharer.available();
// reads last reveided byte (-1 if no data to read)
int lastByte = Sharer.read();
// Empty the stream
Sharer.flush();
// reads data from the serial buffer until the target is found
// returns, true if data is found
bool found = Sharer.find("string to find");
//Returns the next byte of incoming serial data without removing it from the internal serial buffer
// Successive calls to peek() will return the same character
int nextByte = Sharer.peek();
// reads characters from the serial buffer into a String
String str = Sharer.readString();
// Looks for the next valid integer in the incoming serial
int lastInt = Sharer.parseInt();
lastInt = Sharer.parseInt(SKIP_ALL); // all characters other than digits or a minus sign are ignored when scanning the stream for an integer. This is the default mode
lastInt = Sharer.parseInt(SKIP_NONE); // nothing is skipped, and the stream is not touched unless the first waiting character is valid.
lastInt = Sharer.parseInt(SKIP_WHITESPACE); // only tabs, spaces, line feeds, and carriage returns are skipped.
// returns the first valid floating point number from the Serial buffer
float lastFloat = Sharer.parseFloat();
lastFloat = Sharer.parseFloat(SKIP_ALL); // all characters other than a minus sign, decimal point, or digits are ignored when scanning the stream for a floating point number. This is the default mode.
lastFloat = Sharer.parseFloat(SKIP_NONE); // Nothing is skipped, and the stream is not touched unless the first waiting character is valid.
lastFloat = Sharer.parseFloat(SKIP_WHITESPACE); // only tabs, spaces, line feeds, and carriage returns are skipped.
// Prints ASCII encoded string
Sharer.print(85); // sends the string "85"
Sharer.print(1.23456); // sends the string "1.23"
Sharer.print('N'); // sends the string "N"
Sharer.print("Hello world."); // sends the string "Hello world."
Sharer.print(78, BIN); // sends the string "1001110"
Sharer.print(78, OCT); // sends the string "116"
Sharer.print(78, DEC); // sends the string "78"
Sharer.print(78, HEX); // sends the string "4E"
Sharer.print(1.23456, 0); // sends the string "1"
Sharer.print(1.23456, 2); // sends the string "1.23"
Sharer.print(1.23456, 4); // sends the string "1.2345"
Sharer.println("Hello ;)"); // send the string and a new line "Hello ;)\n"
Sharer.println(); // just sends a new line
// Write a single byte
Sharer.write(0x12);
}
You can change the limits of Sharer by editing the constants of file C:\Program Files (x86)\Arduino\libraries\Sharer\src\SharerConfig.h
.
// maximum number of shared functions
#define _SHARER_MAX_FUNCTION_COUNT 16
// maximum number of shared variables
#define _SHARER_MAX_VARIABLE_COUNT 32
// circle buffer size for user serial message
#define _SHARER_USER_RECEIVE_BUFFER_SIZE 64
// maximum number of call to Serial.read() each time Sharer.Run() is called
#define _SHARER_MAX_BYTE_READ_PER_RUN 1000
You can find here the full documentation of Sharer.NET : /Sharer.NET/Sharer.NET.Documentation.md.
A command takes less than 10ms to be called on a Arduino Uno (command + response) at 115200 bauds. The protocole is optimized in order not to have strings to decode, just a binary stream to interprete, byte by byte. Memory footprint has been optimized by using F() macro and PROGMEM to store names of variable, function and arguments in flash.
Sharer uses a unique protocole called the Sharer Protocole. Every serial commands received by the Arduino are interprated.
To be continued, I promise... π
I have some ideas to extend Sharer features like :
- Develop Sharer client for other languages like Python and Node.js
- Add LabView and Matlab examples for research purposes
- Add other transport layers like TCP and UDP
- Add a HTTP REST API based on Sharer that exposes endpoints to call functions, and read/write variables.
If you are interested in helping me with Sharer development, I will be happy to receive feature requests. Fork is also welcome ;)