| 1 | /* | 
|---|
| 2 | * Web Server - for the FACT lid slow control | 
|---|
| 3 | * Need an Ethernet Shield over Arduino. | 
|---|
| 4 | * | 
|---|
| 5 | * based on the work of | 
|---|
| 6 | * Martyn Woerner | 
|---|
| 7 | * Alessandro Calzavara, alessandro(dot)calzavara(at)gmail(dot)com | 
|---|
| 8 | * and Alberto Capponi, bebbo(at)fast-labs net | 
|---|
| 9 | * for Arduino community! :-) | 
|---|
| 10 | * | 
|---|
| 11 | * | 
|---|
| 12 | */ | 
|---|
| 13 | #include <SPI.h> | 
|---|
| 14 | #include <Ethernet.h> | 
|---|
| 15 | #include <avr/pgmspace.h> | 
|---|
| 16 |  | 
|---|
| 17 | #include "ShutterController.h" | 
|---|
| 18 |  | 
|---|
| 19 | #define USE_DHCP_FOR_IP_ADDRESS | 
|---|
| 20 | #define ENABLE_ETHERNET | 
|---|
| 21 | #define SAMPLES  100 | 
|---|
| 22 |  | 
|---|
| 23 | // Define MAC and IP addresses | 
|---|
| 24 | byte _mac[] = { 0x90, 0xA2, 0xDA, 0x00, 0x5C, 0x91 }; | 
|---|
| 25 | //byte _mac[] = { 0x80, 0x49, 0x71, 0x0f, 0x80, 0xd0 }; | 
|---|
| 26 | #if !defined USE_DHCP_FOR_IP_ADDRESS | 
|---|
| 27 | // ip represents the fixed IP address to use if DHCP is disabled. | 
|---|
| 28 | IPAddress _ip(10,0,100,36); | 
|---|
| 29 | #endif | 
|---|
| 30 | // | 
|---|
| 31 |  | 
|---|
| 32 | // Each value is the real current value in the motors; | 
|---|
| 33 | // Define Current Limits in [A] - Offset is 0.5A for no load on the motors | 
|---|
| 34 | // pushing coefficient ~100 Kg/A | 
|---|
| 35 | const double _ZeroCurrent         PROGMEM = 0.25; // [A] | 
|---|
| 36 | const double _CurrentPushingLimit PROGMEM = 0.75; // 0.7-0.5 = 0.2 -> 20  +/- 5 kg | 
|---|
| 37 | const double _OverCurrent         PROGMEM = 1.50; // 1.5-0.5 = 1   -> 100 +/- 5 kg | 
|---|
| 38 |  | 
|---|
| 39 | const int _StartPoint      =     0; | 
|---|
| 40 | const int _StartPointLimit =    70; | 
|---|
| 41 | const int _EndPoint        =  1024; | 
|---|
| 42 | const int _EndPointLimit   =   775; // This must me 740 for the mockup and >770 for the FACT shutter | 
|---|
| 43 |  | 
|---|
| 44 | // Define Lid Status levels and labels | 
|---|
| 45 | const int _UNKNOWN        =  0; | 
|---|
| 46 | const int _CLOSED         =  1; | 
|---|
| 47 | const int _OPEN           =  2; | 
|---|
| 48 | const int _STEADY         =  3; | 
|---|
| 49 | const int _MOVING         =  4; | 
|---|
| 50 | const int _CLOSING        =  5; | 
|---|
| 51 | const int _OPENING        =  6; | 
|---|
| 52 | const int _JAMMED         =  7; | 
|---|
| 53 | const int _MOTOR_FAULT    =  8; | 
|---|
| 54 | const int _POWER_PROBLEM  =  9; | 
|---|
| 55 | const int _OVER_CURRENT   = 10; | 
|---|
| 56 |  | 
|---|
| 57 | const char* _StatusLabel[]  = {"Unknown", | 
|---|
| 58 | "Closed", | 
|---|
| 59 | "Open", | 
|---|
| 60 | "Steady", | 
|---|
| 61 | "Moving", | 
|---|
| 62 | "Closing", | 
|---|
| 63 | "Opening", | 
|---|
| 64 | "Jammed", | 
|---|
| 65 | "Motor Fault", | 
|---|
| 66 | "Power Problem", | 
|---|
| 67 | "Overcurrent"}; | 
|---|
| 68 | // Define Arduino pins | 
|---|
| 69 | const int _pinPWM[2]  = {5,   6}; | 
|---|
| 70 | const int _pinDA[2]   = {2,   7}; | 
|---|
| 71 | const int _pinDB[2]   = {3,   8}; | 
|---|
| 72 | //unsigned char _ENDIAG[2]   = {A4, A5}; | 
|---|
| 73 |  | 
|---|
| 74 |  | 
|---|
| 75 | // Define conversion coefficients | 
|---|
| 76 | double _ADC2V  = 5. / 1024. ; // ADC channel to Volt | 
|---|
| 77 | double _V2A    = 0.140;       // 140 mV/A conversion factor for the | 
|---|
| 78 |  | 
|---|
| 79 |  | 
|---|
| 80 | // Define sensor value and lid status variables | 
|---|
| 81 | double  _sensorValue[2]    = {0,0}; | 
|---|
| 82 | double  _currentValue[2]   = {0,0}; | 
|---|
| 83 | uint8_t _LidStatus[2]      = {0,0}; | 
|---|
| 84 |  | 
|---|
| 85 | extern int  __bss_end; | 
|---|
| 86 | extern void *__brkval; | 
|---|
| 87 |  | 
|---|
| 88 | // Http header token delimiters | 
|---|
| 89 | char *pSpDelimiters = " \r\n"; | 
|---|
| 90 | char *pStxDelimiter = "\002";    // STX - ASCII start of text character | 
|---|
| 91 |  | 
|---|
| 92 | /********************************************************************************************************************** | 
|---|
| 93 | *                                   Strings stored in flash of the HTML we will be transmitting | 
|---|
| 94 | ***********************************************************************************************************************/ | 
|---|
| 95 |  | 
|---|
| 96 | // HTTP Request message | 
|---|
| 97 | prog_char content_404[]        PROGMEM = "HTTP/1.1 404 Not Found\nServer: arduino\nContent-Type: text/html\n\n<html>" | 
|---|
| 98 | "<head><title>Arduino Web Server - Error 404</title></head>" | 
|---|
| 99 | "<body><h1>Error 404: Sorry, that page cannot be found!</h1></body>"; | 
|---|
| 100 | PROGMEM const char *page_404[]               = { content_404 }; // table with 404 page | 
|---|
| 101 |  | 
|---|
| 102 | // HTML Header for pages | 
|---|
| 103 | const prog_char content_main_header[] PROGMEM= "HTTP/1.0 200 OK\nServer: arduino\nCache-Control: no-store, no-cache, must-revalidate\n" | 
|---|
| 104 | "Pragma: no-cache\nConnection: close\nContent-Type: text/html\n"; | 
|---|
| 105 |  | 
|---|
| 106 | const prog_char content_main_top[]    PROGMEM   = "<html><head><meta http-equiv=\"refresh\" content=\"5\"/><title>Arduino Web Server</title>" | 
|---|
| 107 | "<style type=\"text/css\">table{border-collapse:collapse;}td{padding:0.25em 0.5em;border:0.5em solid #C8C8C8;}</style>" | 
|---|
| 108 | "</head><body><h1>Arduino Web Server</h1>"; | 
|---|
| 109 |  | 
|---|
| 110 |  | 
|---|
| 111 | const prog_char content_main_menu[]   PROGMEM = "<table width=\"500\"><tr><td align=\"center\"><a href=\"/\">Page 1</a></td></tr></table>"; | 
|---|
| 112 | const prog_char content_main_footer[] PROGMEM = "</body></html>"; | 
|---|
| 113 | PGM_P contents_main[] PROGMEM = { content_main_header, content_main_top, content_main_menu, content_main_footer }; // table with 404 page | 
|---|
| 114 |  | 
|---|
| 115 | #define CONT_HEADER 0 | 
|---|
| 116 | #define CONT_TOP 1 | 
|---|
| 117 | #define CONT_MENU 2 | 
|---|
| 118 | #define CONT_FOOTER 3 | 
|---|
| 119 |  | 
|---|
| 120 |  | 
|---|
| 121 | // Page 1 | 
|---|
| 122 | const PROGMEM prog_char http_uri1[] = "/"; | 
|---|
| 123 | const PROGMEM prog_char content_title1[] = "<h2>Shutter Lid Control - Beta</h2>"; | 
|---|
| 124 | #ifdef DEBUG | 
|---|
| 125 | // To Be Fixed | 
|---|
| 126 | const PROGMEM prog_char content_page1[] = "<hr /><form action=\"/__output__\" method=\"POST\">" | 
|---|
| 127 | "<button name=\"Button1\" value=\"valueButton1\" type=\"submit\">Move Motor 1 OUT </button><p>" | 
|---|
| 128 | "<button name=\"Button2\" value=\"valueButton2\" type=\"submit\">Move Motor 1 IN </button><p>" | 
|---|
| 129 | "<button name=\"Button3\" value=\"valueButton3\" type=\"submit\">Move Motor 2 OUT </button><p>" | 
|---|
| 130 | "<button name=\"Button4\" value=\"valueButton4\" type=\"submit\">Move Motor 2 IN </button><p>" | 
|---|
| 131 | "<button name=\"Button5\" value=\"valueButton5\" type=\"submit\">Open Lid</button><p>" | 
|---|
| 132 | "<button name=\"Button6\" value=\"valueButton6\" type=\"submit\">Close Lid</button><p>" | 
|---|
| 133 | "Motor Current 1 = \002 A <p> Motor Current 2 = \002 A <p> Hall Sensor 1 = \002 ADC counts <p> Hall Sensor 2 = \002 ADC counts<p>" | 
|---|
| 134 | "</form></p>"; | 
|---|
| 135 | #else | 
|---|
| 136 | const PROGMEM prog_char content_page1[] = "<hr/><form action=\"/__output__\" method=\"POST\">" | 
|---|
| 137 | "<button name=\"Button5\" value=\"valueButton5\" type=\"submit\">Open Lid</button><p/>" | 
|---|
| 138 | "<button name=\"Button6\" value=\"valueButton6\" type=\"submit\">Close Lid</button><p/>" | 
|---|
| 139 | "<span id=\"cur1\" value=\"\002\"> Motor Current 1 = \002 A </span><p/><span id=\"cur2\" value=\"\002\"> Motor Current 2 = \002 A </span><p/>" | 
|---|
| 140 | "<span id=\"pos1\" value=\"\002\"> Hall Sensor 1 = \002 ADC counts </span><p/><span id=\"pos2\" value=\"\002\"> Hall Sensor 2 = \002 ADC counts</span><p/>" | 
|---|
| 141 | "</form></p>"; | 
|---|
| 142 | //<br/></body></html> | 
|---|
| 143 | #endif | 
|---|
| 144 |  | 
|---|
| 145 | // Page 5 | 
|---|
| 146 |  | 
|---|
| 147 | const PROGMEM prog_char http_uri5[] = "/__output__"; | 
|---|
| 148 | const PROGMEM prog_char content_title5[] = "<h2>Shutter Lid Control - Beta</h2>"; | 
|---|
| 149 | #ifdef DEBUG | 
|---|
| 150 | // To Be Fixed | 
|---|
| 151 | const PROGMEM prog_char content_page5[] = "<hr /><form action=\"/__output__\" method=\"POST\">" | 
|---|
| 152 | "<button name=\"Button1\" value=\"valueButton1\" type=\"submit\">Move Motor 1 OUT </button><p>" | 
|---|
| 153 | "<button name=\"Button2\" value=\"valueButton2\" type=\"submit\">Move Motor 1 IN </button><p>" | 
|---|
| 154 | "<button name=\"Button3\" value=\"valueButton3\" type=\"submit\">Move Motor 2 OUT </button><p>" | 
|---|
| 155 | "<button name=\"Button4\" value=\"valueButton4\" type=\"submit\">Move Motor 2 IN </button><p>" | 
|---|
| 156 | "<button name=\"Button5\" value=\"valueButton5\" type=\"submit\">Open Lid</button><p>" | 
|---|
| 157 | "<button name=\"Button6\" value=\"valueButton6\" type=\"submit\">Close Lid</button><p>" | 
|---|
| 158 | "Motor Current 1 = \"\002\" A <p> Motor Current 2 = \"\002\" A <p> Hall Sensor 1 = \"\002\" ADC ounts <p> Hall Sensor 2 = \"\002\" ADC counts<p>" | 
|---|
| 159 | "Lid 1 Status : \"\002\"<p>" | 
|---|
| 160 | "Lid 2 Status : \"\002\"<p><p>" | 
|---|
| 161 | "received a POST request</p></form></p>"; | 
|---|
| 162 | #else | 
|---|
| 163 | const PROGMEM prog_char content_page5[] = "<hr/><form action=\"/__output__\" method=\"POST\">" | 
|---|
| 164 | "<button name=\"Button5\" value=\"valueButton5\" type=\"submit\">Open Lid</button><p/>" | 
|---|
| 165 | "<button name=\"Button6\" value=\"valueButton6\" type=\"submit\">Close Lid</button><p/>" | 
|---|
| 166 | "<span id=\"cur1\" value=\"\002\"> Motor Current 1 = \002 A </span><p/><span id=\"cur2\" value=\"\002\"> Motor Current 2 = \002 A </span><p/>" | 
|---|
| 167 | "<span id=\"pos1\" value=\"\002\"> Hall Sensor 1 = \002 ADC ounts </span><p/><span id=\"pos2\" value=\"\002\"> Hall Sensor 2 = \002 ADC counts</span><p/>" | 
|---|
| 168 | "<span id=\"lid1\" value=\"\002\"> Lid 1 Status : \002</span><p/>" | 
|---|
| 169 | "<span id=\"lid2\" value=\"\002\"> Lid 2 Status : \002</span><p/>" | 
|---|
| 170 | "<p/>received a POST request</form><p/>"; | 
|---|
| 171 | //                                    <br/></body></html> | 
|---|
| 172 | #endif | 
|---|
| 173 |  | 
|---|
| 174 | // declare tables for the pages | 
|---|
| 175 | PGM_P contents_titles[] PROGMEM = { content_title1,  content_title5 }; // titles | 
|---|
| 176 | PGM_P contents_pages [] PROGMEM = { content_page1,  content_page5 }; // real content | 
|---|
| 177 |  | 
|---|
| 178 | #ifdef USE_IMAGES | 
|---|
| 179 |  | 
|---|
| 180 | /********************************************************************************************************************** | 
|---|
| 181 | *                     Image strings and data stored in flash for the image UISs we will be transmitting | 
|---|
| 182 | ***********************************************************************************************************************/ | 
|---|
| 183 |  | 
|---|
| 184 | // A Favicon is a little custom icon that appears next to a website's URL in the address bar of a web browser. | 
|---|
| 185 | // They also show up in your bookmarked sites, on the tabs in tabbed browsers, and as the icon for Internet shortcuts | 
|---|
| 186 | // on your desktop or other folders in Windows. | 
|---|
| 187 | const PROGMEM prog_char http_uri6[] = "/favicon.ico";    // favicon Request message | 
|---|
| 188 |  | 
|---|
| 189 | const PROGMEM prog_char content_image_header[] = "HTTP/1.1 200 OK\nServer: arduino\nContent-Length: \002\nContent-Type: image/\002\n\r\n"; | 
|---|
| 190 |  | 
|---|
| 191 | #ifdef USE_ARDUINO_ICON | 
|---|
| 192 |  | 
|---|
| 193 | const PROGMEM prog_char content_favicon_data[] = { | 
|---|
| 194 | 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x10, 0x10, 0x10, 0x00, 0x01, 0x00, 0x04, 0x00, 0x28, 0x01, 0x00, 0x00, 0x16, 0x00, | 
|---|
| 195 | 0x00, 0x00, 0x28, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x20, 0x00, 0x00, 0x00, 0x01, 0x00, 0x04, 0x00, 0x00, 0x00, | 
|---|
| 196 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 197 | 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x80, 0x00, 0xC0, 0xC0, 0xC0, 0x00, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, | 
|---|
| 198 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 199 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 200 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x44, 0x44, 0x41, 0x00, 0x00, 0x14, 0x44, 0x44, 0x44, 0x41, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 201 | 0x14, 0x44, 0x44, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x44, 0x41, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x14, 0x40, 0x23, | 
|---|
| 202 | 0x33, 0x20, 0x02, 0x33, 0x32, 0x04, 0x12, 0x30, 0x00, 0x32, 0x23, 0x00, 0x03, 0x21, 0x03, 0x00, 0x00, 0x03, 0x30, 0x03, | 
|---|
| 203 | 0x00, 0x30, 0x03, 0x03, 0x33, 0x03, 0x30, 0x33, 0x30, 0x30, 0x03, 0x00, 0x00, 0x03, 0x30, 0x03, 0x00, 0x30, 0x02, 0x30, | 
|---|
| 204 | 0x00, 0x32, 0x23, 0x00, 0x03, 0x20, 0x10, 0x23, 0x33, 0x20, 0x02, 0x33, 0x32, 0x01, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 205 | 0x00, 0x04, 0x41, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x14, 0x44, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x44, 0x44, 0x41, | 
|---|
| 206 | 0x00, 0x00, 0x00, 0x00, 0x14, 0x44, 0x44, 0x44, 0x41, 0x00, 0x00, 0x14, 0x44, 0x44, 0xF8, 0x1F, 0x00, 0x00, 0xE0, 0x07, | 
|---|
| 207 | 0x00, 0x00, 0xC0, 0x03, 0x00, 0x00, 0x80, 0x01, 0x00, 0x00, 0x80, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 208 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 0x01, | 
|---|
| 209 | 0x00, 0x00, 0x80, 0x01, 0x00, 0x00, 0xC0, 0x03, 0x00, 0x00, 0xE0, 0x07, 0x00, 0x00, 0xF8, 0x1F, 0x00, 0x00 | 
|---|
| 210 | }; | 
|---|
| 211 | #else | 
|---|
| 212 |  | 
|---|
| 213 | const PROGMEM prog_char content_favicon_data[] = { | 
|---|
| 214 | 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x10, 0x10, 0x10, 0x00, 0x01, 0x00, 0x04, 0x00, 0x28, 0x01, 0x00, 0x00, 0x16, 0x00, | 
|---|
| 215 | 0x00, 0x00, 0x28, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x20, 0x00, 0x00, 0x00, 0x01, 0x00, 0x04, 0x00, 0x00, 0x00, | 
|---|
| 216 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 217 | 0x00, 0x00, 0x00, 0x00, 0x66, 0x00, 0x00, 0x33, 0x66, 0x00, 0x33, 0x33, 0x66, 0x00, 0x66, 0x66, 0x99, 0x00, 0x99, 0x99, | 
|---|
| 218 | 0x99, 0x00, 0x99, 0x99, 0xCC, 0x00, 0xCC, 0xCC, 0xCC, 0x00, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 219 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 220 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x05, 0x62, 0x00, | 
|---|
| 221 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x47, 0x76, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x67, 0x77, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 222 | 0x00, 0x26, 0x63, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0x50, 0x00, | 
|---|
| 223 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x05, 0x60, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x70, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 224 | 0x00, 0x07, 0x72, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x37, 0x75, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x67, 0x76, 0x00, | 
|---|
| 225 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x67, 0x77, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x67, 0x77, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 226 | 0x00, 0x37, 0x73, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 227 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 228 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | 
|---|
| 229 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 | 
|---|
| 230 | }; | 
|---|
| 231 | #endif  // USE_ARDUINO_ICON | 
|---|
| 232 |  | 
|---|
| 233 | // declare tables for the images | 
|---|
| 234 | PGM_P image_header PROGMEM = content_image_header; | 
|---|
| 235 | PGM_P data_for_images [] PROGMEM = { content_favicon_data}; // real data | 
|---|
| 236 | const int   size_for_images [] PROGMEM = { sizeof(content_favicon_data)}; | 
|---|
| 237 | // declare table for all URIs | 
|---|
| 238 | PGM_P http_uris[] PROGMEM = { http_uri1, http_uri5, http_uri6 }; // URIs | 
|---|
| 239 |  | 
|---|
| 240 |  | 
|---|
| 241 | #else | 
|---|
| 242 | PGM_P image_header PROGMEM = NULL; | 
|---|
| 243 | PGM_P data_for_images [] PROGMEM = { }; // real data | 
|---|
| 244 | const int   size_for_images [] PROGMEM = { }; | 
|---|
| 245 | // declare table for all URIs | 
|---|
| 246 | PGM_P http_uris[] PROGMEM = { http_uri1, http_uri5 }; // URIs | 
|---|
| 247 |  | 
|---|
| 248 | #endif  // USE_IMAGES | 
|---|
| 249 |  | 
|---|
| 250 | #define NUM_PAGES  sizeof(contents_pages)  / sizeof(PGM_P) | 
|---|
| 251 | #define NUM_IMAGES sizeof(data_for_images) / sizeof(PGM_P)    // favicon or png format | 
|---|
| 252 | #define NUM_URIS  NUM_PAGES + NUM_IMAGES  // Pages URIs + favicon URI, etc | 
|---|
| 253 |  | 
|---|
| 254 | /********************************************************************************************************************** | 
|---|
| 255 | *                                                 Shared variable and Setup() | 
|---|
| 256 | ***********************************************************************************************************************/ | 
|---|
| 257 | EthernetServer server(80); | 
|---|
| 258 |  | 
|---|
| 259 | void setup() | 
|---|
| 260 | { | 
|---|
| 261 | Serial.begin(115200); // DEBUG | 
|---|
| 262 |  | 
|---|
| 263 | #ifdef ENABLE_ETHERNET | 
|---|
| 264 | #ifdef USE_DHCP_FOR_IP_ADDRESS | 
|---|
| 265 | Serial.println("Attempting to obtain a DHCP lease..."); | 
|---|
| 266 |  | 
|---|
| 267 | Ethernet.begin(_mac);  // Use DHCP to get an IP address | 
|---|
| 268 | #else | 
|---|
| 269 | Serial.println("Using hard-coded ip..."); | 
|---|
| 270 | Ethernet.begin(_mac, _ip); | 
|---|
| 271 | #endif | 
|---|
| 272 |  | 
|---|
| 273 |  | 
|---|
| 274 | Serial.println("A DHCP lease has been obtained."); | 
|---|
| 275 |  | 
|---|
| 276 | Serial.print("My IP address is "); | 
|---|
| 277 | Serial.println(Ethernet.localIP()); | 
|---|
| 278 |  | 
|---|
| 279 |  | 
|---|
| 280 |  | 
|---|
| 281 | server.begin(); | 
|---|
| 282 | #endif | 
|---|
| 283 | //_Motor.init(); | 
|---|
| 284 |  | 
|---|
| 285 | //For Arduino Motor Shield set pin 4,5,6,7 to output mode | 
|---|
| 286 | pinMode(2, OUTPUT); | 
|---|
| 287 | pinMode(4, OUTPUT); | 
|---|
| 288 | pinMode(5, OUTPUT); | 
|---|
| 289 | pinMode(6, OUTPUT); | 
|---|
| 290 | pinMode(7, OUTPUT); | 
|---|
| 291 | pinMode(8, OUTPUT); | 
|---|
| 292 |  | 
|---|
| 293 | pinMode(A0, INPUT);  // | 
|---|
| 294 | pinMode(A1, INPUT);  // | 
|---|
| 295 | pinMode(A2, INPUT);  // | 
|---|
| 296 | pinMode(A3, INPUT);  // | 
|---|
| 297 |  | 
|---|
| 298 | // First clear all three prescaler bits: | 
|---|
| 299 | int prescalerVal = 0x07; // create a variable called prescalerVal and set it equal to the binary number "00000111" | 
|---|
| 300 |  | 
|---|
| 301 | TCCR0B &= ~prescalerVal; // AND the value in TCCR0B with binary number "11111000" | 
|---|
| 302 |  | 
|---|
| 303 | //Now set the appropriate prescaler bits: | 
|---|
| 304 | prescalerVal = 3; // set prescalerVal equal to binary number "00000001" | 
|---|
| 305 | TCCR0B |= prescalerVal; // OR the value in TCCR0B with binary number "00000001" | 
|---|
| 306 |  | 
|---|
| 307 | //  TCCR0A = _BV(COM0A0) | _BV(COM0B1) | _BV(WGM01) | _BV(WGM00); | 
|---|
| 308 | //  TCCR0B = _BV(WGM02) | _BV(CS02); | 
|---|
| 309 | //  OCR0A = 180; | 
|---|
| 310 | //  OCR0B =  50; | 
|---|
| 311 |  | 
|---|
| 312 | //pinMode(ledPin, OUTPUT); | 
|---|
| 313 | //setLed(true); | 
|---|
| 314 | } | 
|---|
| 315 |  | 
|---|
| 316 | /********************************************************************************************************************** | 
|---|
| 317 | *                                                           Main loop | 
|---|
| 318 | ***********************************************************************************************************************/ | 
|---|
| 319 |  | 
|---|
| 320 | void loop() | 
|---|
| 321 | { | 
|---|
| 322 | #ifdef ENABLE_ETHERNET | 
|---|
| 323 | EthernetClient client = server.available(); | 
|---|
| 324 | #endif | 
|---|
| 325 |  | 
|---|
| 326 |  | 
|---|
| 327 | #ifdef ENABLE_ETHERNET | 
|---|
| 328 | if (client) | 
|---|
| 329 | { | 
|---|
| 330 | // now client is connected to arduino we need to extract the | 
|---|
| 331 | // following fields from the HTTP request. | 
|---|
| 332 | int    nUriIndex;  // Gives the index into table of recognized URIs or -1 for not found. | 
|---|
| 333 | BUFFER requestContent;    // Request content as a null-terminated string. | 
|---|
| 334 | MethodType eMethod = readHttpRequest(client, nUriIndex, requestContent); | 
|---|
| 335 |  | 
|---|
| 336 | #ifdef DEBUG | 
|---|
| 337 | Serial.print(" o- Read Request type: "); | 
|---|
| 338 | Serial.print(eMethod); | 
|---|
| 339 | Serial.print(" Uri index: "); | 
|---|
| 340 | Serial.print(nUriIndex); | 
|---|
| 341 | Serial.print(" content: "); | 
|---|
| 342 | Serial.print(requestContent); | 
|---|
| 343 | Serial.print("\n"); | 
|---|
| 344 | #endif | 
|---|
| 345 | if (nUriIndex < 0) | 
|---|
| 346 | { | 
|---|
| 347 | // URI not found | 
|---|
| 348 | sendProgMemAsString(client, (char*)pgm_read_word(&(page_404[0]))); | 
|---|
| 349 | } | 
|---|
| 350 | else if (nUriIndex < NUM_PAGES) | 
|---|
| 351 | { | 
|---|
| 352 | // Normal page request, may depend on content of the request | 
|---|
| 353 | #ifdef DEBUG | 
|---|
| 354 | Serial.println(" o- Sending page"); | 
|---|
| 355 | #endif | 
|---|
| 356 | sendPage(client, nUriIndex, requestContent); | 
|---|
| 357 | } | 
|---|
| 358 | else | 
|---|
| 359 | { | 
|---|
| 360 | // Image request | 
|---|
| 361 | sendImage(client, nUriIndex, requestContent); | 
|---|
| 362 | } | 
|---|
| 363 |  | 
|---|
| 364 | // give the web browser time to receive the data | 
|---|
| 365 | //  delay(1); | 
|---|
| 366 | delay(100); | 
|---|
| 367 |  | 
|---|
| 368 | client.stop(); | 
|---|
| 369 | } | 
|---|
| 370 | #endif | 
|---|
| 371 |  | 
|---|
| 372 |  | 
|---|
| 373 | } | 
|---|
| 374 |  | 
|---|
| 375 | #ifdef ENABLE_ETHERNET | 
|---|
| 376 | /********************************************************************************************************************** | 
|---|
| 377 | *                                              Method for read HTTP Header Request from web client | 
|---|
| 378 | * | 
|---|
| 379 | * The HTTP request format is defined at http://www.w3.org/Protocols/HTTP/1.0/spec.html#Message-Types | 
|---|
| 380 | * and shows the following structure: | 
|---|
| 381 | *  Full-Request and Full-Response use the generic message format of RFC 822 [7] for transferring entities. Both messages may include optional header fields | 
|---|
| 382 | *  (also known as "headers") and an entity body. The entity body is separated from the headers by a null line (i.e., a line with nothing preceding the CRLF). | 
|---|
| 383 | *      Full-Request   = Request-Line | 
|---|
| 384 | *                       *( General-Header | 
|---|
| 385 | *                        | Request-Header | 
|---|
| 386 | *                        | Entity-Header ) | 
|---|
| 387 | *                       CRLF | 
|---|
| 388 | *                       [ Entity-Body ] | 
|---|
| 389 | * | 
|---|
| 390 | * The Request-Line begins with a method token, followed by the Request-URI and the protocol version, and ending with CRLF. The elements are separated by SP characters. | 
|---|
| 391 | * No CR or LF are allowed except in the final CRLF sequence. | 
|---|
| 392 | *      Request-Line   = Method SP Request-URI SP HTTP-Version CRLF | 
|---|
| 393 | * HTTP header fields, which include General-Header, Request-Header, Response-Header, and Entity-Header fields, follow the same generic format. | 
|---|
| 394 | * Each header field consists of a name followed immediately by a colon (":"), a single space (SP) character, and the field value. | 
|---|
| 395 | * Field names are case-insensitive. Header fields can be extended over multiple lines by preceding each extra line with at least one SP or HT, though this is not recommended. | 
|---|
| 396 | *      HTTP-header    = field-name ":" [ field-value ] CRLF | 
|---|
| 397 | ***********************************************************************************************************************/ | 
|---|
| 398 | // Read HTTP request, setting Uri Index, the requestContent and returning the method type. | 
|---|
| 399 | MethodType readHttpRequest(EthernetClient & client, int & nUriIndex, BUFFER & requestContent) | 
|---|
| 400 | { | 
|---|
| 401 |  | 
|---|
| 402 |  | 
|---|
| 403 | BUFFER readBuffer;    // Just a work buffer into which we can read records | 
|---|
| 404 | int nContentLength = 0; | 
|---|
| 405 | bool bIsUrlEncoded; | 
|---|
| 406 |  | 
|---|
| 407 | requestContent[0] = 0;    // Initialize as an empty string | 
|---|
| 408 | // Read the first line: Request-Line setting Uri Index and returning the method type. | 
|---|
| 409 | MethodType eMethod = readRequestLine(client, readBuffer, nUriIndex, requestContent); | 
|---|
| 410 | // Read any following, non-empty headers setting content length. | 
|---|
| 411 | readRequestHeaders(client, readBuffer, nContentLength, bIsUrlEncoded); | 
|---|
| 412 |  | 
|---|
| 413 | if (nContentLength > 0) | 
|---|
| 414 | { | 
|---|
| 415 | // If there is some content then read it and do an elementary decode. | 
|---|
| 416 | readEntityBody(client, nContentLength, requestContent); | 
|---|
| 417 | if (bIsUrlEncoded) | 
|---|
| 418 | { | 
|---|
| 419 | // The '+' encodes for a space, so decode it within the string | 
|---|
| 420 | for (char * pChar = requestContent; (pChar = strchr(pChar, '+')) != NULL; ) | 
|---|
| 421 | *pChar = ' ';    // Found a '+' so replace with a space | 
|---|
| 422 | } | 
|---|
| 423 | } | 
|---|
| 424 |  | 
|---|
| 425 | return eMethod; | 
|---|
| 426 | } | 
|---|
| 427 |  | 
|---|
| 428 | // Read the first line of the HTTP request, setting Uri Index and returning the method type. | 
|---|
| 429 | // If it is a GET method then we set the requestContent to whatever follows the '?'. For a other | 
|---|
| 430 | // methods there is no content except it may get set later, after the headers for a POST method. | 
|---|
| 431 | MethodType readRequestLine(EthernetClient & client, BUFFER & readBuffer, int & nUriIndex, BUFFER & requestContent) | 
|---|
| 432 | { | 
|---|
| 433 | MethodType eMethod; | 
|---|
| 434 | // Get first line of request: | 
|---|
| 435 | // Request-Line = Method SP Request-URI SP HTTP-Version CRLF | 
|---|
| 436 | getNextHttpLine(client, readBuffer); | 
|---|
| 437 | // Split it into the 3 tokens | 
|---|
| 438 | char * pMethod  = strtok(readBuffer, pSpDelimiters); | 
|---|
| 439 | char * pUri     = strtok(NULL, pSpDelimiters); | 
|---|
| 440 | char * pVersion = strtok(NULL, pSpDelimiters); | 
|---|
| 441 | // URI may optionally comprise the URI of a queryable object a '?' and a query | 
|---|
| 442 | // see http://www.ietf.org/rfc/rfc1630.txt | 
|---|
| 443 | strtok(pUri, "?"); | 
|---|
| 444 | char * pQuery   = strtok(NULL, "?"); | 
|---|
| 445 | if (pQuery != NULL) | 
|---|
| 446 | { | 
|---|
| 447 | strcpy(requestContent, pQuery); | 
|---|
| 448 | // The '+' encodes for a space, so decode it within the string | 
|---|
| 449 | for (pQuery = requestContent; (pQuery = strchr(pQuery, '+')) != NULL; ) | 
|---|
| 450 | *pQuery = ' ';    // Found a '+' so replace with a space | 
|---|
| 451 |  | 
|---|
| 452 | //    Serial.print("Get query string: "); | 
|---|
| 453 | //    Serial.println(requestContent); | 
|---|
| 454 | } | 
|---|
| 455 | if (strcmp(pMethod, "GET") == 0){ | 
|---|
| 456 | eMethod = MethodGet; | 
|---|
| 457 | #ifdef DEBUG | 
|---|
| 458 | Serial.println("readRequestLine-> GET"); | 
|---|
| 459 | #endif | 
|---|
| 460 | } | 
|---|
| 461 | else if (strcmp(pMethod, "POST") == 0){ | 
|---|
| 462 | eMethod = MethodPost; | 
|---|
| 463 | #ifdef DEBUG | 
|---|
| 464 | Serial.println("readRequestLine-> POST"); | 
|---|
| 465 | #endif | 
|---|
| 466 | } | 
|---|
| 467 | else if (strcmp(pMethod, "HEAD") == 0){ | 
|---|
| 468 | eMethod = MethodHead; | 
|---|
| 469 | #ifdef DEBUG | 
|---|
| 470 | Serial.println("readRequestLine-> HEAD"); | 
|---|
| 471 | #endif | 
|---|
| 472 | } | 
|---|
| 473 | else | 
|---|
| 474 | eMethod = MethodUnknown; | 
|---|
| 475 |  | 
|---|
| 476 | // See if we recognize the URI and get its index | 
|---|
| 477 | nUriIndex = GetUriIndex(pUri); | 
|---|
| 478 |  | 
|---|
| 479 | return eMethod; | 
|---|
| 480 | } | 
|---|
| 481 |  | 
|---|
| 482 | // Read each header of the request till we get the terminating CRLF | 
|---|
| 483 | void readRequestHeaders(EthernetClient & client, BUFFER & readBuffer, int & nContentLength, bool & bIsUrlEncoded) | 
|---|
| 484 | { | 
|---|
| 485 | nContentLength = 0;      // Default is zero in cate there is no content length. | 
|---|
| 486 | bIsUrlEncoded  = true;   // Default encoding | 
|---|
| 487 | // Read various headers, each terminated by CRLF. | 
|---|
| 488 | // The CRLF gets removed and the buffer holds each header as a string. | 
|---|
| 489 | // An empty header of zero length terminates the list. | 
|---|
| 490 | do | 
|---|
| 491 | { | 
|---|
| 492 | getNextHttpLine(client, readBuffer); | 
|---|
| 493 | //    Serial.println(readBuffer); // DEBUG | 
|---|
| 494 | // Process a header. We only need to extract the (optionl) content | 
|---|
| 495 | // length for the binary content that follows all these headers. | 
|---|
| 496 | // General-Header = Date | Pragma | 
|---|
| 497 | // Request-Header = Authorization | From | If-Modified-Since | Referer | User-Agent | 
|---|
| 498 | // Entity-Header  = Allow | Content-Encoding | Content-Length | Content-Type | 
|---|
| 499 | //                | Expires | Last-Modified | extension-header | 
|---|
| 500 | // extension-header = HTTP-header | 
|---|
| 501 | //       HTTP-header    = field-name ":" [ field-value ] CRLF | 
|---|
| 502 | //       field-name     = token | 
|---|
| 503 | //       field-value    = *( field-content | LWS ) | 
|---|
| 504 | //       field-content  = <the OCTETs making up the field-value | 
|---|
| 505 | //                        and consisting of either *TEXT or combinations | 
|---|
| 506 | //                        of token, tspecials, and quoted-string> | 
|---|
| 507 | char * pFieldName  = strtok(readBuffer, pSpDelimiters); | 
|---|
| 508 | char * pFieldValue = strtok(NULL, pSpDelimiters); | 
|---|
| 509 |  | 
|---|
| 510 | if (strcmp(pFieldName, "Content-Length:") == 0) | 
|---|
| 511 | { | 
|---|
| 512 | nContentLength = atoi(pFieldValue); | 
|---|
| 513 | } | 
|---|
| 514 | else if (strcmp(pFieldName, "Content-Type:") == 0) | 
|---|
| 515 | { | 
|---|
| 516 | if (strcmp(pFieldValue, "application/x-www-form-urlencoded") != 0) | 
|---|
| 517 | bIsUrlEncoded = false; | 
|---|
| 518 | } | 
|---|
| 519 | } while (strlen(readBuffer) > 0);    // empty string terminates | 
|---|
| 520 | } | 
|---|
| 521 |  | 
|---|
| 522 | // Read the entity body of given length (after all the headers) into the buffer. | 
|---|
| 523 | void readEntityBody(EthernetClient & client, int nContentLength, BUFFER & content) | 
|---|
| 524 | { | 
|---|
| 525 | int i; | 
|---|
| 526 | char c; | 
|---|
| 527 |  | 
|---|
| 528 | if (nContentLength >= sizeof(content)) | 
|---|
| 529 | nContentLength = sizeof(content) - 1;  // Should never happen! | 
|---|
| 530 |  | 
|---|
| 531 | for (i = 0; i < nContentLength; ++i) | 
|---|
| 532 | { | 
|---|
| 533 | c = client.read(); | 
|---|
| 534 | //    Serial.print(c); // DEBUG | 
|---|
| 535 | content[i] = c; | 
|---|
| 536 | } | 
|---|
| 537 |  | 
|---|
| 538 | content[nContentLength] = 0;  // Null string terminator | 
|---|
| 539 |  | 
|---|
| 540 | //  Serial.print("Content: "); | 
|---|
| 541 | //  Serial.println(content); | 
|---|
| 542 | } | 
|---|
| 543 |  | 
|---|
| 544 | // See if we recognize the URI and get its index; or -1 if we don't recognize it. | 
|---|
| 545 | int GetUriIndex(char * pUri) | 
|---|
| 546 | { | 
|---|
| 547 |  | 
|---|
| 548 | #ifdef DEBUG | 
|---|
| 549 | Serial.print("GetUriIndex("); | 
|---|
| 550 | Serial.print(pUri); | 
|---|
| 551 | Serial.print(")\n"); | 
|---|
| 552 | #endif | 
|---|
| 553 |  | 
|---|
| 554 | int nUriIndex = -1; | 
|---|
| 555 | // select the page from the buffer (GET and POST) [start] | 
|---|
| 556 | for (int i = 0; i < NUM_URIS; i++) | 
|---|
| 557 | { | 
|---|
| 558 | if (strcmp_P(pUri, (PGM_P)pgm_read_word(&(http_uris[i]))) == 0) | 
|---|
| 559 | { | 
|---|
| 560 | nUriIndex = i; | 
|---|
| 561 |  | 
|---|
| 562 | #ifdef DEBUG | 
|---|
| 563 | Serial.print(" o- URI: "); | 
|---|
| 564 | Serial.println(pUri); | 
|---|
| 565 | #endif | 
|---|
| 566 |  | 
|---|
| 567 | break; | 
|---|
| 568 | } | 
|---|
| 569 | } | 
|---|
| 570 | //  Serial.print("URI: "); | 
|---|
| 571 | //  Serial.print(pUri); | 
|---|
| 572 | //  Serial.print(" Page: "); | 
|---|
| 573 | //  Serial.println(nUriIndex); | 
|---|
| 574 |  | 
|---|
| 575 | return nUriIndex; | 
|---|
| 576 | } | 
|---|
| 577 |  | 
|---|
| 578 | /********************************************************************************************************************** | 
|---|
| 579 | * Read the next HTTP header record which is CRLF delimited.  We replace CRLF with string terminating null. | 
|---|
| 580 | ***********************************************************************************************************************/ | 
|---|
| 581 | void getNextHttpLine(EthernetClient & client, BUFFER & readBuffer) | 
|---|
| 582 | { | 
|---|
| 583 | char c; | 
|---|
| 584 | int bufindex = 0; // reset buffer | 
|---|
| 585 |  | 
|---|
| 586 | // reading next header of HTTP request | 
|---|
| 587 | if (client.connected() && client.available()) | 
|---|
| 588 | { | 
|---|
| 589 | // read a line terminated by CRLF | 
|---|
| 590 | readBuffer[0] = client.read(); | 
|---|
| 591 | readBuffer[1] = client.read(); | 
|---|
| 592 | bufindex = 2; | 
|---|
| 593 | for (int i = 2; readBuffer[i - 2] != '\r' && readBuffer[i - 1] != '\n'; ++i) | 
|---|
| 594 | { | 
|---|
| 595 | // read full line and save it in buffer, up to the buffer size | 
|---|
| 596 | c = client.read(); | 
|---|
| 597 | if (bufindex < sizeof(readBuffer)) | 
|---|
| 598 | readBuffer[bufindex++] = c; | 
|---|
| 599 | } | 
|---|
| 600 | readBuffer[bufindex - 2] = 0;  // Null string terminator overwrites '\r' | 
|---|
| 601 | } | 
|---|
| 602 | } | 
|---|
| 603 |  | 
|---|
| 604 | /********************************************************************************************************************** | 
|---|
| 605 | *                                                              Send Pages | 
|---|
| 606 | Full-Response  = Status-Line | 
|---|
| 607 | *( General-Header | 
|---|
| 608 | | Response-Header | 
|---|
| 609 | | Entity-Header ) | 
|---|
| 610 | CRLF | 
|---|
| 611 | [ Entity-Body ] | 
|---|
| 612 |  | 
|---|
| 613 | Status-Line = HTTP-Version SP Status-Code SP Reason-Phrase CRLF | 
|---|
| 614 | General-Header = Date | Pragma | 
|---|
| 615 | Response-Header = Location | Server | WWW-Authenticate | 
|---|
| 616 | Entity-Header  = Allow | Content-Encoding | Content-Length | Content-Type | 
|---|
| 617 | | Expires | Last-Modified | extension-header | 
|---|
| 618 | * | 
|---|
| 619 | ***********************************************************************************************************************/ | 
|---|
| 620 | void sendPage(EthernetClient & client, int nUriIndex, BUFFER & requestContent) | 
|---|
| 621 | { | 
|---|
| 622 | // Read Sensor values for every page reload | 
|---|
| 623 | ReadSensorM(-1,100); | 
|---|
| 624 | ReadCurrentM(-1,100); | 
|---|
| 625 |  | 
|---|
| 626 | #ifdef DEBUG | 
|---|
| 627 | Serial.print("sendPage("); | 
|---|
| 628 | Serial.print(nUriIndex);        Serial.print(", "); | 
|---|
| 629 | Serial.print(requestContent); | 
|---|
| 630 | Serial.println(")"); | 
|---|
| 631 | #endif | 
|---|
| 632 |  | 
|---|
| 633 | if (strncmp(requestContent, "Button1=", 8) == 0){ | 
|---|
| 634 | //Action1(strncmp(&requestContent[9], "true", 4) == 0); | 
|---|
| 635 | MoveTo(0, _EndPoint, 255); | 
|---|
| 636 | } | 
|---|
| 637 | else if (strncmp(requestContent, "Button2=", 8) == 0){ | 
|---|
| 638 | //Action2(strncmp(&requestContent[9], "true", 4) == 0); | 
|---|
| 639 | MoveTo(0, _StartPoint, 255); | 
|---|
| 640 | } | 
|---|
| 641 | else if (strncmp(requestContent, "Button3=", 8) == 0){ | 
|---|
| 642 | //Action3(strncmp(&requestContent[9], "true", 4) == 0); | 
|---|
| 643 | MoveTo(1, _EndPoint, 255); | 
|---|
| 644 | } | 
|---|
| 645 | else if (strncmp(requestContent, "Button4=", 8) == 0){ | 
|---|
| 646 | //Action4(strncmp(&requestContent[9], "true", 4) == 0); | 
|---|
| 647 | MoveTo(1, _StartPoint, 255); | 
|---|
| 648 | } | 
|---|
| 649 | else if (strncmp(requestContent, "Button5=", 8) == 0){ | 
|---|
| 650 | MoveTo(1, _StartPoint, 255); | 
|---|
| 651 | delay(100); | 
|---|
| 652 | MoveTo(0, _StartPoint, 255); | 
|---|
| 653 | } | 
|---|
| 654 | else if (strncmp(requestContent, "Button6=", 8) == 0){ | 
|---|
| 655 | MoveTo(0, _EndPoint, 255); | 
|---|
| 656 | delay(100); | 
|---|
| 657 | MoveTo(1, _EndPoint, 255); | 
|---|
| 658 | } | 
|---|
| 659 | #ifdef DEBUG | 
|---|
| 660 | else if (strncmp(requestContent, "", 1) == 0) | 
|---|
| 661 | Serial.print("-> Refresh\n"); | 
|---|
| 662 | else | 
|---|
| 663 | Serial.print("-> not recognized\n"); | 
|---|
| 664 | #endif | 
|---|
| 665 |  | 
|---|
| 666 | // send HTML header | 
|---|
| 667 | // sendProgMemAsString(client,(char*)pgm_read_word(&(contents_main[CONT_HEADER]))); | 
|---|
| 668 | sendProgMemAsString(client, (char*)pgm_read_word(&(contents_main[CONT_TOP]))); | 
|---|
| 669 |  | 
|---|
| 670 | // send menu | 
|---|
| 671 | sendProgMemAsString(client, (char*)pgm_read_word(&(contents_main[CONT_MENU]))); | 
|---|
| 672 |  | 
|---|
| 673 | // send title | 
|---|
| 674 | sendProgMemAsString(client, (char*)pgm_read_word(&(contents_titles[nUriIndex]))); | 
|---|
| 675 |  | 
|---|
| 676 | // send the body for the requested page | 
|---|
| 677 | sendUriContentByIndex(client, nUriIndex, requestContent); | 
|---|
| 678 |  | 
|---|
| 679 | // Append the data sent in the original HTTP request | 
|---|
| 680 | client.print("<br/>"); | 
|---|
| 681 | // send POST variables | 
|---|
| 682 | client.print(requestContent); | 
|---|
| 683 |  | 
|---|
| 684 | // send footer | 
|---|
| 685 | sendProgMemAsString(client,(char*)pgm_read_word(&(contents_main[CONT_FOOTER]))); | 
|---|
| 686 | } | 
|---|
| 687 |  | 
|---|
| 688 | /********************************************************************************************************************** | 
|---|
| 689 | *                                                              Send Images | 
|---|
| 690 | ***********************************************************************************************************************/ | 
|---|
| 691 | void sendImage(EthernetClient & client, int nUriIndex, BUFFER & requestContent) | 
|---|
| 692 | { | 
|---|
| 693 | int nImageIndex = nUriIndex - NUM_PAGES; | 
|---|
| 694 |  | 
|---|
| 695 | // send the header for the requested image | 
|---|
| 696 | sendUriContentByIndex(client, nUriIndex, requestContent); | 
|---|
| 697 |  | 
|---|
| 698 | // send the image data | 
|---|
| 699 | sendProgMemAsBinary(client, (char *)pgm_read_word(&(data_for_images[nImageIndex])), (int)pgm_read_word(&(size_for_images[nImageIndex]))); | 
|---|
| 700 | } | 
|---|
| 701 |  | 
|---|
| 702 | /********************************************************************************************************************** | 
|---|
| 703 | *                                                              Send content split by buffer size | 
|---|
| 704 | ***********************************************************************************************************************/ | 
|---|
| 705 | // If we provide string data then we don't need specify an explicit size and can do a string copy | 
|---|
| 706 | void sendProgMemAsString(EthernetClient & client, const char *realword) | 
|---|
| 707 | { | 
|---|
| 708 | sendProgMemAsBinary(client, realword, strlen_P(realword)); | 
|---|
| 709 | } | 
|---|
| 710 |  | 
|---|
| 711 | // Non-string data needs to provide an explicit size | 
|---|
| 712 | void sendProgMemAsBinary(EthernetClient & client, const char* realword, int realLen) | 
|---|
| 713 | { | 
|---|
| 714 | int remaining = realLen; | 
|---|
| 715 | const char * offsetPtr = realword; | 
|---|
| 716 | int nSize = sizeof(BUFFER); | 
|---|
| 717 | BUFFER buffer; | 
|---|
| 718 |  | 
|---|
| 719 | while (remaining > 0) | 
|---|
| 720 | { | 
|---|
| 721 | // print content | 
|---|
| 722 | if (nSize > remaining) | 
|---|
| 723 | nSize = remaining;      // Partial buffer left to send | 
|---|
| 724 |  | 
|---|
| 725 | memcpy_P(buffer, offsetPtr, nSize); | 
|---|
| 726 |  | 
|---|
| 727 | if (client.write((const uint8_t *)buffer, nSize) != nSize) | 
|---|
| 728 | Serial.println("Failed to send data"); | 
|---|
| 729 |  | 
|---|
| 730 | // more content to print? | 
|---|
| 731 | remaining -= nSize; | 
|---|
| 732 | offsetPtr += nSize; | 
|---|
| 733 | } | 
|---|
| 734 | } | 
|---|
| 735 |  | 
|---|
| 736 | /********************************************************************************************************************** | 
|---|
| 737 | *                                                              Send real page content | 
|---|
| 738 | ***********************************************************************************************************************/ | 
|---|
| 739 | // This method takes the contents page identified by nUriIndex, divides it up into buffer-sized | 
|---|
| 740 | // strings, passes it on for STX substitution and finally sending to the client. | 
|---|
| 741 | void sendUriContentByIndex(EthernetClient client, int nUriIndex, BUFFER & requestContent) | 
|---|
| 742 | { | 
|---|
| 743 | // Locate the page data for the URI and prepare to process in buffer-sized chunks. | 
|---|
| 744 | const char * offsetPtr;               // Pointer to offset within URI for data to be copied to buffer and sent. | 
|---|
| 745 | char *pNextString; | 
|---|
| 746 | int nSubstituteIndex = -1;            // Count of substitutions so far for this URI | 
|---|
| 747 | int remaining;                        // Total bytes (of URI) remaining to be sent | 
|---|
| 748 | int nSize = sizeof(BUFFER) - 1;       // Effective size of buffer allowing last char as string terminator | 
|---|
| 749 | BUFFER buffer; | 
|---|
| 750 |  | 
|---|
| 751 | if (nUriIndex < NUM_PAGES) | 
|---|
| 752 | offsetPtr = (char*)pgm_read_word(&(contents_pages[nUriIndex])); | 
|---|
| 753 | else | 
|---|
| 754 | offsetPtr = (char*)pgm_read_word(&(image_header)); | 
|---|
| 755 |  | 
|---|
| 756 | buffer[nSize] = 0;  // ensure there is always a string terminator | 
|---|
| 757 | remaining = strlen_P(offsetPtr);  // Set total bytes of URI remaining | 
|---|
| 758 |  | 
|---|
| 759 | while (remaining > 0) | 
|---|
| 760 | { | 
|---|
| 761 | // print content | 
|---|
| 762 | if (nSize > remaining) | 
|---|
| 763 | { | 
|---|
| 764 | // Set whole buffer to string terminator before copying remainder. | 
|---|
| 765 | memset(buffer, 0, STRING_BUFFER_SIZE); | 
|---|
| 766 | nSize = remaining;      // Partial buffer left to send | 
|---|
| 767 | } | 
|---|
| 768 | memcpy_P(buffer, offsetPtr, nSize); | 
|---|
| 769 | offsetPtr += nSize; | 
|---|
| 770 | // We have a buffer's worth of page to check for substitution markers/delimiters. | 
|---|
| 771 | // Scan the buffer for markers, dividing it up into separate strings. | 
|---|
| 772 | if (buffer[0] == *pStxDelimiter)    // First char is delimiter | 
|---|
| 773 | { | 
|---|
| 774 | sendSubstitute(client, nUriIndex, ++nSubstituteIndex, requestContent); | 
|---|
| 775 | --remaining; | 
|---|
| 776 | } | 
|---|
| 777 | // First string is either terminated by the null at the end of the buffer | 
|---|
| 778 | // or by a substitution delimiter.  So simply send it to the client. | 
|---|
| 779 | pNextString = strtok(buffer, pStxDelimiter); | 
|---|
| 780 | client.print(pNextString); | 
|---|
| 781 | remaining -= strlen(pNextString); | 
|---|
| 782 | // Scan for strings between delimiters | 
|---|
| 783 | for (pNextString = strtok(NULL, pStxDelimiter); pNextString != NULL && remaining > 0; pNextString = strtok(NULL, pStxDelimiter)) | 
|---|
| 784 | { | 
|---|
| 785 | // pNextString is pointing to the next string AFTER a delimiter | 
|---|
| 786 | sendSubstitute(client, nUriIndex, ++nSubstituteIndex, requestContent); | 
|---|
| 787 | --remaining; | 
|---|
| 788 | client.print(pNextString); | 
|---|
| 789 | remaining -= strlen(pNextString); | 
|---|
| 790 | } | 
|---|
| 791 | } | 
|---|
| 792 | } | 
|---|
| 793 |  | 
|---|
| 794 | // Call this method in response to finding a substitution character '\002' within some | 
|---|
| 795 | // URI content to send the appropriate replacement text, depending on the URI index and | 
|---|
| 796 | // the substitution index within the content. | 
|---|
| 797 | void sendSubstitute(EthernetClient client, int nUriIndex, int nSubstituteIndex, BUFFER & requestContent) | 
|---|
| 798 | { | 
|---|
| 799 | #ifdef DEBUG | 
|---|
| 800 | Serial.print("sendSubstitute("); | 
|---|
| 801 | Serial.print(nUriIndex);        Serial.print(", "); | 
|---|
| 802 | Serial.print(nSubstituteIndex); Serial.print(", "); | 
|---|
| 803 | Serial.print(requestContent); | 
|---|
| 804 | Serial.print(")\n"); | 
|---|
| 805 | #endif | 
|---|
| 806 | if (nUriIndex < NUM_PAGES) | 
|---|
| 807 | { | 
|---|
| 808 | // Page request | 
|---|
| 809 | switch (nUriIndex) | 
|---|
| 810 | { | 
|---|
| 811 | case 1:  // page 2 | 
|---|
| 812 | #ifdef DEBUG | 
|---|
| 813 | Serial.println(" -> Case 1"); | 
|---|
| 814 | #endif | 
|---|
| 815 | if (nSubstituteIndex < 4){ | 
|---|
| 816 | //client.print("<b>"); | 
|---|
| 817 | client.print(_currentValue[nSubstituteIndex/2]); | 
|---|
| 818 | //client.print("</b>"); | 
|---|
| 819 | } | 
|---|
| 820 | else if ( (nSubstituteIndex >= 4) && | 
|---|
| 821 | (nSubstituteIndex <  8)     ) { | 
|---|
| 822 | //client.print("<b>"); | 
|---|
| 823 | client.print(_sensorValue[nSubstituteIndex/2-2]); | 
|---|
| 824 | //client.print("</b>"); | 
|---|
| 825 | } | 
|---|
| 826 | else if ( (nSubstituteIndex >= 8) && | 
|---|
| 827 | (nSubstituteIndex <  12)     ) { | 
|---|
| 828 | //client.print("<b>"); | 
|---|
| 829 | client.print(_StatusLabel[_LidStatus[nSubstituteIndex/2-4]]); | 
|---|
| 830 | //client.print("</b>"); | 
|---|
| 831 |  | 
|---|
| 832 | } | 
|---|
| 833 | break; | 
|---|
| 834 | case 2:  // page 3 | 
|---|
| 835 | Serial.println(" -> Case 2"); | 
|---|
| 836 | // | 
|---|
| 837 | //        switch (nSubstituteIndex) | 
|---|
| 838 | //        { | 
|---|
| 839 | //          case 0:  // LedOn button send value | 
|---|
| 840 | //#ifdef DEBUG | 
|---|
| 841 | //            Serial.println(requestContent); | 
|---|
| 842 | //#endif | 
|---|
| 843 | //            if (strncmp(requestContent, "Button1=", 8) == 0) | 
|---|
| 844 | //              Action1(strncmp(&requestContent[9], "true", 4) == 0); | 
|---|
| 845 | // | 
|---|
| 846 | //            client.print(Action1 ? "false" : "true"); | 
|---|
| 847 | //            break; | 
|---|
| 848 | //          case 1:  // LedOn button legend | 
|---|
| 849 | //            //client.print(isLedOn ? "Off" : "On"); | 
|---|
| 850 | //            break; | 
|---|
| 851 | //          case 2:  // LedOn partial image name | 
|---|
| 852 | //            //client.print(isLedOn ? "on" : "off"); | 
|---|
| 853 | //            break; | 
|---|
| 854 | //          default: | 
|---|
| 855 | //            break; | 
|---|
| 856 | //         } | 
|---|
| 857 | break; | 
|---|
| 858 | } | 
|---|
| 859 | } | 
|---|
| 860 | else | 
|---|
| 861 | { | 
|---|
| 862 | // Image request | 
|---|
| 863 | int nImageIndex = nUriIndex - NUM_PAGES; | 
|---|
| 864 |  | 
|---|
| 865 | switch (nSubstituteIndex) | 
|---|
| 866 | { | 
|---|
| 867 | case 0: | 
|---|
| 868 | // Content-Length value - ie. image size | 
|---|
| 869 | char strSize[6];  // Up to 5 digits plus null terminator | 
|---|
| 870 | itoa((int)pgm_read_word(&(size_for_images[nImageIndex])), strSize, 10); | 
|---|
| 871 | //Serial.println(strSize);    // Debug | 
|---|
| 872 | client.print(strSize); | 
|---|
| 873 | break; | 
|---|
| 874 | case 1: | 
|---|
| 875 | // Content-Type partial value | 
|---|
| 876 | switch (nImageIndex) | 
|---|
| 877 | { | 
|---|
| 878 | case 0:  // favicon | 
|---|
| 879 | client.print("x-icon"); | 
|---|
| 880 | break; | 
|---|
| 881 | case 1:  // led on image | 
|---|
| 882 | case 2:  // led off image | 
|---|
| 883 | client.print("png"); | 
|---|
| 884 | break; | 
|---|
| 885 | } | 
|---|
| 886 | } | 
|---|
| 887 | } | 
|---|
| 888 | } | 
|---|
| 889 | #endif | 
|---|
| 890 | // | 
|---|
| 891 | //void Action1(bool argument) | 
|---|
| 892 | //{ | 
|---|
| 893 | //  Serial.print("Action->Action1("); | 
|---|
| 894 | //  Serial.print(argument); | 
|---|
| 895 | //  Serial.print(")\n"); | 
|---|
| 896 | // | 
|---|
| 897 | //  // Move motor 0 out | 
|---|
| 898 | //  int m=0; | 
|---|
| 899 | //  MoveTo(m, 1023, 200); | 
|---|
| 900 | // | 
|---|
| 901 | //} | 
|---|
| 902 |  | 
|---|
| 903 | //void Action2(bool argument) | 
|---|
| 904 | //{ | 
|---|
| 905 | //  Serial.print("Action->Action2("); | 
|---|
| 906 | //  Serial.print(argument); | 
|---|
| 907 | //  Serial.print(")\n"); | 
|---|
| 908 | // | 
|---|
| 909 | //  // Move motor 1 out | 
|---|
| 910 | //  int m=0; | 
|---|
| 911 | //  MoveTo(m, 0, 200); | 
|---|
| 912 | //} | 
|---|
| 913 | // | 
|---|
| 914 | //void Action3(bool argument) | 
|---|
| 915 | //{ | 
|---|
| 916 | //  Serial.print("Action->Action3("); | 
|---|
| 917 | //  Serial.print(argument); | 
|---|
| 918 | //  Serial.print(")\n"); | 
|---|
| 919 | // | 
|---|
| 920 | //  // Move motor 0 in | 
|---|
| 921 | //  int m=1; | 
|---|
| 922 | //  MoveTo(m, 1023, 200); | 
|---|
| 923 | //} | 
|---|
| 924 | // | 
|---|
| 925 | //void Action4(bool argument) | 
|---|
| 926 | //{ | 
|---|
| 927 | //  Serial.print("Action->Action4("); | 
|---|
| 928 | //  Serial.print(argument); | 
|---|
| 929 | //  Serial.print(")\n"); | 
|---|
| 930 | // | 
|---|
| 931 | //  // Move motor 1 in | 
|---|
| 932 | //  int m=1; | 
|---|
| 933 | //  MoveTo(m, 0, 200); | 
|---|
| 934 | //} | 
|---|
| 935 | // | 
|---|
| 936 | // position in [%]-> [0-1] Closed-Open | 
|---|
| 937 | // | 
|---|
| 938 | void MoveTo(int motor, double target_position, int mySpeed){ | 
|---|
| 939 |  | 
|---|
| 940 | // define tmp value for the speed | 
|---|
| 941 | int speedTmp = 0; | 
|---|
| 942 |  | 
|---|
| 943 | // define variable containing the current actuator position | 
|---|
| 944 | // the travel to be done to rech the target position and the | 
|---|
| 945 | // motor current | 
|---|
| 946 | double current_position; | 
|---|
| 947 | double err_current_position; | 
|---|
| 948 |  | 
|---|
| 949 | double original_position; | 
|---|
| 950 | double err_original_position; | 
|---|
| 951 |  | 
|---|
| 952 | double motor_current; | 
|---|
| 953 | double err_motor_current; | 
|---|
| 954 |  | 
|---|
| 955 | double travel; | 
|---|
| 956 |  | 
|---|
| 957 | // only one reading to define the position is not sufficient | 
|---|
| 958 | // current_position = ReadSensor(motor); | 
|---|
| 959 |  | 
|---|
| 960 | // Calculate average final position | 
|---|
| 961 | double tmp=0; | 
|---|
| 962 | double tmpM=0; | 
|---|
| 963 | double tmpS=0; | 
|---|
| 964 | int    steps=0; | 
|---|
| 965 |  | 
|---|
| 966 | for (int i=0;i<SAMPLES;i++){ | 
|---|
| 967 | tmp=ReadSensor(motor); | 
|---|
| 968 | tmpM += tmp; | 
|---|
| 969 | tmpS += tmp*tmp; | 
|---|
| 970 | } | 
|---|
| 971 | tmpM /= SAMPLES; | 
|---|
| 972 | tmpS  = sqrt(tmpS/SAMPLES - tmpM*tmpM); | 
|---|
| 973 |  | 
|---|
| 974 |  | 
|---|
| 975 | int tmpS_int = (int) tmpS; | 
|---|
| 976 |  | 
|---|
| 977 | // round the mean to it to the closest integer | 
|---|
| 978 | current_position = (int) (tmpM+0.5); | 
|---|
| 979 | if (((int)tmpS) < 1){ | 
|---|
| 980 | err_current_position =     1.; | 
|---|
| 981 | } | 
|---|
| 982 | else{ | 
|---|
| 983 | err_current_position = tmpS; | 
|---|
| 984 | } | 
|---|
| 985 |  | 
|---|
| 986 | original_position      = current_position; | 
|---|
| 987 | err_original_position  = err_current_position; | 
|---|
| 988 |  | 
|---|
| 989 | // calculate the travel needed to reach the target position | 
|---|
| 990 | travel = target_position - current_position; | 
|---|
| 991 |  | 
|---|
| 992 | Serial.print("Moving motor "); | 
|---|
| 993 | Serial.print(motor); | 
|---|
| 994 | Serial.print(" from pos. "); | 
|---|
| 995 | Serial.print(current_position,3); | 
|---|
| 996 | Serial.print(" to position "); | 
|---|
| 997 | Serial.print(target_position,3); | 
|---|
| 998 | Serial.print(" - Travel distance = "); | 
|---|
| 999 | Serial.println(travel,3); | 
|---|
| 1000 |  | 
|---|
| 1001 | Serial.print(" - The current position of the actuator is "); | 
|---|
| 1002 | Serial.print(tmpM,3); | 
|---|
| 1003 | Serial.print( " +/- " ); | 
|---|
| 1004 | Serial.println(tmpS,3); | 
|---|
| 1005 |  | 
|---|
| 1006 | Serial.print(" - The corrected position of the actuator is "); | 
|---|
| 1007 | Serial.print(current_position); | 
|---|
| 1008 | Serial.print( " +/- " ); | 
|---|
| 1009 | Serial.println(err_current_position); | 
|---|
| 1010 |  | 
|---|
| 1011 |  | 
|---|
| 1012 | // [IF] the travel is bigger than +2*(absolute position error) try to move out the motor | 
|---|
| 1013 | if (travel > 2*err_current_position){ | 
|---|
| 1014 | // Try to place here (if you have time) a speed self adjucting algorithm | 
|---|
| 1015 | // base on the trave which the actuator has still to do | 
|---|
| 1016 | if( _LidStatus[motor] != _CLOSED){ | 
|---|
| 1017 |  | 
|---|
| 1018 | Serial.println(" - going out (lid closing)"); | 
|---|
| 1019 | speedTmp = mySpeed; // positive speed | 
|---|
| 1020 | steps++; | 
|---|
| 1021 | _LidStatus[motor] = _CLOSING; | 
|---|
| 1022 |  | 
|---|
| 1023 | // Accelerate motor from 0 to speedTmp linearly | 
|---|
| 1024 | for (int j=0;j<speedTmp;j++){ | 
|---|
| 1025 | Motor(motor, j); | 
|---|
| 1026 | delay(1); | 
|---|
| 1027 | } | 
|---|
| 1028 | } | 
|---|
| 1029 | else{ | 
|---|
| 1030 | Serial.println(" - already closed"); | 
|---|
| 1031 |  | 
|---|
| 1032 | _LidStatus[motor] = _CLOSED; | 
|---|
| 1033 | return; | 
|---|
| 1034 | } | 
|---|
| 1035 | } | 
|---|
| 1036 | // [ELSE IF] travel is between -2*(absolute position error) and +2*(absolute position error) | 
|---|
| 1037 | //           consider yourself already in position | 
|---|
| 1038 | else if (travel <=  2*err_current_position && | 
|---|
| 1039 | travel >= -2*err_current_position    ){ | 
|---|
| 1040 | Serial.println(" - already in place"); | 
|---|
| 1041 |  | 
|---|
| 1042 | _LidStatus[motor] = _STEADY; | 
|---|
| 1043 | return; | 
|---|
| 1044 | // already in place don't bother moving more. | 
|---|
| 1045 | } | 
|---|
| 1046 | // [ELSE} if the travel is smaller than -2*(absolute position error) try to move in the motor | 
|---|
| 1047 | else{ | 
|---|
| 1048 | Serial.println(" - going in (lid opening)"); | 
|---|
| 1049 | speedTmp = -mySpeed; // negative speed | 
|---|
| 1050 | steps++; | 
|---|
| 1051 | _LidStatus[motor] = _OPENING; | 
|---|
| 1052 |  | 
|---|
| 1053 | // Accelerate motor from 0 to -speedTmp linearly | 
|---|
| 1054 | for (int j=0;j>speedTmp;j--){ | 
|---|
| 1055 | Motor(motor, j); | 
|---|
| 1056 | delay(1); | 
|---|
| 1057 | } | 
|---|
| 1058 | } | 
|---|
| 1059 |  | 
|---|
| 1060 |  | 
|---|
| 1061 | // Start the main loop which checks the motors while they are mooving | 
|---|
| 1062 | tmp=0; | 
|---|
| 1063 | tmpM=0; | 
|---|
| 1064 | tmpS=0; | 
|---|
| 1065 | for (steps=1;abs(travel) != 0 ;steps++){ | 
|---|
| 1066 | //Read Current | 
|---|
| 1067 | motor_current = ReadCurrentM(motor,10); | 
|---|
| 1068 |  | 
|---|
| 1069 | // If Overcurrent stop | 
|---|
| 1070 | if (motor_current > _OverCurrent){ | 
|---|
| 1071 | Motor(motor, 0); // Stop Motor | 
|---|
| 1072 |  | 
|---|
| 1073 | Serial.print(" - WARNING!!! Overcurrent "); | 
|---|
| 1074 | Serial.print(motor_current,3); | 
|---|
| 1075 | Serial.print(" [A] at position "); | 
|---|
| 1076 | Serial.println(current_position,3); | 
|---|
| 1077 |  | 
|---|
| 1078 | _LidStatus[motor] = _OVER_CURRENT; | 
|---|
| 1079 | return; | 
|---|
| 1080 | } | 
|---|
| 1081 |  | 
|---|
| 1082 | // If Fault stop | 
|---|
| 1083 | if (getFault(motor)){ | 
|---|
| 1084 | Motor(motor, 0); // Stop Motor | 
|---|
| 1085 | Serial.print(" - GetFault - at position "); | 
|---|
| 1086 | Serial.println(current_position,3); | 
|---|
| 1087 | _LidStatus[motor] = _MOTOR_FAULT; | 
|---|
| 1088 | break; | 
|---|
| 1089 | } | 
|---|
| 1090 |  | 
|---|
| 1091 | // Average Current around the steps | 
|---|
| 1092 | tmp   =  motor_current; | 
|---|
| 1093 | tmpM += tmp; | 
|---|
| 1094 | tmpS += tmp*tmp; | 
|---|
| 1095 |  | 
|---|
| 1096 | // Read current position | 
|---|
| 1097 | // it doesn't make sense to read it here more time as the actuars are moving | 
|---|
| 1098 | current_position = ReadSensorM(motor,10); | 
|---|
| 1099 |  | 
|---|
| 1100 | // Calculate travel distance | 
|---|
| 1101 | travel           = target_position - current_position; | 
|---|
| 1102 |  | 
|---|
| 1103 | // Read current absorbed the motor and append the values for the calculation | 
|---|
| 1104 | // of the mean value and its error | 
|---|
| 1105 |  | 
|---|
| 1106 |  | 
|---|
| 1107 | // [IF] the current drops below ~0.07 A might means that the end swirch | 
|---|
| 1108 | //      stopped the motor check also the position to determine if this is true. | 
|---|
| 1109 | if (motor_current < _ZeroCurrent){ | 
|---|
| 1110 | // Closing | 
|---|
| 1111 | if ( current_position > _EndPointLimit && target_position > _EndPointLimit ){ | 
|---|
| 1112 | Serial.print(" - Reached End of Actuator. Pos = "); | 
|---|
| 1113 | Serial.println(current_position, 3); | 
|---|
| 1114 |  | 
|---|
| 1115 | _LidStatus[motor] = _CLOSED; | 
|---|
| 1116 | break; //Exit from the for loop | 
|---|
| 1117 | } | 
|---|
| 1118 | // Opening | 
|---|
| 1119 | else if (current_position < _StartPointLimit && target_position < _StartPointLimit ){ | 
|---|
| 1120 | Serial.print(" - Reached Beginning of Actuator. Pos = "); | 
|---|
| 1121 | Serial.println(current_position, 3); | 
|---|
| 1122 |  | 
|---|
| 1123 | _LidStatus[motor] = _OPEN; | 
|---|
| 1124 | break; //Exit from the for loop | 
|---|
| 1125 | } | 
|---|
| 1126 | // Error | 
|---|
| 1127 | else  { | 
|---|
| 1128 | Serial.print(" - Error!! No current in motor "); | 
|---|
| 1129 | Serial.print(motor); | 
|---|
| 1130 | Serial.print(". I= "); | 
|---|
| 1131 | Serial.print(motor_current); | 
|---|
| 1132 | Serial.print("A . Pos = "); | 
|---|
| 1133 | Serial.println(current_position, 3); | 
|---|
| 1134 |  | 
|---|
| 1135 | _LidStatus[motor]  = _POWER_PROBLEM; | 
|---|
| 1136 | break; //Exit from the for loop | 
|---|
| 1137 | } | 
|---|
| 1138 | } | 
|---|
| 1139 |  | 
|---|
| 1140 | if (_LidStatus[motor] == _CLOSING && motor_current > _CurrentPushingLimit && current_position > _EndPointLimit){ | 
|---|
| 1141 | Serial.print(" - step "); | 
|---|
| 1142 | Serial.print(steps); | 
|---|
| 1143 | // double travel_done= (55. * (target_position - original_position))/1024.; | 
|---|
| 1144 | // Serial.print(" - travelled for "); | 
|---|
| 1145 | // Serial.print(travel_done); | 
|---|
| 1146 | Serial.print(" - pushing -> I = "); | 
|---|
| 1147 | Serial.print(motor_current, 3); | 
|---|
| 1148 | Serial.println(" A"); | 
|---|
| 1149 | _LidStatus[motor] = _CLOSED; | 
|---|
| 1150 | break; | 
|---|
| 1151 | } | 
|---|
| 1152 |  | 
|---|
| 1153 | // If too many cycles of the loop print a message and check the position and the current | 
|---|
| 1154 | if (steps %50 == 0){ | 
|---|
| 1155 | Serial.print(" - step "); | 
|---|
| 1156 | Serial.print(steps); | 
|---|
| 1157 | Serial.print(" - still in loop. Pos = "); | 
|---|
| 1158 | Serial.print(current_position, 3); | 
|---|
| 1159 | Serial.print(" - Istantaneous current = "); | 
|---|
| 1160 | Serial.print(motor_current, 3); | 
|---|
| 1161 | Serial.print(" A"); | 
|---|
| 1162 | Serial.print(" - Average current = "); | 
|---|
| 1163 | Serial.print(tmpM/steps, 3); | 
|---|
| 1164 | Serial.println(" A"); | 
|---|
| 1165 |  | 
|---|
| 1166 |  | 
|---|
| 1167 | if( _LidStatus[motor] != _CLOSING && | 
|---|
| 1168 | _LidStatus[motor] != _OPENING) | 
|---|
| 1169 | _LidStatus[motor] = _MOVING; | 
|---|
| 1170 |  | 
|---|
| 1171 | // Redefine better those limits... they might not be necessary with the new low | 
|---|
| 1172 | // current limits | 
|---|
| 1173 | //if (steps %500 && current_position > 1000 && target_position > 1000 ){ | 
|---|
| 1174 | //  Serial.print(" - Reached End of Actuator. Pos = "); | 
|---|
| 1175 | //  Serial.println(current_position, 3); | 
|---|
| 1176 | //  break; | 
|---|
| 1177 | //} | 
|---|
| 1178 | //if (steps %500 && current_position < 80 && target_position < 80 ){ | 
|---|
| 1179 | //  Serial.print(" - Reached Beginning of Actuator. Pos = "); | 
|---|
| 1180 | //  Serial.println(current_position, 3); | 
|---|
| 1181 | //  break; | 
|---|
| 1182 | //} | 
|---|
| 1183 | } | 
|---|
| 1184 |  | 
|---|
| 1185 |  | 
|---|
| 1186 |  | 
|---|
| 1187 | // minimum delay between a cycle and the other is 1 ms | 
|---|
| 1188 | delay (10); | 
|---|
| 1189 | } | 
|---|
| 1190 |  | 
|---|
| 1191 | // At this stage the motor should be stopped in any case | 
|---|
| 1192 | Motor(motor, 0); | 
|---|
| 1193 | Serial.print(" - motor reached the destination - pos. "); | 
|---|
| 1194 | Serial.println(current_position,3); | 
|---|
| 1195 |  | 
|---|
| 1196 | // Calculate current average and sigma | 
|---|
| 1197 | tmpM /= steps; | 
|---|
| 1198 | tmpS  = sqrt(tmpS/steps - tmpM*tmpM); | 
|---|
| 1199 | Serial.print(" - average current over the loop "); | 
|---|
| 1200 | Serial.print(tmpM,3); | 
|---|
| 1201 | Serial.print( " +/- " ); | 
|---|
| 1202 | Serial.println(tmpS,3); | 
|---|
| 1203 |  | 
|---|
| 1204 | // Wait 100 ms then calculate average final position | 
|---|
| 1205 | delay(100); | 
|---|
| 1206 | tmp=0; tmpM=0; tmpS=0; | 
|---|
| 1207 |  | 
|---|
| 1208 | for (int i=0;i<SAMPLES;i++){ | 
|---|
| 1209 | tmp=ReadSensor(motor); | 
|---|
| 1210 | tmpM += tmp; | 
|---|
| 1211 | tmpS += tmp*tmp; | 
|---|
| 1212 | } | 
|---|
| 1213 | tmpM /= SAMPLES; | 
|---|
| 1214 | tmpS  = sqrt(tmpS/SAMPLES - tmpM*tmpM); | 
|---|
| 1215 |  | 
|---|
| 1216 | Serial.print(" - final position is "); | 
|---|
| 1217 | Serial.print(tmpM,3); | 
|---|
| 1218 | Serial.print( " +/- " ); | 
|---|
| 1219 | Serial.println(tmpS,3); | 
|---|
| 1220 |  | 
|---|
| 1221 | Serial.print(" - Lid staus is "); | 
|---|
| 1222 | Serial.println(_StatusLabel[_LidStatus[motor]]); | 
|---|
| 1223 |  | 
|---|
| 1224 | Serial.print("AvailableMemory()="); | 
|---|
| 1225 | Serial.println(availableMemory()); | 
|---|
| 1226 |  | 
|---|
| 1227 |  | 
|---|
| 1228 | } | 
|---|
| 1229 |  | 
|---|
| 1230 | int availableMemory() { | 
|---|
| 1231 | int free_memory; | 
|---|
| 1232 | if ((int)__brkval==0) | 
|---|
| 1233 | free_memory = ((int)&free_memory) - ((int) &__bss_end); | 
|---|
| 1234 | else | 
|---|
| 1235 | free_memory = ((int)&free_memory) - ((int) &__brkval); | 
|---|
| 1236 |  | 
|---|
| 1237 | return free_memory; | 
|---|
| 1238 | } | 
|---|
| 1239 |  | 
|---|
| 1240 |  | 
|---|
| 1241 | // ReadSensor(-1); - Read all sensor | 
|---|
| 1242 | // ReadSensor(0);  - Read sensor 0 | 
|---|
| 1243 | // ReadSensor(1);  - Read sensor 1 | 
|---|
| 1244 | double ReadSensor(int motor){ | 
|---|
| 1245 | #ifdef DEBUG | 
|---|
| 1246 | Serial.println("Action->ReadSensors()"); | 
|---|
| 1247 | #endif | 
|---|
| 1248 |  | 
|---|
| 1249 | switch (motor){ | 
|---|
| 1250 | case -1: // Read all of them | 
|---|
| 1251 | _sensorValue[0] =  analogRead(A2);//*_ADC2V - _Voffset;// *_Compensation;          // Actuator 1 position | 
|---|
| 1252 | _sensorValue[1] =  analogRead(A3);//*_ADC2V - _Voffset;// *_Compensation;          // Actuator 2 position | 
|---|
| 1253 | return -1; | 
|---|
| 1254 | case 0: | 
|---|
| 1255 | _sensorValue[motor] =  analogRead(A2);//*_ADC2V - _Voffset; //*_Compensation; | 
|---|
| 1256 | return _sensorValue[motor]; // Actuator 1 position | 
|---|
| 1257 | case 1: | 
|---|
| 1258 | _sensorValue[motor] =  analogRead(A3);//*_ADC2V - _Voffset;//_Compensation; | 
|---|
| 1259 | return _sensorValue[motor]; // Actuator 1 position | 
|---|
| 1260 | } | 
|---|
| 1261 | } | 
|---|
| 1262 |  | 
|---|
| 1263 | double ReadSensorM(int motor, int samples){ | 
|---|
| 1264 | #ifdef DEBUG | 
|---|
| 1265 | Serial.println("Action->ReadSensorsM()"); | 
|---|
| 1266 | #endif | 
|---|
| 1267 | switch (motor){ | 
|---|
| 1268 | case -1: // Read all of them | 
|---|
| 1269 | _sensorValue[0] = 0; | 
|---|
| 1270 | _sensorValue[1] = 0; | 
|---|
| 1271 | for (int j=0;j<samples;j++){ | 
|---|
| 1272 | _sensorValue[0] +=  analogRead(A2); | 
|---|
| 1273 | _sensorValue[1] +=  analogRead(A3); | 
|---|
| 1274 | } | 
|---|
| 1275 | _sensorValue[0]/=samples; | 
|---|
| 1276 | _sensorValue[1]/=samples; | 
|---|
| 1277 | return -1; | 
|---|
| 1278 |  | 
|---|
| 1279 | case 0: | 
|---|
| 1280 | _sensorValue[motor]=0; | 
|---|
| 1281 | for (int j=0;j<samples;j++) | 
|---|
| 1282 | _sensorValue[motor] +=  analogRead(A2);//*_ADC2V - _Voffset; //*_Compensation; | 
|---|
| 1283 | _sensorValue[motor] /= samples; | 
|---|
| 1284 | return _sensorValue[motor]; // Actuator 1 position | 
|---|
| 1285 |  | 
|---|
| 1286 | case 1: | 
|---|
| 1287 | _sensorValue[motor]=0; | 
|---|
| 1288 | for (int j=0;j<samples;j++) | 
|---|
| 1289 | _sensorValue[motor] +=  analogRead(A3);//*_ADC2V - _Voffset;//_Compensation; | 
|---|
| 1290 | _sensorValue[motor] /= samples; | 
|---|
| 1291 | return _sensorValue[motor]; // Actuator 1 position | 
|---|
| 1292 | } | 
|---|
| 1293 |  | 
|---|
| 1294 | } | 
|---|
| 1295 |  | 
|---|
| 1296 |  | 
|---|
| 1297 | // ReadCurrent(-1); - Read current for all the motors | 
|---|
| 1298 | // ReadCurrent(0);  - Read motor 0 current | 
|---|
| 1299 | // ReadCurrent(1);  - Read motor 1 current | 
|---|
| 1300 | double ReadCurrent(int motor){ | 
|---|
| 1301 | switch (motor){ | 
|---|
| 1302 | case -1: // Read all of them | 
|---|
| 1303 | _currentValue[0] =  analogRead(A4)*_ADC2V/(_V2A*10.); | 
|---|
| 1304 | _currentValue[1] =  analogRead(A5)*_ADC2V/(_V2A*10.); | 
|---|
| 1305 | Serial.println("Action->ReadSensors()"); | 
|---|
| 1306 | return -1; | 
|---|
| 1307 | case 0: | 
|---|
| 1308 | _currentValue[motor] =  analogRead(A4)*_ADC2V/(_V2A*10.); | 
|---|
| 1309 | return _currentValue[motor]; // Current of the motor | 
|---|
| 1310 | case 1: | 
|---|
| 1311 | _currentValue[motor] =  analogRead(A5)*_ADC2V/(_V2A*10.); | 
|---|
| 1312 | return _currentValue[motor]; // Current of the motor | 
|---|
| 1313 | } | 
|---|
| 1314 | } | 
|---|
| 1315 |  | 
|---|
| 1316 | double ReadCurrentM(int motor,  int samples){ | 
|---|
| 1317 | switch (motor){ | 
|---|
| 1318 | case -1: // Read all of them | 
|---|
| 1319 | _currentValue[0] = 0; | 
|---|
| 1320 | _currentValue[1] = 0; | 
|---|
| 1321 | for (int j=0;j<samples;j++){ | 
|---|
| 1322 | _currentValue[0] +=  analogRead(A4)*_ADC2V/(_V2A*10.); | 
|---|
| 1323 | _currentValue[1] +=  analogRead(A5)*_ADC2V/(_V2A*10.); | 
|---|
| 1324 | } | 
|---|
| 1325 | _currentValue[0]/=samples; | 
|---|
| 1326 | _currentValue[1]/=samples; | 
|---|
| 1327 | return -1; | 
|---|
| 1328 | case 0: | 
|---|
| 1329 | _currentValue[motor] = 0; | 
|---|
| 1330 | for (int j=0;j<samples;j++) | 
|---|
| 1331 | _currentValue[motor] +=  analogRead(A4)*_ADC2V/(_V2A*10.); | 
|---|
| 1332 | _currentValue[motor]/=samples; | 
|---|
| 1333 | return _currentValue[motor]; | 
|---|
| 1334 | case 1: | 
|---|
| 1335 | _currentValue[motor] = 0; | 
|---|
| 1336 | for (int j=0;j<samples;j++) | 
|---|
| 1337 | _currentValue[motor] +=  analogRead(A5)*_ADC2V/(_V2A*10.); | 
|---|
| 1338 | _currentValue[motor]/=samples; | 
|---|
| 1339 | return _currentValue[motor]; | 
|---|
| 1340 |  | 
|---|
| 1341 | } | 
|---|
| 1342 |  | 
|---|
| 1343 | } | 
|---|
| 1344 |  | 
|---|
| 1345 |  | 
|---|
| 1346 | // Return error status for motor | 
|---|
| 1347 | unsigned char getFault(int motor) | 
|---|
| 1348 | { | 
|---|
| 1349 | return 0; //!digitalRead(_ENDIAG[motor]); | 
|---|
| 1350 | } | 
|---|
| 1351 |  | 
|---|
| 1352 | // control motor, -255 < pwm < 255 | 
|---|
| 1353 | // | 
|---|
| 1354 | void Motor(int motor, int pwm){ | 
|---|
| 1355 | bool reverse = false; | 
|---|
| 1356 |  | 
|---|
| 1357 | // Check sign and direction | 
|---|
| 1358 | if (pwm < 0){ | 
|---|
| 1359 | pwm = -pwm; | 
|---|
| 1360 | reverse = true; | 
|---|
| 1361 | } | 
|---|
| 1362 |  | 
|---|
| 1363 | // Check max speed | 
|---|
| 1364 | if      (pwm >  255) pwm =  255; | 
|---|
| 1365 | else if (pwm < -255) pwm = -255; | 
|---|
| 1366 |  | 
|---|
| 1367 | // Activate motors | 
|---|
| 1368 | analogWrite(_pinPWM[motor], pwm); //set pwm control, 0 for stop, and 255 for maximum speed | 
|---|
| 1369 | if (pwm != 0){ | 
|---|
| 1370 | if(reverse) { | 
|---|
| 1371 | digitalWrite(_pinDA[motor], HIGH); | 
|---|
| 1372 | digitalWrite(_pinDB[motor], LOW); | 
|---|
| 1373 | } | 
|---|
| 1374 | else { | 
|---|
| 1375 | digitalWrite(_pinDA[motor], LOW); | 
|---|
| 1376 | digitalWrite(_pinDB[motor], HIGH); | 
|---|
| 1377 | } | 
|---|
| 1378 | } | 
|---|
| 1379 | else { | 
|---|
| 1380 | digitalWrite(_pinDA[motor], LOW); | 
|---|
| 1381 | digitalWrite(_pinDB[motor], LOW); | 
|---|
| 1382 | } | 
|---|
| 1383 | } | 
|---|
| 1384 |  | 
|---|
| 1385 |  | 
|---|