├── ArduinoFiles └── README.TXT ├── README.md ├── ScratchFiles ├── ExtensionDescriptors │ ├── P4K_Extension.s2e │ ├── s2a_fm.s2e │ ├── s2a_fm_DE.s2e │ ├── s2a_fm_Es.s2e │ ├── s2a_fm_Fr.s2e │ ├── s2a_fm_GR.s2e │ ├── s2a_fm_It.s2e │ ├── s2a_fm_NL.s2e │ ├── s2a_fm_Pt.s2e │ ├── s2a_fm_ja.s2e │ ├── s2a_fm_ja2.s2e │ ├── s2a_fm_ko.s2e │ ├── s2a_fm_zh_cn.s2e │ └── s2a_fm_zh_tw.s2e └── ScratchProjects │ ├── P4K_Extension.sb2 │ ├── Turn On LED On Pin 6.sb2 │ ├── s2a_fm_base.sb2 │ ├── s2a_fm_base_DE.sb2 │ ├── s2a_fm_base_Es.sb2 │ ├── s2a_fm_base_Fr.sb2 │ ├── s2a_fm_base_GR.sb2 │ ├── s2a_fm_base_It.sb2 │ ├── s2a_fm_base_Pt.sb2 │ ├── s2a_fm_base_ja.sb2 │ ├── s2a_fm_base_ko.sb2 │ ├── s2a_fm_base_nl.sb2 │ ├── s2a_fm_base_zh_cn.sb2 │ ├── s2a_fm_base_zh_tw.sb2 │ ├── sonarTest.sb2 │ └── spinning_cat.sb2 ├── Snap!Files ├── Español │ ├── blink.xml │ ├── blink_tiempo_variable.xml │ ├── boton.xml │ ├── boton_con_imagen.xml │ ├── contador1.xml │ ├── contador2.xml │ ├── s2a_fm_Snap_base_Es.xml │ ├── semaforo_tiempo_variable.xml │ ├── servo1.xml │ └── servo2.xml ├── S2a_fm_Pt.xml ├── Snap!Mobile │ ├── StandardFirmataYun │ │ └── StandardFirmataYun │ │ │ └── StandardFirmataYun.ino │ ├── arduino │ │ ├── PyMata │ │ │ ├── __init__.py │ │ │ ├── pymata.py │ │ │ ├── pymata_command_handler.py │ │ │ └── pymata_serial.py │ │ ├── log │ │ │ └── s2a_fm_debugging.log │ │ ├── s2a_fm.py │ │ ├── scratch_command_handlers.py │ │ ├── scratch_http_server.py │ │ ├── serial │ │ │ ├── __init__.py │ │ │ ├── rfc2217.py │ │ │ ├── serialcli.py │ │ │ ├── serialjava.py │ │ │ ├── serialposix.py │ │ │ ├── serialutil.py │ │ │ ├── serialwin32.py │ │ │ ├── sermsdos.py │ │ │ ├── tools │ │ │ │ ├── __init__.py │ │ │ │ ├── list_ports.py │ │ │ │ ├── list_ports_linux.py │ │ │ │ ├── list_ports_osx.py │ │ │ │ ├── list_ports_posix.py │ │ │ │ ├── list_ports_windows.py │ │ │ │ └── miniterm.py │ │ │ ├── urlhandler │ │ │ │ ├── __init__.py │ │ │ │ ├── protocol_hwgrep.py │ │ │ │ ├── protocol_loop.py │ │ │ │ ├── protocol_rfc2217.py │ │ │ │ └── protocol_socket.py │ │ │ └── win32.py │ │ └── xlate.cfg │ └── wiredMotorTest.xml ├── Turn On LED On Pin 6.xml ├── blink.xml ├── s2a_fm_Snap_base.xml ├── s2a_fm_Snap_base_DE.xml ├── s2a_fm_Snap_base_Es.xml ├── s2a_fm_Snap_base_FR.xml ├── s2a_fm_Snap_base_GR.xml ├── s2a_fm_Snap_base_It.xml ├── s2a_fm_Snap_base_dutch.xml ├── s2a_fm_Snap_base_ko.xml ├── s2a_fm_Snap_base_zh_cn.xml ├── s2a_fm_Snap_base_zh_tw.xml └── spinning_sprite.xml ├── documentation ├── Español │ └── s2a_fm_Espanish_tutorial.pdf ├── LED_EXAMPLE.png ├── pot1.png ├── s2a_fm_reference - 完成稿.pdf ├── s2a_fm_reference.pdf ├── s2a_fm_參考文件.pdf ├── scratch_blocks.png ├── scratch_langs.png └── snap_blocks.png ├── extra_goodies ├── linux │ └── s2a_fm.sh └── windows │ └── s2a_fm.bat ├── license.txt ├── s2a_fm.py ├── scratch_command_handlers.py ├── scratch_http_server.py └── xlate.cfg /ArduinoFiles/README.TXT: -------------------------------------------------------------------------------- 1 | All custom Arduino Firmata sketches may be found in PyMata @ https://github.com/MrYsLab/PyMata 2 | 3 | You can also use StandardFirmata supplied with the Arduino IDE, if you do not need the extended features of the custom 4 | sketches. -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | s2a_fm 2 | ====== 3 | 4 | [![Join the chat at https://gitter.im/MrYsLab/s2a_fm](https://badges.gitter.im/MrYsLab/s2a_fm.svg)](https://gitter.im/MrYsLab/s2a_fm?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge) 5 |

6 | Sublime's custom image 7 |

8 | 9 | # A New Version For Python 3.5, called s2aio, Is Available [Here](https://github.com/MrYsLab/s2aio/wiki). 10 | 11 | Arduino users! Would you like to configure and control your Arduino micro-controller without having 12 | to write a single line of Arduino sketch code and at the same time have access to a graphical user 13 | interface? 14 | 15 | Scratch and Snap! programmers! Would you like to control and communicate with an Arduino board using 16 | Scratch? Imagine, using Scratch to control physical devices such as LEDs, motors, and relays while 17 | monitoring devices, such as temperature sensors, potentiometers, and light sensors. What would you 18 | create? 19 | 20 | Want to do motor control using Snap! and Wifi? Check out our article on Physical Computing. 21 | http://www.instructables.com/id/The-SnapMobile-Start-Your-Physical-Computing-Engin/ 22 | 23 | All the code is included here! 24 | 25 | s2a_fm is a Scratch/Snap! hardware extension written in Python allowing Scratch or Snap! and an Arduino 26 | micro-controller to communicate seamlessly. 27 | 28 | Program Block translations are included for Chinese, Dutch, English (default), French, German, Greek, Italian, Korean, Portuguese, Spanish, and Taiwanese. 29 | 30 | Make sure you get the latest version of the Scratch Off-Line editor if you are using Scratch (version 404 or greater). 31 | 32 | Check Out Our Blog 33 | ------------------ 34 | http://mryslab.blogspot.com/ 35 | 36 | Installation Instructions 37 | -------------------------- 38 | The s2a_fm Reference Manual, s2a_fm_reference.pdf, located in the documentation directory of this distribution, 39 | provides full installation instructions. 40 | 41 | Alternative Japanese Block Translation May 30, 2017 42 | -------------------------------------- 43 | Translations provided through the generosity of Hiroaki Kawashima 44 | 45 | Japanese Block Translation Oct 17 2016 46 | -------------------------------------- 47 | Translations provided through the generosity of Antoine Choppin 48 | 49 | Greek Block Translation Feb 3, 2015 50 | ----------------------------------- 51 | Translations provided through the generosity of Alexandros Moskofidis 52 | 53 | Italian Block Translation Dec 12, 2014 54 | --------------------------------------- 55 | Translations provided through the generosity of Gianfranco Zuliani 56 | 57 | New Language Translation October 9, 2014 58 | ------------------------ 59 | Taiwanese translation provided through the courtesy of Hsu Jen-Chieh. 60 | 61 | Version 1.5 Apr 12, 2014 62 | ------------------------ 63 | New Features for 1.5 64 | 65 | Korean translation provided through the generosity of Dr. Eungil Kim 66 | 67 | Portuguese and German translations of Block Text provided through the generosity of 68 | Aldo von Wangenheim, from www.computacaonaescola.ufsc.br 69 | 70 | A Chinese translation of the reference manual was graciously provided by Yufangjun 发自我的小米手机 71 | 72 | Version 1.4 Mar 23, 2014 73 | ----------------------- 74 | New Feature for 1.4 75 | 76 | Chinese translation of Block Text provided through the generosity of Professor YuFangjun 77 | 78 | French translation of Block Text provided through the generosity of Professor Sebastien Canet 79 | . 80 | 81 | Version 1.3 Feb 23, 2014 82 | ------------------------ 83 | New Features for 1.3 84 | 85 | 1. Support for the upcoming Snap!Mobile Physical Computing Project in "Instructables" 86 | 87 | 2. Spanish Translation of Block Text and a Spanish Tutorial provided courtesy of Professor 88 | José Manuel Ruiz 89 | 90 | 91 | Version 1.2 Jan 1, 2014 92 | ----------------------- 93 | New Features for 1.2: 94 | 95 | 1. Support for Snap! 4.0 provided. 96 | 97 | 2. Support for up to 4 simultaneous HC-SR04 type "Ping" Sensors. 98 | (This requires using PyMata version 1.54 or greater and the FirmataPlus Arduino sketch supplied with PyMata 1.54). 99 | 100 | 3. Dutch translation for the Scratch/Snap! Block Text included (Thanks to Sjoerd Dirk Meijer). 101 | 102 | 4. Provision to translate Scratch/Snap! Block Text to any language. 103 | 104 | Version 1.1 December 19, 2013 105 | ----------------------------- 106 | 107 | New Features for 1.1: 108 | 109 | 1. When enabling a digital pin, the pin capabilty is verified for the requested mode. 110 | 111 | 2. A new Scratch debugger command block has been added to help debug Scratch scripts. 112 | 113 | Known Snap! 4.0 Extension Issues 114 | ---------------------- 115 | None. 116 | 117 | 118 | Known Scratch 2.0 Extension Issues 119 | ------------ 120 | All previous major issues have been resolved as of version 404 of the Off-Line editor. 121 | 122 | None. 123 | 124 | Scratch s2a_fm Extension Blocks 125 | ------------------------------- 126 | 127 | ![ScreenShot](https://raw.github.com/MrYsLab/s2a_fm/master/documentation/scratch_blocks.png) 128 | 129 | Snap! s2a_fm Extension Blocks 130 | ----------------------------- 131 | ![ScreenShot](https://raw.github.com/MrYsLab/s2a_fm/master/documentation/snap_blocks.png) 132 | 133 | Wiring Diagrams for Examples 134 | ---------------------------- 135 | 136 | ![ScreenShot](https://raw.github.com/MrYsLab/s2a_fm/master/documentation/LED_EXAMPLE.png) 137 | 138 | ![ScreenShot](https://raw.github.com/MrYsLab/s2a_fm/master/documentation/pot1.png) 139 | 140 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/P4K_Extension.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "P4K Extension", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/s2a_fm", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "%m.pin_state Digital Pin %n for %m.digital_pin_mode", 9 | "digital_pin_mode", 10 | "Enable", 11 | "PIN", 12 | "Input" 13 | ], 14 | [ 15 | " ", 16 | "%m.pin_state Analog Pin (A) %n for Input", 17 | "analog_pin_mode", 18 | "Enable", 19 | "PIN" 20 | ], 21 | [ 22 | "", 23 | "DigitalWrite Set Pin %n to %m.high_low", 24 | "digital_write", 25 | "PIN", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "AnalogWrite (PWM) Set Pin %n to %n", 31 | "analog_write", 32 | "PIN", 33 | "VAL" 34 | ], 35 | [ 36 | "", 37 | "Play Tone on Pin: %n HZ: %n ms: %n", 38 | "play_tone", 39 | "PIN", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "Turn Tone Off for Pin: %n", 46 | "tone_off", 47 | "PIN" 48 | ], 49 | [ 50 | "", 51 | "Move Servo on Pin %n Deg: %n", 52 | "set_servo_position", 53 | "PIN", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "Read Digital Pin %n", 59 | "digital_read", 60 | "PIN" 61 | ], 62 | [ 63 | "r", 64 | "Read Analog Pin (A) %n", 65 | "analog_read", 66 | "PIN" 67 | ], 68 | [ 69 | "", 70 | "Debugger %m.off_on", 71 | "debugger", 72 | "Off" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "Enable", 78 | "Disable" 79 | ], 80 | "digital_pin_mode": [ 81 | "Input", 82 | "Output", 83 | "PWM", 84 | "Servo", 85 | "Tone", 86 | "SONAR" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "Off", 94 | "On" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "s2a_fm - Scratch to Arduino", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "%m.pin_state Digital Pin %n for %m.digital_pin_mode", 9 | "digital_pin_mode", 10 | "Enable", 11 | "PIN", 12 | "Input" 13 | ], 14 | [ 15 | " ", 16 | "%m.pin_state Analog Pin (A) %n for Input", 17 | "analog_pin_mode", 18 | "Enable", 19 | "PIN" 20 | ], 21 | [ 22 | "", 23 | "DigitalWrite Set Pin %n to %m.high_low", 24 | "digital_write", 25 | "PIN", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "AnalogWrite (PWM) Set Pin %n to %n", 31 | "analog_write", 32 | "PIN", 33 | "VAL" 34 | ], 35 | [ 36 | "", 37 | "Play Tone on Pin: %n HZ: %n ms: %n", 38 | "play_tone", 39 | "PIN", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "Turn Tone Off for Pin: %n", 46 | "tone_off", 47 | "PIN" 48 | ], 49 | [ 50 | "", 51 | "Move Servo on Pin %n Deg: %n", 52 | "set_servo_position", 53 | "PIN", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "Read Digital Pin %n", 59 | "digital_read", 60 | "PIN" 61 | ], 62 | [ 63 | "r", 64 | "Read Analog Pin (A) %n", 65 | "analog_read", 66 | "PIN" 67 | ], 68 | [ 69 | "", 70 | "Debugger %m.off_on", 71 | "debugger", 72 | "Off" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "Enable", 78 | "Disable" 79 | ], 80 | "digital_pin_mode": [ 81 | "Input", 82 | "Output", 83 | "PWM", 84 | "Servo", 85 | "Tone", 86 | "SONAR" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "Off", 94 | "On" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_DE.s2e: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ExtensionDescriptors/s2a_fm_DE.s2e -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_Es.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "s2a_fm - Scratch para Arduino", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "%m.pin_state PIN Digital %n como %m.digital_pin_mode", 9 | "digital_pin_mode", 10 | "Activar", 11 | "PIN", 12 | "Entrada" 13 | ], 14 | [ 15 | " ", 16 | "%m.pin_state PIN Anal�gico (A) %n como entrada", 17 | "analog_pin_mode", 18 | "Activar", 19 | "PIN" 20 | ], 21 | [ 22 | "", 23 | "Valor Digital en PIN %n poner a %m.high_low", 24 | "digital_write", 25 | "PIN", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "Escribir valor Anal�gico(PWM) en Pin %n poner a %n", 31 | "analog_write", 32 | "PIN", 33 | "VAL" 34 | ], 35 | [ 36 | "", 37 | "Generar Tono en PIN: %n HZ: %n MS: %n", 38 | "play_tone", 39 | "PIN", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "Apagar Tono en PIN: %n", 46 | "tone_off", 47 | "PIN" 48 | ], 49 | [ 50 | "", 51 | "Mover Servo en Pin %n Grados: %n", 52 | "set_servo_position", 53 | "PIN", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "Leer PIN Digital %n", 59 | "digital_read", 60 | "PIN" 61 | ], 62 | [ 63 | "r", 64 | "Leer PIN Anal�gico(A) %n", 65 | "analog_read", 66 | "PIN" 67 | ], 68 | [ 69 | "", 70 | "Depurar %m.off_on", 71 | "debugger", 72 | "No" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "Activar", 78 | "Desactivar" 79 | ], 80 | "digital_pin_mode": [ 81 | "Entrada", 82 | "Salida", 83 | "PWM", 84 | "Servo", 85 | "Tono", 86 | "SONAR" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "No", 94 | "Si" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_Fr.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "Scratch pour Arduino", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "%m.pin_state la broche Digital %n comme une %m.digital_pin_mode", 9 | "digital_pin_mode", 10 | "Activer", 11 | "PIN", 12 | "entrée." 13 | ], 14 | [ 15 | " ", 16 | "%m.pin_state la broche Analogique A%n comme une entrée", 17 | "analog_pin_mode", 18 | "Activer", 19 | "PIN" 20 | ], 21 | [ 22 | "", 23 | "Mettre l'état logique de la broche Digital %n à %m.high_low", 24 | "digital_write", 25 | "PIN", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "Ecrire sur la broche PWM %n la valeur %n", 31 | "analog_write", 32 | "PIN", 33 | "VAL" 34 | ], 35 | [ 36 | "", 37 | "Jouer le son sur la broche %n , fréquence (Hz) : %n , durée (ms) : %n", 38 | "play_tone", 39 | "PIN", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "Arrêter le son sur la broche : %n", 46 | "tone_off", 47 | "PIN" 48 | ], 49 | [ 50 | "", 51 | "Tourner le servo-moteur sur la broche %n de : %n degrés", 52 | "set_servo_position", 53 | "PIN", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "l'état logique de la broche Digital %n", 59 | "digital_read", 60 | "PIN" 61 | ], 62 | [ 63 | "r", 64 | "la valeur sur la broche Analogique A%n", 65 | "analog_read", 66 | "PIN" 67 | ], 68 | [ 69 | "", 70 | "Debugger %m.off_on", 71 | "debugger", 72 | "Arrêt" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "Activer", 78 | "Désactiver" 79 | ], 80 | "digital_pin_mode": [ 81 | "entrée.", 82 | "sortie.", 83 | "impulsion PWM.", 84 | "rotation Servo-moteur.", 85 | "sortie Son.", 86 | "mesure Sonar." 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "Arrêt", 94 | "Marche" 95 | ] 96 | } 97 | } -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_GR.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "s2a_fm - Scratch για το Arduino", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "%m.pin_state Ψηφιακό PIN %n για %m.digital_pin_mode", 9 | "digital_pin_mode", 10 | "Ενεργοποίηση", 11 | "PIN", 12 | "Είσοδο" 13 | ], 14 | [ 15 | " ", 16 | "%m.pin_state Αναλογικό PIN (A) %n για Είσοδο", 17 | "analog_pin_mode", 18 | "Ενεργοποίηση", 19 | "PIN" 20 | ], 21 | [ 22 | "", 23 | "Ψηφιακή Εγγραφή όρισε Pin %n σε %m.high_low", 24 | "digital_write", 25 | "PIN", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "Αναλογική Εγγραφή (PWM) όρισε Pin %n σε %n", 31 | "analog_write", 32 | "PIN", 33 | "VAL" 34 | ], 35 | [ 36 | "", 37 | "Παίξε τόνο στο Pin: %n HZ: %n για ms: %n", 38 | "play_tone", 39 | "PIN", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "Σταμάτησε τόνο στο Pin: %n", 46 | "tone_off", 47 | "PIN" 48 | ], 49 | [ 50 | "", 51 | "Μετακίνησε το Σέρβο στο Pin %n κατά μοίρες: %n", 52 | "set_servo_position", 53 | "PIN", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "Διάβασε Ψηφιακή Είσοδο στο Pin %n", 59 | "digital_read", 60 | "PIN" 61 | ], 62 | [ 63 | "r", 64 | "Διάβασε Αναλογική Είσοδο στο Pin (A) %n", 65 | "analog_read", 66 | "PIN" 67 | ], 68 | [ 69 | "", 70 | "Αποσφαλματωτής %m.off_on", 71 | "debugger", 72 | "Κλείσιμο" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "Ενεργοποίηση", 78 | "Απενεργοποίηση" 79 | ], 80 | "digital_pin_mode": [ 81 | "Είσοδο", 82 | "Έξοδο", 83 | "PWM", 84 | "Σέρβο", 85 | "Τόνος", 86 | "Σόναρ" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "Κλείσιμο", 94 | "Άνοιγμα" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_It.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "s2a_fm - Scratch per Arduino", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "%m.pin_state Pin Digitale %n come %m.digital_pin_mode", 9 | "digital_pin_mode", 10 | "Attiva", 11 | "PIN", 12 | "Ingresso" 13 | ], 14 | [ 15 | " ", 16 | "%m.pin_state Pin Analogico (A) %n come Ingresso", 17 | "analog_pin_mode", 18 | "Attiva", 19 | "PIN" 20 | ], 21 | [ 22 | "", 23 | "Imposta Pin Digitale %n a %m.high_low", 24 | "digital_write", 25 | "PIN", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "Imposta Pin Analogico (PWM) %n a %n", 31 | "analog_write", 32 | "PIN", 33 | "VAL" 34 | ], 35 | [ 36 | "", 37 | "Esegui Tono su Pin %n Frequenza: %n Hz Durata: %n ms", 38 | "play_tone", 39 | "PIN", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "Interrompi Tono su Pin %n", 46 | "tone_off", 47 | "PIN" 48 | ], 49 | [ 50 | "", 51 | "Sposta Servo su Pin %n Posizione: %n gradi", 52 | "set_servo_position", 53 | "PIN", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "Livello Logico Pin Digitale %n", 59 | "digital_read", 60 | "PIN" 61 | ], 62 | [ 63 | "r", 64 | "Valore Pin Analogico (A) %n", 65 | "analog_read", 66 | "PIN" 67 | ], 68 | [ 69 | "", 70 | "Debugger %m.off_on", 71 | "debugger", 72 | "No" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "Attiva", 78 | "Disattiva" 79 | ], 80 | "digital_pin_mode": [ 81 | "Ingresso", 82 | "Uscita", 83 | "PWM", 84 | "Servo", 85 | "Tono", 86 | "Sonar" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "No", 94 | "Si" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_NL.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "s2a_fm - Scratch voor Arduino", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "Zet %m.pin_state : Digitale pin %n als %m.digital_pin_mode", 9 | "digital_pin_mode", 10 | "aan", 11 | "PIN", 12 | "ingang" 13 | ], 14 | [ 15 | " ", 16 | "Zet %m.pin_state : Analoge pin (A) %n als ingang", 17 | "analog_pin_mode", 18 | "aan", 19 | "PIN" 20 | ], 21 | [ 22 | "", 23 | "DigitalWrite: Maak pin %n %m.high_low", 24 | "digital_write", 25 | "PIN", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "AnalogWrite (PWM): Maak pin %n %n", 31 | "analog_write", 32 | "PIN", 33 | "VAL" 34 | ], 35 | [ 36 | "", 37 | "Speel toon op pin %n , Hz: %n , ms: %n", 38 | "play_tone", 39 | "PIN", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "Zet toon uit op pin %n", 46 | "tone_off", 47 | "PIN" 48 | ], 49 | [ 50 | "", 51 | "Draai servo op pin %n %n graden", 52 | "set_servo_position", 53 | "PIN", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "Lees digitale pin %n uit", 59 | "digital_read", 60 | "PIN" 61 | ], 62 | [ 63 | "r", 64 | "Lees analoge pin (A) %n uit", 65 | "analog_read", 66 | "PIN" 67 | ], 68 | [ 69 | "", 70 | "Debugger %m.off_on", 71 | "debugger", 72 | "uitgeschakeld" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "aan", 78 | "uit" 79 | ], 80 | "digital_pin_mode": [ 81 | "ingang", 82 | "uitgang", 83 | "PWM", 84 | "servo", 85 | "toon", 86 | "afstand" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "ingeschakeld", 94 | "uitgeschakeld" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_Pt.s2e: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ExtensionDescriptors/s2a_fm_Pt.s2e -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_ja.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "s2a_fm - アルドゥイーノ用スクラッチ", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "%n 番目のデジタルピンを %m.digital_pin_mode として %m.pin_state にする", 9 | "digital_pin_mode_ja", 10 | "1", 11 | "入力", 12 | "有効" 13 | ], 14 | [ 15 | " ", 16 | "%n 番目のアナログピンを入力として %m.pin_state にする", 17 | "analog_pin_mode_ja", 18 | "1", 19 | "有効" 20 | ], 21 | [ 22 | "", 23 | "%n 番目のデジタルピンを %m.high_low に設定する", 24 | "digital_write", 25 | "1", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "%n 番目のPWMピンに %n と言う値を書き込む", 31 | "analog_write", 32 | "1", 33 | "10" 34 | ], 35 | [ 36 | "", 37 | "%n 番目のピンに次の音を鳴らす、周波数: %n ハーツ、長さ: %n ミリ秒", 38 | "play_tone", 39 | "1", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "%n 番目のピンで音を止める", 46 | "tone_off", 47 | "1" 48 | ], 49 | [ 50 | "", 51 | "%n 番目のピンのサーボモーターを %n 度で回す", 52 | "set_servo_position", 53 | "1", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "%n 番目のデジタルピンの状態", 59 | "digital_read", 60 | "1" 61 | ], 62 | [ 63 | "r", 64 | "%n 番目のアナログピンの値", 65 | "analog_read", 66 | "1" 67 | ], 68 | [ 69 | "", 70 | "デバッガーを %m.off_on", 71 | "debugger", 72 | "止める" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "有効", 78 | "無効" 79 | ], 80 | "digital_pin_mode": [ 81 | "入力", 82 | "出力", 83 | "PWM波", 84 | "サーボモーターの回転", 85 | "音声出力", 86 | "超音波の距離測定" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "止める", 94 | "動かす" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_ja2.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "s2a_fm - Arduino", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "デジタルピン %n を %m.digital_pin_mode に %m.pin_state", 9 | "digital_pin_mode_ja", 10 | "1", 11 | "入力", 12 | "使う" 13 | ], 14 | [ 15 | " ", 16 | "アナログピン %n を入力に %m.pin_state", 17 | "analog_pin_mode_ja", 18 | "1", 19 | "使う" 20 | ], 21 | [ 22 | "", 23 | "デジタルピン %n を %m.high_low にする", 24 | "digital_write", 25 | "1", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "PWMピン %n を値 %n にする", 31 | "analog_write", 32 | "1", 33 | "10" 34 | ], 35 | [ 36 | "", 37 | "ピン %n の音を鳴らす: %n Hz, %n ミリ秒", 38 | "play_tone", 39 | "1", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "ピン %n の音を止める", 46 | "tone_off", 47 | "1" 48 | ], 49 | [ 50 | "", 51 | "ピン %n のサーボを %n 度にする", 52 | "set_servo_position", 53 | "1", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "デジタルピン %n の状態", 59 | "digital_read", 60 | "1" 61 | ], 62 | [ 63 | "r", 64 | "アナログピン %n の値", 65 | "analog_read", 66 | "1" 67 | ], 68 | [ 69 | "", 70 | "デバッガーを %m.off_on", 71 | "debugger", 72 | "止める" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "使う", 78 | "使わない" 79 | ], 80 | "digital_pin_mode": [ 81 | "入力", 82 | "出力", 83 | "PWM", 84 | "サーボ", 85 | "音", 86 | "ソナー" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "止める", 94 | "動かす" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_ko.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "스크래치2&아두이노", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "%m.pin_state : 디지털 %n 번핀 / %m.digital_pin_mode 모드", 9 | "digital_pin_mode", 10 | "사용", 11 | "0", 12 | "입력" 13 | ], 14 | [ 15 | " ", 16 | "%m.pin_state : 아날로그 %n 번핀", 17 | "analog_pin_mode", 18 | "사용", 19 | "0" 20 | ], 21 | [ 22 | "", 23 | "디지털 %n 번핀에 %m.high_low 보내기", 24 | "digital_write", 25 | "0", 26 | "0" 27 | ], 28 | [ 29 | "", 30 | "디지털(전류조절) %n 번핀에 %n 보내기", 31 | "analog_write", 32 | "0", 33 | "0" 34 | ], 35 | [ 36 | "", 37 | "신호음 %n 번핀에서 %n 주파수로 %n 동안 재생하기", 38 | "play_tone", 39 | "0", 40 | "HZ", 41 | "ms" 42 | ], 43 | [ 44 | "", 45 | "%n 번핀의 신호음 끄기", 46 | "tone_off", 47 | "0" 48 | ], 49 | [ 50 | "", 51 | "서보모터 %n 번핀을 %n ° 만큼 회전시키기", 52 | "set_servo_position", 53 | "0", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "디지털 %n 번핀 값 읽어오기", 59 | "digital_read", 60 | "0" 61 | ], 62 | [ 63 | "r", 64 | "아날로그 %n 번핀 값 읽어보기", 65 | "analog_read", 66 | "0" 67 | ], 68 | [ 69 | "", 70 | "디버거 %m.off_on", 71 | "debugger", 72 | "끄기" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "사용", 78 | "미사용" 79 | ], 80 | "digital_pin_mode": [ 81 | "입력", 82 | "출력", 83 | "전류조절", 84 | "서보모터", 85 | "신호음", 86 | "음파탐지" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "끄기", 94 | "켜기" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_zh_cn.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "s2a_fm - Scratch to Arduino", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | " 设置 %m.pin_state : 数字引脚 %n 为 %m.digital_pin_mode", 9 | "digital_pin_mode", 10 | "允许", 11 | "引脚号", 12 | "输入" 13 | ], 14 | [ 15 | " ", 16 | "设置 %m.pin_state : 模拟引脚(A) %n 为 输入", 17 | "analog_pin_mode", 18 | "允许", 19 | "引脚号" 20 | ], 21 | [ 22 | "", 23 | "数字脚输出: 设置引脚 %n 为 %m.high_low", 24 | "digital_write", 25 | "引脚号", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "模拟输出 (PWM): 设置引脚 %n 的值为 %n", 31 | "analog_write", 32 | "引脚号", 33 | "数量值" 34 | ], 35 | [ 36 | "", 37 | "在引脚 %n 演奏 ,频率为: %n Hz ,时长为 : %n ms", 38 | "play_tone", 39 | "引脚号", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "关闭引脚 %n 的音调", 46 | "tone_off", 47 | "引脚号" 48 | ], 49 | [ 50 | "", 51 | "设置第 %n 脚为舵机输出 转动角度为 %n", 52 | "set_servo_position", 53 | "引脚号", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "读取数字脚 %n 的值", 59 | "digital_read", 60 | "引脚号" 61 | ], 62 | [ 63 | "r", 64 | "读取模拟脚(A) %n 的值", 65 | "analog_read", 66 | "引脚号" 67 | ], 68 | [ 69 | "", 70 | "调试 %m.off_on", 71 | "debugger", 72 | "关" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "允许", 78 | "禁止" 79 | ], 80 | "digital_pin_mode": [ 81 | "输入", 82 | "输出", 83 | "PWM", 84 | "舵机", 85 | "音调", 86 | "超声波" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "关", 94 | "开" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ExtensionDescriptors/s2a_fm_zh_tw.s2e: -------------------------------------------------------------------------------- 1 | { 2 | "extensionName": "s2a_fm - Scratch to Arduino", 3 | "extensionPort": 50209, 4 | "url": "https://github.com/MrYsLab/PyMata", 5 | "blockSpecs": [ 6 | [ 7 | " ", 8 | "%m.pin_state : 數位腳位 %n 為 %m.digital_pin_mode", 9 | "digital_pin_mode", 10 | "啟用", 11 | "號碼", 12 | "輸入" 13 | ], 14 | [ 15 | " ", 16 | "%m.pin_state : 類比腳位(A) %n 為 輸入", 17 | "analog_pin_mode", 18 | "啟用", 19 | "號碼" 20 | ], 21 | [ 22 | "", 23 | "數位輸出: 設定腳位 %n 為 %m.high_low", 24 | "digital_write", 25 | "號碼", 26 | 0 27 | ], 28 | [ 29 | "", 30 | "模擬類比輸出(PWM): 設定腳位 %n 的值為 %n", 31 | "analog_write", 32 | "號碼", 33 | "數量值" 34 | ], 35 | [ 36 | "", 37 | "在腳位 %n 播放音調, 頻率為: %n Hz, 時間為: %n ms", 38 | "play_tone", 39 | "號碼", 40 | 1000, 41 | 500 42 | ], 43 | [ 44 | "", 45 | "關閉腳位 %n 的音調", 46 | "tone_off", 47 | "號碼" 48 | ], 49 | [ 50 | "", 51 | "設定第 %n 腳位為伺服機輸出 轉動角度為 %n", 52 | "set_servo_position", 53 | "號碼", 54 | 90 55 | ], 56 | [ 57 | "r", 58 | "讀取數位腳位 %n 的值", 59 | "digital_read", 60 | "號碼" 61 | ], 62 | [ 63 | "r", 64 | "讀取類比腳位(A) %n 的值", 65 | "analog_read", 66 | "號碼" 67 | ], 68 | [ 69 | "", 70 | "除錯工具 %m.off_on", 71 | "debugger", 72 | "關" 73 | ] 74 | ], 75 | "menus": { 76 | "pin_state": [ 77 | "啟用", 78 | "停用" 79 | ], 80 | "digital_pin_mode": [ 81 | "輸入", 82 | "輸出", 83 | "PWM", 84 | "伺服機", 85 | "音調", 86 | "超音波" 87 | ], 88 | "high_low": [ 89 | "0", 90 | "1" 91 | ], 92 | "off_on": [ 93 | "關", 94 | "開" 95 | ] 96 | } 97 | } 98 | -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/P4K_Extension.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/P4K_Extension.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/Turn On LED On Pin 6.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/Turn On LED On Pin 6.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_DE.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_DE.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_Es.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_Es.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_Fr.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_Fr.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_GR.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_GR.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_It.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_It.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_Pt.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_Pt.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_ja.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_ja.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_ko.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_ko.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_nl.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_nl.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_zh_cn.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_zh_cn.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/s2a_fm_base_zh_tw.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/s2a_fm_base_zh_tw.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/sonarTest.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/sonarTest.sb2 -------------------------------------------------------------------------------- /ScratchFiles/ScratchProjects/spinning_cat.sb2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/ScratchFiles/ScratchProjects/spinning_cat.sb2 -------------------------------------------------------------------------------- /Snap!Files/Español/blink.xml: -------------------------------------------------------------------------------- 1 | :image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAeAAAAFoCAYAAACPNyggAAACtUlEQVR4nO3BMQEAAADCoPVPbQwfoAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA+Bo3+AAF/RMkcAAAAAElFTkSuQmCC
1datamapmany1data lists
1
1
110i
1
cont
catchtag
cont
catchtag
ActivarActivar 2 | Desactivar5EntradaEntrada 3 | Salida 4 | PWM 5 | Servo 6 | Tono 7 | SONAR
ActivarActivar 8 | Desactivar5
5
5
500 9 | 1
5128
51000500
5
590
NoNo 10 | Si
-------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/PyMata/__init__.py: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/PyMata/pymata_serial.py: -------------------------------------------------------------------------------- 1 | __author__ = 'Copyright (c) 2013 Alan Yorinks All rights reserved.' 2 | """ 3 | Created on Tue Sep 3 07:12:01 2013 4 | 5 | @author: Alan Yorinks 6 | Copyright (c) 2013-14 Alan Yorinks All rights reserved. 7 | 8 | This program is free software; you can redistribute it and/or 9 | modify it under the terms of the GNU General Public 10 | License as published by the Free Software Foundation; either 11 | version 3 of the License, or (at your option) any later version. 12 | 13 | This library is distributed in the hope that it will be useful, 14 | but WITHOUT ANY WARRANTY; without even the implied warranty of 15 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 16 | Lesser General Public License for more details. 17 | 18 | You should have received a copy of the GNU Lesser General Public 19 | License along with this library; if not, write to the Free Software 20 | Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA 21 | """ 22 | 23 | import threading 24 | import serial 25 | 26 | 27 | class PyMataSerial(threading.Thread): 28 | """ 29 | This class manages the serial port for Arduino serial communications 30 | """ 31 | 32 | # class variables 33 | arduino = serial.Serial() 34 | 35 | port_id = "" 36 | baud_rate = 115200 37 | timeout = 1 38 | command_deque = None 39 | 40 | 41 | def __init__(self, port_id, command_deque): 42 | """ 43 | Constructor: 44 | @param command_deque: A reference to the deque shared with the _command_handler 45 | """ 46 | self.port_id = port_id 47 | self.command_deque = command_deque 48 | 49 | threading.Thread.__init__(self) 50 | self.daemon = True 51 | self.arduino = serial.Serial(self.port_id, self.baud_rate, 52 | timeout=int(self.timeout)) 53 | 54 | 55 | def open(self): 56 | """ 57 | open the serial port using the configuration data 58 | returns a reference to this instance 59 | """ 60 | # open a serial port 61 | print '\nOpening Arduino Serial port %s ' % self.port_id 62 | 63 | try: 64 | 65 | # in case the port is already open, let's close it and then 66 | #reopen it 67 | self.arduino.close() 68 | self.arduino.open() 69 | return self.arduino 70 | 71 | except Exception: 72 | # opened failed - will report back to caller 73 | raise 74 | 75 | def close(self): 76 | """ 77 | Close the serial port 78 | return: None 79 | """ 80 | self.arduino.close() 81 | 82 | def write(self, data): 83 | """ 84 | write the data to the serial port 85 | return: None 86 | """ 87 | self.arduino.write(data) 88 | 89 | def run(self): 90 | """ 91 | This method continually runs. If an incoming character is available on the serial port 92 | it is read and placed on the _command_deque 93 | @return: Never Returns 94 | """ 95 | while 1: 96 | # we can get an OSError: [Errno9] Bad file descriptor when shutting down 97 | # just ignore it 98 | try: 99 | if self.arduino.inWaiting(): 100 | c = self.arduino.read() 101 | self.command_deque.append(ord(c)) 102 | except OSError: 103 | pass 104 | 105 | 106 | 107 | 108 | 109 | 110 | 111 | 112 | 113 | 114 | 115 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/log/s2a_fm_debugging.log: -------------------------------------------------------------------------------- 1 | INFO:root:s2a_fm version 1.2 Copyright(C) 2013-14 Alan Yorinks All Rights Reserved 2 | INFO:root:com port = /dev/ttyATH0 3 | INFO:root:30 Total Pins and 12 Analog Pins Found 4 | INFO:root:scratch_http_server.py: keyboard interrupt exception 5 | INFO:root:s2a_fm.py: keyboard interrupt exception 6 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/s2a_fm.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | 3 | # -*- coding: utf-8 -*- 4 | 5 | """ 6 | Created on Wed Nov 25 13:17:15 2013 7 | 8 | @author: Alan Yorinks 9 | Copyright (c) 2013-14 Alan Yorinks All right reserved. 10 | 11 | This program is free software; you can redistribute it and/or 12 | modify it under the terms of the GNU Lesser General Public 13 | License as published by the Free Software Foundation; either 14 | version 2.1 of the License, or (at your option) any later version. 15 | 16 | This library is distributed in the hope that it will be useful, 17 | but WITHOUT ANY WARRANTY; without even the implied warranty of 18 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 19 | Lesser General Public License for more details. 20 | 21 | You should have received a copy of the GNU Lesser General Public 22 | License along with this library; if not, write to the Free Software 23 | Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA 24 | 25 | """ 26 | import os 27 | import sys 28 | import logging 29 | from PyMata.pymata import PyMata 30 | import scratch_http_server 31 | from scratch_command_handlers import ScratchCommandHandlers 32 | import time 33 | 34 | 35 | #noinspection PyBroadException 36 | def s2a_fm(): 37 | 38 | """ 39 | This is the "main" function of the program. 40 | It will instantiate PyMata for communication with an Arduino micro-controller 41 | and the command handlers class. 42 | It will the start the HTTP server to communicate with Scratch 2.0 43 | @return : This is the main loop and should never return 44 | """ 45 | # total number of pins on arduino board 46 | total_pins_discovered = 0 47 | # number of pins that are analog 48 | number_of_analog_pins_discovered = 0 49 | 50 | # make sure we have a log directory and if not, create it. 51 | if not os.path.exists('log'): 52 | os.makedirs('log') 53 | 54 | # turn on logging 55 | logging.basicConfig(filename='./log/s2a_fm_debugging.log', filemode='w', level=logging.DEBUG) 56 | logging.info('s2a_fm version 1.2 Copyright(C) 2013-14 Alan Yorinks All Rights Reserved ') 57 | print 's2a_fm version 1.2 Copyright(C) 2013-14 Alan Yorinks All Rights Reserved ' 58 | 59 | # get the com_port from the command line or default if none given 60 | # if user specified the com port on the command line, use that when invoking PyMata, 61 | # else use '/dev/ttyACM0' 62 | if len(sys.argv) == 2: 63 | com_port= str(sys.argv[1]) 64 | else: 65 | com_port = '/dev/ttyATH0' 66 | logging.info('com port = %s' % com_port) 67 | 68 | try: 69 | # instantiate PyMata 70 | firmata = PyMata(com_port) # pragma: no cover 71 | except Exception: 72 | print 'Could not instantiate PyMata - is your Arduino plugged in?' 73 | logging.exception('Could not instantiate PyMata - is your Arduino plugged in?') 74 | logging.debug("Exiting s2a_fm") 75 | return 76 | 77 | # determine the total number of pins and the number of analog pins for the Arduino 78 | # get the arduino analog pin map 79 | # it will contain an entry for all the pins with non-analog set to firmata.IGNORE 80 | firmata.analog_mapping_query() 81 | 82 | capability_map = firmata.get_analog_mapping_request_results() 83 | 84 | firmata.capability_query() 85 | print "Please wait for Total Arduino Pin Discovery to complete. This can take up to 30 additional seconds." 86 | 87 | # count the pins 88 | for pin in capability_map: 89 | total_pins_discovered += 1 90 | # non analog pins will be marked as IGNORE 91 | if pin != firmata.IGNORE: 92 | number_of_analog_pins_discovered += 1 93 | 94 | # log the number of pins found 95 | logging.info('%d Total Pins and %d Analog Pins Found' % (total_pins_discovered, 96 | number_of_analog_pins_discovered)) 97 | 98 | # instantiate the command handler 99 | scratch_command_handler = ScratchCommandHandlers(firmata, com_port, total_pins_discovered, 100 | number_of_analog_pins_discovered) 101 | 102 | # wait for a maximum of 30 seconds to retrieve the Arduino capability query 103 | start_time = time.time() 104 | 105 | pin_capability = firmata.get_capability_query_results() 106 | while not pin_capability: 107 | if time.time() - start_time > 30: 108 | print '' 109 | print "Could not determine pin capability - exiting." 110 | firmata.close() 111 | # keep sending out a capability query until there is a response 112 | pin_capability = firmata.get_capability_query_results() 113 | time.sleep(.5) 114 | 115 | # we've got the capability, now build a dictionary with pin as the key and a list of all the capabilities 116 | # for the pin as the key's value 117 | pin_list = [] 118 | total_pins_discovered = 0 119 | for entry in pin_capability: 120 | # bump up pin counter each time IGNORE is found 121 | if entry == firmata.IGNORE: 122 | scratch_command_handler.pin_map[total_pins_discovered] = pin_list 123 | total_pins_discovered += 1 124 | pin_list = [] 125 | else: 126 | pin_list.append(entry) 127 | 128 | print "Arduino Total Pin Discovery completed in %d seconds" % (int(time.time() - start_time)) 129 | 130 | 131 | try: 132 | # start the server passing it the handle to PyMata and the command handler. 133 | scratch_http_server.start_server(firmata, scratch_command_handler) 134 | 135 | except Exception: 136 | logging.debug('Exception in s2a_fm.py %s' % str(Exception)) 137 | firmata.close() 138 | return 139 | 140 | except KeyboardInterrupt: 141 | # give control back to the shell that started us 142 | logging.info('s2a_fm.py: keyboard interrupt exception') 143 | firmata.close() 144 | return 145 | 146 | if __name__ == "__main__": 147 | s2a_fm() 148 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/scratch_http_server.py: -------------------------------------------------------------------------------- 1 | # -*- coding: utf-8 -*- 2 | """ 3 | Created on Mon Nov 25 14:45:49 2013 4 | 5 | @author: Alan Yorinks 6 | Copyright (c) 2013-14 Alan Yorinks All right reserved. 7 | 8 | This program is free software; you can redistribute it and/or 9 | modify it under the terms of the GNU Lesser General Public 10 | License as published by the Free Software Foundation; either 11 | version 2.1 of the License, or (at your option) any later version. 12 | 13 | This library is distributed in the hope that it will be useful, 14 | but WITHOUT ANY WARRANTY; without even the implied warranty of 15 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 16 | Lesser General Public License for more details. 17 | 18 | You should have received a copy of the GNU Lesser General Public 19 | License along with this library; if not, write to the Free Software 20 | Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA 21 | """ 22 | 23 | import logging 24 | from BaseHTTPServer import BaseHTTPRequestHandler 25 | from BaseHTTPServer import HTTPServer 26 | from string import split 27 | 28 | 29 | class GetHandler(BaseHTTPRequestHandler): 30 | """ 31 | This class contains the HTTP server that Scratch2 communicates with 32 | Scratch sends HTTP GET requests and this class processes the requests. 33 | 34 | HTTP GET requests are accepted and the appropriate command handler is 35 | called to process the command. 36 | """ 37 | 38 | firmata = None 39 | 40 | # tcp server port - must match that in the .s2e descriptor file 41 | port = 50209 42 | 43 | # instance handle for the scratch command handler 44 | scratch_command_handler = None 45 | 46 | #indicator so that we can tell user Scratch is ready to go 47 | waiting_for_first_scratch_poll = True 48 | 49 | # this is a 'classmethod' because we need to set data before starting 50 | # the HTTP server. 51 | #noinspection PyMethodParameters 52 | @classmethod 53 | def set_items(self, firmata, scratch_command_handler): 54 | """ 55 | This method stores the input parameters for later use. 56 | It is a class method, because these values need to established 57 | prior to instantiating the class 58 | """ 59 | # instance variable for PyMata 60 | #noinspection PyAttributeOutsideInit 61 | self.firmata = firmata 62 | 63 | # instance variable for scratch command handler 64 | #noinspection PyAttributeOutsideInit 65 | self.command_handler = scratch_command_handler 66 | 67 | #noinspection PyPep8Naming 68 | def do_GET(self): 69 | """ 70 | Scratch2 only sends HTTP GET commands. This method processes them. 71 | It differentiates between a "normal" command request and a request 72 | to send policy information to keep Flash happy on Scratch. 73 | (This may change when Scratch is converted to HTML 5 74 | """ 75 | 76 | # skip over the / in the command 77 | cmd = self.path[1:] 78 | # create a list containing the command and all of its parameters 79 | cmd_list = split(cmd, '/') 80 | 81 | # get the command handler method for the command and call the handler 82 | # cmd_list[0] contains the command. look up the command method 83 | s = self.command_handler.do_command(cmd_list) 84 | 85 | self.send_resp(s) 86 | 87 | 88 | # we can't use the standard send_response since we don't conform to its 89 | # standards, so we craft our own response handler here 90 | def send_resp(self, response): 91 | """ 92 | This method sends Scratch an HTTP response to an HTTP GET command. 93 | """ 94 | 95 | crlf = "\r\n" 96 | # http_response = str(response + crlf) 97 | http_response = "HTTP/1.1 200 OK" + crlf 98 | http_response += "Content-Type: text/html; charset=ISO-8859-1" + crlf 99 | http_response += "Content-Length" + str(len(response)) + crlf 100 | http_response += "Access-Control-Allow-Origin: *" + crlf 101 | http_response += crlf 102 | #add the response to the nonsense above 103 | if response != 'okay': 104 | http_response += str(response + crlf) 105 | # send it out the door to Scratch 106 | self.wfile.write(http_response) 107 | 108 | def start_server(firmata, command_handler): 109 | """ 110 | This function populates class variables with essential data and 111 | instantiates the HTTP Server 112 | """ 113 | 114 | GetHandler.set_items(firmata, command_handler) 115 | try: 116 | server = HTTPServer(('192.168.2.189', 50209), GetHandler) 117 | print 'Starting HTTP Server!' 118 | print 'Use to exit the extension\n' 119 | print 'Please start Scratch or Snap!' 120 | except Exception: 121 | logging.debug('Exception in scratch_http_server.py: HTTP Socket may already be in use - restart Scratch') 122 | print 'HTTP Socket may already be in use - restart Scratch' 123 | raise 124 | try: 125 | #start the server 126 | server.serve_forever() 127 | except KeyboardInterrupt: 128 | logging.info('scratch_http_server.py: keyboard interrupt exception') 129 | print "Goodbye !" 130 | raise KeyboardInterrupt 131 | except Exception: 132 | logging.debug('scratch_http_server.py: Exception %s' % str(Exception)) 133 | raise -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/__init__.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | 3 | # portable serial port access with python 4 | # this is a wrapper module for different platform implementations 5 | # 6 | # (C) 2001-2010 Chris Liechti 7 | # this is distributed under a free software license, see license.txt 8 | 9 | VERSION = '2.7' 10 | 11 | import sys 12 | 13 | if sys.platform == 'cli': 14 | from serial.serialcli import * 15 | else: 16 | import os 17 | # chose an implementation, depending on os 18 | if os.name == 'nt': #sys.platform == 'win32': 19 | from serial.serialwin32 import * 20 | elif os.name == 'posix': 21 | from serial.serialposix import * 22 | elif os.name == 'java': 23 | from serial.serialjava import * 24 | else: 25 | raise ImportError("Sorry: no implementation for your platform ('%s') available" % (os.name,)) 26 | 27 | 28 | protocol_handler_packages = [ 29 | 'serial.urlhandler', 30 | ] 31 | 32 | def serial_for_url(url, *args, **kwargs): 33 | """\ 34 | Get an instance of the Serial class, depending on port/url. The port is not 35 | opened when the keyword parameter 'do_not_open' is true, by default it 36 | is. All other parameters are directly passed to the __init__ method when 37 | the port is instantiated. 38 | 39 | The list of package names that is searched for protocol handlers is kept in 40 | ``protocol_handler_packages``. 41 | 42 | e.g. we want to support a URL ``foobar://``. A module 43 | ``my_handlers.protocol_foobar`` is provided by the user. Then 44 | ``protocol_handler_packages.append("my_handlers")`` would extend the search 45 | path so that ``serial_for_url("foobar://"))`` would work. 46 | """ 47 | # check remove extra parameter to not confuse the Serial class 48 | do_open = 'do_not_open' not in kwargs or not kwargs['do_not_open'] 49 | if 'do_not_open' in kwargs: del kwargs['do_not_open'] 50 | # the default is to use the native version 51 | klass = Serial # 'native' implementation 52 | # check port type and get class 53 | try: 54 | url_nocase = url.lower() 55 | except AttributeError: 56 | # it's not a string, use default 57 | pass 58 | else: 59 | if '://' in url_nocase: 60 | protocol = url_nocase.split('://', 1)[0] 61 | for package_name in protocol_handler_packages: 62 | module_name = '%s.protocol_%s' % (package_name, protocol,) 63 | try: 64 | handler_module = __import__(module_name) 65 | except ImportError: 66 | pass 67 | else: 68 | klass = sys.modules[module_name].Serial 69 | break 70 | else: 71 | raise ValueError('invalid URL, protocol %r not known' % (protocol,)) 72 | else: 73 | klass = Serial # 'native' implementation 74 | # instantiate and open when desired 75 | instance = klass(None, *args, **kwargs) 76 | instance.port = url 77 | if do_open: 78 | instance.open() 79 | return instance 80 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/serialcli.py: -------------------------------------------------------------------------------- 1 | #! python 2 | # Python Serial Port Extension for Win32, Linux, BSD, Jython and .NET/Mono 3 | # serial driver for .NET/Mono (IronPython), .NET >= 2 4 | # see __init__.py 5 | # 6 | # (C) 2008 Chris Liechti 7 | # this is distributed under a free software license, see license.txt 8 | 9 | import clr 10 | import System 11 | import System.IO.Ports 12 | from serial.serialutil import * 13 | 14 | 15 | def device(portnum): 16 | """Turn a port number into a device name""" 17 | return System.IO.Ports.SerialPort.GetPortNames()[portnum] 18 | 19 | 20 | # must invoke function with byte array, make a helper to convert strings 21 | # to byte arrays 22 | sab = System.Array[System.Byte] 23 | def as_byte_array(string): 24 | return sab([ord(x) for x in string]) # XXX will require adaption when run with a 3.x compatible IronPython 25 | 26 | class IronSerial(SerialBase): 27 | """Serial port implementation for .NET/Mono.""" 28 | 29 | BAUDRATES = (50, 75, 110, 134, 150, 200, 300, 600, 1200, 1800, 2400, 4800, 30 | 9600, 19200, 38400, 57600, 115200) 31 | 32 | def open(self): 33 | """Open port with current settings. This may throw a SerialException 34 | if the port cannot be opened.""" 35 | if self._port is None: 36 | raise SerialException("Port must be configured before it can be used.") 37 | if self._isOpen: 38 | raise SerialException("Port is already open.") 39 | try: 40 | self._port_handle = System.IO.Ports.SerialPort(self.portstr) 41 | except Exception, msg: 42 | self._port_handle = None 43 | raise SerialException("could not open port %s: %s" % (self.portstr, msg)) 44 | 45 | self._reconfigurePort() 46 | self._port_handle.Open() 47 | self._isOpen = True 48 | if not self._rtscts: 49 | self.setRTS(True) 50 | self.setDTR(True) 51 | self.flushInput() 52 | self.flushOutput() 53 | 54 | def _reconfigurePort(self): 55 | """Set communication parameters on opened port.""" 56 | if not self._port_handle: 57 | raise SerialException("Can only operate on a valid port handle") 58 | 59 | #~ self._port_handle.ReceivedBytesThreshold = 1 60 | 61 | if self._timeout is None: 62 | self._port_handle.ReadTimeout = System.IO.Ports.SerialPort.InfiniteTimeout 63 | else: 64 | self._port_handle.ReadTimeout = int(self._timeout*1000) 65 | 66 | # if self._timeout != 0 and self._interCharTimeout is not None: 67 | # timeouts = (int(self._interCharTimeout * 1000),) + timeouts[1:] 68 | 69 | if self._writeTimeout is None: 70 | self._port_handle.WriteTimeout = System.IO.Ports.SerialPort.InfiniteTimeout 71 | else: 72 | self._port_handle.WriteTimeout = int(self._writeTimeout*1000) 73 | 74 | 75 | # Setup the connection info. 76 | try: 77 | self._port_handle.BaudRate = self._baudrate 78 | except IOError, e: 79 | # catch errors from illegal baudrate settings 80 | raise ValueError(str(e)) 81 | 82 | if self._bytesize == FIVEBITS: 83 | self._port_handle.DataBits = 5 84 | elif self._bytesize == SIXBITS: 85 | self._port_handle.DataBits = 6 86 | elif self._bytesize == SEVENBITS: 87 | self._port_handle.DataBits = 7 88 | elif self._bytesize == EIGHTBITS: 89 | self._port_handle.DataBits = 8 90 | else: 91 | raise ValueError("Unsupported number of data bits: %r" % self._bytesize) 92 | 93 | if self._parity == PARITY_NONE: 94 | self._port_handle.Parity = getattr(System.IO.Ports.Parity, 'None') # reserved keyword in Py3k 95 | elif self._parity == PARITY_EVEN: 96 | self._port_handle.Parity = System.IO.Ports.Parity.Even 97 | elif self._parity == PARITY_ODD: 98 | self._port_handle.Parity = System.IO.Ports.Parity.Odd 99 | elif self._parity == PARITY_MARK: 100 | self._port_handle.Parity = System.IO.Ports.Parity.Mark 101 | elif self._parity == PARITY_SPACE: 102 | self._port_handle.Parity = System.IO.Ports.Parity.Space 103 | else: 104 | raise ValueError("Unsupported parity mode: %r" % self._parity) 105 | 106 | if self._stopbits == STOPBITS_ONE: 107 | self._port_handle.StopBits = System.IO.Ports.StopBits.One 108 | elif self._stopbits == STOPBITS_ONE_POINT_FIVE: 109 | self._port_handle.StopBits = System.IO.Ports.StopBits.OnePointFive 110 | elif self._stopbits == STOPBITS_TWO: 111 | self._port_handle.StopBits = System.IO.Ports.StopBits.Two 112 | else: 113 | raise ValueError("Unsupported number of stop bits: %r" % self._stopbits) 114 | 115 | if self._rtscts and self._xonxoff: 116 | self._port_handle.Handshake = System.IO.Ports.Handshake.RequestToSendXOnXOff 117 | elif self._rtscts: 118 | self._port_handle.Handshake = System.IO.Ports.Handshake.RequestToSend 119 | elif self._xonxoff: 120 | self._port_handle.Handshake = System.IO.Ports.Handshake.XOnXOff 121 | else: 122 | self._port_handle.Handshake = getattr(System.IO.Ports.Handshake, 'None') # reserved keyword in Py3k 123 | 124 | #~ def __del__(self): 125 | #~ self.close() 126 | 127 | def close(self): 128 | """Close port""" 129 | if self._isOpen: 130 | if self._port_handle: 131 | try: 132 | self._port_handle.Close() 133 | except System.IO.Ports.InvalidOperationException: 134 | # ignore errors. can happen for unplugged USB serial devices 135 | pass 136 | self._port_handle = None 137 | self._isOpen = False 138 | 139 | def makeDeviceName(self, port): 140 | try: 141 | return device(port) 142 | except TypeError, e: 143 | raise SerialException(str(e)) 144 | 145 | # - - - - - - - - - - - - - - - - - - - - - - - - 146 | 147 | def inWaiting(self): 148 | """Return the number of characters currently in the input buffer.""" 149 | if not self._port_handle: raise portNotOpenError 150 | return self._port_handle.BytesToRead 151 | 152 | def read(self, size=1): 153 | """Read size bytes from the serial port. If a timeout is set it may 154 | return less characters as requested. With no timeout it will block 155 | until the requested number of bytes is read.""" 156 | if not self._port_handle: raise portNotOpenError 157 | # must use single byte reads as this is the only way to read 158 | # without applying encodings 159 | data = bytearray() 160 | while size: 161 | try: 162 | data.append(self._port_handle.ReadByte()) 163 | except System.TimeoutException, e: 164 | break 165 | else: 166 | size -= 1 167 | return bytes(data) 168 | 169 | def write(self, data): 170 | """Output the given string over the serial port.""" 171 | if not self._port_handle: raise portNotOpenError 172 | if not isinstance(data, (bytes, bytearray)): 173 | raise TypeError('expected %s or bytearray, got %s' % (bytes, type(data))) 174 | try: 175 | # must call overloaded method with byte array argument 176 | # as this is the only one not applying encodings 177 | self._port_handle.Write(as_byte_array(data), 0, len(data)) 178 | except System.TimeoutException, e: 179 | raise writeTimeoutError 180 | return len(data) 181 | 182 | def flushInput(self): 183 | """Clear input buffer, discarding all that is in the buffer.""" 184 | if not self._port_handle: raise portNotOpenError 185 | self._port_handle.DiscardInBuffer() 186 | 187 | def flushOutput(self): 188 | """Clear output buffer, aborting the current output and 189 | discarding all that is in the buffer.""" 190 | if not self._port_handle: raise portNotOpenError 191 | self._port_handle.DiscardOutBuffer() 192 | 193 | def sendBreak(self, duration=0.25): 194 | """Send break condition. Timed, returns to idle state after given duration.""" 195 | if not self._port_handle: raise portNotOpenError 196 | import time 197 | self._port_handle.BreakState = True 198 | time.sleep(duration) 199 | self._port_handle.BreakState = False 200 | 201 | def setBreak(self, level=True): 202 | """Set break: Controls TXD. When active, to transmitting is possible.""" 203 | if not self._port_handle: raise portNotOpenError 204 | self._port_handle.BreakState = bool(level) 205 | 206 | def setRTS(self, level=True): 207 | """Set terminal status line: Request To Send""" 208 | if not self._port_handle: raise portNotOpenError 209 | self._port_handle.RtsEnable = bool(level) 210 | 211 | def setDTR(self, level=True): 212 | """Set terminal status line: Data Terminal Ready""" 213 | if not self._port_handle: raise portNotOpenError 214 | self._port_handle.DtrEnable = bool(level) 215 | 216 | def getCTS(self): 217 | """Read terminal status line: Clear To Send""" 218 | if not self._port_handle: raise portNotOpenError 219 | return self._port_handle.CtsHolding 220 | 221 | def getDSR(self): 222 | """Read terminal status line: Data Set Ready""" 223 | if not self._port_handle: raise portNotOpenError 224 | return self._port_handle.DsrHolding 225 | 226 | def getRI(self): 227 | """Read terminal status line: Ring Indicator""" 228 | if not self._port_handle: raise portNotOpenError 229 | #~ return self._port_handle.XXX 230 | return False #XXX an error would be better 231 | 232 | def getCD(self): 233 | """Read terminal status line: Carrier Detect""" 234 | if not self._port_handle: raise portNotOpenError 235 | return self._port_handle.CDHolding 236 | 237 | # - - platform specific - - - - 238 | # none 239 | 240 | 241 | # assemble Serial class with the platform specific implementation and the base 242 | # for file-like behavior. for Python 2.6 and newer, that provide the new I/O 243 | # library, derive from io.RawIOBase 244 | try: 245 | import io 246 | except ImportError: 247 | # classic version with our own file-like emulation 248 | class Serial(IronSerial, FileLike): 249 | pass 250 | else: 251 | # io library present 252 | class Serial(IronSerial, io.RawIOBase): 253 | pass 254 | 255 | 256 | # Nur Testfunktion!! 257 | if __name__ == '__main__': 258 | import sys 259 | 260 | s = Serial(0) 261 | sys.stdio.write('%s\n' % s) 262 | 263 | s = Serial() 264 | sys.stdio.write('%s\n' % s) 265 | 266 | 267 | s.baudrate = 19200 268 | s.databits = 7 269 | s.close() 270 | s.port = 0 271 | s.open() 272 | sys.stdio.write('%s\n' % s) 273 | 274 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/serialjava.py: -------------------------------------------------------------------------------- 1 | #!jython 2 | # 3 | # Python Serial Port Extension for Win32, Linux, BSD, Jython 4 | # module for serial IO for Jython and JavaComm 5 | # see __init__.py 6 | # 7 | # (C) 2002-2008 Chris Liechti 8 | # this is distributed under a free software license, see license.txt 9 | 10 | from serial.serialutil import * 11 | 12 | def my_import(name): 13 | mod = __import__(name) 14 | components = name.split('.') 15 | for comp in components[1:]: 16 | mod = getattr(mod, comp) 17 | return mod 18 | 19 | 20 | def detect_java_comm(names): 21 | """try given list of modules and return that imports""" 22 | for name in names: 23 | try: 24 | mod = my_import(name) 25 | mod.SerialPort 26 | return mod 27 | except (ImportError, AttributeError): 28 | pass 29 | raise ImportError("No Java Communications API implementation found") 30 | 31 | 32 | # Java Communications API implementations 33 | # http://mho.republika.pl/java/comm/ 34 | 35 | comm = detect_java_comm([ 36 | 'javax.comm', # Sun/IBM 37 | 'gnu.io', # RXTX 38 | ]) 39 | 40 | 41 | def device(portnumber): 42 | """Turn a port number into a device name""" 43 | enum = comm.CommPortIdentifier.getPortIdentifiers() 44 | ports = [] 45 | while enum.hasMoreElements(): 46 | el = enum.nextElement() 47 | if el.getPortType() == comm.CommPortIdentifier.PORT_SERIAL: 48 | ports.append(el) 49 | return ports[portnumber].getName() 50 | 51 | 52 | class JavaSerial(SerialBase): 53 | """Serial port class, implemented with Java Communications API and 54 | thus usable with jython and the appropriate java extension.""" 55 | 56 | def open(self): 57 | """Open port with current settings. This may throw a SerialException 58 | if the port cannot be opened.""" 59 | if self._port is None: 60 | raise SerialException("Port must be configured before it can be used.") 61 | if self._isOpen: 62 | raise SerialException("Port is already open.") 63 | if type(self._port) == type(''): # strings are taken directly 64 | portId = comm.CommPortIdentifier.getPortIdentifier(self._port) 65 | else: 66 | portId = comm.CommPortIdentifier.getPortIdentifier(device(self._port)) # numbers are transformed to a comport id obj 67 | try: 68 | self.sPort = portId.open("python serial module", 10) 69 | except Exception, msg: 70 | self.sPort = None 71 | raise SerialException("Could not open port: %s" % msg) 72 | self._reconfigurePort() 73 | self._instream = self.sPort.getInputStream() 74 | self._outstream = self.sPort.getOutputStream() 75 | self._isOpen = True 76 | 77 | def _reconfigurePort(self): 78 | """Set communication parameters on opened port.""" 79 | if not self.sPort: 80 | raise SerialException("Can only operate on a valid port handle") 81 | 82 | self.sPort.enableReceiveTimeout(30) 83 | if self._bytesize == FIVEBITS: 84 | jdatabits = comm.SerialPort.DATABITS_5 85 | elif self._bytesize == SIXBITS: 86 | jdatabits = comm.SerialPort.DATABITS_6 87 | elif self._bytesize == SEVENBITS: 88 | jdatabits = comm.SerialPort.DATABITS_7 89 | elif self._bytesize == EIGHTBITS: 90 | jdatabits = comm.SerialPort.DATABITS_8 91 | else: 92 | raise ValueError("unsupported bytesize: %r" % self._bytesize) 93 | 94 | if self._stopbits == STOPBITS_ONE: 95 | jstopbits = comm.SerialPort.STOPBITS_1 96 | elif stopbits == STOPBITS_ONE_POINT_FIVE: 97 | self._jstopbits = comm.SerialPort.STOPBITS_1_5 98 | elif self._stopbits == STOPBITS_TWO: 99 | jstopbits = comm.SerialPort.STOPBITS_2 100 | else: 101 | raise ValueError("unsupported number of stopbits: %r" % self._stopbits) 102 | 103 | if self._parity == PARITY_NONE: 104 | jparity = comm.SerialPort.PARITY_NONE 105 | elif self._parity == PARITY_EVEN: 106 | jparity = comm.SerialPort.PARITY_EVEN 107 | elif self._parity == PARITY_ODD: 108 | jparity = comm.SerialPort.PARITY_ODD 109 | elif self._parity == PARITY_MARK: 110 | jparity = comm.SerialPort.PARITY_MARK 111 | elif self._parity == PARITY_SPACE: 112 | jparity = comm.SerialPort.PARITY_SPACE 113 | else: 114 | raise ValueError("unsupported parity type: %r" % self._parity) 115 | 116 | jflowin = jflowout = 0 117 | if self._rtscts: 118 | jflowin |= comm.SerialPort.FLOWCONTROL_RTSCTS_IN 119 | jflowout |= comm.SerialPort.FLOWCONTROL_RTSCTS_OUT 120 | if self._xonxoff: 121 | jflowin |= comm.SerialPort.FLOWCONTROL_XONXOFF_IN 122 | jflowout |= comm.SerialPort.FLOWCONTROL_XONXOFF_OUT 123 | 124 | self.sPort.setSerialPortParams(self._baudrate, jdatabits, jstopbits, jparity) 125 | self.sPort.setFlowControlMode(jflowin | jflowout) 126 | 127 | if self._timeout >= 0: 128 | self.sPort.enableReceiveTimeout(self._timeout*1000) 129 | else: 130 | self.sPort.disableReceiveTimeout() 131 | 132 | def close(self): 133 | """Close port""" 134 | if self._isOpen: 135 | if self.sPort: 136 | self._instream.close() 137 | self._outstream.close() 138 | self.sPort.close() 139 | self.sPort = None 140 | self._isOpen = False 141 | 142 | def makeDeviceName(self, port): 143 | return device(port) 144 | 145 | # - - - - - - - - - - - - - - - - - - - - - - - - 146 | 147 | def inWaiting(self): 148 | """Return the number of characters currently in the input buffer.""" 149 | if not self.sPort: raise portNotOpenError 150 | return self._instream.available() 151 | 152 | def read(self, size=1): 153 | """Read size bytes from the serial port. If a timeout is set it may 154 | return less characters as requested. With no timeout it will block 155 | until the requested number of bytes is read.""" 156 | if not self.sPort: raise portNotOpenError 157 | read = bytearray() 158 | if size > 0: 159 | while len(read) < size: 160 | x = self._instream.read() 161 | if x == -1: 162 | if self.timeout >= 0: 163 | break 164 | else: 165 | read.append(x) 166 | return bytes(read) 167 | 168 | def write(self, data): 169 | """Output the given string over the serial port.""" 170 | if not self.sPort: raise portNotOpenError 171 | if not isinstance(data, (bytes, bytearray)): 172 | raise TypeError('expected %s or bytearray, got %s' % (bytes, type(data))) 173 | self._outstream.write(data) 174 | return len(data) 175 | 176 | def flushInput(self): 177 | """Clear input buffer, discarding all that is in the buffer.""" 178 | if not self.sPort: raise portNotOpenError 179 | self._instream.skip(self._instream.available()) 180 | 181 | def flushOutput(self): 182 | """Clear output buffer, aborting the current output and 183 | discarding all that is in the buffer.""" 184 | if not self.sPort: raise portNotOpenError 185 | self._outstream.flush() 186 | 187 | def sendBreak(self, duration=0.25): 188 | """Send break condition. Timed, returns to idle state after given duration.""" 189 | if not self.sPort: raise portNotOpenError 190 | self.sPort.sendBreak(duration*1000.0) 191 | 192 | def setBreak(self, level=1): 193 | """Set break: Controls TXD. When active, to transmitting is possible.""" 194 | if self.fd is None: raise portNotOpenError 195 | raise SerialException("The setBreak function is not implemented in java.") 196 | 197 | def setRTS(self, level=1): 198 | """Set terminal status line: Request To Send""" 199 | if not self.sPort: raise portNotOpenError 200 | self.sPort.setRTS(level) 201 | 202 | def setDTR(self, level=1): 203 | """Set terminal status line: Data Terminal Ready""" 204 | if not self.sPort: raise portNotOpenError 205 | self.sPort.setDTR(level) 206 | 207 | def getCTS(self): 208 | """Read terminal status line: Clear To Send""" 209 | if not self.sPort: raise portNotOpenError 210 | self.sPort.isCTS() 211 | 212 | def getDSR(self): 213 | """Read terminal status line: Data Set Ready""" 214 | if not self.sPort: raise portNotOpenError 215 | self.sPort.isDSR() 216 | 217 | def getRI(self): 218 | """Read terminal status line: Ring Indicator""" 219 | if not self.sPort: raise portNotOpenError 220 | self.sPort.isRI() 221 | 222 | def getCD(self): 223 | """Read terminal status line: Carrier Detect""" 224 | if not self.sPort: raise portNotOpenError 225 | self.sPort.isCD() 226 | 227 | 228 | # assemble Serial class with the platform specific implementation and the base 229 | # for file-like behavior. for Python 2.6 and newer, that provide the new I/O 230 | # library, derive from io.RawIOBase 231 | try: 232 | import io 233 | except ImportError: 234 | # classic version with our own file-like emulation 235 | class Serial(JavaSerial, FileLike): 236 | pass 237 | else: 238 | # io library present 239 | class Serial(JavaSerial, io.RawIOBase): 240 | pass 241 | 242 | 243 | if __name__ == '__main__': 244 | s = Serial(0, 245 | baudrate=19200, # baudrate 246 | bytesize=EIGHTBITS, # number of databits 247 | parity=PARITY_EVEN, # enable parity checking 248 | stopbits=STOPBITS_ONE, # number of stopbits 249 | timeout=3, # set a timeout value, None for waiting forever 250 | xonxoff=0, # enable software flow control 251 | rtscts=0, # enable RTS/CTS flow control 252 | ) 253 | s.setRTS(1) 254 | s.setDTR(1) 255 | s.flushInput() 256 | s.flushOutput() 257 | s.write('hello') 258 | sys.stdio.write('%r\n' % s.read(5)) 259 | sys.stdio.write('%s\n' % s.inWaiting()) 260 | del s 261 | 262 | 263 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/serialwin32.py: -------------------------------------------------------------------------------- 1 | #! python 2 | # Python Serial Port Extension for Win32, Linux, BSD, Jython 3 | # serial driver for win32 4 | # see __init__.py 5 | # 6 | # (C) 2001-2011 Chris Liechti 7 | # this is distributed under a free software license, see license.txt 8 | # 9 | # Initial patch to use ctypes by Giovanni Bajo 10 | 11 | import ctypes 12 | from serial import win32 13 | 14 | from serial.serialutil import * 15 | 16 | 17 | def device(portnum): 18 | """Turn a port number into a device name""" 19 | return 'COM%d' % (portnum+1) # numbers are transformed to a string 20 | 21 | 22 | class Win32Serial(SerialBase): 23 | """Serial port implementation for Win32 based on ctypes.""" 24 | 25 | BAUDRATES = (50, 75, 110, 134, 150, 200, 300, 600, 1200, 1800, 2400, 4800, 26 | 9600, 19200, 38400, 57600, 115200) 27 | 28 | def __init__(self, *args, **kwargs): 29 | self.hComPort = None 30 | self._overlappedRead = None 31 | self._overlappedWrite = None 32 | self._rtsToggle = False 33 | 34 | self._rtsState = win32.RTS_CONTROL_ENABLE 35 | self._dtrState = win32.DTR_CONTROL_ENABLE 36 | 37 | 38 | SerialBase.__init__(self, *args, **kwargs) 39 | 40 | def open(self): 41 | """Open port with current settings. This may throw a SerialException 42 | if the port cannot be opened.""" 43 | if self._port is None: 44 | raise SerialException("Port must be configured before it can be used.") 45 | if self._isOpen: 46 | raise SerialException("Port is already open.") 47 | # the "\\.\COMx" format is required for devices other than COM1-COM8 48 | # not all versions of windows seem to support this properly 49 | # so that the first few ports are used with the DOS device name 50 | port = self.portstr 51 | try: 52 | if port.upper().startswith('COM') and int(port[3:]) > 8: 53 | port = '\\\\.\\' + port 54 | except ValueError: 55 | # for like COMnotanumber 56 | pass 57 | self.hComPort = win32.CreateFile(port, 58 | win32.GENERIC_READ | win32.GENERIC_WRITE, 59 | 0, # exclusive access 60 | None, # no security 61 | win32.OPEN_EXISTING, 62 | win32.FILE_ATTRIBUTE_NORMAL | win32.FILE_FLAG_OVERLAPPED, 63 | 0) 64 | if self.hComPort == win32.INVALID_HANDLE_VALUE: 65 | self.hComPort = None # 'cause __del__ is called anyway 66 | raise SerialException("could not open port %r: %r" % (self.portstr, ctypes.WinError())) 67 | 68 | try: 69 | self._overlappedRead = win32.OVERLAPPED() 70 | self._overlappedRead.hEvent = win32.CreateEvent(None, 1, 0, None) 71 | self._overlappedWrite = win32.OVERLAPPED() 72 | #~ self._overlappedWrite.hEvent = win32.CreateEvent(None, 1, 0, None) 73 | self._overlappedWrite.hEvent = win32.CreateEvent(None, 0, 0, None) 74 | 75 | # Setup a 4k buffer 76 | win32.SetupComm(self.hComPort, 4096, 4096) 77 | 78 | # Save original timeout values: 79 | self._orgTimeouts = win32.COMMTIMEOUTS() 80 | win32.GetCommTimeouts(self.hComPort, ctypes.byref(self._orgTimeouts)) 81 | 82 | self._reconfigurePort() 83 | 84 | # Clear buffers: 85 | # Remove anything that was there 86 | win32.PurgeComm(self.hComPort, 87 | win32.PURGE_TXCLEAR | win32.PURGE_TXABORT | 88 | win32.PURGE_RXCLEAR | win32.PURGE_RXABORT) 89 | except: 90 | try: 91 | self._close() 92 | except: 93 | # ignore any exception when closing the port 94 | # also to keep original exception that happened when setting up 95 | pass 96 | self.hComPort = None 97 | raise 98 | else: 99 | self._isOpen = True 100 | 101 | 102 | def _reconfigurePort(self): 103 | """Set communication parameters on opened port.""" 104 | if not self.hComPort: 105 | raise SerialException("Can only operate on a valid port handle") 106 | 107 | # Set Windows timeout values 108 | # timeouts is a tuple with the following items: 109 | # (ReadIntervalTimeout,ReadTotalTimeoutMultiplier, 110 | # ReadTotalTimeoutConstant,WriteTotalTimeoutMultiplier, 111 | # WriteTotalTimeoutConstant) 112 | if self._timeout is None: 113 | timeouts = (0, 0, 0, 0, 0) 114 | elif self._timeout == 0: 115 | timeouts = (win32.MAXDWORD, 0, 0, 0, 0) 116 | else: 117 | timeouts = (0, 0, int(self._timeout*1000), 0, 0) 118 | if self._timeout != 0 and self._interCharTimeout is not None: 119 | timeouts = (int(self._interCharTimeout * 1000),) + timeouts[1:] 120 | 121 | if self._writeTimeout is None: 122 | pass 123 | elif self._writeTimeout == 0: 124 | timeouts = timeouts[:-2] + (0, win32.MAXDWORD) 125 | else: 126 | timeouts = timeouts[:-2] + (0, int(self._writeTimeout*1000)) 127 | win32.SetCommTimeouts(self.hComPort, ctypes.byref(win32.COMMTIMEOUTS(*timeouts))) 128 | 129 | win32.SetCommMask(self.hComPort, win32.EV_ERR) 130 | 131 | # Setup the connection info. 132 | # Get state and modify it: 133 | comDCB = win32.DCB() 134 | win32.GetCommState(self.hComPort, ctypes.byref(comDCB)) 135 | comDCB.BaudRate = self._baudrate 136 | 137 | if self._bytesize == FIVEBITS: 138 | comDCB.ByteSize = 5 139 | elif self._bytesize == SIXBITS: 140 | comDCB.ByteSize = 6 141 | elif self._bytesize == SEVENBITS: 142 | comDCB.ByteSize = 7 143 | elif self._bytesize == EIGHTBITS: 144 | comDCB.ByteSize = 8 145 | else: 146 | raise ValueError("Unsupported number of data bits: %r" % self._bytesize) 147 | 148 | if self._parity == PARITY_NONE: 149 | comDCB.Parity = win32.NOPARITY 150 | comDCB.fParity = 0 # Disable Parity Check 151 | elif self._parity == PARITY_EVEN: 152 | comDCB.Parity = win32.EVENPARITY 153 | comDCB.fParity = 1 # Enable Parity Check 154 | elif self._parity == PARITY_ODD: 155 | comDCB.Parity = win32.ODDPARITY 156 | comDCB.fParity = 1 # Enable Parity Check 157 | elif self._parity == PARITY_MARK: 158 | comDCB.Parity = win32.MARKPARITY 159 | comDCB.fParity = 1 # Enable Parity Check 160 | elif self._parity == PARITY_SPACE: 161 | comDCB.Parity = win32.SPACEPARITY 162 | comDCB.fParity = 1 # Enable Parity Check 163 | else: 164 | raise ValueError("Unsupported parity mode: %r" % self._parity) 165 | 166 | if self._stopbits == STOPBITS_ONE: 167 | comDCB.StopBits = win32.ONESTOPBIT 168 | elif self._stopbits == STOPBITS_ONE_POINT_FIVE: 169 | comDCB.StopBits = win32.ONE5STOPBITS 170 | elif self._stopbits == STOPBITS_TWO: 171 | comDCB.StopBits = win32.TWOSTOPBITS 172 | else: 173 | raise ValueError("Unsupported number of stop bits: %r" % self._stopbits) 174 | 175 | comDCB.fBinary = 1 # Enable Binary Transmission 176 | # Char. w/ Parity-Err are replaced with 0xff (if fErrorChar is set to TRUE) 177 | if self._rtscts: 178 | comDCB.fRtsControl = win32.RTS_CONTROL_HANDSHAKE 179 | elif self._rtsToggle: 180 | comDCB.fRtsControl = win32.RTS_CONTROL_TOGGLE 181 | else: 182 | comDCB.fRtsControl = self._rtsState 183 | if self._dsrdtr: 184 | comDCB.fDtrControl = win32.DTR_CONTROL_HANDSHAKE 185 | else: 186 | comDCB.fDtrControl = self._dtrState 187 | 188 | if self._rtsToggle: 189 | comDCB.fOutxCtsFlow = 0 190 | else: 191 | comDCB.fOutxCtsFlow = self._rtscts 192 | comDCB.fOutxDsrFlow = self._dsrdtr 193 | comDCB.fOutX = self._xonxoff 194 | comDCB.fInX = self._xonxoff 195 | comDCB.fNull = 0 196 | comDCB.fErrorChar = 0 197 | comDCB.fAbortOnError = 0 198 | comDCB.XonChar = XON 199 | comDCB.XoffChar = XOFF 200 | 201 | if not win32.SetCommState(self.hComPort, ctypes.byref(comDCB)): 202 | raise ValueError("Cannot configure port, some setting was wrong. Original message: %r" % ctypes.WinError()) 203 | 204 | #~ def __del__(self): 205 | #~ self.close() 206 | 207 | 208 | def _close(self): 209 | """internal close port helper""" 210 | if self.hComPort: 211 | # Restore original timeout values: 212 | win32.SetCommTimeouts(self.hComPort, self._orgTimeouts) 213 | # Close COM-Port: 214 | win32.CloseHandle(self.hComPort) 215 | if self._overlappedRead is not None: 216 | win32.CloseHandle(self._overlappedRead.hEvent) 217 | self._overlappedRead = None 218 | if self._overlappedWrite is not None: 219 | win32.CloseHandle(self._overlappedWrite.hEvent) 220 | self._overlappedWrite = None 221 | self.hComPort = None 222 | 223 | def close(self): 224 | """Close port""" 225 | if self._isOpen: 226 | self._close() 227 | self._isOpen = False 228 | 229 | def makeDeviceName(self, port): 230 | return device(port) 231 | 232 | # - - - - - - - - - - - - - - - - - - - - - - - - 233 | 234 | def inWaiting(self): 235 | """Return the number of characters currently in the input buffer.""" 236 | flags = win32.DWORD() 237 | comstat = win32.COMSTAT() 238 | if not win32.ClearCommError(self.hComPort, ctypes.byref(flags), ctypes.byref(comstat)): 239 | raise SerialException('call to ClearCommError failed') 240 | return comstat.cbInQue 241 | 242 | def read(self, size=1): 243 | """Read size bytes from the serial port. If a timeout is set it may 244 | return less characters as requested. With no timeout it will block 245 | until the requested number of bytes is read.""" 246 | if not self.hComPort: raise portNotOpenError 247 | if size > 0: 248 | win32.ResetEvent(self._overlappedRead.hEvent) 249 | flags = win32.DWORD() 250 | comstat = win32.COMSTAT() 251 | if not win32.ClearCommError(self.hComPort, ctypes.byref(flags), ctypes.byref(comstat)): 252 | raise SerialException('call to ClearCommError failed') 253 | if self.timeout == 0: 254 | n = min(comstat.cbInQue, size) 255 | if n > 0: 256 | buf = ctypes.create_string_buffer(n) 257 | rc = win32.DWORD() 258 | err = win32.ReadFile(self.hComPort, buf, n, ctypes.byref(rc), ctypes.byref(self._overlappedRead)) 259 | if not err and win32.GetLastError() != win32.ERROR_IO_PENDING: 260 | raise SerialException("ReadFile failed (%r)" % ctypes.WinError()) 261 | err = win32.WaitForSingleObject(self._overlappedRead.hEvent, win32.INFINITE) 262 | read = buf.raw[:rc.value] 263 | else: 264 | read = bytes() 265 | else: 266 | buf = ctypes.create_string_buffer(size) 267 | rc = win32.DWORD() 268 | err = win32.ReadFile(self.hComPort, buf, size, ctypes.byref(rc), ctypes.byref(self._overlappedRead)) 269 | if not err and win32.GetLastError() != win32.ERROR_IO_PENDING: 270 | raise SerialException("ReadFile failed (%r)" % ctypes.WinError()) 271 | err = win32.GetOverlappedResult(self.hComPort, ctypes.byref(self._overlappedRead), ctypes.byref(rc), True) 272 | read = buf.raw[:rc.value] 273 | else: 274 | read = bytes() 275 | return bytes(read) 276 | 277 | def write(self, data): 278 | """Output the given string over the serial port.""" 279 | if not self.hComPort: raise portNotOpenError 280 | #~ if not isinstance(data, (bytes, bytearray)): 281 | #~ raise TypeError('expected %s or bytearray, got %s' % (bytes, type(data))) 282 | # convert data (needed in case of memoryview instance: Py 3.1 io lib), ctypes doesn't like memoryview 283 | data = to_bytes(data) 284 | if data: 285 | #~ win32event.ResetEvent(self._overlappedWrite.hEvent) 286 | n = win32.DWORD() 287 | err = win32.WriteFile(self.hComPort, data, len(data), ctypes.byref(n), self._overlappedWrite) 288 | if not err and win32.GetLastError() != win32.ERROR_IO_PENDING: 289 | raise SerialException("WriteFile failed (%r)" % ctypes.WinError()) 290 | if self._writeTimeout != 0: # if blocking (None) or w/ write timeout (>0) 291 | # Wait for the write to complete. 292 | #~ win32.WaitForSingleObject(self._overlappedWrite.hEvent, win32.INFINITE) 293 | err = win32.GetOverlappedResult(self.hComPort, self._overlappedWrite, ctypes.byref(n), True) 294 | if n.value != len(data): 295 | raise writeTimeoutError 296 | return n.value 297 | else: 298 | return 0 299 | 300 | def flush(self): 301 | """Flush of file like objects. In this case, wait until all data 302 | is written.""" 303 | while self.outWaiting(): 304 | time.sleep(0.05) 305 | # XXX could also use WaitCommEvent with mask EV_TXEMPTY, but it would 306 | # require overlapped IO and its also only possible to set a single mask 307 | # on the port--- 308 | 309 | def flushInput(self): 310 | """Clear input buffer, discarding all that is in the buffer.""" 311 | if not self.hComPort: raise portNotOpenError 312 | win32.PurgeComm(self.hComPort, win32.PURGE_RXCLEAR | win32.PURGE_RXABORT) 313 | 314 | def flushOutput(self): 315 | """Clear output buffer, aborting the current output and 316 | discarding all that is in the buffer.""" 317 | if not self.hComPort: raise portNotOpenError 318 | win32.PurgeComm(self.hComPort, win32.PURGE_TXCLEAR | win32.PURGE_TXABORT) 319 | 320 | def sendBreak(self, duration=0.25): 321 | """Send break condition. Timed, returns to idle state after given duration.""" 322 | if not self.hComPort: raise portNotOpenError 323 | import time 324 | win32.SetCommBreak(self.hComPort) 325 | time.sleep(duration) 326 | win32.ClearCommBreak(self.hComPort) 327 | 328 | def setBreak(self, level=1): 329 | """Set break: Controls TXD. When active, to transmitting is possible.""" 330 | if not self.hComPort: raise portNotOpenError 331 | if level: 332 | win32.SetCommBreak(self.hComPort) 333 | else: 334 | win32.ClearCommBreak(self.hComPort) 335 | 336 | def setRTS(self, level=1): 337 | """Set terminal status line: Request To Send""" 338 | # remember level for reconfigure 339 | if level: 340 | self._rtsState = win32.RTS_CONTROL_ENABLE 341 | else: 342 | self._rtsState = win32.RTS_CONTROL_DISABLE 343 | # also apply now if port is open 344 | if self.hComPort: 345 | if level: 346 | win32.EscapeCommFunction(self.hComPort, win32.SETRTS) 347 | else: 348 | win32.EscapeCommFunction(self.hComPort, win32.CLRRTS) 349 | 350 | def setDTR(self, level=1): 351 | """Set terminal status line: Data Terminal Ready""" 352 | # remember level for reconfigure 353 | if level: 354 | self._dtrState = win32.DTR_CONTROL_ENABLE 355 | else: 356 | self._dtrState = win32.DTR_CONTROL_DISABLE 357 | # also apply now if port is open 358 | if self.hComPort: 359 | if level: 360 | win32.EscapeCommFunction(self.hComPort, win32.SETDTR) 361 | else: 362 | win32.EscapeCommFunction(self.hComPort, win32.CLRDTR) 363 | 364 | def _GetCommModemStatus(self): 365 | stat = win32.DWORD() 366 | win32.GetCommModemStatus(self.hComPort, ctypes.byref(stat)) 367 | return stat.value 368 | 369 | def getCTS(self): 370 | """Read terminal status line: Clear To Send""" 371 | if not self.hComPort: raise portNotOpenError 372 | return win32.MS_CTS_ON & self._GetCommModemStatus() != 0 373 | 374 | def getDSR(self): 375 | """Read terminal status line: Data Set Ready""" 376 | if not self.hComPort: raise portNotOpenError 377 | return win32.MS_DSR_ON & self._GetCommModemStatus() != 0 378 | 379 | def getRI(self): 380 | """Read terminal status line: Ring Indicator""" 381 | if not self.hComPort: raise portNotOpenError 382 | return win32.MS_RING_ON & self._GetCommModemStatus() != 0 383 | 384 | def getCD(self): 385 | """Read terminal status line: Carrier Detect""" 386 | if not self.hComPort: raise portNotOpenError 387 | return win32.MS_RLSD_ON & self._GetCommModemStatus() != 0 388 | 389 | # - - platform specific - - - - 390 | 391 | def setBufferSize(self, rx_size=4096, tx_size=None): 392 | """\ 393 | Recommend a buffer size to the driver (device driver can ignore this 394 | vlaue). Must be called before the port is opended. 395 | """ 396 | if tx_size is None: tx_size = rx_size 397 | win32.SetupComm(self.hComPort, rx_size, tx_size) 398 | 399 | def setXON(self, level=True): 400 | """\ 401 | Manually control flow - when software flow control is enabled. 402 | This will send XON (true) and XOFF (false) to the other device. 403 | WARNING: this function is not portable to different platforms! 404 | """ 405 | if not self.hComPort: raise portNotOpenError 406 | if level: 407 | win32.EscapeCommFunction(self.hComPort, win32.SETXON) 408 | else: 409 | win32.EscapeCommFunction(self.hComPort, win32.SETXOFF) 410 | 411 | def outWaiting(self): 412 | """return how many characters the in the outgoing buffer""" 413 | flags = win32.DWORD() 414 | comstat = win32.COMSTAT() 415 | if not win32.ClearCommError(self.hComPort, ctypes.byref(flags), ctypes.byref(comstat)): 416 | raise SerialException('call to ClearCommError failed') 417 | return comstat.cbOutQue 418 | 419 | # functions useful for RS-485 adapters 420 | def setRtsToggle(self, rtsToggle): 421 | """Change RTS toggle control setting.""" 422 | self._rtsToggle = rtsToggle 423 | if self._isOpen: self._reconfigurePort() 424 | 425 | def getRtsToggle(self): 426 | """Get the current RTS toggle control setting.""" 427 | return self._rtsToggle 428 | 429 | rtsToggle = property(getRtsToggle, setRtsToggle, doc="RTS toggle control setting") 430 | 431 | 432 | # assemble Serial class with the platform specific implementation and the base 433 | # for file-like behavior. for Python 2.6 and newer, that provide the new I/O 434 | # library, derive from io.RawIOBase 435 | try: 436 | import io 437 | except ImportError: 438 | # classic version with our own file-like emulation 439 | class Serial(Win32Serial, FileLike): 440 | pass 441 | else: 442 | # io library present 443 | class Serial(Win32Serial, io.RawIOBase): 444 | pass 445 | 446 | 447 | # Nur Testfunktion!! 448 | if __name__ == '__main__': 449 | s = Serial(0) 450 | sys.stdout.write("%s\n" % s) 451 | 452 | s = Serial() 453 | sys.stdout.write("%s\n" % s) 454 | 455 | s.baudrate = 19200 456 | s.databits = 7 457 | s.close() 458 | s.port = 0 459 | s.open() 460 | sys.stdout.write("%s\n" % s) 461 | 462 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/sermsdos.py: -------------------------------------------------------------------------------- 1 | # sermsdos.py 2 | # 3 | # History: 4 | # 5 | # 3rd September 2002 Dave Haynes 6 | # 1. First defined 7 | # 8 | # Although this code should run under the latest versions of 9 | # Python, on DOS-based platforms such as Windows 95 and 98, 10 | # it has been specifically written to be compatible with 11 | # PyDOS, available at: 12 | # http://www.python.org/ftp/python/wpy/dos.html 13 | # 14 | # PyDOS is a stripped-down version of Python 1.5.2 for 15 | # DOS machines. Therefore, in making changes to this file, 16 | # please respect Python 1.5.2 syntax. In addition, please 17 | # limit the width of this file to 60 characters. 18 | # 19 | # Note also that the modules in PyDOS contain fewer members 20 | # than other versions, so we are restricted to using the 21 | # following: 22 | # 23 | # In module os: 24 | # ------------- 25 | # environ, chdir, getcwd, getpid, umask, fdopen, close, 26 | # dup, dup2, fstat, lseek, open, read, write, O_RDONLY, 27 | # O_WRONLY, O_RDWR, O_APPEND, O_CREAT, O_EXCL, O_TRUNC, 28 | # access, F_OK, R_OK, W_OK, X_OK, chmod, listdir, mkdir, 29 | # remove, rename, renames, rmdir, stat, unlink, utime, 30 | # execl, execle, execlp, execlpe, execvp, execvpe, _exit, 31 | # system. 32 | # 33 | # In module os.path: 34 | # ------------------ 35 | # curdir, pardir, sep, altsep, pathsep, defpath, linesep. 36 | # 37 | 38 | import os 39 | import sys 40 | import string 41 | import serial.serialutil 42 | 43 | BAUD_RATES = { 44 | 110: "11", 45 | 150: "15", 46 | 300: "30", 47 | 600: "60", 48 | 1200: "12", 49 | 2400: "24", 50 | 4800: "48", 51 | 9600: "96", 52 | 19200: "19"} 53 | 54 | (PARITY_NONE, PARITY_EVEN, PARITY_ODD, PARITY_MARK, 55 | PARITY_SPACE) = (0, 1, 2, 3, 4) 56 | (STOPBITS_ONE, STOPBITS_ONEANDAHALF, 57 | STOPBITS_TWO) = (1, 1.5, 2) 58 | FIVEBITS, SIXBITS, SEVENBITS, EIGHTBITS = (5, 6, 7, 8) 59 | (RETURN_ERROR, RETURN_BUSY, RETURN_RETRY, RETURN_READY, 60 | RETURN_NONE) = ('E', 'B', 'P', 'R', 'N') 61 | portNotOpenError = ValueError('port not open') 62 | 63 | def device(portnum): 64 | return 'COM%d' % (portnum+1) 65 | 66 | class Serial(serialutil.FileLike): 67 | """ 68 | port: number of device; numbering starts at 69 | zero. if everything fails, the user can 70 | specify a device string, note that this 71 | isn't portable any more 72 | baudrate: baud rate 73 | bytesize: number of databits 74 | parity: enable parity checking 75 | stopbits: number of stopbits 76 | timeout: set a timeout (None for waiting forever) 77 | xonxoff: enable software flow control 78 | rtscts: enable RTS/CTS flow control 79 | retry: DOS retry mode 80 | """ 81 | def __init__(self, 82 | port, 83 | baudrate = 9600, 84 | bytesize = EIGHTBITS, 85 | parity = PARITY_NONE, 86 | stopbits = STOPBITS_ONE, 87 | timeout = None, 88 | xonxoff = 0, 89 | rtscts = 0, 90 | retry = RETURN_RETRY 91 | ): 92 | 93 | if type(port) == type(''): 94 | # strings are taken directly 95 | self.portstr = port 96 | else: 97 | # numbers are transformed to a string 98 | self.portstr = device(port+1) 99 | 100 | self.baud = BAUD_RATES[baudrate] 101 | self.bytesize = str(bytesize) 102 | 103 | if parity == PARITY_NONE: 104 | self.parity = 'N' 105 | elif parity == PARITY_EVEN: 106 | self.parity = 'E' 107 | elif parity == PARITY_ODD: 108 | self.parity = 'O' 109 | elif parity == PARITY_MARK: 110 | self.parity = 'M' 111 | elif parity == PARITY_SPACE: 112 | self.parity = 'S' 113 | 114 | self.stop = str(stopbits) 115 | self.retry = retry 116 | self.filename = "sermsdos.tmp" 117 | 118 | self._config(self.portstr, self.baud, self.parity, 119 | self.bytesize, self.stop, self.retry, self.filename) 120 | 121 | def __del__(self): 122 | self.close() 123 | 124 | def close(self): 125 | pass 126 | 127 | def _config(self, port, baud, parity, data, stop, retry, 128 | filename): 129 | comString = string.join(("MODE ", port, ":" 130 | , " BAUD= ", baud, " PARITY= ", parity 131 | , " DATA= ", data, " STOP= ", stop, " RETRY= ", 132 | retry, " > ", filename ), '') 133 | os.system(comString) 134 | 135 | def setBaudrate(self, baudrate): 136 | self._config(self.portstr, BAUD_RATES[baudrate], 137 | self.parity, self.bytesize, self.stop, self.retry, 138 | self.filename) 139 | 140 | def inWaiting(self): 141 | """returns the number of bytes waiting to be read""" 142 | raise NotImplementedError 143 | 144 | def read(self, num = 1): 145 | """Read num bytes from serial port""" 146 | handle = os.open(self.portstr, 147 | os.O_RDONLY | os.O_BINARY) 148 | rv = os.read(handle, num) 149 | os.close(handle) 150 | return rv 151 | 152 | def write(self, s): 153 | """Write string to serial port""" 154 | handle = os.open(self.portstr, 155 | os.O_WRONLY | os.O_BINARY) 156 | rv = os.write(handle, s) 157 | os.close(handle) 158 | return rv 159 | 160 | def flushInput(self): 161 | raise NotImplementedError 162 | 163 | def flushOutput(self): 164 | raise NotImplementedError 165 | 166 | def sendBreak(self): 167 | raise NotImplementedError 168 | 169 | def setRTS(self,level=1): 170 | """Set terminal status line""" 171 | raise NotImplementedError 172 | 173 | def setDTR(self,level=1): 174 | """Set terminal status line""" 175 | raise NotImplementedError 176 | 177 | def getCTS(self): 178 | """Eead terminal status line""" 179 | raise NotImplementedError 180 | 181 | def getDSR(self): 182 | """Eead terminal status line""" 183 | raise NotImplementedError 184 | 185 | def getRI(self): 186 | """Eead terminal status line""" 187 | raise NotImplementedError 188 | 189 | def getCD(self): 190 | """Eead terminal status line""" 191 | raise NotImplementedError 192 | 193 | def __repr__(self): 194 | return string.join(( ": ", self.portstr 195 | , self.baud, self.parity, self.bytesize, self.stop, 196 | self.retry , self.filename), ' ') 197 | 198 | if __name__ == '__main__': 199 | s = Serial(0) 200 | sys.stdio.write('%s %s\n' % (__name__, s)) 201 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/tools/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/Snap!Files/Snap!Mobile/arduino/serial/tools/__init__.py -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/tools/list_ports.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | 3 | # portable serial port access with python 4 | # this is a wrapper module for different platform implementations of the 5 | # port enumeration feature 6 | # 7 | # (C) 2011-2013 Chris Liechti 8 | # this is distributed under a free software license, see license.txt 9 | 10 | """\ 11 | This module will provide a function called comports that returns an 12 | iterable (generator or list) that will enumerate available com ports. Note that 13 | on some systems non-existent ports may be listed. 14 | 15 | Additionally a grep function is supplied that can be used to search for ports 16 | based on their descriptions or hardware ID. 17 | """ 18 | 19 | import sys, os, re 20 | 21 | # chose an implementation, depending on os 22 | #~ if sys.platform == 'cli': 23 | #~ else: 24 | import os 25 | # chose an implementation, depending on os 26 | if os.name == 'nt': #sys.platform == 'win32': 27 | from serial.tools.list_ports_windows import * 28 | elif os.name == 'posix': 29 | from serial.tools.list_ports_posix import * 30 | #~ elif os.name == 'java': 31 | else: 32 | raise ImportError("Sorry: no implementation for your platform ('%s') available" % (os.name,)) 33 | 34 | # - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 35 | 36 | def grep(regexp): 37 | """\ 38 | Search for ports using a regular expression. Port name, description and 39 | hardware ID are searched. The function returns an iterable that returns the 40 | same tuples as comport() would do. 41 | """ 42 | r = re.compile(regexp, re.I) 43 | for port, desc, hwid in comports(): 44 | if r.search(port) or r.search(desc) or r.search(hwid): 45 | yield port, desc, hwid 46 | 47 | 48 | # - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 49 | def main(): 50 | import optparse 51 | 52 | parser = optparse.OptionParser( 53 | usage = "%prog [options] []", 54 | description = "Miniterm - A simple terminal program for the serial port." 55 | ) 56 | 57 | parser.add_option("--debug", 58 | help="print debug messages and tracebacks (development mode)", 59 | dest="debug", 60 | default=False, 61 | action='store_true') 62 | 63 | parser.add_option("-v", "--verbose", 64 | help="show more messages (can be given multiple times)", 65 | dest="verbose", 66 | default=1, 67 | action='count') 68 | 69 | parser.add_option("-q", "--quiet", 70 | help="suppress all messages", 71 | dest="verbose", 72 | action='store_const', 73 | const=0) 74 | 75 | (options, args) = parser.parse_args() 76 | 77 | 78 | hits = 0 79 | # get iteraror w/ or w/o filter 80 | if args: 81 | if len(args) > 1: 82 | parser.error('more than one regexp not supported') 83 | print "Filtered list with regexp: %r" % (args[0],) 84 | iterator = sorted(grep(args[0])) 85 | else: 86 | iterator = sorted(comports()) 87 | # list them 88 | for port, desc, hwid in iterator: 89 | print("%-20s" % (port,)) 90 | if options.verbose > 1: 91 | print(" desc: %s" % (desc,)) 92 | print(" hwid: %s" % (hwid,)) 93 | hits += 1 94 | if options.verbose: 95 | if hits: 96 | print("%d ports found" % (hits,)) 97 | else: 98 | print("no ports found") 99 | 100 | # - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 101 | # test 102 | if __name__ == '__main__': 103 | main() 104 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/tools/list_ports_linux.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | 3 | # portable serial port access with python 4 | # 5 | # This is a module that gathers a list of serial ports including details on 6 | # GNU/Linux systems 7 | # 8 | # (C) 2011-2013 Chris Liechti 9 | # this is distributed under a free software license, see license.txt 10 | 11 | import glob 12 | import sys 13 | import os 14 | import re 15 | 16 | try: 17 | import subprocess 18 | except ImportError: 19 | def popen(argv): 20 | try: 21 | si, so = os.popen4(' '.join(argv)) 22 | return so.read().strip() 23 | except: 24 | raise IOError('lsusb failed') 25 | else: 26 | def popen(argv): 27 | try: 28 | return subprocess.check_output(argv, stderr=subprocess.STDOUT).strip() 29 | except: 30 | raise IOError('lsusb failed') 31 | 32 | 33 | # The comports function is expected to return an iterable that yields tuples of 34 | # 3 strings: port name, human readable description and a hardware ID. 35 | # 36 | # as currently no method is known to get the second two strings easily, they 37 | # are currently just identical to the port name. 38 | 39 | # try to detect the OS so that a device can be selected... 40 | plat = sys.platform.lower() 41 | 42 | def read_line(filename): 43 | """help function to read a single line from a file. returns none""" 44 | try: 45 | f = open(filename) 46 | line = f.readline().strip() 47 | f.close() 48 | return line 49 | except IOError: 50 | return None 51 | 52 | def re_group(regexp, text): 53 | """search for regexp in text, return 1st group on match""" 54 | if sys.version < '3': 55 | m = re.search(regexp, text) 56 | else: 57 | # text is bytes-like 58 | m = re.search(regexp, text.decode('ascii', 'replace')) 59 | if m: return m.group(1) 60 | 61 | 62 | # try to extract descriptions from sysfs. this was done by experimenting, 63 | # no guarantee that it works for all devices or in the future... 64 | 65 | def usb_sysfs_hw_string(sysfs_path): 66 | """given a path to a usb device in sysfs, return a string describing it""" 67 | bus, dev = os.path.basename(os.path.realpath(sysfs_path)).split('-') 68 | snr = read_line(sysfs_path+'/serial') 69 | if snr: 70 | snr_txt = ' SNR=%s' % (snr,) 71 | else: 72 | snr_txt = '' 73 | return 'USB VID:PID=%s:%s%s' % ( 74 | read_line(sysfs_path+'/idVendor'), 75 | read_line(sysfs_path+'/idProduct'), 76 | snr_txt 77 | ) 78 | 79 | def usb_lsusb_string(sysfs_path): 80 | base = os.path.basename(os.path.realpath(sysfs_path)) 81 | bus = base.split('-')[0] 82 | try: 83 | dev = int(read_line(os.path.join(sysfs_path, 'devnum'))) 84 | desc = popen(['lsusb', '-v', '-s', '%s:%s' % (bus, dev)]) 85 | # descriptions from device 86 | iManufacturer = re_group('iManufacturer\s+\w+ (.+)', desc) 87 | iProduct = re_group('iProduct\s+\w+ (.+)', desc) 88 | iSerial = re_group('iSerial\s+\w+ (.+)', desc) or '' 89 | # descriptions from kernel 90 | idVendor = re_group('idVendor\s+0x\w+ (.+)', desc) 91 | idProduct = re_group('idProduct\s+0x\w+ (.+)', desc) 92 | # create descriptions. prefer text from device, fall back to the others 93 | return '%s %s %s' % (iManufacturer or idVendor, iProduct or idProduct, iSerial) 94 | except IOError: 95 | return base 96 | 97 | def describe(device): 98 | """\ 99 | Get a human readable description. 100 | For USB-Serial devices try to run lsusb to get a human readable description. 101 | For USB-CDC devices read the description from sysfs. 102 | """ 103 | base = os.path.basename(device) 104 | # USB-Serial devices 105 | sys_dev_path = '/sys/class/tty/%s/device/driver/%s' % (base, base) 106 | if os.path.exists(sys_dev_path): 107 | sys_usb = os.path.dirname(os.path.dirname(os.path.realpath(sys_dev_path))) 108 | return usb_lsusb_string(sys_usb) 109 | # USB-CDC devices 110 | sys_dev_path = '/sys/class/tty/%s/device/interface' % (base,) 111 | if os.path.exists(sys_dev_path): 112 | return read_line(sys_dev_path) 113 | return base 114 | 115 | def hwinfo(device): 116 | """Try to get a HW identification using sysfs""" 117 | base = os.path.basename(device) 118 | if os.path.exists('/sys/class/tty/%s/device' % (base,)): 119 | # PCI based devices 120 | sys_id_path = '/sys/class/tty/%s/device/id' % (base,) 121 | if os.path.exists(sys_id_path): 122 | return read_line(sys_id_path) 123 | # USB-Serial devices 124 | sys_dev_path = '/sys/class/tty/%s/device/driver/%s' % (base, base) 125 | if os.path.exists(sys_dev_path): 126 | sys_usb = os.path.dirname(os.path.dirname(os.path.realpath(sys_dev_path))) 127 | return usb_sysfs_hw_string(sys_usb) 128 | # USB-CDC devices 129 | if base.startswith('ttyACM'): 130 | sys_dev_path = '/sys/class/tty/%s/device' % (base,) 131 | if os.path.exists(sys_dev_path): 132 | return usb_sysfs_hw_string(sys_dev_path + '/..') 133 | return 'n/a' # XXX directly remove these from the list? 134 | 135 | def comports(): 136 | devices = glob.glob('/dev/ttyS*') + glob.glob('/dev/ttyUSB*') + glob.glob('/dev/ttyACM*') 137 | return [(d, describe(d), hwinfo(d)) for d in devices] 138 | 139 | # - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 140 | # test 141 | if __name__ == '__main__': 142 | for port, desc, hwid in sorted(comports()): 143 | print "%s: %s [%s]" % (port, desc, hwid) 144 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/tools/list_ports_osx.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | 3 | # portable serial port access with python 4 | # 5 | # This is a module that gathers a list of serial ports including details on OSX 6 | # 7 | # code originally from https://github.com/makerbot/pyserial/tree/master/serial/tools 8 | # with contributions from cibomahto, dgs3, FarMcKon, tedbrandston 9 | # and modifications by cliechti 10 | # 11 | # this is distributed under a free software license, see license.txt 12 | 13 | 14 | 15 | # List all of the callout devices in OS/X by querying IOKit. 16 | 17 | # See the following for a reference of how to do this: 18 | # http://developer.apple.com/library/mac/#documentation/DeviceDrivers/Conceptual/WorkingWSerial/WWSerial_SerialDevs/SerialDevices.html#//apple_ref/doc/uid/TP30000384-CIHGEAFD 19 | 20 | # More help from darwin_hid.py 21 | 22 | # Also see the 'IORegistryExplorer' for an idea of what we are actually searching 23 | 24 | import ctypes 25 | from ctypes import util 26 | import re 27 | 28 | iokit = ctypes.cdll.LoadLibrary(ctypes.util.find_library('IOKit')) 29 | cf = ctypes.cdll.LoadLibrary(ctypes.util.find_library('CoreFoundation')) 30 | 31 | kIOMasterPortDefault = ctypes.c_void_p.in_dll(iokit, "kIOMasterPortDefault") 32 | kCFAllocatorDefault = ctypes.c_void_p.in_dll(cf, "kCFAllocatorDefault") 33 | 34 | kCFStringEncodingMacRoman = 0 35 | 36 | iokit.IOServiceMatching.restype = ctypes.c_void_p 37 | 38 | iokit.IOServiceGetMatchingServices.argtypes = [ctypes.c_void_p, ctypes.c_void_p, ctypes.c_void_p] 39 | iokit.IOServiceGetMatchingServices.restype = ctypes.c_void_p 40 | 41 | iokit.IORegistryEntryGetParentEntry.argtypes = [ctypes.c_void_p, ctypes.c_void_p, ctypes.c_void_p] 42 | 43 | iokit.IORegistryEntryCreateCFProperty.argtypes = [ctypes.c_void_p, ctypes.c_void_p, ctypes.c_void_p, ctypes.c_uint32] 44 | iokit.IORegistryEntryCreateCFProperty.restype = ctypes.c_void_p 45 | 46 | iokit.IORegistryEntryGetPath.argtypes = [ctypes.c_void_p, ctypes.c_void_p, ctypes.c_void_p] 47 | iokit.IORegistryEntryGetPath.restype = ctypes.c_void_p 48 | 49 | iokit.IORegistryEntryGetName.argtypes = [ctypes.c_void_p, ctypes.c_void_p] 50 | iokit.IORegistryEntryGetName.restype = ctypes.c_void_p 51 | 52 | iokit.IOObjectGetClass.argtypes = [ctypes.c_void_p, ctypes.c_void_p] 53 | iokit.IOObjectGetClass.restype = ctypes.c_void_p 54 | 55 | iokit.IOObjectRelease.argtypes = [ctypes.c_void_p] 56 | 57 | 58 | cf.CFStringCreateWithCString.argtypes = [ctypes.c_void_p, ctypes.c_char_p, ctypes.c_int32] 59 | cf.CFStringCreateWithCString.restype = ctypes.c_void_p 60 | 61 | cf.CFStringGetCStringPtr.argtypes = [ctypes.c_void_p, ctypes.c_uint32] 62 | cf.CFStringGetCStringPtr.restype = ctypes.c_char_p 63 | 64 | cf.CFNumberGetValue.argtypes = [ctypes.c_void_p, ctypes.c_uint32, ctypes.c_void_p] 65 | cf.CFNumberGetValue.restype = ctypes.c_void_p 66 | 67 | def get_string_property(device_t, property): 68 | """ Search the given device for the specified string property 69 | 70 | @param device_t Device to search 71 | @param property String to search for. 72 | @return Python string containing the value, or None if not found. 73 | """ 74 | key = cf.CFStringCreateWithCString( 75 | kCFAllocatorDefault, 76 | property.encode("mac_roman"), 77 | kCFStringEncodingMacRoman 78 | ) 79 | 80 | CFContainer = iokit.IORegistryEntryCreateCFProperty( 81 | device_t, 82 | key, 83 | kCFAllocatorDefault, 84 | 0 85 | ); 86 | 87 | output = None 88 | 89 | if CFContainer: 90 | output = cf.CFStringGetCStringPtr(CFContainer, 0) 91 | 92 | return output 93 | 94 | def get_int_property(device_t, property): 95 | """ Search the given device for the specified string property 96 | 97 | @param device_t Device to search 98 | @param property String to search for. 99 | @return Python string containing the value, or None if not found. 100 | """ 101 | key = cf.CFStringCreateWithCString( 102 | kCFAllocatorDefault, 103 | property.encode("mac_roman"), 104 | kCFStringEncodingMacRoman 105 | ) 106 | 107 | CFContainer = iokit.IORegistryEntryCreateCFProperty( 108 | device_t, 109 | key, 110 | kCFAllocatorDefault, 111 | 0 112 | ); 113 | 114 | number = ctypes.c_uint16() 115 | 116 | if CFContainer: 117 | output = cf.CFNumberGetValue(CFContainer, 2, ctypes.byref(number)) 118 | 119 | return number.value 120 | 121 | def IORegistryEntryGetName(device): 122 | pathname = ctypes.create_string_buffer(100) # TODO: Is this ok? 123 | iokit.IOObjectGetClass( 124 | device, 125 | ctypes.byref(pathname) 126 | ) 127 | 128 | return pathname.value 129 | 130 | def GetParentDeviceByType(device, parent_type): 131 | """ Find the first parent of a device that implements the parent_type 132 | @param IOService Service to inspect 133 | @return Pointer to the parent type, or None if it was not found. 134 | """ 135 | # First, try to walk up the IOService tree to find a parent of this device that is a IOUSBDevice. 136 | while IORegistryEntryGetName(device) != parent_type: 137 | parent = ctypes.c_void_p() 138 | response = iokit.IORegistryEntryGetParentEntry( 139 | device, 140 | "IOService".encode("mac_roman"), 141 | ctypes.byref(parent) 142 | ) 143 | 144 | # If we weren't able to find a parent for the device, we're done. 145 | if response != 0: 146 | return None 147 | 148 | device = parent 149 | 150 | return device 151 | 152 | def GetIOServicesByType(service_type): 153 | """ 154 | """ 155 | serial_port_iterator = ctypes.c_void_p() 156 | 157 | response = iokit.IOServiceGetMatchingServices( 158 | kIOMasterPortDefault, 159 | iokit.IOServiceMatching(service_type), 160 | ctypes.byref(serial_port_iterator) 161 | ) 162 | 163 | services = [] 164 | while iokit.IOIteratorIsValid(serial_port_iterator): 165 | service = iokit.IOIteratorNext(serial_port_iterator) 166 | if not service: 167 | break 168 | services.append(service) 169 | 170 | iokit.IOObjectRelease(serial_port_iterator) 171 | 172 | return services 173 | 174 | def comports(): 175 | # Scan for all iokit serial ports 176 | services = GetIOServicesByType('IOSerialBSDClient') 177 | 178 | ports = [] 179 | for service in services: 180 | info = [] 181 | 182 | # First, add the callout device file. 183 | info.append(get_string_property(service, "IOCalloutDevice")) 184 | 185 | # If the serial port is implemented by a 186 | usb_device = GetParentDeviceByType(service, "IOUSBDevice") 187 | if usb_device != None: 188 | info.append(get_string_property(usb_device, "USB Product Name")) 189 | 190 | info.append( 191 | "USB VID:PID=%x:%x SNR=%s"%( 192 | get_int_property(usb_device, "idVendor"), 193 | get_int_property(usb_device, "idProduct"), 194 | get_string_property(usb_device, "USB Serial Number")) 195 | ) 196 | else: 197 | info.append('n/a') 198 | info.append('n/a') 199 | 200 | ports.append(info) 201 | 202 | return ports 203 | 204 | # test 205 | if __name__ == '__main__': 206 | for port, desc, hwid in sorted(comports()): 207 | print "%s: %s [%s]" % (port, desc, hwid) 208 | 209 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/tools/list_ports_posix.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | 3 | # portable serial port access with python 4 | 5 | # This is a module that gathers a list of serial ports on POSIXy systems. 6 | # For some specific implementations, see also list_ports_linux, list_ports_osx 7 | # 8 | # this is a wrapper module for different platform implementations of the 9 | # port enumeration feature 10 | # 11 | # (C) 2011-2013 Chris Liechti 12 | # this is distributed under a free software license, see license.txt 13 | 14 | """\ 15 | The ``comports`` function is expected to return an iterable that yields tuples 16 | of 3 strings: port name, human readable description and a hardware ID. 17 | 18 | As currently no method is known to get the second two strings easily, they are 19 | currently just identical to the port name. 20 | """ 21 | 22 | import glob 23 | import sys 24 | import os 25 | 26 | # try to detect the OS so that a device can be selected... 27 | plat = sys.platform.lower() 28 | 29 | if plat[:5] == 'linux': # Linux (confirmed) 30 | from serial.tools.list_ports_linux import comports 31 | 32 | elif plat == 'cygwin': # cygwin/win32 33 | def comports(): 34 | devices = glob.glob('/dev/com*') 35 | return [(d, d, d) for d in devices] 36 | 37 | elif plat[:7] == 'openbsd': # OpenBSD 38 | def comports(): 39 | devices = glob.glob('/dev/cua*') 40 | return [(d, d, d) for d in devices] 41 | 42 | elif plat[:3] == 'bsd' or \ 43 | plat[:7] == 'freebsd': 44 | 45 | def comports(): 46 | devices = glob.glob('/dev/cuad*') 47 | return [(d, d, d) for d in devices] 48 | 49 | elif plat[:6] == 'darwin': # OS X (confirmed) 50 | from serial.tools.list_ports_osx import comports 51 | 52 | elif plat[:6] == 'netbsd': # NetBSD 53 | def comports(): 54 | """scan for available ports. return a list of device names.""" 55 | devices = glob.glob('/dev/dty*') 56 | return [(d, d, d) for d in devices] 57 | 58 | elif plat[:4] == 'irix': # IRIX 59 | def comports(): 60 | """scan for available ports. return a list of device names.""" 61 | devices = glob.glob('/dev/ttyf*') 62 | return [(d, d, d) for d in devices] 63 | 64 | elif plat[:2] == 'hp': # HP-UX (not tested) 65 | def comports(): 66 | """scan for available ports. return a list of device names.""" 67 | devices = glob.glob('/dev/tty*p0') 68 | return [(d, d, d) for d in devices] 69 | 70 | elif plat[:5] == 'sunos': # Solaris/SunOS 71 | def comports(): 72 | """scan for available ports. return a list of device names.""" 73 | devices = glob.glob('/dev/tty*c') 74 | return [(d, d, d) for d in devices] 75 | 76 | elif plat[:3] == 'aix': # AIX 77 | def comports(): 78 | """scan for available ports. return a list of device names.""" 79 | devices = glob.glob('/dev/tty*') 80 | return [(d, d, d) for d in devices] 81 | 82 | else: 83 | # platform detection has failed... 84 | sys.stderr.write("""\ 85 | don't know how to enumerate ttys on this system. 86 | ! I you know how the serial ports are named send this information to 87 | ! the author of this module: 88 | 89 | sys.platform = %r 90 | os.name = %r 91 | pySerial version = %s 92 | 93 | also add the naming scheme of the serial ports and with a bit luck you can get 94 | this module running... 95 | """ % (sys.platform, os.name, serial.VERSION)) 96 | raise ImportError("Sorry: no implementation for your platform ('%s') available" % (os.name,)) 97 | 98 | # test 99 | if __name__ == '__main__': 100 | for port, desc, hwid in sorted(comports()): 101 | print "%s: %s [%s]" % (port, desc, hwid) 102 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/tools/list_ports_windows.py: -------------------------------------------------------------------------------- 1 | import ctypes 2 | import re 3 | 4 | def ValidHandle(value, func, arguments): 5 | if value == 0: 6 | raise ctypes.WinError() 7 | return value 8 | 9 | import serial 10 | from serial.win32 import ULONG_PTR, is_64bit 11 | from ctypes.wintypes import HANDLE 12 | from ctypes.wintypes import BOOL 13 | from ctypes.wintypes import HWND 14 | from ctypes.wintypes import DWORD 15 | from ctypes.wintypes import WORD 16 | from ctypes.wintypes import LONG 17 | from ctypes.wintypes import ULONG 18 | from ctypes.wintypes import LPCSTR 19 | from ctypes.wintypes import HKEY 20 | from ctypes.wintypes import BYTE 21 | 22 | NULL = 0 23 | HDEVINFO = ctypes.c_void_p 24 | PCTSTR = ctypes.c_char_p 25 | PTSTR = ctypes.c_void_p 26 | CHAR = ctypes.c_char 27 | LPDWORD = PDWORD = ctypes.POINTER(DWORD) 28 | #~ LPBYTE = PBYTE = ctypes.POINTER(BYTE) 29 | LPBYTE = PBYTE = ctypes.c_void_p # XXX avoids error about types 30 | 31 | ACCESS_MASK = DWORD 32 | REGSAM = ACCESS_MASK 33 | 34 | 35 | def byte_buffer(length): 36 | """Get a buffer for a string""" 37 | return (BYTE*length)() 38 | 39 | def string(buffer): 40 | s = [] 41 | for c in buffer: 42 | if c == 0: break 43 | s.append(chr(c & 0xff)) # "& 0xff": hack to convert signed to unsigned 44 | return ''.join(s) 45 | 46 | 47 | class GUID(ctypes.Structure): 48 | _fields_ = [ 49 | ('Data1', DWORD), 50 | ('Data2', WORD), 51 | ('Data3', WORD), 52 | ('Data4', BYTE*8), 53 | ] 54 | def __str__(self): 55 | return "{%08x-%04x-%04x-%s-%s}" % ( 56 | self.Data1, 57 | self.Data2, 58 | self.Data3, 59 | ''.join(["%02x" % d for d in self.Data4[:2]]), 60 | ''.join(["%02x" % d for d in self.Data4[2:]]), 61 | ) 62 | 63 | class SP_DEVINFO_DATA(ctypes.Structure): 64 | _fields_ = [ 65 | ('cbSize', DWORD), 66 | ('ClassGuid', GUID), 67 | ('DevInst', DWORD), 68 | ('Reserved', ULONG_PTR), 69 | ] 70 | def __str__(self): 71 | return "ClassGuid:%s DevInst:%s" % (self.ClassGuid, self.DevInst) 72 | PSP_DEVINFO_DATA = ctypes.POINTER(SP_DEVINFO_DATA) 73 | 74 | PSP_DEVICE_INTERFACE_DETAIL_DATA = ctypes.c_void_p 75 | 76 | setupapi = ctypes.windll.LoadLibrary("setupapi") 77 | SetupDiDestroyDeviceInfoList = setupapi.SetupDiDestroyDeviceInfoList 78 | SetupDiDestroyDeviceInfoList.argtypes = [HDEVINFO] 79 | SetupDiDestroyDeviceInfoList.restype = BOOL 80 | 81 | SetupDiClassGuidsFromName = setupapi.SetupDiClassGuidsFromNameA 82 | SetupDiClassGuidsFromName.argtypes = [PCTSTR, ctypes.POINTER(GUID), DWORD, PDWORD] 83 | SetupDiClassGuidsFromName.restype = BOOL 84 | 85 | SetupDiEnumDeviceInfo = setupapi.SetupDiEnumDeviceInfo 86 | SetupDiEnumDeviceInfo.argtypes = [HDEVINFO, DWORD, PSP_DEVINFO_DATA] 87 | SetupDiEnumDeviceInfo.restype = BOOL 88 | 89 | SetupDiGetClassDevs = setupapi.SetupDiGetClassDevsA 90 | SetupDiGetClassDevs.argtypes = [ctypes.POINTER(GUID), PCTSTR, HWND, DWORD] 91 | SetupDiGetClassDevs.restype = HDEVINFO 92 | SetupDiGetClassDevs.errcheck = ValidHandle 93 | 94 | SetupDiGetDeviceRegistryProperty = setupapi.SetupDiGetDeviceRegistryPropertyA 95 | SetupDiGetDeviceRegistryProperty.argtypes = [HDEVINFO, PSP_DEVINFO_DATA, DWORD, PDWORD, PBYTE, DWORD, PDWORD] 96 | SetupDiGetDeviceRegistryProperty.restype = BOOL 97 | 98 | SetupDiGetDeviceInstanceId = setupapi.SetupDiGetDeviceInstanceIdA 99 | SetupDiGetDeviceInstanceId.argtypes = [HDEVINFO, PSP_DEVINFO_DATA, PTSTR, DWORD, PDWORD] 100 | SetupDiGetDeviceInstanceId.restype = BOOL 101 | 102 | SetupDiOpenDevRegKey = setupapi.SetupDiOpenDevRegKey 103 | SetupDiOpenDevRegKey.argtypes = [HDEVINFO, PSP_DEVINFO_DATA, DWORD, DWORD, DWORD, REGSAM] 104 | SetupDiOpenDevRegKey.restype = HKEY 105 | 106 | advapi32 = ctypes.windll.LoadLibrary("Advapi32") 107 | RegCloseKey = advapi32.RegCloseKey 108 | RegCloseKey.argtypes = [HKEY] 109 | RegCloseKey.restype = LONG 110 | 111 | RegQueryValueEx = advapi32.RegQueryValueExA 112 | RegQueryValueEx.argtypes = [HKEY, LPCSTR, LPDWORD, LPDWORD, LPBYTE, LPDWORD] 113 | RegQueryValueEx.restype = LONG 114 | 115 | 116 | DIGCF_PRESENT = 2 117 | DIGCF_DEVICEINTERFACE = 16 118 | INVALID_HANDLE_VALUE = 0 119 | ERROR_INSUFFICIENT_BUFFER = 122 120 | SPDRP_HARDWAREID = 1 121 | SPDRP_FRIENDLYNAME = 12 122 | DICS_FLAG_GLOBAL = 1 123 | DIREG_DEV = 0x00000001 124 | KEY_READ = 0x20019 125 | 126 | # workaround for compatibility between Python 2.x and 3.x 127 | Ports = serial.to_bytes([80, 111, 114, 116, 115]) # "Ports" 128 | PortName = serial.to_bytes([80, 111, 114, 116, 78, 97, 109, 101]) # "PortName" 129 | 130 | def comports(): 131 | GUIDs = (GUID*8)() # so far only seen one used, so hope 8 are enough... 132 | guids_size = DWORD() 133 | if not SetupDiClassGuidsFromName( 134 | Ports, 135 | GUIDs, 136 | ctypes.sizeof(GUIDs), 137 | ctypes.byref(guids_size)): 138 | raise ctypes.WinError() 139 | 140 | # repeat for all possible GUIDs 141 | for index in range(guids_size.value): 142 | g_hdi = SetupDiGetClassDevs( 143 | ctypes.byref(GUIDs[index]), 144 | None, 145 | NULL, 146 | DIGCF_PRESENT) # was DIGCF_PRESENT|DIGCF_DEVICEINTERFACE which misses CDC ports 147 | 148 | devinfo = SP_DEVINFO_DATA() 149 | devinfo.cbSize = ctypes.sizeof(devinfo) 150 | index = 0 151 | while SetupDiEnumDeviceInfo(g_hdi, index, ctypes.byref(devinfo)): 152 | index += 1 153 | 154 | # get the real com port name 155 | hkey = SetupDiOpenDevRegKey( 156 | g_hdi, 157 | ctypes.byref(devinfo), 158 | DICS_FLAG_GLOBAL, 159 | 0, 160 | DIREG_DEV, # DIREG_DRV for SW info 161 | KEY_READ) 162 | port_name_buffer = byte_buffer(250) 163 | port_name_length = ULONG(ctypes.sizeof(port_name_buffer)) 164 | RegQueryValueEx( 165 | hkey, 166 | PortName, 167 | None, 168 | None, 169 | ctypes.byref(port_name_buffer), 170 | ctypes.byref(port_name_length)) 171 | RegCloseKey(hkey) 172 | 173 | # unfortunately does this method also include parallel ports. 174 | # we could check for names starting with COM or just exclude LPT 175 | # and hope that other "unknown" names are serial ports... 176 | if string(port_name_buffer).startswith('LPT'): 177 | continue 178 | 179 | # hardware ID 180 | szHardwareID = byte_buffer(250) 181 | # try to get ID that includes serial number 182 | if not SetupDiGetDeviceInstanceId( 183 | g_hdi, 184 | ctypes.byref(devinfo), 185 | ctypes.byref(szHardwareID), 186 | ctypes.sizeof(szHardwareID) - 1, 187 | None): 188 | # fall back to more generic hardware ID if that would fail 189 | if not SetupDiGetDeviceRegistryProperty( 190 | g_hdi, 191 | ctypes.byref(devinfo), 192 | SPDRP_HARDWAREID, 193 | None, 194 | ctypes.byref(szHardwareID), 195 | ctypes.sizeof(szHardwareID) - 1, 196 | None): 197 | # Ignore ERROR_INSUFFICIENT_BUFFER 198 | if ctypes.GetLastError() != ERROR_INSUFFICIENT_BUFFER: 199 | raise ctypes.WinError() 200 | # stringify 201 | szHardwareID_str = string(szHardwareID) 202 | 203 | # in case of USB, make a more readable string, similar to that form 204 | # that we also generate on other platforms 205 | if szHardwareID_str.startswith('USB'): 206 | m = re.search(r'VID_([0-9a-f]{4})&PID_([0-9a-f]{4})(\\(\w+))?', szHardwareID_str, re.I) 207 | if m: 208 | if m.group(4): 209 | szHardwareID_str = 'USB VID:PID=%s:%s SNR=%s' % (m.group(1), m.group(2), m.group(4)) 210 | else: 211 | szHardwareID_str = 'USB VID:PID=%s:%s' % (m.group(1), m.group(2)) 212 | 213 | # friendly name 214 | szFriendlyName = byte_buffer(250) 215 | if not SetupDiGetDeviceRegistryProperty( 216 | g_hdi, 217 | ctypes.byref(devinfo), 218 | SPDRP_FRIENDLYNAME, 219 | #~ SPDRP_DEVICEDESC, 220 | None, 221 | ctypes.byref(szFriendlyName), 222 | ctypes.sizeof(szFriendlyName) - 1, 223 | None): 224 | # Ignore ERROR_INSUFFICIENT_BUFFER 225 | #~ if ctypes.GetLastError() != ERROR_INSUFFICIENT_BUFFER: 226 | #~ raise IOError("failed to get details for %s (%s)" % (devinfo, szHardwareID.value)) 227 | # ignore errors and still include the port in the list, friendly name will be same as port name 228 | yield string(port_name_buffer), 'n/a', szHardwareID_str 229 | else: 230 | yield string(port_name_buffer), string(szFriendlyName), szHardwareID_str 231 | 232 | SetupDiDestroyDeviceInfoList(g_hdi) 233 | 234 | # - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 235 | # test 236 | if __name__ == '__main__': 237 | import serial 238 | 239 | for port, desc, hwid in sorted(comports()): 240 | print "%s: %s [%s]" % (port, desc, hwid) 241 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/urlhandler/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/Snap!Files/Snap!Mobile/arduino/serial/urlhandler/__init__.py -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/urlhandler/protocol_hwgrep.py: -------------------------------------------------------------------------------- 1 | #! python 2 | # 3 | # Python Serial Port Extension for Win32, Linux, BSD, Jython 4 | # see __init__.py 5 | # 6 | # This module implements a special URL handler that uses the port listing to 7 | # find ports by searching the string descriptions. 8 | # 9 | # (C) 2011 Chris Liechti 10 | # this is distributed under a free software license, see license.txt 11 | # 12 | # URL format: hwgrep://regexp 13 | 14 | import serial 15 | import serial.tools.list_ports 16 | 17 | class Serial(serial.Serial): 18 | """Just inherit the native Serial port implementation and patch the open function.""" 19 | 20 | def setPort(self, value): 21 | """translate port name before storing it""" 22 | if isinstance(value, basestring) and value.startswith('hwgrep://'): 23 | serial.Serial.setPort(self, self.fromURL(value)) 24 | else: 25 | serial.Serial.setPort(self, value) 26 | 27 | def fromURL(self, url): 28 | """extract host and port from an URL string""" 29 | if url.lower().startswith("hwgrep://"): url = url[9:] 30 | # use a for loop to get the 1st element from the generator 31 | for port, desc, hwid in serial.tools.list_ports.grep(url): 32 | return port 33 | else: 34 | raise serial.SerialException('no ports found matching regexp %r' % (url,)) 35 | 36 | # override property 37 | port = property(serial.Serial.getPort, setPort, doc="Port setting") 38 | 39 | # - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 40 | if __name__ == '__main__': 41 | #~ s = Serial('hwgrep://ttyS0') 42 | s = Serial(None) 43 | s.port = 'hwgrep://ttyS0' 44 | print s 45 | 46 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/urlhandler/protocol_loop.py: -------------------------------------------------------------------------------- 1 | #! python 2 | # 3 | # Python Serial Port Extension for Win32, Linux, BSD, Jython 4 | # see __init__.py 5 | # 6 | # This module implements a loop back connection receiving itself what it sent. 7 | # 8 | # The purpose of this module is.. well... You can run the unit tests with it. 9 | # and it was so easy to implement ;-) 10 | # 11 | # (C) 2001-2011 Chris Liechti 12 | # this is distributed under a free software license, see license.txt 13 | # 14 | # URL format: loop://[option[/option...]] 15 | # options: 16 | # - "debug" print diagnostic messages 17 | 18 | from serial.serialutil import * 19 | import threading 20 | import time 21 | import logging 22 | 23 | # map log level names to constants. used in fromURL() 24 | LOGGER_LEVELS = { 25 | 'debug': logging.DEBUG, 26 | 'info': logging.INFO, 27 | 'warning': logging.WARNING, 28 | 'error': logging.ERROR, 29 | } 30 | 31 | 32 | class LoopbackSerial(SerialBase): 33 | """Serial port implementation that simulates a loop back connection in plain software.""" 34 | 35 | BAUDRATES = (50, 75, 110, 134, 150, 200, 300, 600, 1200, 1800, 2400, 4800, 36 | 9600, 19200, 38400, 57600, 115200) 37 | 38 | def open(self): 39 | """Open port with current settings. This may throw a SerialException 40 | if the port cannot be opened.""" 41 | if self._isOpen: 42 | raise SerialException("Port is already open.") 43 | self.logger = None 44 | self.buffer_lock = threading.Lock() 45 | self.loop_buffer = bytearray() 46 | self.cts = False 47 | self.dsr = False 48 | 49 | if self._port is None: 50 | raise SerialException("Port must be configured before it can be used.") 51 | # not that there is anything to open, but the function applies the 52 | # options found in the URL 53 | self.fromURL(self.port) 54 | 55 | # not that there anything to configure... 56 | self._reconfigurePort() 57 | # all things set up get, now a clean start 58 | self._isOpen = True 59 | if not self._rtscts: 60 | self.setRTS(True) 61 | self.setDTR(True) 62 | self.flushInput() 63 | self.flushOutput() 64 | 65 | def _reconfigurePort(self): 66 | """Set communication parameters on opened port. for the loop:// 67 | protocol all settings are ignored!""" 68 | # not that's it of any real use, but it helps in the unit tests 69 | if not isinstance(self._baudrate, (int, long)) or not 0 < self._baudrate < 2**32: 70 | raise ValueError("invalid baudrate: %r" % (self._baudrate)) 71 | if self.logger: 72 | self.logger.info('_reconfigurePort()') 73 | 74 | def close(self): 75 | """Close port""" 76 | if self._isOpen: 77 | self._isOpen = False 78 | # in case of quick reconnects, give the server some time 79 | time.sleep(0.3) 80 | 81 | def makeDeviceName(self, port): 82 | raise SerialException("there is no sensible way to turn numbers into URLs") 83 | 84 | def fromURL(self, url): 85 | """extract host and port from an URL string""" 86 | if url.lower().startswith("loop://"): url = url[7:] 87 | try: 88 | # process options now, directly altering self 89 | for option in url.split('/'): 90 | if '=' in option: 91 | option, value = option.split('=', 1) 92 | else: 93 | value = None 94 | if not option: 95 | pass 96 | elif option == 'logging': 97 | logging.basicConfig() # XXX is that good to call it here? 98 | self.logger = logging.getLogger('pySerial.loop') 99 | self.logger.setLevel(LOGGER_LEVELS[value]) 100 | self.logger.debug('enabled logging') 101 | else: 102 | raise ValueError('unknown option: %r' % (option,)) 103 | except ValueError, e: 104 | raise SerialException('expected a string in the form "[loop://][option[/option...]]": %s' % e) 105 | 106 | # - - - - - - - - - - - - - - - - - - - - - - - - 107 | 108 | def inWaiting(self): 109 | """Return the number of characters currently in the input buffer.""" 110 | if not self._isOpen: raise portNotOpenError 111 | if self.logger: 112 | # attention the logged value can differ from return value in 113 | # threaded environments... 114 | self.logger.debug('inWaiting() -> %d' % (len(self.loop_buffer),)) 115 | return len(self.loop_buffer) 116 | 117 | def read(self, size=1): 118 | """Read size bytes from the serial port. If a timeout is set it may 119 | return less characters as requested. With no timeout it will block 120 | until the requested number of bytes is read.""" 121 | if not self._isOpen: raise portNotOpenError 122 | if self._timeout is not None: 123 | timeout = time.time() + self._timeout 124 | else: 125 | timeout = None 126 | data = bytearray() 127 | while size > 0: 128 | self.buffer_lock.acquire() 129 | try: 130 | block = to_bytes(self.loop_buffer[:size]) 131 | del self.loop_buffer[:size] 132 | finally: 133 | self.buffer_lock.release() 134 | data += block 135 | size -= len(block) 136 | # check for timeout now, after data has been read. 137 | # useful for timeout = 0 (non blocking) read 138 | if timeout and time.time() > timeout: 139 | break 140 | return bytes(data) 141 | 142 | def write(self, data): 143 | """Output the given string over the serial port. Can block if the 144 | connection is blocked. May raise SerialException if the connection is 145 | closed.""" 146 | if not self._isOpen: raise portNotOpenError 147 | # ensure we're working with bytes 148 | data = to_bytes(data) 149 | # calculate aprox time that would be used to send the data 150 | time_used_to_send = 10.0*len(data) / self._baudrate 151 | # when a write timeout is configured check if we would be successful 152 | # (not sending anything, not even the part that would have time) 153 | if self._writeTimeout is not None and time_used_to_send > self._writeTimeout: 154 | time.sleep(self._writeTimeout) # must wait so that unit test succeeds 155 | raise writeTimeoutError 156 | self.buffer_lock.acquire() 157 | try: 158 | self.loop_buffer += data 159 | finally: 160 | self.buffer_lock.release() 161 | return len(data) 162 | 163 | def flushInput(self): 164 | """Clear input buffer, discarding all that is in the buffer.""" 165 | if not self._isOpen: raise portNotOpenError 166 | if self.logger: 167 | self.logger.info('flushInput()') 168 | self.buffer_lock.acquire() 169 | try: 170 | del self.loop_buffer[:] 171 | finally: 172 | self.buffer_lock.release() 173 | 174 | def flushOutput(self): 175 | """Clear output buffer, aborting the current output and 176 | discarding all that is in the buffer.""" 177 | if not self._isOpen: raise portNotOpenError 178 | if self.logger: 179 | self.logger.info('flushOutput()') 180 | 181 | def sendBreak(self, duration=0.25): 182 | """Send break condition. Timed, returns to idle state after given 183 | duration.""" 184 | if not self._isOpen: raise portNotOpenError 185 | 186 | def setBreak(self, level=True): 187 | """Set break: Controls TXD. When active, to transmitting is 188 | possible.""" 189 | if not self._isOpen: raise portNotOpenError 190 | if self.logger: 191 | self.logger.info('setBreak(%r)' % (level,)) 192 | 193 | def setRTS(self, level=True): 194 | """Set terminal status line: Request To Send""" 195 | if not self._isOpen: raise portNotOpenError 196 | if self.logger: 197 | self.logger.info('setRTS(%r) -> state of CTS' % (level,)) 198 | self.cts = level 199 | 200 | def setDTR(self, level=True): 201 | """Set terminal status line: Data Terminal Ready""" 202 | if not self._isOpen: raise portNotOpenError 203 | if self.logger: 204 | self.logger.info('setDTR(%r) -> state of DSR' % (level,)) 205 | self.dsr = level 206 | 207 | def getCTS(self): 208 | """Read terminal status line: Clear To Send""" 209 | if not self._isOpen: raise portNotOpenError 210 | if self.logger: 211 | self.logger.info('getCTS() -> state of RTS (%r)' % (self.cts,)) 212 | return self.cts 213 | 214 | def getDSR(self): 215 | """Read terminal status line: Data Set Ready""" 216 | if not self._isOpen: raise portNotOpenError 217 | if self.logger: 218 | self.logger.info('getDSR() -> state of DTR (%r)' % (self.dsr,)) 219 | return self.dsr 220 | 221 | def getRI(self): 222 | """Read terminal status line: Ring Indicator""" 223 | if not self._isOpen: raise portNotOpenError 224 | if self.logger: 225 | self.logger.info('returning dummy for getRI()') 226 | return False 227 | 228 | def getCD(self): 229 | """Read terminal status line: Carrier Detect""" 230 | if not self._isOpen: raise portNotOpenError 231 | if self.logger: 232 | self.logger.info('returning dummy for getCD()') 233 | return True 234 | 235 | # - - - platform specific - - - 236 | # None so far 237 | 238 | 239 | # assemble Serial class with the platform specific implementation and the base 240 | # for file-like behavior. for Python 2.6 and newer, that provide the new I/O 241 | # library, derive from io.RawIOBase 242 | try: 243 | import io 244 | except ImportError: 245 | # classic version with our own file-like emulation 246 | class Serial(LoopbackSerial, FileLike): 247 | pass 248 | else: 249 | # io library present 250 | class Serial(LoopbackSerial, io.RawIOBase): 251 | pass 252 | 253 | 254 | # simple client test 255 | if __name__ == '__main__': 256 | import sys 257 | s = Serial('loop://') 258 | sys.stdout.write('%s\n' % s) 259 | 260 | sys.stdout.write("write...\n") 261 | s.write("hello\n") 262 | s.flush() 263 | sys.stdout.write("read: %s\n" % s.read(5)) 264 | 265 | s.close() 266 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/urlhandler/protocol_rfc2217.py: -------------------------------------------------------------------------------- 1 | #! python 2 | # 3 | # Python Serial Port Extension for Win32, Linux, BSD, Jython 4 | # see ../__init__.py 5 | # 6 | # This is a thin wrapper to load the rfc2271 implementation. 7 | # 8 | # (C) 2011 Chris Liechti 9 | # this is distributed under a free software license, see license.txt 10 | 11 | from serial.rfc2217 import Serial 12 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/urlhandler/protocol_socket.py: -------------------------------------------------------------------------------- 1 | #! python 2 | # 3 | # Python Serial Port Extension for Win32, Linux, BSD, Jython 4 | # see __init__.py 5 | # 6 | # This module implements a simple socket based client. 7 | # It does not support changing any port parameters and will silently ignore any 8 | # requests to do so. 9 | # 10 | # The purpose of this module is that applications using pySerial can connect to 11 | # TCP/IP to serial port converters that do not support RFC 2217. 12 | # 13 | # (C) 2001-2011 Chris Liechti 14 | # this is distributed under a free software license, see license.txt 15 | # 16 | # URL format: socket://:[/option[/option...]] 17 | # options: 18 | # - "debug" print diagnostic messages 19 | 20 | from serial.serialutil import * 21 | import time 22 | import socket 23 | import logging 24 | 25 | # map log level names to constants. used in fromURL() 26 | LOGGER_LEVELS = { 27 | 'debug': logging.DEBUG, 28 | 'info': logging.INFO, 29 | 'warning': logging.WARNING, 30 | 'error': logging.ERROR, 31 | } 32 | 33 | POLL_TIMEOUT = 2 34 | 35 | class SocketSerial(SerialBase): 36 | """Serial port implementation for plain sockets.""" 37 | 38 | BAUDRATES = (50, 75, 110, 134, 150, 200, 300, 600, 1200, 1800, 2400, 4800, 39 | 9600, 19200, 38400, 57600, 115200) 40 | 41 | def open(self): 42 | """Open port with current settings. This may throw a SerialException 43 | if the port cannot be opened.""" 44 | self.logger = None 45 | if self._port is None: 46 | raise SerialException("Port must be configured before it can be used.") 47 | if self._isOpen: 48 | raise SerialException("Port is already open.") 49 | try: 50 | # XXX in future replace with create_connection (py >=2.6) 51 | self._socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) 52 | self._socket.connect(self.fromURL(self.portstr)) 53 | except Exception, msg: 54 | self._socket = None 55 | raise SerialException("Could not open port %s: %s" % (self.portstr, msg)) 56 | 57 | self._socket.settimeout(POLL_TIMEOUT) # used for write timeout support :/ 58 | 59 | # not that there anything to configure... 60 | self._reconfigurePort() 61 | # all things set up get, now a clean start 62 | self._isOpen = True 63 | if not self._rtscts: 64 | self.setRTS(True) 65 | self.setDTR(True) 66 | self.flushInput() 67 | self.flushOutput() 68 | 69 | def _reconfigurePort(self): 70 | """Set communication parameters on opened port. for the socket:// 71 | protocol all settings are ignored!""" 72 | if self._socket is None: 73 | raise SerialException("Can only operate on open ports") 74 | if self.logger: 75 | self.logger.info('ignored port configuration change') 76 | 77 | def close(self): 78 | """Close port""" 79 | if self._isOpen: 80 | if self._socket: 81 | try: 82 | self._socket.shutdown(socket.SHUT_RDWR) 83 | self._socket.close() 84 | except: 85 | # ignore errors. 86 | pass 87 | self._socket = None 88 | self._isOpen = False 89 | # in case of quick reconnects, give the server some time 90 | time.sleep(0.3) 91 | 92 | def makeDeviceName(self, port): 93 | raise SerialException("there is no sensible way to turn numbers into URLs") 94 | 95 | def fromURL(self, url): 96 | """extract host and port from an URL string""" 97 | if url.lower().startswith("socket://"): url = url[9:] 98 | try: 99 | # is there a "path" (our options)? 100 | if '/' in url: 101 | # cut away options 102 | url, options = url.split('/', 1) 103 | # process options now, directly altering self 104 | for option in options.split('/'): 105 | if '=' in option: 106 | option, value = option.split('=', 1) 107 | else: 108 | value = None 109 | if option == 'logging': 110 | logging.basicConfig() # XXX is that good to call it here? 111 | self.logger = logging.getLogger('pySerial.socket') 112 | self.logger.setLevel(LOGGER_LEVELS[value]) 113 | self.logger.debug('enabled logging') 114 | else: 115 | raise ValueError('unknown option: %r' % (option,)) 116 | # get host and port 117 | host, port = url.split(':', 1) # may raise ValueError because of unpacking 118 | port = int(port) # and this if it's not a number 119 | if not 0 <= port < 65536: raise ValueError("port not in range 0...65535") 120 | except ValueError, e: 121 | raise SerialException('expected a string in the form "[rfc2217://]:[/option[/option...]]": %s' % e) 122 | return (host, port) 123 | 124 | # - - - - - - - - - - - - - - - - - - - - - - - - 125 | 126 | def inWaiting(self): 127 | """Return the number of characters currently in the input buffer.""" 128 | if not self._isOpen: raise portNotOpenError 129 | if self.logger: 130 | # set this one to debug as the function could be called often... 131 | self.logger.debug('WARNING: inWaiting returns dummy value') 132 | return 0 # hmmm, see comment in read() 133 | 134 | def read(self, size=1): 135 | """Read size bytes from the serial port. If a timeout is set it may 136 | return less characters as requested. With no timeout it will block 137 | until the requested number of bytes is read.""" 138 | if not self._isOpen: raise portNotOpenError 139 | data = bytearray() 140 | if self._timeout is not None: 141 | timeout = time.time() + self._timeout 142 | else: 143 | timeout = None 144 | while len(data) < size and (timeout is None or time.time() < timeout): 145 | try: 146 | # an implementation with internal buffer would be better 147 | # performing... 148 | t = time.time() 149 | block = self._socket.recv(size - len(data)) 150 | duration = time.time() - t 151 | if block: 152 | data.extend(block) 153 | else: 154 | # no data -> EOF (connection probably closed) 155 | break 156 | except socket.timeout: 157 | # just need to get out of recv from time to time to check if 158 | # still alive 159 | continue 160 | except socket.error, e: 161 | # connection fails -> terminate loop 162 | raise SerialException('connection failed (%s)' % e) 163 | return bytes(data) 164 | 165 | def write(self, data): 166 | """Output the given string over the serial port. Can block if the 167 | connection is blocked. May raise SerialException if the connection is 168 | closed.""" 169 | if not self._isOpen: raise portNotOpenError 170 | try: 171 | self._socket.sendall(to_bytes(data)) 172 | except socket.error, e: 173 | # XXX what exception if socket connection fails 174 | raise SerialException("socket connection failed: %s" % e) 175 | return len(data) 176 | 177 | def flushInput(self): 178 | """Clear input buffer, discarding all that is in the buffer.""" 179 | if not self._isOpen: raise portNotOpenError 180 | if self.logger: 181 | self.logger.info('ignored flushInput') 182 | 183 | def flushOutput(self): 184 | """Clear output buffer, aborting the current output and 185 | discarding all that is in the buffer.""" 186 | if not self._isOpen: raise portNotOpenError 187 | if self.logger: 188 | self.logger.info('ignored flushOutput') 189 | 190 | def sendBreak(self, duration=0.25): 191 | """Send break condition. Timed, returns to idle state after given 192 | duration.""" 193 | if not self._isOpen: raise portNotOpenError 194 | if self.logger: 195 | self.logger.info('ignored sendBreak(%r)' % (duration,)) 196 | 197 | def setBreak(self, level=True): 198 | """Set break: Controls TXD. When active, to transmitting is 199 | possible.""" 200 | if not self._isOpen: raise portNotOpenError 201 | if self.logger: 202 | self.logger.info('ignored setBreak(%r)' % (level,)) 203 | 204 | def setRTS(self, level=True): 205 | """Set terminal status line: Request To Send""" 206 | if not self._isOpen: raise portNotOpenError 207 | if self.logger: 208 | self.logger.info('ignored setRTS(%r)' % (level,)) 209 | 210 | def setDTR(self, level=True): 211 | """Set terminal status line: Data Terminal Ready""" 212 | if not self._isOpen: raise portNotOpenError 213 | if self.logger: 214 | self.logger.info('ignored setDTR(%r)' % (level,)) 215 | 216 | def getCTS(self): 217 | """Read terminal status line: Clear To Send""" 218 | if not self._isOpen: raise portNotOpenError 219 | if self.logger: 220 | self.logger.info('returning dummy for getCTS()') 221 | return True 222 | 223 | def getDSR(self): 224 | """Read terminal status line: Data Set Ready""" 225 | if not self._isOpen: raise portNotOpenError 226 | if self.logger: 227 | self.logger.info('returning dummy for getDSR()') 228 | return True 229 | 230 | def getRI(self): 231 | """Read terminal status line: Ring Indicator""" 232 | if not self._isOpen: raise portNotOpenError 233 | if self.logger: 234 | self.logger.info('returning dummy for getRI()') 235 | return False 236 | 237 | def getCD(self): 238 | """Read terminal status line: Carrier Detect""" 239 | if not self._isOpen: raise portNotOpenError 240 | if self.logger: 241 | self.logger.info('returning dummy for getCD()') 242 | return True 243 | 244 | # - - - platform specific - - - 245 | # None so far 246 | 247 | 248 | # assemble Serial class with the platform specific implementation and the base 249 | # for file-like behavior. for Python 2.6 and newer, that provide the new I/O 250 | # library, derive from io.RawIOBase 251 | try: 252 | import io 253 | except ImportError: 254 | # classic version with our own file-like emulation 255 | class Serial(SocketSerial, FileLike): 256 | pass 257 | else: 258 | # io library present 259 | class Serial(SocketSerial, io.RawIOBase): 260 | pass 261 | 262 | 263 | # simple client test 264 | if __name__ == '__main__': 265 | import sys 266 | s = Serial('socket://localhost:7000') 267 | sys.stdout.write('%s\n' % s) 268 | 269 | sys.stdout.write("write...\n") 270 | s.write("hello\n") 271 | s.flush() 272 | sys.stdout.write("read: %s\n" % s.read(5)) 273 | 274 | s.close() 275 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/serial/win32.py: -------------------------------------------------------------------------------- 1 | from ctypes import * 2 | from ctypes.wintypes import HANDLE 3 | from ctypes.wintypes import BOOL 4 | from ctypes.wintypes import LPCWSTR 5 | _stdcall_libraries = {} 6 | _stdcall_libraries['kernel32'] = WinDLL('kernel32') 7 | from ctypes.wintypes import DWORD 8 | from ctypes.wintypes import WORD 9 | from ctypes.wintypes import BYTE 10 | 11 | INVALID_HANDLE_VALUE = HANDLE(-1).value 12 | 13 | # some details of the windows API differ between 32 and 64 bit systems.. 14 | def is_64bit(): 15 | """Returns true when running on a 64 bit system""" 16 | return sizeof(c_ulong) != sizeof(c_void_p) 17 | 18 | # ULONG_PTR is a an ordinary number, not a pointer and contrary to the name it 19 | # is either 32 or 64 bits, depending on the type of windows... 20 | # so test if this a 32 bit windows... 21 | if is_64bit(): 22 | # assume 64 bits 23 | ULONG_PTR = c_int64 24 | else: 25 | # 32 bits 26 | ULONG_PTR = c_ulong 27 | 28 | 29 | class _SECURITY_ATTRIBUTES(Structure): 30 | pass 31 | LPSECURITY_ATTRIBUTES = POINTER(_SECURITY_ATTRIBUTES) 32 | 33 | 34 | try: 35 | CreateEventW = _stdcall_libraries['kernel32'].CreateEventW 36 | except AttributeError: 37 | # Fallback to non wide char version for old OS... 38 | from ctypes.wintypes import LPCSTR 39 | CreateEventA = _stdcall_libraries['kernel32'].CreateEventA 40 | CreateEventA.restype = HANDLE 41 | CreateEventA.argtypes = [LPSECURITY_ATTRIBUTES, BOOL, BOOL, LPCSTR] 42 | CreateEvent=CreateEventA 43 | 44 | CreateFileA = _stdcall_libraries['kernel32'].CreateFileA 45 | CreateFileA.restype = HANDLE 46 | CreateFileA.argtypes = [LPCSTR, DWORD, DWORD, LPSECURITY_ATTRIBUTES, DWORD, DWORD, HANDLE] 47 | CreateFile = CreateFileA 48 | else: 49 | CreateEventW.restype = HANDLE 50 | CreateEventW.argtypes = [LPSECURITY_ATTRIBUTES, BOOL, BOOL, LPCWSTR] 51 | CreateEvent = CreateEventW # alias 52 | 53 | CreateFileW = _stdcall_libraries['kernel32'].CreateFileW 54 | CreateFileW.restype = HANDLE 55 | CreateFileW.argtypes = [LPCWSTR, DWORD, DWORD, LPSECURITY_ATTRIBUTES, DWORD, DWORD, HANDLE] 56 | CreateFile = CreateFileW # alias 57 | 58 | class _OVERLAPPED(Structure): 59 | pass 60 | OVERLAPPED = _OVERLAPPED 61 | 62 | class _COMSTAT(Structure): 63 | pass 64 | COMSTAT = _COMSTAT 65 | 66 | class _DCB(Structure): 67 | pass 68 | DCB = _DCB 69 | 70 | class _COMMTIMEOUTS(Structure): 71 | pass 72 | COMMTIMEOUTS = _COMMTIMEOUTS 73 | 74 | GetLastError = _stdcall_libraries['kernel32'].GetLastError 75 | GetLastError.restype = DWORD 76 | GetLastError.argtypes = [] 77 | 78 | LPOVERLAPPED = POINTER(_OVERLAPPED) 79 | LPDWORD = POINTER(DWORD) 80 | 81 | GetOverlappedResult = _stdcall_libraries['kernel32'].GetOverlappedResult 82 | GetOverlappedResult.restype = BOOL 83 | GetOverlappedResult.argtypes = [HANDLE, LPOVERLAPPED, LPDWORD, BOOL] 84 | 85 | ResetEvent = _stdcall_libraries['kernel32'].ResetEvent 86 | ResetEvent.restype = BOOL 87 | ResetEvent.argtypes = [HANDLE] 88 | 89 | LPCVOID = c_void_p 90 | 91 | WriteFile = _stdcall_libraries['kernel32'].WriteFile 92 | WriteFile.restype = BOOL 93 | WriteFile.argtypes = [HANDLE, LPCVOID, DWORD, LPDWORD, LPOVERLAPPED] 94 | 95 | LPVOID = c_void_p 96 | 97 | ReadFile = _stdcall_libraries['kernel32'].ReadFile 98 | ReadFile.restype = BOOL 99 | ReadFile.argtypes = [HANDLE, LPVOID, DWORD, LPDWORD, LPOVERLAPPED] 100 | 101 | CloseHandle = _stdcall_libraries['kernel32'].CloseHandle 102 | CloseHandle.restype = BOOL 103 | CloseHandle.argtypes = [HANDLE] 104 | 105 | ClearCommBreak = _stdcall_libraries['kernel32'].ClearCommBreak 106 | ClearCommBreak.restype = BOOL 107 | ClearCommBreak.argtypes = [HANDLE] 108 | 109 | LPCOMSTAT = POINTER(_COMSTAT) 110 | 111 | ClearCommError = _stdcall_libraries['kernel32'].ClearCommError 112 | ClearCommError.restype = BOOL 113 | ClearCommError.argtypes = [HANDLE, LPDWORD, LPCOMSTAT] 114 | 115 | SetupComm = _stdcall_libraries['kernel32'].SetupComm 116 | SetupComm.restype = BOOL 117 | SetupComm.argtypes = [HANDLE, DWORD, DWORD] 118 | 119 | EscapeCommFunction = _stdcall_libraries['kernel32'].EscapeCommFunction 120 | EscapeCommFunction.restype = BOOL 121 | EscapeCommFunction.argtypes = [HANDLE, DWORD] 122 | 123 | GetCommModemStatus = _stdcall_libraries['kernel32'].GetCommModemStatus 124 | GetCommModemStatus.restype = BOOL 125 | GetCommModemStatus.argtypes = [HANDLE, LPDWORD] 126 | 127 | LPDCB = POINTER(_DCB) 128 | 129 | GetCommState = _stdcall_libraries['kernel32'].GetCommState 130 | GetCommState.restype = BOOL 131 | GetCommState.argtypes = [HANDLE, LPDCB] 132 | 133 | LPCOMMTIMEOUTS = POINTER(_COMMTIMEOUTS) 134 | 135 | GetCommTimeouts = _stdcall_libraries['kernel32'].GetCommTimeouts 136 | GetCommTimeouts.restype = BOOL 137 | GetCommTimeouts.argtypes = [HANDLE, LPCOMMTIMEOUTS] 138 | 139 | PurgeComm = _stdcall_libraries['kernel32'].PurgeComm 140 | PurgeComm.restype = BOOL 141 | PurgeComm.argtypes = [HANDLE, DWORD] 142 | 143 | SetCommBreak = _stdcall_libraries['kernel32'].SetCommBreak 144 | SetCommBreak.restype = BOOL 145 | SetCommBreak.argtypes = [HANDLE] 146 | 147 | SetCommMask = _stdcall_libraries['kernel32'].SetCommMask 148 | SetCommMask.restype = BOOL 149 | SetCommMask.argtypes = [HANDLE, DWORD] 150 | 151 | SetCommState = _stdcall_libraries['kernel32'].SetCommState 152 | SetCommState.restype = BOOL 153 | SetCommState.argtypes = [HANDLE, LPDCB] 154 | 155 | SetCommTimeouts = _stdcall_libraries['kernel32'].SetCommTimeouts 156 | SetCommTimeouts.restype = BOOL 157 | SetCommTimeouts.argtypes = [HANDLE, LPCOMMTIMEOUTS] 158 | 159 | WaitForSingleObject = _stdcall_libraries['kernel32'].WaitForSingleObject 160 | WaitForSingleObject.restype = DWORD 161 | WaitForSingleObject.argtypes = [HANDLE, DWORD] 162 | 163 | ONESTOPBIT = 0 # Variable c_int 164 | TWOSTOPBITS = 2 # Variable c_int 165 | ONE5STOPBITS = 1 166 | 167 | NOPARITY = 0 # Variable c_int 168 | ODDPARITY = 1 # Variable c_int 169 | EVENPARITY = 2 # Variable c_int 170 | MARKPARITY = 3 171 | SPACEPARITY = 4 172 | 173 | RTS_CONTROL_HANDSHAKE = 2 # Variable c_int 174 | RTS_CONTROL_DISABLE = 0 # Variable c_int 175 | RTS_CONTROL_ENABLE = 1 # Variable c_int 176 | RTS_CONTROL_TOGGLE = 3 # Variable c_int 177 | SETRTS = 3 178 | CLRRTS = 4 179 | 180 | DTR_CONTROL_HANDSHAKE = 2 # Variable c_int 181 | DTR_CONTROL_DISABLE = 0 # Variable c_int 182 | DTR_CONTROL_ENABLE = 1 # Variable c_int 183 | SETDTR = 5 184 | CLRDTR = 6 185 | 186 | MS_DSR_ON = 32 # Variable c_ulong 187 | EV_RING = 256 # Variable c_int 188 | EV_PERR = 512 # Variable c_int 189 | EV_ERR = 128 # Variable c_int 190 | SETXOFF = 1 # Variable c_int 191 | EV_RXCHAR = 1 # Variable c_int 192 | GENERIC_WRITE = 1073741824 # Variable c_long 193 | PURGE_TXCLEAR = 4 # Variable c_int 194 | FILE_FLAG_OVERLAPPED = 1073741824 # Variable c_int 195 | EV_DSR = 16 # Variable c_int 196 | MAXDWORD = 4294967295L # Variable c_uint 197 | EV_RLSD = 32 # Variable c_int 198 | ERROR_IO_PENDING = 997 # Variable c_long 199 | MS_CTS_ON = 16 # Variable c_ulong 200 | EV_EVENT1 = 2048 # Variable c_int 201 | EV_RX80FULL = 1024 # Variable c_int 202 | PURGE_RXABORT = 2 # Variable c_int 203 | FILE_ATTRIBUTE_NORMAL = 128 # Variable c_int 204 | PURGE_TXABORT = 1 # Variable c_int 205 | SETXON = 2 # Variable c_int 206 | OPEN_EXISTING = 3 # Variable c_int 207 | MS_RING_ON = 64 # Variable c_ulong 208 | EV_TXEMPTY = 4 # Variable c_int 209 | EV_RXFLAG = 2 # Variable c_int 210 | MS_RLSD_ON = 128 # Variable c_ulong 211 | GENERIC_READ = 2147483648L # Variable c_ulong 212 | EV_EVENT2 = 4096 # Variable c_int 213 | EV_CTS = 8 # Variable c_int 214 | EV_BREAK = 64 # Variable c_int 215 | PURGE_RXCLEAR = 8 # Variable c_int 216 | INFINITE = 0xFFFFFFFFL 217 | 218 | 219 | class N11_OVERLAPPED4DOLLAR_48E(Union): 220 | pass 221 | class N11_OVERLAPPED4DOLLAR_484DOLLAR_49E(Structure): 222 | pass 223 | N11_OVERLAPPED4DOLLAR_484DOLLAR_49E._fields_ = [ 224 | ('Offset', DWORD), 225 | ('OffsetHigh', DWORD), 226 | ] 227 | 228 | PVOID = c_void_p 229 | 230 | N11_OVERLAPPED4DOLLAR_48E._anonymous_ = ['_0'] 231 | N11_OVERLAPPED4DOLLAR_48E._fields_ = [ 232 | ('_0', N11_OVERLAPPED4DOLLAR_484DOLLAR_49E), 233 | ('Pointer', PVOID), 234 | ] 235 | _OVERLAPPED._anonymous_ = ['_0'] 236 | _OVERLAPPED._fields_ = [ 237 | ('Internal', ULONG_PTR), 238 | ('InternalHigh', ULONG_PTR), 239 | ('_0', N11_OVERLAPPED4DOLLAR_48E), 240 | ('hEvent', HANDLE), 241 | ] 242 | _SECURITY_ATTRIBUTES._fields_ = [ 243 | ('nLength', DWORD), 244 | ('lpSecurityDescriptor', LPVOID), 245 | ('bInheritHandle', BOOL), 246 | ] 247 | _COMSTAT._fields_ = [ 248 | ('fCtsHold', DWORD, 1), 249 | ('fDsrHold', DWORD, 1), 250 | ('fRlsdHold', DWORD, 1), 251 | ('fXoffHold', DWORD, 1), 252 | ('fXoffSent', DWORD, 1), 253 | ('fEof', DWORD, 1), 254 | ('fTxim', DWORD, 1), 255 | ('fReserved', DWORD, 25), 256 | ('cbInQue', DWORD), 257 | ('cbOutQue', DWORD), 258 | ] 259 | _DCB._fields_ = [ 260 | ('DCBlength', DWORD), 261 | ('BaudRate', DWORD), 262 | ('fBinary', DWORD, 1), 263 | ('fParity', DWORD, 1), 264 | ('fOutxCtsFlow', DWORD, 1), 265 | ('fOutxDsrFlow', DWORD, 1), 266 | ('fDtrControl', DWORD, 2), 267 | ('fDsrSensitivity', DWORD, 1), 268 | ('fTXContinueOnXoff', DWORD, 1), 269 | ('fOutX', DWORD, 1), 270 | ('fInX', DWORD, 1), 271 | ('fErrorChar', DWORD, 1), 272 | ('fNull', DWORD, 1), 273 | ('fRtsControl', DWORD, 2), 274 | ('fAbortOnError', DWORD, 1), 275 | ('fDummy2', DWORD, 17), 276 | ('wReserved', WORD), 277 | ('XonLim', WORD), 278 | ('XoffLim', WORD), 279 | ('ByteSize', BYTE), 280 | ('Parity', BYTE), 281 | ('StopBits', BYTE), 282 | ('XonChar', c_char), 283 | ('XoffChar', c_char), 284 | ('ErrorChar', c_char), 285 | ('EofChar', c_char), 286 | ('EvtChar', c_char), 287 | ('wReserved1', WORD), 288 | ] 289 | _COMMTIMEOUTS._fields_ = [ 290 | ('ReadIntervalTimeout', DWORD), 291 | ('ReadTotalTimeoutMultiplier', DWORD), 292 | ('ReadTotalTimeoutConstant', DWORD), 293 | ('WriteTotalTimeoutMultiplier', DWORD), 294 | ('WriteTotalTimeoutConstant', DWORD), 295 | ] 296 | __all__ = ['GetLastError', 'MS_CTS_ON', 'FILE_ATTRIBUTE_NORMAL', 297 | 'DTR_CONTROL_ENABLE', '_COMSTAT', 'MS_RLSD_ON', 298 | 'GetOverlappedResult', 'SETXON', 'PURGE_TXABORT', 299 | 'PurgeComm', 'N11_OVERLAPPED4DOLLAR_48E', 'EV_RING', 300 | 'ONESTOPBIT', 'SETXOFF', 'PURGE_RXABORT', 'GetCommState', 301 | 'RTS_CONTROL_ENABLE', '_DCB', 'CreateEvent', 302 | '_COMMTIMEOUTS', '_SECURITY_ATTRIBUTES', 'EV_DSR', 303 | 'EV_PERR', 'EV_RXFLAG', 'OPEN_EXISTING', 'DCB', 304 | 'FILE_FLAG_OVERLAPPED', 'EV_CTS', 'SetupComm', 305 | 'LPOVERLAPPED', 'EV_TXEMPTY', 'ClearCommBreak', 306 | 'LPSECURITY_ATTRIBUTES', 'SetCommBreak', 'SetCommTimeouts', 307 | 'COMMTIMEOUTS', 'ODDPARITY', 'EV_RLSD', 308 | 'GetCommModemStatus', 'EV_EVENT2', 'PURGE_TXCLEAR', 309 | 'EV_BREAK', 'EVENPARITY', 'LPCVOID', 'COMSTAT', 'ReadFile', 310 | 'PVOID', '_OVERLAPPED', 'WriteFile', 'GetCommTimeouts', 311 | 'ResetEvent', 'EV_RXCHAR', 'LPCOMSTAT', 'ClearCommError', 312 | 'ERROR_IO_PENDING', 'EscapeCommFunction', 'GENERIC_READ', 313 | 'RTS_CONTROL_HANDSHAKE', 'OVERLAPPED', 314 | 'DTR_CONTROL_HANDSHAKE', 'PURGE_RXCLEAR', 'GENERIC_WRITE', 315 | 'LPDCB', 'CreateEventW', 'SetCommMask', 'EV_EVENT1', 316 | 'SetCommState', 'LPVOID', 'CreateFileW', 'LPDWORD', 317 | 'EV_RX80FULL', 'TWOSTOPBITS', 'LPCOMMTIMEOUTS', 'MAXDWORD', 318 | 'MS_DSR_ON', 'MS_RING_ON', 319 | 'N11_OVERLAPPED4DOLLAR_484DOLLAR_49E', 'EV_ERR', 320 | 'ULONG_PTR', 'CreateFile', 'NOPARITY', 'CloseHandle'] 321 | -------------------------------------------------------------------------------- /Snap!Files/Snap!Mobile/arduino/xlate.cfg: -------------------------------------------------------------------------------- 1 | # @author: Alan Yorinks 2 | # Copyright (c) 2013 Alan Yorinks All right reserved. 3 | # 4 | # @co-author: Sjoerd Dirk Meijer, fromScratchEd.nl (language support) 5 | # @co-author: Professor José Manuel Ruiz (Spanish translation) 6 | # 7 | # This program is free software; you can redistribute it and/or 8 | # modify it under the terms of the GNU Lesser General Public 9 | # License as published by the Free Software Foundation; either 10 | #version 2.1 of the License, or (at your option) any later version. 11 | # 12 | # This library is distributed in the hope that it will be useful, 13 | # but WITHOUT ANY WARRANTY; without even the implied warranty of 14 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15 | # Lesser General Public License for more details. 16 | # 17 | # You should have received a copy of the GNU Lesser General Public 18 | # License along with this library; if not, write to the Free Software 19 | # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA 20 | 21 | [translation_lists] 22 | ln_languages = English,Dutch (NL),Spanish 23 | ln_ENABLE = Enable,aan,Activar 24 | ln_DISABLE =Disable,uit,Desactivar 25 | ln_INPUT = Input,ingang,Entrada 26 | ln_OUTPUT = Output,uitgang,Salida 27 | ln_PWM = PWM,PWM,PWM 28 | ln_SERVO = Servo,servo,Servo 29 | ln_TONE = Tone,toon,Tono 30 | ln_SONAR = SONAR,afstand,Sonar 31 | ln_OFF = Off,uitgeschakeld,No 32 | ln_ON = On,ingeschakeld,Si 33 | -------------------------------------------------------------------------------- /documentation/Español/s2a_fm_Espanish_tutorial.pdf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/documentation/Español/s2a_fm_Espanish_tutorial.pdf -------------------------------------------------------------------------------- /documentation/LED_EXAMPLE.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/documentation/LED_EXAMPLE.png -------------------------------------------------------------------------------- /documentation/pot1.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/documentation/pot1.png -------------------------------------------------------------------------------- /documentation/s2a_fm_reference - 完成稿.pdf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/documentation/s2a_fm_reference - 完成稿.pdf -------------------------------------------------------------------------------- /documentation/s2a_fm_reference.pdf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/documentation/s2a_fm_reference.pdf -------------------------------------------------------------------------------- /documentation/s2a_fm_參考文件.pdf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/documentation/s2a_fm_參考文件.pdf -------------------------------------------------------------------------------- /documentation/scratch_blocks.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/documentation/scratch_blocks.png -------------------------------------------------------------------------------- /documentation/scratch_langs.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/documentation/scratch_langs.png -------------------------------------------------------------------------------- /documentation/snap_blocks.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MrYsLab/s2a_fm/4a00749da24dae9152698c872a4f30d9a25cc214/documentation/snap_blocks.png -------------------------------------------------------------------------------- /extra_goodies/linux/s2a_fm.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | # Modify this file for your serial port designator. 4 | # Move this file to the directory that contains s2a_fm.py 5 | # and execute it by typing "./s2a_fm.sh" 6 | # Make sure that this file is executable as well as s2a_fm.py 7 | 8 | python s2a_fm.py /dev/ttyACM0 9 | 10 | -------------------------------------------------------------------------------- /extra_goodies/windows/s2a_fm.bat: -------------------------------------------------------------------------------- 1 | REM Modify this file for your serial port designator and for the location of 2 | REM python. 3 | REM Move this file to the directory that contains s2a_fm.py 4 | REM and execute it by opening the file explorer and double clicking. 5 | REM You can also create a desktop shortcut if you like. 6 | REM Right click on this file in the Windows File Explorer and select shortcut to create the shortcut. 7 | REM Make sure that this file is executable as well as s2a_fm.py 8 | 9 | c:\python27\python s2a_fm.py com3 10 | 11 | -------------------------------------------------------------------------------- /s2a_fm.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | 3 | # -*- coding: utf-8 -*- 4 | 5 | """ 6 | Created on Wed Nov 25 13:17:15 2013 7 | 8 | @author: Alan Yorinks 9 | Copyright (c) 2013-14 Alan Yorinks All right reserved. 10 | 11 | This program is free software; you can redistribute it and/or 12 | modify it under the terms of the GNU Lesser General Public 13 | License as published by the Free Software Foundation; either 14 | version 2.1 of the License, or (at your option) any later version. 15 | 16 | This library is distributed in the hope that it will be useful, 17 | but WITHOUT ANY WARRANTY; without even the implied warranty of 18 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 19 | Lesser General Public License for more details. 20 | 21 | You should have received a copy of the GNU Lesser General Public 22 | License along with this library; if not, write to the Free Software 23 | Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA 24 | 25 | """ 26 | import os 27 | import sys 28 | import logging 29 | from PyMata.pymata import PyMata 30 | import scratch_http_server 31 | from scratch_command_handlers import ScratchCommandHandlers 32 | import time 33 | 34 | 35 | #noinspection PyBroadException 36 | def s2a_fm(): 37 | 38 | """ 39 | This is the "main" function of the program. 40 | It will instantiate PyMata for communication with an Arduino micro-controller 41 | and the command handlers class. 42 | It will the start the HTTP server to communicate with Scratch 2.0 43 | @return : This is the main loop and should never return 44 | """ 45 | # total number of pins on arduino board 46 | total_pins_discovered = 0 47 | # number of pins that are analog 48 | number_of_analog_pins_discovered = 0 49 | 50 | # make sure we have a log directory and if not, create it. 51 | if not os.path.exists('log'): 52 | os.makedirs('log') 53 | 54 | # turn on logging 55 | logging.basicConfig(filename='./log/s2a_fm_debugging.log', filemode='w', level=logging.DEBUG) 56 | logging.info('s2a_fm version 1.5 Copyright(C) 2013-14 Alan Yorinks All Rights Reserved ') 57 | print 's2a_fm version 1.5 Copyright(C) 2013-14 Alan Yorinks All Rights Reserved ' 58 | 59 | # get the com_port from the command line or default if none given 60 | # if user specified the com port on the command line, use that when invoking PyMata, 61 | # else use '/dev/ttyACM0' 62 | if len(sys.argv) == 2: 63 | com_port = str(sys.argv[1]) 64 | else: 65 | com_port = '/dev/ttyACM0' 66 | logging.info('com port = %s' % com_port) 67 | 68 | try: 69 | # instantiate PyMata 70 | firmata = PyMata(com_port) # pragma: no cover 71 | except Exception: 72 | print 'Could not instantiate PyMata - is your Arduino plugged in?' 73 | logging.exception('Could not instantiate PyMata - is your Arduino plugged in?') 74 | logging.debug("Exiting s2a_fm") 75 | return 76 | 77 | # determine the total number of pins and the number of analog pins for the Arduino 78 | # get the arduino analog pin map 79 | # it will contain an entry for all the pins with non-analog set to firmata.IGNORE 80 | firmata.analog_mapping_query() 81 | 82 | capability_map = firmata.get_analog_mapping_request_results() 83 | 84 | firmata.capability_query() 85 | print "Please wait for Total Arduino Pin Discovery to complete. This can take up to 30 additional seconds." 86 | 87 | # count the pins 88 | for pin in capability_map: 89 | total_pins_discovered += 1 90 | # non analog pins will be marked as IGNORE 91 | if pin != firmata.IGNORE: 92 | number_of_analog_pins_discovered += 1 93 | 94 | # log the number of pins found 95 | logging.info('%d Total Pins and %d Analog Pins Found' % (total_pins_discovered, number_of_analog_pins_discovered)) 96 | 97 | # instantiate the command handler 98 | scratch_command_handler = ScratchCommandHandlers(firmata, com_port, total_pins_discovered, 99 | number_of_analog_pins_discovered) 100 | 101 | # wait for a maximum of 30 seconds to retrieve the Arduino capability query 102 | start_time = time.time() 103 | 104 | pin_capability = firmata.get_capability_query_results() 105 | while not pin_capability: 106 | if time.time() - start_time > 30: 107 | print '' 108 | print "Could not determine pin capability - exiting." 109 | firmata.close() 110 | # keep sending out a capability query until there is a response 111 | pin_capability = firmata.get_capability_query_results() 112 | time.sleep(.1) 113 | 114 | # we've got the capability, now build a dictionary with pin as the key and a list of all the capabilities 115 | # for the pin as the key's value 116 | pin_list = [] 117 | total_pins_discovered = 0 118 | for entry in pin_capability: 119 | # bump up pin counter each time IGNORE is found 120 | if entry == firmata.IGNORE: 121 | scratch_command_handler.pin_map[total_pins_discovered] = pin_list 122 | total_pins_discovered += 1 123 | pin_list = [] 124 | else: 125 | pin_list.append(entry) 126 | 127 | print "Arduino Total Pin Discovery completed in %d seconds" % (int(time.time() - start_time)) 128 | 129 | try: 130 | # start the server passing it the handle to PyMata and the command handler. 131 | scratch_http_server.start_server(firmata, scratch_command_handler) 132 | 133 | except Exception: 134 | logging.debug('Exception in s2a_fm.py %s' % str(Exception)) 135 | firmata.close() 136 | return 137 | 138 | except KeyboardInterrupt: 139 | # give control back to the shell that started us 140 | logging.info('s2a_fm.py: keyboard interrupt exception') 141 | firmata.close() 142 | return 143 | 144 | if __name__ == "__main__": 145 | s2a_fm() -------------------------------------------------------------------------------- /scratch_http_server.py: -------------------------------------------------------------------------------- 1 | # -*- coding: utf-8 -*- 2 | """ 3 | Created on Mon Nov 25 14:45:49 2013 4 | 5 | @author: Alan Yorinks 6 | Copyright (c) 2013-14 Alan Yorinks All right reserved. 7 | 8 | This program is free software; you can redistribute it and/or 9 | modify it under the terms of the GNU Lesser General Public 10 | License as published by the Free Software Foundation; either 11 | version 2.1 of the License, or (at your option) any later version. 12 | 13 | This library is distributed in the hope that it will be useful, 14 | but WITHOUT ANY WARRANTY; without even the implied warranty of 15 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 16 | Lesser General Public License for more details. 17 | 18 | You should have received a copy of the GNU Lesser General Public 19 | License along with this library; if not, write to the Free Software 20 | Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA 21 | """ 22 | 23 | import logging 24 | from BaseHTTPServer import BaseHTTPRequestHandler 25 | from BaseHTTPServer import HTTPServer 26 | from string import split 27 | 28 | 29 | class GetHandler(BaseHTTPRequestHandler): 30 | """ 31 | This class contains the HTTP server that Scratch2 communicates with 32 | Scratch sends HTTP GET requests and this class processes the requests. 33 | 34 | HTTP GET requests are accepted and the appropriate command handler is 35 | called to process the command. 36 | """ 37 | 38 | firmata = None 39 | 40 | # tcp server port - must match that in the .s2e descriptor file 41 | port = 50209 42 | 43 | # instance handle for the scratch command handler 44 | scratch_command_handler = None 45 | 46 | #indicator so that we can tell user Scratch is ready to go 47 | waiting_for_first_scratch_poll = True 48 | 49 | # this is a 'classmethod' because we need to set data before starting 50 | # the HTTP server. 51 | #noinspection PyMethodParameters 52 | @classmethod 53 | def set_items(self, firmata, scratch_command_handler): 54 | """ 55 | This method stores the input parameters for later use. 56 | It is a class method, because these values need to established 57 | prior to instantiating the class 58 | """ 59 | # instance variable for PyMata 60 | #noinspection PyAttributeOutsideInit 61 | self.firmata = firmata 62 | 63 | # instance variable for scratch command handler 64 | #noinspection PyAttributeOutsideInit 65 | self.command_handler = scratch_command_handler 66 | 67 | #noinspection PyPep8Naming 68 | def do_GET(self): 69 | """ 70 | Scratch2 only sends HTTP GET commands. This method processes them. 71 | It differentiates between a "normal" command request and a request 72 | to send policy information to keep Flash happy on Scratch. 73 | (This may change when Scratch is converted to HTML 5 74 | """ 75 | 76 | # skip over the / in the command 77 | cmd = self.path[1:] 78 | # create a list containing the command and all of its parameters 79 | cmd_list = split(cmd, '/') 80 | 81 | # get the command handler method for the command and call the handler 82 | # cmd_list[0] contains the command. look up the command method 83 | 84 | s = self.command_handler.do_command(cmd_list) 85 | 86 | # if pin was not enabled for reporter block, a "NoneType" can be returned by the command_handler 87 | if (s is None) or (len(s) == 0): 88 | 89 | err_statement = ("do_GET: Do you have all active pins enabled? " + str(cmd_list)) 90 | logging.info(err_statement) 91 | print err_statement 92 | return 93 | else: 94 | self.send_resp(s) 95 | 96 | # we can't use the standard send_response since we don't conform to its 97 | # standards, so we craft our own response handler here 98 | def send_resp(self, response): 99 | """ 100 | This method sends Scratch an HTTP response to an HTTP GET command. 101 | """ 102 | 103 | crlf = "\r\n" 104 | # http_response = str(response + crlf) 105 | http_response = "HTTP/1.1 200 OK" + crlf 106 | http_response += "Content-Type: text/html; charset=ISO-8859-1" + crlf 107 | http_response += "Content-Length" + str(len(response)) + crlf 108 | http_response += "Access-Control-Allow-Origin: *" + crlf 109 | http_response += crlf 110 | #add the response to the nonsense above 111 | if response != 'okay': 112 | http_response += str(response + crlf) 113 | # send it out the door to Scratch 114 | self.wfile.write(http_response) 115 | 116 | 117 | def start_server(firmata, command_handler): 118 | """ 119 | This function populates class variables with essential data and 120 | instantiates the HTTP Server 121 | """ 122 | 123 | GetHandler.set_items(firmata, command_handler) 124 | try: 125 | server = HTTPServer(('localhost', 50209), GetHandler) 126 | print 'Starting HTTP Server!' 127 | print 'Use to exit the extension\n' 128 | print 'Please start Scratch or Snap!' 129 | except Exception: 130 | logging.debug('Exception in scratch_http_server.py: HTTP Socket may already be in use - restart Scratch') 131 | print 'HTTP Socket may already be in use - restart Scratch' 132 | raise 133 | try: 134 | #start the server 135 | server.serve_forever() 136 | except KeyboardInterrupt: 137 | logging.info('scratch_http_server.py: keyboard interrupt exception') 138 | print "Goodbye !" 139 | raise KeyboardInterrupt 140 | except Exception: 141 | logging.debug('scratch_http_server.py: Exception %s' % str(Exception)) 142 | raise -------------------------------------------------------------------------------- /xlate.cfg: -------------------------------------------------------------------------------- 1 | # @author: Alan Yorinks 2 | # Copyright (c) 2013-2014 Alan Yorinks All right reserved. 3 | # 4 | # @co-author: Sjoerd Dirk Meijer, fromScratchEd.nl (language support) 5 | # @co-author: Professor José Manuel Ruiz (Spanish translation) 6 | # @co-author: Professor yufangjun (Chinese translation) 7 | # @co-author: Sebastien Canet (French translation) 8 | # @co-author: Aldo von Wangenheim, from www.computacaonaescola.ufsc.br (Portuguese translation) 9 | # @co-author: Aldo von Wangenheim, from www.computacaonaescola.ufsc.br (German translation) 10 | # @co-author: Dr. Eungil Kim (Korean Translation) 11 | # @co-author: Hsu Jen-Chieh (Chinese-TW translation) 12 | # @co-author: Gianfranco Zuliani (Italian translation) 13 | # @co-author: Alexandros Moskofidis (Greek translation) 14 | # @co-author: Antoine Choppin (Japanese translation) 15 | # 16 | # This program is free software; you can redistribute it and/or 17 | # modify it under the terms of the GNU Lesser General Public 18 | # License as published by the Free Software Foundation; either 19 | #version 2.1 of the License, or (at your option) any later version. 20 | # 21 | # This library is distributed in the hope that it will be useful, 22 | # but WITHOUT ANY WARRANTY; without even the implied warranty of 23 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 24 | # Lesser General Public License for more details. 25 | # 26 | # You should have received a copy of the GNU Lesser General Public License 27 | # along with this library; if not, write to the Free Software 28 | # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA 29 | 30 | # NOTE: The Chinese and French translation strings are the values that are sent by the 31 | # Scratch program. Also, for French and Chinese, the translations in Snap! are written directly 32 | # into the blocks. 33 | 34 | [translation_lists] 35 | ln_languages = English,Dutch (NL),Spanish,��������(zh_cn),French(FR),Portuguese(PT),German(DE),Korean(KO), Chinese(zh_tw),Italian(IT),Greek(GR),Japanese(JA),Japanese2(JA2) 36 | ln_ENABLE = Enable,aan,Activar,%E5%85%81%E8%AE%B8,Activer,Ativar,Setze,%EC%82%AC%EC%9A%A9,%E5%95%9F%E7%94%A8,Attiva,%CE%95%CE%BD%CE%B5%CF%81%CE%B3%CE%BF%CF%80%CE%BF%CE%AF%CE%B7%CF%83%CE%B7,%E6%9C%89%E5%8A%B9,%E4%BD%BF%E3%81%86 37 | ln_DISABLE =Disable,uit,Desactivar,%E7%A6%81%E6%AD%A2,Desactiver,Desativar,Deaktiviere,%EB%AF%B8%EC%82%AC%EC%9A%A9,%E5%81%9C%E7%94%A8,Disattiva,%CE%91%CF%80%CE%B5%CE%BD%CE%B5%CF%81%CE%B3%CE%BF%CF%80%CE%BF%CE%AF%CE%B7%CF%83%CE%B7,%E7%84%A1%E5%8A%B9,%E4%BD%BF%E3%82%8F%E3%81%AA%E3%81%84 38 | ln_INPUT = Input,ingang,Entrada,%E8%BE%93%E5%85%A5,entree,Entrada,Eingang,%EC%9E%85%EB%A0%A5,%E8%BC%B8%E5%85%A5,Ingresso,%CE%95%CE%AF%CF%83%CE%BF%CE%B4%CE%BF,%E5%85%A5%E5%8A%9B,%E5%85%A5%E5%8A%9B 39 | ln_OUTPUT = Output,uitgang,Salida,%E8%BE%93%E5%87%BA,sortie,Sa%C3%ADda,Ausgang,%EC%B6%9C%EB%A0%A5,%E8%BC%B8%E5%87%BA,Uscita,%CE%88%CE%BE%CE%BF%CE%B4%CE%BF,%E5%87%BA%E5%8A%9B,%E5%87%BA%E5%8A%9B 40 | ln_PWM = PWM,PWM,PWM,PWM,impulsion%20PWM,PWM,PWM,%EC%A0%84%EB%A5%98%EC%A1%B0%EC%A0%88,PWM,PWM,PWM,PWM%E6%B3%A2,PWM 41 | ln_SERVO = Servo,servo,Servo,%E8%88%B5%E6%9C%BA,rotation%20Servo-moteur.,Servo,Servo,%EC%84%9C%EB%B3%B4%EB%AA%A8%ED%84%B0,%E4%BC%BA%E6%9C%8D%E6%A9%9F,Servo,%CE%A3%CE%AD%CF%81%CE%B2%CE%BF,%E3%82%B5%E3%83%BC%E3%83%9C%E3%83%A2%E3%83%BC%E3%82%BF%E3%83%BC%E3%81%AE%E5%9B%9E%E8%BB%A2,%E3%82%B5%E3%83%BC%E3%83%9C 42 | ln_TONE = Tone,toon,Tono,%E9%9F%B3%E8%B0%83,sortie%20Son.,Som,Ton,%EC%8B%A0%ED%98%B8%EC%9D%8C,%E9%9F%B3%E8%AA%BF,Tono,%CE%A4%CF%8C%CE%BD%CE%BF%CF%82,%E9%9F%B3%E5%A3%B0%E5%87%BA%E5%8A%9B,%E9%9F%B3 43 | ln_SONAR = SONAR,afstand,Sonar,%E8%B6%85%E5%A3%B0%E6%B3%A2,mesure%20Sonar.,Ultrassom,Ultraschall,%EC%9D%8C%ED%8C%8C%ED%83%90%EC%A7%80,%E8%B6%85%E9%9F%B3%E6%B3%A2,Sonar,%CE%A3%CF%8C%CE%BD%CE%B1%CF%81,%E8%B6%85%E9%9F%B3%E6%B3%A2%E3%81%AE%E8%B7%9D%E9%9B%A2%E6%B8%AC%E5%AE%9A,%E3%82%BD%E3%83%8A%E3%83%BC 44 | ln_OFF = Off,uitgeschakeld,No,%E5%85%B3,Arret,N%C3%A3o,Nein,%EB%81%84%EA%B8%B0,%E9%97%9C,No,%CE%9A%CE%BB%CE%B5%CE%AF%CF%83%CE%B9%CE%BC%CE%BF,%E6%AD%A2%E3%82%81%E3%82%8B,%E6%AD%A2%E3%82%81%E3%82%8B 45 | ln_ON = On,ingeschakeld,Si,%E5%BC%80,Marche,Sim,Ja,%EC%BC%9C%EA%B8%B0,%E9%96%8B,Si,%CE%86%CE%BD%CE%BF%CE%B9%CE%B3%CE%BC%CE%B1,%E5%8B%95%E3%81%8B%E3%81%99,%E5%8B%95%E3%81%8B%E3%81%99 46 | --------------------------------------------------------------------------------