#ifdef ESP32 #include #include // #include // #include "USBCDC.h" #include "driver/temp_sensor.h" #elif defined(ESP8266) #include #include #endif #include // #include "w5500/esp32_w5500.h" // #include #include #include // #include "ESPDMX.h" #include #include #include #include "routes/config.h" #include "routes/networks.h" #include "routes/status.h" // DMXESPSerial dmx1; // DMXESPSerial dmx2; dmx_port_t dmx1 = DMX_NUM_0; // for esp32s2 dmx_port_t dmx2 = DMX_NUM_1; byte dmx1_data[DMX_PACKET_SIZE]; byte dmx2_data[DMX_PACKET_SIZE]; unsigned long dmx1_lastUpdate = millis(); unsigned long dmx2_lastUpdate = millis(); bool dmx1_IsConnected = false; bool dmx2_IsConnected = false; // Button #define PIN_LED 7 #define PIN_BUTTON 5 uint8_t brightness_led = 20; bool status_led; /* hw_timer_t *timer = NULL; // H/W timer defining (Pointer to the Structure) portMUX_TYPE timerMux = portMUX_INITIALIZER_UNLOCKED; void IRAM_ATTR onTimer() { // Defining interrupt function with IRAM_ATTR for faster access portENTER_CRITICAL_ISR(&timerMux); status_led = !status_led; if (!status_led) { analogWrite(PIN_LED, brightness_led); } else { analogWrite(PIN_LED, 0); } portEXIT_CRITICAL_ISR(&timerMux); } */ // Ethernet stuff #define ETH_SCK 36 #define ETH_SS 34 #define ETH_MOSI 35 #define ETH_MISO 37 #define ETH_INT 38 #define ETH_SPI_CLOCK_MHZ 25 byte mac[6]; AsyncWebServer server(80); ArtnetWiFi artnet; String broadcastIp; uint8_t universe1; uint8_t universe2; Direction direction1; Direction direction2; /* void ledBlink(int ms) { if (timer == NULL) { timer = timerBegin(0, 80, true); // timer 0, prescalar: 80, UP counting timerAttachInterrupt(timer, &onTimer, true); // Attach interrupt } if (ms == 0) { timerAlarmDisable(timer); analogWrite(PIN_LED, brightness_led); } else { ms = ms * 1000; timerAlarmWrite(timer, ms, true); // Match value= 1000000 for 1 sec. delay. timerAlarmEnable(timer); // Enable Timer with interrupt (Alarm Enable) } } */ void onButtonPress() { config.begin("dmx", true); ButtonAction action = static_cast(config.getUInt("button-action", DEFAULT_BUTTON_ACTION)); config.end(); switch (action) { case ResetConfig: config.begin("dmx", false); config.clear(); config.end(); ESP.restart(); break; case Restart: config.begin("dmx", false); config.putBool("restart-via-btn", true); config.end(); ESP.restart(); break; case None: // do nothing break; } } void setup() { Serial.begin(9600); // Get ETH mac delay(1000); esp_efuse_mac_get_default(mac); esp_read_mac(mac, ESP_MAC_ETH); Serial.printf("%02x:%02x:%02x:%02x:%02x:%02x ESP MAC ETH\n", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]); esp_read_mac(mac, ESP_MAC_WIFI_SOFTAP); Serial.printf("%02x:%02x:%02x:%02x:%02x:%02x ESP MAC SOFTAP\n", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]); esp_read_mac(mac, ESP_MAC_WIFI_STA); // ESP_MAC_BASE Serial.printf("%02x:%02x:%02x:%02x:%02x:%02x ESP MAC BASE\n", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]); // LED config.begin("dmx", true); brightness_led = config.getUInt("led-brightness", DEFAULT_LED_BRIGHTNESS); bool restartViaButton = config.getBool("restart-via-btn", false); config.end(); analogWrite(PIN_LED, brightness_led); // Button pinMode(PIN_BUTTON, INPUT_PULLUP); if (digitalRead(PIN_BUTTON) == LOW && !restartViaButton) { // ledBlink(100); unsigned long startTime = millis(); while (digitalRead(PIN_BUTTON) == LOW && (millis() - startTime <= 3000)) { } if (digitalRead(PIN_BUTTON) == LOW) { // ledBlink(0); Serial.println("Reset config"); config.begin("dmx", false); config.clear(); config.end(); delay(2000); } } config.begin("dmx", false); config.putBool("restart-via-btn", false); config.end(); // ledBlink(500); attachInterrupt(PIN_BUTTON, onButtonPress, FALLING); // wait for serial monitor delay(5000); Serial.println("Starting DMX-Interface..."); config.begin("dmx", true); universe1 = config.getUInt("universe-1", DEFAULT_UNIVERSE1); universe2 = config.getUInt("universe-2", DEFAULT_UNIVERSE2); direction1 = static_cast(config.getUInt("direction-1", DEFAULT_DIRECTION1)); direction2 = static_cast(config.getUInt("direction-2", DEFAULT_DIRECTION2)); Serial.print("Port A: Universe "); Serial.print(universe1); Serial.print(" "); Serial.println((direction1 == Input) ? "DMX -> Art-Net" : "Art-Net -> DMX"); Serial.print("Port B: Universe "); Serial.print(universe2); Serial.print(" "); Serial.println((direction2 == Input) ? "DMX -> Art-Net" : "Art-Net -> DMX"); Connection connection = static_cast(config.getUInt("connection", DEFAULT_CONNECTION)); IpMethod ipMethod = static_cast(config.getUInt("ip-method"), DEFAULT_IP_METHOD); char hostname[30]; snprintf(hostname, sizeof(hostname), "ChaosDMX-%02X%02X", mac[4], mac[5]); DEFAULT_SSID = hostname; Serial.print("Hostname: "); Serial.println(hostname); String ssid = config.getString("ssid", DEFAULT_SSID); String pwd = config.getString("password", DEFAULT_PASSWORD); // Default IP as defined in standard https://art-net.org.uk/downloads/art-net.pdf, page 13 IPAddress ip = config.getUInt("ip", DEFAULT_IP); IPAddress subnet = config.getUInt("subnet", DEFAULT_SUBNET); IPAddress gateway = config.getUInt("gateway", 0); config.end(); switch (connection) { case WiFiSta: Serial.println("Initialize as WiFi Station"); WiFi.setHostname(hostname); WiFi.begin(ssid, pwd); Serial.println("SSID: " + ssid + ", pwd: " + pwd); if (ipMethod == Static) { WiFi.config(ip, gateway, subnet); Serial.println("IP: " + ip.toString() + ", gateway: " + gateway + ", subnet: " + subnet); } while (WiFi.status() != WL_CONNECTED) { Serial.print("."); delay(500); } broadcastIp = String(WiFi.broadcastIP().toString().c_str()); Serial.println(""); Serial.print("WiFi connected, IP = "); Serial.println(WiFi.localIP()); Serial.print("MAC address: "); Serial.println(WiFi.macAddress()); Serial.print("Broadcast IP: "); Serial.println(broadcastIp); break; case Ethernet: { Serial.println("Initialize as ETH"); ESP32_W5500_onEvent(); if (ETH.begin(ETH_MISO, ETH_MOSI, ETH_SCK, ETH_SS, ETH_INT, ETH_SPI_CLOCK_MHZ, SPI2_HOST, mac)) { // Dynamic IP setup } else { Serial.println("Failed to configure Ethernet"); } ETH.setHostname(hostname); // ESP32_W5500_waitForConnect(); uint8_t timeout = 5; // in s Serial.print("Wait for connect"); while (!ESP32_W5500_eth_connected && timeout > 0) { delay(1000); timeout--; Serial.print("."); } Serial.println(); if (ESP32_W5500_eth_connected) { Serial.println("DHCP OK!"); } else { Serial.println("Set static IP"); ETH.config(ip, gateway, subnet); } broadcastIp = ETH.broadcastIP().toString(); Serial.print("Local IP : "); Serial.println(ETH.localIP()); Serial.print("Subnet Mask : "); Serial.println(ETH.subnetMask()); Serial.print("Gateway IP : "); Serial.println(ETH.gatewayIP()); Serial.print("DNS Server : "); Serial.println(ETH.dnsIP()); Serial.print("MAC address : "); Serial.println(ETH.macAddress()); Serial.print("Broadcast IP: "); Serial.println(broadcastIp); Serial.println("Ethernet Successfully Initialized"); break; } default: Serial.println("Initialize as WiFi AccessPoint"); WiFi.softAPsetHostname(hostname); WiFi.softAP(ssid, pwd); // AP always with DHCP // WiFi.softAPConfig(ip, gateway, subnet); broadcastIp = WiFi.softAPBroadcastIP().toString(); Serial.print("WiFi AP enabled, IP = "); Serial.println(WiFi.softAPIP()); Serial.print("MAC address: "); Serial.println(WiFi.softAPmacAddress()); Serial.print("Broadcast IP: "); Serial.println(broadcastIp); break; } // Initialize DMX ports Serial.println("Initialize DMX..."); #ifdef CONFIG_IDF_TARGET_ESP32S2 // dmx1.init(21, 33, Serial0); // dmx2.init(17, 18, Serial1); Serial.print("DMX driver 1 installed: "); Serial.println(dmx_driver_is_installed(dmx1)); Serial.print("DMX driver 2 installed: "); Serial.println(dmx_driver_is_installed(dmx2)); dmx_config_t dmx_config = DMX_CONFIG_DEFAULT; dmx_personality_t personalities[] = {}; /*dmx_personality_t personalities[] = { {1, "Default Personality"} };*/ /*int personality_count = 1;*/ int personality_count = 0; dmx_driver_install(dmx1, &dmx_config, personalities, personality_count); dmx_set_pin(dmx1, 21, 33, -1); dmx_driver_install(dmx2, &dmx_config, personalities, personality_count); dmx_set_pin(dmx2, 17, 18, -1); Serial.print("DMX driver 1 installed: "); Serial.println(dmx_driver_is_installed(dmx1)); Serial.print("DMX driver 2 installed: "); Serial.println(dmx_driver_is_installed(dmx2)); Serial.print("DMX driver 1 enabled: "); Serial.println(dmx_driver_is_enabled(dmx1)); Serial.print("DMX driver 2 enabled: "); Serial.println(dmx_driver_is_enabled(dmx2)); // TX/RX Pins und Serial0/Serial1 ausgeben /* Now set the DMX hardware pins to the pins that we want to use and setup will be complete! */ #else dmx1.init(21, 33, Serial1); dmx2.init(17, 18, Serial2); #endif // Initialize Art-Net Serial.println("Initialize Art-Net..."); artnet.begin(); // if Artnet packet comes to this universe, this function is called if (direction1 == Output) { Serial.println("DMX1 as out"); artnet.subscribeArtDmxUniverse(universe1, [&](const uint8_t *data, uint16_t size, const ArtDmxMetadata &metadata, const ArtNetRemoteInfo &remote) { dmx_write_offset(dmx1, 1, data, size); dmx_send(dmx1); dmx_wait_sent(dmx1, DMX_TIMEOUT_TICK); }); } if (direction2 == Output) { Serial.println("DMX2 as out"); artnet.subscribeArtDmxUniverse(universe2, [&](const uint8_t *data, uint16_t size, const ArtDmxMetadata &metadata, const ArtNetRemoteInfo &remote) { dmx_write_offset(dmx2, 1, data, size); dmx_send(dmx2); dmx_wait_sent(dmx2, DMX_TIMEOUT_TICK); }); } // if Artnet packet comes, this function is called to every universe // artnet.subscribeArtDmx([&](const uint8_t *data, uint16_t size, const ArtDmxMetadata &metadata, const ArtNetRemoteInfo &remote) {}); if (!LittleFS.begin(true)) { Serial.println("An Error has occurred while mounting LittleFS"); return; } server.serveStatic("/", LittleFS, "/").setDefaultFile("index.html"); server.on("/config", HTTP_GET, [](AsyncWebServerRequest *request) { onGetConfig(request); }); server.on("/config", HTTP_DELETE, [](AsyncWebServerRequest *request) { config.begin("dmx", false); config.clear(); config.end(); // respond with default config onGetConfig(request); ESP.restart(); }); server.on("/networks", HTTP_GET, [](AsyncWebServerRequest *request) { onGetNetworks(request); }); server.on("/status", HTTP_GET, [](AsyncWebServerRequest *request) { onGetStatus(request); }); server.onRequestBody([](AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total) { if (request->url() == "/config" && request->method() == HTTP_PUT) { onPutConfig(request, data, len, index, total); Serial.println("restarting ESP..."); ESP.restart(); } }); delay(1000); server.begin(); Serial.println("Server started!"); // scan networks and cache them WiFi.scanNetworks(true); // ledBlink(0); // Internal temperature RP2040 /*float tempC = analogReadTemp(); // Get internal temperature Serial.print("Temperature Celsius (ºC): "); Serial.println(tempC);*/ // Internal temperature ESP32 https://www.espboards.dev/blog/esp32-inbuilt-temperature-sensor/ Serial.print("Temperature: "); float result = 0; temp_sensor_read_celsius(&result); Serial.print(result); Serial.println(" °C"); Serial.printf("Internal Total heap %d, internal Free Heap %d\n", ESP.getHeapSize(), ESP.getFreeHeap()); Serial.printf("SPIRam Total heap %d, SPIRam Free Heap %d\n", ESP.getPsramSize(), ESP.getFreePsram()); Serial.printf("ChipRevision %d, Cpu Freq %d, SDK Version %s\n", ESP.getChipRevision(), ESP.getCpuFreqMHz(), ESP.getSdkVersion()); Serial.printf("Flash Size %d, Flash Speed %d\n", ESP.getFlashChipSize(), ESP.getFlashChipSpeed()); } void loop() { // check if artnet packet has come and execute callback artnet.parse(); /* We need a place to store information about the DMX packets we receive. We will use a dmx_packet_t to store that packet information. */ dmx_packet_t dmx1_packet; dmx_packet_t dmx2_packet; /* And now we wait! The DMX standard defines the amount of time until DMX officially times out. That amount of time is converted into ESP32 clock ticks using the constant `DMX_TIMEOUT_TICK`. If it takes longer than that amount of time to receive data, this if statement will evaluate to false. */ if (direction1 == Input && dmx_receive(dmx1, &dmx1_packet, 0)) { // Serial.println("Recv DMX1"); /* If this code gets called, it means we've received DMX data! */ dmx_read_offset(dmx1, 1, dmx1_data, 512); artnet.sendArtDmx(broadcastIp, universe1, dmx1_data, 512); /* Get the current time since boot in milliseconds so that we can find out how long it has been since we last updated data and printed to the Serial Monitor. */ unsigned long now = millis(); /* We should check to make sure that there weren't any DMX errors. */ if (!dmx1_packet.err) { /* Don't forget we need to actually read the DMX data into our buffer so that we can print it out. */ /*dmx_read_offset(dmx1, 1, dmx1_data, dmx1_packet.size); artnet.sendArtDmx(broadcastIp, universe1, dmx1_data, 512);*/ if (now - dmx1_lastUpdate > 1000) { /* Print the received start code - it's usually 0. */ // Serial.printf("Start code is 0x%02X and slot 1 is 0x%02X\n", dmx1_data[0], dmx1_data[1]); dmx1_lastUpdate = now; } } else { /* Oops! A DMX error occurred! Don't worry, this can happen when you first connect or disconnect your DMX devices. If you are consistently getting DMX errors, then something may have gone wrong with your code or something is seriously wrong with your DMX transmitter. */ Serial.println("A DMX 1 error occurred."); } } if (direction2 == Input && dmx_receive(dmx2, &dmx2_packet, 0)) { // Serial.println("Recv DMX2"); /* If this code gets called, it means we've received DMX data! */ dmx_read_offset(dmx2, 1, dmx2_data, 512); artnet.sendArtDmx(broadcastIp, universe2, dmx2_data, 512); /* Get the current time since boot in milliseconds so that we can find out how long it has been since we last updated data and printed to the Serial Monitor. */ unsigned long now = millis(); /* We should check to make sure that there weren't any DMX errors. */ if (!dmx2_packet.err) { /* Don't forget we need to actually read the DMX data into our buffer so that we can print it out. */ /*dmx_read_offset(dmx2, 1, dmx2_data, dmx2_packet.size); artnet.sendArtDmx(broadcastIp, universe2, dmx2_data, 512);*/ if (now - dmx2_lastUpdate > 1000) { /* Print the received start code - it's usually 0. */ // Serial.printf("Start code is 0x%02X and slot 1 is 0x%02X\n", dmx2_data[0], dmx2_data[1]); dmx2_lastUpdate = now; } } else { /* Oops! A DMX error occurred! Don't worry, this can happen when you first connect or disconnect your DMX devices. If you are consistently getting DMX errors, then something may have gone wrong with your code or something is seriously wrong with your DMX transmitter. */ Serial.println("A DMX 2 error occurred."); } } }