Compare commits
	
		
			3 Commits
		
	
	
		
			32ba413380
			...
			22a0c86d15
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 22a0c86d15 | |||
| 55cc608454 | |||
| 09e8e75edc | 
							
								
								
									
										316
									
								
								src/util.h
									
									
									
									
									
								
							
							
						
						
									
										316
									
								
								src/util.h
									
									
									
									
									
								
							| @@ -1,18 +1,310 @@ | ||||
| #pragma once | ||||
|  | ||||
| enum Brightness { | ||||
|     Normal, | ||||
|     Bright, | ||||
| // TODO: reverse engineer builtin and custom bitmaps | ||||
|  | ||||
| // seems like our sign listens to messages on this address | ||||
| // it is also the software's default value | ||||
| static constexpr auto SIGN_ADDRESS = 128; | ||||
|  | ||||
| enum class FontType { | ||||
|     Font5x6Short = 'q', | ||||
|     Font5x11ShortAndWide = 'r', | ||||
|     Font7x6Default = 's', | ||||
|     Font7x11Wide = 't', | ||||
|     Font7x9 = 'u', | ||||
|     Font7x17ExtraWide = 'v', | ||||
|     FontSmallFont = 'w', | ||||
| }; | ||||
|  | ||||
| void sendTextToSign(String text, Brightness brightness = Bright) { | ||||
|     // debugging | ||||
|     Serial.print("Sending text \""); | ||||
|     Serial.print(text); | ||||
|     Serial.println("\""); | ||||
| enum class Brightness { | ||||
|     Normal = 'a', | ||||
|     Bright = 'b', | ||||
| }; | ||||
|  | ||||
|     // TODO: find out how the brightness attribute in the LED sign is encoded in the UART signal | ||||
|     Serial1.print("f01A\\s"); | ||||
|     Serial1.print(text); | ||||
|     Serial1.print("\r\r\r"); | ||||
| enum class Method { | ||||
|     Cyclic = 'A', | ||||
|     Immediate = 'B', | ||||
|     OpenFromRight = 'C', | ||||
|     OpenFromLeft = 'D', | ||||
|     OpenFromCenter = 'E', | ||||
|     OpenToCenter = 'F', | ||||
|     CoverFromCenter = 'G', | ||||
|     CoverFromRight = 'H', | ||||
|     CoverFromLeft = 'I', | ||||
|     CoverToCenter = 'J', | ||||
|     ScrollUp = 'K', | ||||
|     ScrollDown = 'L', | ||||
|     InterlaceToCenter = 'M', | ||||
|     InterlaceCover = 'N', | ||||
|     CoverUp = 'O', | ||||
|     CoverDown = 'P', | ||||
|     ScanLine = 'Q', | ||||
|     Explode = 'R', | ||||
|     PacMan = 'S', | ||||
|     FallAndStack = 'T', | ||||
|     Shoot = 'U', | ||||
|     Flash = 'V', | ||||
|     Random = 'W', | ||||
|     SlideIn = 'X', | ||||
| }; | ||||
|  | ||||
| // macros are literal characters prefixed by ^ | ||||
| enum class Macros { | ||||
|     CurrentTime12hWithoutAmPm = 'D', | ||||
|     CurrentTime12hWithAmPm = 'E', | ||||
|     CurrentTime24h = 'F', | ||||
|     CurrentDateMmDdYyyy = 'A', | ||||
|     CurrentDateYyyy_Mm_Dd = 'B', | ||||
|     CurrentDateMm_Dd_Yyyy = 'C', | ||||
|     CurrentDateDd_Mm_Yyyy = 'a', | ||||
|     Beep1 = 'q', | ||||
|     Beep2 = 'r', | ||||
|     Beep3 = 's', | ||||
| }; | ||||
|  | ||||
| // symbols should be inserted as string representations of their numeric value prefixed by ^ | ||||
| enum class Symbols { | ||||
|     Sun = 66, | ||||
|     CloudOrWind = 67, | ||||
|     UmbrellaRaining = 68, | ||||
|     WallClock = 69, | ||||
|     Telephone = 70, | ||||
|     Glasses = 71, | ||||
|     Faucet = 72, | ||||
|     RocketOrSyringeOrSo = 73, | ||||
|     WhatTheHell = 74, | ||||
|     Key = 75, | ||||
|     Shirt = 76, | ||||
|     Helicopter = 77, | ||||
|     Car = 78, | ||||
|     Tank = 79, | ||||
|     House = 80, | ||||
|     Teapot = 81, | ||||
|     Trees = 82, | ||||
|     Duck = 83, | ||||
|     Scooter = 84, | ||||
|     Bicycle = 85, | ||||
|     Crown = 86, | ||||
|     AppleOrSo = 87, | ||||
|     ArrowRight = 88, | ||||
|     ArrowLeft = 89, | ||||
|     ArrowDownLeft = 90, | ||||
|     ArrowUpLeft = 91, | ||||
|     Cup = 92, | ||||
|     Chair = 93, | ||||
|     HighHeel = 94, | ||||
|     PrizeCup = 95, | ||||
| }; | ||||
|  | ||||
| // cartoons are represented as these characters prefixed by ^ | ||||
| enum class Cartoon { | ||||
|     MerryXMas = 'i', | ||||
|     HappyNewYear = 'j', | ||||
|     FourthOfJuly = 'k', | ||||
|     HappyEaster = 'l', | ||||
|     HappyHalloween = 'm', | ||||
|     DontDrinkAndDrive = 'n', | ||||
|     NoSmoking = 'o', | ||||
|     Welcome = 'p', | ||||
| }; | ||||
|  | ||||
| // the descriptions originate from the UTF-16 specification, https://www.ssec.wisc.edu/~tomw/java/unicode.html | ||||
| // the extended charset starts where ASCII ends | ||||
| enum class ExtendedChar { | ||||
|     // Ç | ||||
|     LatinCapitalLetterCWithCedilla = 0x80, | ||||
|     // ü | ||||
|     LatinSmallLetterUWithDiaeresis = 0x81, | ||||
|     // é | ||||
|     LatinSmallLetterEWithAcute = 0x82, | ||||
|     // â | ||||
|     LatinSmallLetterAWithCircumflex = 0x83, | ||||
|     // ä | ||||
|     LatinSmallLetterAWithDiaeresis = 0x84, | ||||
|     // á | ||||
|     LatinSmallLetterAWithGrave = 0x85, | ||||
|     // å | ||||
|     LatinSmallLetterAWithRingAbove = 0x86, | ||||
|     // ç | ||||
|     LatinSmallLetterCWithCedilla = 0x87, | ||||
|     // ê | ||||
|     LatinSmallLetterEWithCircumflex = 0x88, | ||||
|     // ë | ||||
|     LatinSmallLetterEWithDiaeresis = 0x89, | ||||
|     // è | ||||
|     LatinSmallLetterEWithGrave = 0x8a, | ||||
|     // Ï | ||||
|     LatinCapitalLetterIWithDiaeresis = 0x8b, | ||||
|     // Î | ||||
|     LatinCapitalLetterIWithCircumflex = 0x8c, | ||||
|     // Ì | ||||
|     LatinCapitalLetterIWithGrave = 0x8d, | ||||
|     // Ä | ||||
|     LatinCapitalLetterAWithDiaeresis = 0x8f, | ||||
|     // Å | ||||
|     LatinCapitalLetterAWithRingAbove = 0x8f, | ||||
|     // É | ||||
|     LatinCapitalLetterEWithAcute = 0x90, | ||||
|     // æ | ||||
|     LatinSmallLetterAe = 0x91, | ||||
|     // Æ | ||||
|     LatinCapitalLetterAe = 0x92, | ||||
|     // ô | ||||
|     LatinSmallLetterOWithCircumflex = 0x93, | ||||
|     // ö | ||||
|     LatinSmallLetterOWithDiaeresis = 0x94, | ||||
|     // ò | ||||
|     LatinSmallLetterOWithGrave = 0x95, | ||||
|     // û | ||||
|     LatinSmallLetterUWithCircumflex = 0x96, | ||||
|     // ù | ||||
|     LatinSmallLetterUWithGrave = 0x97, | ||||
|     // ÿ | ||||
|     LatinSmallLetterYWithDiaeresis = 0x98, | ||||
|     // Ö | ||||
|     LatinCapitalLLetterOWithDiaeresis = 0x99, | ||||
|     // Ü | ||||
|     LatinCapitalLetterUWithDiaeresis = 0x9a, | ||||
|     // ¢ | ||||
|     CentSign = 0x9b, | ||||
|     // £ | ||||
|     PoundSign = 0x9c, | ||||
|     // ¥ | ||||
|     YenSign = 0x9d, | ||||
|     // urm... | ||||
|     UnknownPCurrencySign = 0x9e, | ||||
|     // ƒ | ||||
|     LatinSmallLetterFWithHook = 0x9f, | ||||
|     // á | ||||
|     LatinSmallLetterAWithAcute = 0xa0, | ||||
|     // Í | ||||
|     LatinCapitalLetterIWithAcute = 0xa1, | ||||
|     // Ó | ||||
|     LatinCapitalLetterOWithAcute = 0xa2, | ||||
|     // ú | ||||
|     LatinSmallLetterUWithAcute = 0xa3, | ||||
|     // ñ | ||||
|     LatinSmallLetterNWithTilde = 0xa4, | ||||
|     // Ñ | ||||
|     LatinCapitalLetterNWithTilde = 0xa5, | ||||
|     // a̱ | ||||
|     LatinSmallLetterAWithMacronBelow = 0xa6, | ||||
|     // couldn't find that one in the UTF-16 table | ||||
|     LatinSmallLetterOWithMacronBelow = 0xa7, | ||||
|     // ¿ | ||||
|     InvertedQuestionMark = 0xa8, | ||||
|     // you can probably imagine what it'll look like | ||||
|     LatinCapitalLetterZWithRingAbove = 0xa9, | ||||
|     // Ź | ||||
|     LatinCapitalLetterZWithAcute = 0xaa, | ||||
|     // ń | ||||
|     LatinSmallLetterNWithAcute = 0xab, | ||||
|     // some weird L with a diagonal line | ||||
|     AnotherWeirdSign = 0xac, | ||||
|     // Ś or ś, can't tell for sure, let's assume capital | ||||
|     LatinCapitalLetterSWithAcute = 0xab, | ||||
|     // Ć | ||||
|     LatinCapitalLetterCWithAcute = 0xac, | ||||
|     // ȩ | ||||
|     LatinSmallLetterEWithCedilla = 0xad, | ||||
|     // well, you can imagine what it'll look like | ||||
|     LatinSmallLetterAWithCedilla = 0xb0, | ||||
|     // € | ||||
|     EuroSign = 0xb1, | ||||
|     // E with some vertical line below | ||||
|     LatinCapitalLetterEWithSomeLineBelow = 0xb3, | ||||
|     // a with some vertical line below | ||||
|     LatinSmallLetterAWithSomeWeirdLineBelow = 0xb4, | ||||
|     // α | ||||
|     GreekSmallLetterAlpha = 0xe0, | ||||
|     // β | ||||
|     GreekSmallLetterBeta = 0xe1, | ||||
|     // Γ | ||||
|     GreekCapitalLetterGamma = 0xe2, | ||||
|     // π | ||||
|     GreekSmallLetterPi = 0xe3, | ||||
|     // Σ | ||||
|     GreekCapitalLetterSigma = 0xe4, | ||||
|     // σ | ||||
|     GreekSmallLetterSigma = 0xe5, | ||||
|     // μ | ||||
|     GreekSmallLetterMu = 0xe6, | ||||
|     // τ | ||||
|     GreekSmallLetterTau = 0xe7, | ||||
|     // Φ | ||||
|     GreekCapitalLetterPhi = 0xe8, | ||||
|     // θ | ||||
|     GreekSmallLetterTheta = 0xe9, | ||||
|     // Ω | ||||
|     GreekCapitalLetterOmega = 0xea, | ||||
|     // δ | ||||
|     GreekSmallLetterDelta = 0xeb, | ||||
|     // ∞ | ||||
|     Infinity = 0xec, | ||||
|     // φ | ||||
|     GreekSmallLetterPhi = 0xed, | ||||
| }; | ||||
|  | ||||
| /* | ||||
|  * Message syntax: | ||||
|  * separator: ~ | ||||
|  * address (int string): 128 | ||||
|  * separator: ~ | ||||
|  * random string(?): f01 | ||||
|  * method (single char) | ||||
|  * format string (see below) | ||||
|  * end of message: \r\r\r | ||||
|  * | ||||
|  * the format string contains the message along with control characters that define some of the formatting | ||||
|  * note: we have not tested the UTF-8 capabilities, so far only ASCII | ||||
|  * it supports multiple lines (yet to be tested) | ||||
|  * you can also insert macros (e.g., current time/date), symbols (e.g., a sun) and cartoons | ||||
|  * the specials are preceded by a caret following the respective character | ||||
|  * then, there is an extended charset that supports a small subset of UTF-16 (yes, that's right, it's not just UTF-8) | ||||
|  * it can further contain information that should be used from their occurrence on | ||||
|  * the formatting escape sequences are prefixed by a literal backspace following a char that represents the format | ||||
|  * brightness is \<ab>, speed is Y<1-8>, "stay time" is \Z<1-8>, font type is \[A-X] | ||||
|  * new lines are introduced by a single \r, the end of the message is marked by three of them | ||||
|  * | ||||
|  * default formatting: | ||||
|  *   - method: cyclic | ||||
|  *   - font type: 7x6 | ||||
|  *   - speed: 5 | ||||
|  *   - stay time: 4 | ||||
|  *   - brightness: bright | ||||
|  */ | ||||
|  | ||||
| void sendTextToSign( | ||||
|     const String &text, | ||||
|     const Method method = Method::Cyclic, | ||||
|     const FontType fontType = FontType::Font7x6Default, | ||||
|     const int speed = 5, | ||||
|     const int stayTime = 4, | ||||
|     const Brightness brightness = Brightness::Bright | ||||
| ) { | ||||
|     if (speed < 0 || speed > 8) { | ||||
|         Serial.println("Invalid speed: " + String(speed, DEC)); | ||||
|         return; | ||||
|     } | ||||
|     if (stayTime < 0 || stayTime > 8) { | ||||
|         Serial.println("Invalid stayTime: " + String(stayTime, DEC)); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     Serial.println("Sending text: \"" + text + "\""); | ||||
|  | ||||
|     String toSend = "~" + String(SIGN_ADDRESS, DEC) + "~f01" + (char) method + "\\" + (char) brightness + "\\Y" + String(speed, DEC) + "\\Z" + String(stayTime, DEC) + "\\" + (char) fontType + text + "\r\r\r"; | ||||
|  | ||||
|     Serial1.print(toSend); | ||||
|  | ||||
|     Serial.println("Text sent to device: " + toSend); | ||||
|     Serial.print("Hex dump: "); | ||||
|     for (size_t i = 0; i < toSend.length(); ++i) { | ||||
|         auto hexChar = String(toSend[i], HEX); | ||||
|         if (hexChar.length() == 1) { | ||||
|             hexChar = '0' + hexChar; | ||||
|         } | ||||
|         Serial.print(hexChar + ' '); | ||||
|     } | ||||
|     Serial.println(); | ||||
| } | ||||
|   | ||||
| @@ -28,11 +28,6 @@ namespace { | ||||
|                     <input type="text" name="text"> | ||||
|                 </div> | ||||
|  | ||||
|                 <div> | ||||
|                     <label for="bright">Bright mode</label> | ||||
|                     <input type="checkbox" name="bright"> | ||||
|                 </div> | ||||
|  | ||||
|                 <div> | ||||
|                     <button>Send</button> | ||||
|                 </div> | ||||
| @@ -45,7 +40,14 @@ namespace { | ||||
|     } | ||||
|  | ||||
|     void handleFormSend() { | ||||
|         sendTextToSign("HELLO FORM"); | ||||
|         String text = webServer.arg("text"); | ||||
|  | ||||
|         if (text.isEmpty()) { | ||||
|             webServer.send(404, "text/plain", "empty text"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         sendTextToSign(text); | ||||
|  | ||||
|         // redirect back to root page | ||||
|         webServer.sendHeader("Location", "/"); | ||||
| @@ -54,7 +56,14 @@ namespace { | ||||
|  | ||||
|     // RESTful API endpoint -- returns a proper status code rather than a redirect | ||||
|     void handleApiSend() { | ||||
|         sendTextToSign("HELLO API"); | ||||
|         String text = webServer.arg("text"); | ||||
|  | ||||
|         if (text.isEmpty()) { | ||||
|             webServer.send(404, "text/plain", "empty text"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         sendTextToSign(text); | ||||
|         webServer.send(200, "text/plain", "ok"); | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										42
									
								
								tools/re-server.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								tools/re-server.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| """ | ||||
| Simple socket server used to reverse engineer the messaging protocol the bundled software uses to send data to the | ||||
| LED sign. | ||||
|  | ||||
| The server just dumps incoming messages in a hexdump-like format as well as an easy to read Python bytestring. | ||||
| """ | ||||
|  | ||||
|  | ||||
| import asyncio | ||||
| import string | ||||
|  | ||||
|  | ||||
| async def handle_client(reader, writer): | ||||
|     message = await reader.read(255) | ||||
|  | ||||
|     hex_chars = [hex(j)[2:].zfill(2) for j in message] | ||||
|  | ||||
|     ascii_chars = [] | ||||
|     for j in message: | ||||
|         char = chr(j) | ||||
|         if char in (string.ascii_letters + string.digits + string.punctuation): | ||||
|             ascii_chars.append(char) | ||||
|         else: | ||||
|             ascii_chars.append(".") | ||||
|  | ||||
|     print("Message received: ", end="") | ||||
|     print(" ".join(hex_chars), end=" | ") | ||||
|     print("".join(ascii_chars), end=" | ") | ||||
|     print("".join(repr(message))) | ||||
|  | ||||
|     writer.close() | ||||
|  | ||||
|  | ||||
| async def run_server(): | ||||
|     server = await asyncio.start_server(handle_client, '', 3000) | ||||
|     async with server: | ||||
|         await server.serve_forever() | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(run_server()) | ||||
|  | ||||
		Reference in New Issue
	
	Block a user