diff --git a/roundly/platformio.ini b/roundly/platformio.ini new file mode 100644 index 0000000..15c072e --- /dev/null +++ b/roundly/platformio.ini @@ -0,0 +1,31 @@ +[platformio] +default_envs = d1_mini_pro + +[env] +framework = arduino +upload_speed = 921600 +upload_port = + /dev/ttyUSB0 + /dev/tty.SLAB_USBtoUART +lib_deps = + SPI + Wire + 64 ; ArduinoJson + 1269 ; Painless Mesh + 265 ; AccelStepper + +[env:nodemcuv2] +platform = espressif8266 +board = nodemcuv2 +lib_deps = + ${env.lib_deps} + ESP8266WiFi + +[env:d1_mini_pro] +platform = espressif8266 +board = d1_mini_pro +lib_deps = + ${env.lib_deps} + ESP8266WiFi +upload_speed = 460800 +; 9600, 19200, 38400, 57600, 115200, 230400, 460800, 921600 diff --git a/roundly/src/main.cpp b/roundly/src/main.cpp new file mode 100644 index 0000000..b65197a --- /dev/null +++ b/roundly/src/main.cpp @@ -0,0 +1,447 @@ +// +// wirelessly connected cloud (Wireless Mesh Networking) +// MIDI-like +// spacial +// + +// +// conversation on the ROOT @ SEMA, Seoul +// + +// +// 2020 10 14 +// + +//===================== +// +// 'DISABLE_AP' +// --> disabling AP is for teensy audio samplers. +// they need this to reduce noise from AP beacon signals. +// but, then they cannot build-up net. by themselves. +// we need who can do AP.. +// ==> TODO! just prepare some 'dummy' postmans around. w/ AP activated. +// +// 'DISABLE_I2C_REQ' +// --> a quirk.. due to bi-directional I2C hardship. +// ideally, we want to make this sampler node also speak. +// but, I2C doesn't work. maybe middleware bug.. we later want to change to diff. proto. +// for example, UART or so. +// ==> BEWARE! yet, still we need to take off this.. for 'osc' node. +// +// 'SET_ROOT' +// 'SET_CONTAINSROOT' +// --> for the network stability +// declare 1 root node and branches(constricted to 'contains the root') +// to improve the stability of the net +// +//==================== + +//===================== +#define SET_CONTAINSROOT +//==================== + +//======================== +#define ROUNDLY_A_KEY 200 // A-E-I-O-U-W-Y-N (up to 8 roundlys) - KEY 200 ~ 207 +#define ROUNDLY_E_KEY 201 +#define ROUNDLY_I_KEY 202 +#define ROUNDLY_O_KEY 203 +#define ROUNDLY_U_KEY 204 +#define ROUNDLY_W_KEY 205 +#define ROUNDLY_Y_KEY 206 +#define ROUNDLY_N_KEY 207 +//======================= + +//======================== +#define ID_KEY ROUNDLY_A_KEY +//======================= + +//======================== +#define MESH_SSID "forest-all/around" +#define MESH_PASSWORD "cc*vvvv/kkk" +#define MESH_PORT 5555 +#define MESH_CHANNEL 5 +#define LONELY_TO_DIE (1000) +//======================= + +// +// LED status indication +// phase 0 +// - LED => steady on +// - booted. and running. no connection. scanning. +// phase 1 +// - LED => slow blinking (syncronized) +// - + connected. +// +#if defined(ARDUINO_ESP8266_NODEMCU) // nodemcuv2 +#define LED_PIN 2 +#elif defined(ARDUINO_ESP8266_WEMOS_D1MINIPRO) // d1_mini_pro +#define LED_PIN 2 +#elif defined(ARDUINO_ESP8266_ESP12) // huzzah +#define LED_PIN 2 +#elif defined(ARDUINO_FEATHER_ESP32) // featheresp32 +#define LED_PIN 13 +#elif defined(ARDUINO_NodeMCU_32S) // nodemcu-32s +#define LED_PIN 2 +#elif defined(ARDUINO_ESP32_DEV) // esp32doit-devkit-v1 +#define LED_PIN 2 +#endif +#define LED_PERIOD (1111) +#define LED_ONTIME (1) + +//arduino +#include + +//i2c +#include +#include "../../post.h" + +//painlessmesh +#include +painlessMesh mesh; + +//scheduler +Scheduler runner; + +//task #0 : connection indicator +bool onFlag = false; +bool isConnected = false; +//prototypes +void taskStatusBlink_steadyOn(); +void taskStatusBlink_slowblink_insync(); +void taskStatusBlink_steadyOff(); +//the task +Task statusblinks(0, 1, &taskStatusBlink_steadyOn); // at start, steady on. default == disabled. ==> setup() will enable. +// when disconnected, and trying, steadyon. +void taskStatusBlink_steadyOn() { + onFlag = true; +} +// when connected, blink per 1s. sync-ed. (== default configuration) +void taskStatusBlink_slowblink_insync() { + // toggler + onFlag = !onFlag; + // on-time + statusblinks.delay(LED_ONTIME); + // re-enable & sync. + if (statusblinks.isLastIteration()) { + statusblinks.setIterations(2); //refill iteration counts + statusblinks.enableDelayed(LED_PERIOD - (mesh.getNodeTime() % (LED_PERIOD*1000))/1000); //re-enable with sync-ed delay + } +} +// when connected, steadyoff. (== alternative configuration) +void taskStatusBlink_steadyOff() { + onFlag = false; +} + +//task #1 : happy or lonely +// --> automatic reset after some time of 'loneliness (disconnected from any node)' +void nothappyalone() { + static bool isConnected_prev = false; + static unsigned long lonely_time_start = 0; + // oh.. i m lost the signal(==connection) + if (isConnected_prev != isConnected && isConnected == false) { + lonely_time_start = millis(); + Serial.println("oh.. i m lost!"); + } + // .... how long we've been lonely? + if (isConnected == false) { + if (millis() - lonely_time_start > LONELY_TO_DIE) { + // okay. i m fed up. bye the world. + Serial.println("okay. i m fed up. bye the world."); + Serial.println(); +#if defined(ESP8266) + ESP.reset(); +#else +#error unknown esp. +#endif + } + } + // + isConnected_prev = isConnected; +} +// Task nothappyalone_task(1000, TASK_FOREVER, ¬happyalone, &runner, true); // by default, ENABLED. +Task nothappyalone_task(100, TASK_FOREVER, ¬happyalone); // by default, ENABLED. + +// +#include +#define STEP_MODE_CONSTANT_VEL (0xDE00 + 0x01) +#define STEP_MODE_ACCELERATING (0xDE00 + 0x02) +#define STEP_MODE STEP_MODE_CONSTANT_VEL +// #define STEP_MODE STEP_MODE_ACCELERATING +// NOTE: --> well.. acceleration enabled mode.. is a bit worse. (less torque) +#define STEPS_PER_REV (2048.0) +// speed (rpm) * steps-per-revolution == speed (steps per minute) +// --> speed (steps per minute) / 60 == speed (steps per second) +// --> speed (steps per second) * 60 / steps-per-revolution == speed (rpm) +#define STEPS_PER_SEC_TO_RPM (60.0 / STEPS_PER_REV) +#define RPM_TO_STEPS_PER_SEC (STEPS_PER_REV / 60.0) +// parameter (torque-speed trade-off) +#define STEPS_PER_SEC_MAX (500) +#define RPM_MAX (STEPS_PER_SEC_MAX * STEPS_PER_SEC_TO_RPM) +#define ACCELERATION_MAX (500) +// +AccelStepper stepper(AccelStepper::FULL4WIRE, D5, D6, D7, D8); // N.B. - @esp8266, NEVER use "5, 6, 7, 8" -> do "D5, D6, D7, D8" !! + +// my tasks +extern Task stepping_task; +extern Task rest_task; +int step_target = 0; +int step_duration = 10000; +void stepping() { + // + // if (stepper.distanceToGo() == 0) { + // + float cur_step = stepper.currentPosition(); + float target_step = step_target; + float dur = step_duration; + // float target_step = notes[score_now][note_idx][0]; + // float dur = notes[score_now][note_idx][1]; + float steps = target_step - cur_step; + float velocity = steps / dur * 1000; // unit conv.: (steps/msec) --> (steps/sec) + float speed = fabs(velocity); + // + if (speed > STEPS_PER_SEC_MAX) { + Serial.println("oh.. it might be TOO FAST for me.."); + } else { + Serial.println("okay. i m going."); + } + + // + stepper.moveTo(target_step); + stepper.setSpeed(velocity); + //NOTE: 'setSpeed' should come LATER than 'moveTo'! + // --> 'moveTo' re-calculate the velocity. + // --> so we need to re-override it. + // + // } +} +Task stepping_task(0, TASK_ONCE, &stepping); + +void rest() { +} +Task rest_task(0, TASK_ONCE, &rest); + +// mesh callbacks +void receivedCallback(uint32_t from, String & msg) { // REQUIRED + Serial.print("got msg.: "); + Serial.println(msg); + //parse now. + + //parse letter string. + + // letter frame ( '[' + 30 bytes + ']' ) + // : [123456789012345678901234567890] + + // 'MIDI' letter frame + // : [123456789012345678901234567890] + // : [KKKVVVG.......................] + // : KKK - Key + // .substring(1, 4); + // : VVV - Velocity (volume/amp.) + // .substring(4, 7); + // : G - Gate (note on/off) + // .substring(7, 8); + + String str_key = msg.substring(1, 4); + String str_velocity = msg.substring(4, 7); + String str_gate = msg.substring(7, 8); + + int key = str_key.toInt(); + int velocity = str_velocity.toInt(); // 0 ~ 127 + int gate = str_gate.toInt(); + + // : [_______X......................] + // : X - Extension starter 'X' + // .substring(8, 9); + // Extension (X == 'X') + // : [_______X1111222233344455667788] + // : 1 - data of 4 letters + // .substring(9, 13); + // : 2 - data of 4 letters + // .substring(13, 17); + // : 3 - data of 3 letters + // .substring(17, 20); + // : 4 - data of 3 letters + // .substring(20, 23); + // : 5 - data of 2 letters + // .substring(23, 25); + // : 6 - data of 2 letters + // .substring(25, 27); + // : 7 - data of 2 letters + // .substring(27, 29); + // : 8 - data of 2 letters + // .substring(29, 31); + + String str_ext = msg.substring(8, 9); + String str_x1 = msg.substring(9, 13); + String str_x2 = msg.substring(13, 17); + String str_x3 = msg.substring(17, 20); + String str_x4 = msg.substring(20, 23); + String str_x5 = msg.substring(23, 25); + String str_x6 = msg.substring(25, 27); + String str_x7 = msg.substring(27, 29); + String str_x8 = msg.substring(29, 31); + + if (str_ext == "X") { + step_target = str_x1.toInt(); // 0 ~ 9999 + step_duration = str_x2.toInt(); // 0 ~ 9999 + } + + //is it for me? + if (key == ID_KEY) { + if (gate == 1) { + stepping_task.restartDelayed(10); + } else if (gate == 0) { + rest_task.restartDelayed(10); + } + } +} +void changedConnectionCallback() { + Serial.println(mesh.getNodeList().size()); + // check status -> modify status LED + if (mesh.getNodeList().size() > 0) { + // (still) connected. + onFlag = false; //reset flag stat. + statusblinks.set(LED_PERIOD, 2, &taskStatusBlink_slowblink_insync); + // statusblinks.set(0, 1, &taskStatusBlink_steadyOff); + statusblinks.enable(); + Serial.println("connected!"); + // + isConnected = true; + runner.addTask(nothappyalone_task); + nothappyalone_task.enable(); + } + else { + // disconnected!! + statusblinks.set(0, 1, &taskStatusBlink_steadyOn); + statusblinks.enable(); + // + isConnected = false; + } + // let I2C device know + ///// + Serial.println("hi. client, we ve got a change in the net."); +} +void newConnectionCallback(uint32_t nodeId) { + Serial.println(mesh.getNodeList().size()); + Serial.println("newConnectionCallback."); + changedConnectionCallback(); +} + +void setup() { + //led + pinMode(LED_PIN, OUTPUT); + + //mesh + WiFiMode_t node_type = WIFI_AP_STA; +#if defined(DISABLE_AP) + system_phy_set_max_tpw(0); + node_type = WIFI_STA; +#endif + // mesh.setDebugMsgTypes(ERROR | DEBUG | CONNECTION); + mesh.setDebugMsgTypes( ERROR | STARTUP ); + mesh.init(MESH_SSID, MESH_PASSWORD, &runner, MESH_PORT, node_type, MESH_CHANNEL); + + // + // void init(String ssid, String password, Scheduler *baseScheduler, uint16_t port = 5555, WiFiMode_t connectMode = WIFI_AP_STA, uint8_t channel = 1, uint8_t hidden = 0, uint8_t maxconn = MAX_CONN); + // void init(String ssid, String password, uint16_t port = 5555, WiFiMode_t connectMode = WIFI_AP_STA, uint8_t channel = 1, uint8_t hidden = 0, uint8_t maxconn = MAX_CONN); + // + +#if defined(SET_ROOT) + mesh.setRoot(true); +#endif +#if defined(SET_CONTAINSROOT) + mesh.setContainsRoot(true); +#endif + //callbacks + mesh.onReceive(&receivedCallback); + mesh.onNewConnection(&newConnectionCallback); + mesh.onChangedConnections(&changedConnectionCallback); + Serial.println(mesh.getNodeList().size()); + + //tasks + runner.addTask(statusblinks); + statusblinks.enable(); + + //serial + Serial.begin(115200); + delay(100); + Serial.println("hi, postman ready."); +#if defined(DISABLE_AP) + Serial.println("!NOTE!: we are in the WIFI_STA mode!"); +#endif + + //understanding what is 'the nodeId' ==> last 4 bytes of 'softAPmacAddress' + // uint32_t nodeId = tcp::encodeNodeId(MAC); + Serial.print("nodeId (dec) : "); + Serial.println(mesh.getNodeId(), DEC); + Serial.print("nodeId (hex) : "); + Serial.println(mesh.getNodeId(), HEX); + uint8_t MAC[] = {0, 0, 0, 0, 0, 0}; + if (WiFi.softAPmacAddress(MAC) == 0) { + Serial.println("init(): WiFi.softAPmacAddress(MAC) failed."); + } + Serial.print("MAC : "); + Serial.print(MAC[0], HEX); Serial.print(", "); + Serial.print(MAC[1], HEX); Serial.print(", "); + Serial.print(MAC[2], HEX); Serial.print(", "); + Serial.print(MAC[3], HEX); Serial.print(", "); + Serial.print(MAC[4], HEX); Serial.print(", "); + Serial.println(MAC[5], HEX); + + // for instance, + + // a huzzah board + // nodeId (dec) : 3256120530 + // nodeId (hex) : C21474D2 + // MAC : BE, DD, C2, 14, 74, D2 + + // a esp8266 board (node mcu) + // nodeId (dec) : 758581767 + // nodeId (hex) : 2D370A07 + // MAC : B6, E6, 2D, 37, A, 7 + + //introduction + Serial.print("my ID Key --> "); + Serial.println(ID_KEY); + + //i2c master + Wire.begin(); + + //random seed + randomSeed(analogRead(0)); + + //stepper + pinMode(D5, OUTPUT); + pinMode(D6, OUTPUT); + pinMode(D7, OUTPUT); + pinMode(D8, OUTPUT); + /// " + /// The fastest motor speed that can be reliably supported is about 4000 steps per + /// second at a clock frequency of 16 MHz on Arduino such as Uno etc. + /// " @ AccelStepper.h + stepper.setMaxSpeed(STEPS_PER_SEC_MAX); //steps per second (trade-off between speed vs. torque) +#if (STEP_MODE == STEP_MODE_ACCELERATING) + stepper.setAcceleration(ACCELERATION_MAX); +#endif + + //tasks + runner.addTask(stepping_task); + runner.addTask(rest_task); + + rest_task.restartDelayed(500); +} + +void loop() { + runner.execute(); + mesh.update(); + digitalWrite(LED_PIN, !onFlag); // value == false is ON. so onFlag == true is ON. (pull-up) + + //stepper + if (stepper.distanceToGo() != 0) { +#if (STEP_MODE == STEP_MODE_CONSTANT_VEL) + stepper.runSpeed(); +#elif (STEP_MODE == STEP_MODE_ACCELERATING) + stepper.run(); +#endif + } +}