Add MAX_ENDPOINT_INDEX compile time option for the XMEGA devices.
[pub/lufa.git] / Projects / Webserver / Lib / HTTPServerApp.c
index 01aab76..9ee514d 100644 (file)
-/*\r
-             LUFA Library\r
-     Copyright (C) Dean Camera, 2010.\r
-              \r
-  dean [at] fourwalledcubicle [dot] com\r
-      www.fourwalledcubicle.com\r
-*/\r
-\r
-/*\r
-  Copyright 2010  Dean Camera (dean [at] fourwalledcubicle [dot] com)\r
-\r
-  Permission to use, copy, modify, distribute, and sell this \r
-  software and its documentation for any purpose is hereby granted\r
-  without fee, provided that the above copyright notice appear in \r
-  all copies and that both that the copyright notice and this\r
-  permission notice and warranty disclaimer appear in supporting \r
-  documentation, and that the name of the author not be used in \r
-  advertising or publicity pertaining to distribution of the \r
-  software without specific, written prior permission.\r
-\r
-  The author disclaim all warranties with regard to this\r
-  software, including all implied warranties of merchantability\r
-  and fitness.  In no event shall the author be liable for any\r
-  special, indirect or consequential damages or any damages\r
-  whatsoever resulting from loss of use, data or profits, whether\r
-  in an action of contract, negligence or other tortious action,\r
-  arising out of or in connection with the use or performance of\r
-  this software.\r
-*/\r
-\r
-/** \file\r
- *\r
- *  Simple HTTP Webserver Application. When connected to the uIP stack,\r
- *  this will serve out files to HTTP clients.\r
- */\r
\r
-#include "HTTPServerApp.h"\r
-\r
-/** HTTP server response header, for transmission before the page contents. This indicates to the host that a page exists at the\r
- *  given location, and gives extra connection information.\r
- */\r
-char PROGMEM HTTP200Header[] = "HTTP/1.1 200 OK\r\n"\r
-                               "Server: LUFA RNDIS\r\n"\r
-                               "Content-type: text/html\r\n"\r
-                               "Connection: close\r\n\r\n";\r
-\r
-/** HTTP server response header, for transmission before a resource not found error. This indicates to the host that the given\r
- *  given URL is invalid, and gives extra error information.\r
- */\r
-char PROGMEM HTTP404Header[] = "HTTP/1.1 404 Not Found\r\n"\r
-                               "Server: LUFA RNDIS\r\n"\r
-                               "Connection: close\r\n\r\n"\r
-                                                          "The requested file was not found.";\r
-\r
-/** FAT Fs structure to hold the internal state of the FAT driver for the dataflash contents. */\r
-FATFS DiskFATState;\r
-\r
-/** Initialization function for the simple HTTP webserver. */\r
-void WebserverApp_Init(void)\r
-{\r
-       /* Listen on port 80 for HTTP connections from hosts */\r
-       uip_listen(HTONS(HTTP_SERVER_PORT));\r
-       \r
-       /* Mount the dataflash disk via FatFS */\r
-       f_mount(0, &DiskFATState);\r
-}\r
-\r
-/** uIP stack application callback for the simple HTTP webserver. This function must be called each time the\r
- *  TCP/IP stack needs a TCP packet to be processed.\r
- */\r
-void WebserverApp_Callback(void)\r
-{\r
-       uip_tcp_appstate_t* const AppState    = &uip_conn->appstate;\r
-       char*                     AppData     = (char*)uip_appdata;\r
-       uint16_t                  AppDataSize = 0;\r
-\r
-       if (uip_aborted() || uip_timedout())\r
-       {\r
-               /* Close the file before terminating, if it is open */\r
-               f_close(&AppState->FileToSend);\r
-\r
-               AppState->CurrentState = WEBSERVER_STATE_Closed;\r
-\r
-               return;\r
-       }\r
-       else if (uip_closed())\r
-       {\r
-               /* Completed connection, just return */\r
-               return;\r
-       }\r
-       else if (uip_connected())\r
-       {\r
-               /* New connection - initialize connection state and data pointer to the appropriate HTTP header */\r
-               AppState->CurrentState = WEBSERVER_STATE_OpenRequestedFile;\r
-       }\r
-       \r
-       switch (AppState->CurrentState)\r
-       {\r
-               case WEBSERVER_STATE_OpenRequestedFile:\r
-                       /* Wait for the packet containing the request header */\r
-                       if (uip_datalen())\r
-                       {\r
-                               /* Must be a GET request, abort otherwise */\r
-                               if (strncmp(AppData, "GET ", (sizeof("GET ") - 1)) != 0)\r
-                               {\r
-                                       uip_abort();\r
-                                       break;\r
-                               }\r
-               \r
-                               char FileName[13];\r
-\r
-                               /* Copy over the requested filename from the GET request */\r
-                               for (uint8_t i = 0; i < (sizeof(FileName) - 1); i++)\r
-                               {\r
-                                       FileName[i] = AppData[sizeof("GET ") + i];\r
-                                       \r
-                                       if (FileName[i] == ' ')\r
-                                       {\r
-                                               FileName[i] = 0x00;\r
-                                               break;\r
-                                       }\r
-                               }\r
-                               \r
-                               /* Ensure requested filename is null-terminated */\r
-                               FileName[(sizeof(FileName) - 1)] = 0x00;\r
-                               \r
-                               /* If no filename specified, assume the default of INDEX.HTM */\r
-                               if (FileName[0] == 0x00)\r
-                                 strcpy(FileName, "INDEX.HTM");\r
-                               \r
-                               /* Try to open the file from the Dataflash disk */\r
-                               AppState->FileOpen = (f_open(&AppState->FileToSend, FileName, FA_OPEN_EXISTING | FA_READ) == FR_OK);\r
-\r
-                               AppState->CurrentState = WEBSERVER_STATE_SendHeaders;\r
-                       }\r
-\r
-                       break;\r
-               case WEBSERVER_STATE_SendHeaders:\r
-                       /* Determine what HTTP header should be sent to the client */\r
-                       if (AppState->FileOpen)\r
-                       {\r
-                               AppDataSize = strlen_P(HTTP200Header);\r
-                               strncpy_P(AppData, HTTP200Header, AppDataSize);                         \r
-                       }\r
-                       else\r
-                       {\r
-                               AppDataSize = strlen_P(HTTP404Header);\r
-                               strncpy_P(AppData, HTTP404Header, AppDataSize);                         \r
-                       }\r
-                       \r
-                       uip_send(AppData, AppDataSize);\r
-                       \r
-                       AppState->CurrentState = WEBSERVER_STATE_SendData;\r
-                       break;\r
-               case WEBSERVER_STATE_SendData:\r
-                       /* If end of file/file not open, progress to the close state */\r
-                       if (!(AppState->FileOpen))\r
-                       {\r
-                               f_close(&AppState->FileToSend);\r
-                               uip_close();\r
-                               AppState->CurrentState = WEBSERVER_STATE_Closed;\r
-                               break;\r
-                       }\r
-\r
-                       uint16_t MaxSegSize = uip_mss();\r
-                       \r
-                       /* Read the next chunk of data from the open file */\r
-                       f_read(&AppState->FileToSend, AppData, MaxSegSize, &AppDataSize);\r
-                       AppState->FileOpen = (MaxSegSize == AppDataSize);\r
-\r
-                       /* If data was read, send it to the client */\r
-                       if (AppDataSize)\r
-                         uip_send(AppData, AppDataSize);\r
-                                       \r
-                       break;\r
-       }\r
-}\r
+/*
+             LUFA Library
+     Copyright (C) Dean Camera, 2012.
+
+  dean [at] fourwalledcubicle [dot] com
+           www.lufa-lib.org
+*/
+
+/*
+  Copyright 2012  Dean Camera (dean [at] fourwalledcubicle [dot] com)
+
+  Permission to use, copy, modify, distribute, and sell this
+  software and its documentation for any purpose is hereby granted
+  without fee, provided that the above copyright notice appear in
+  all copies and that both that the copyright notice and this
+  permission notice and warranty disclaimer appear in supporting
+  documentation, and that the name of the author not be used in
+  advertising or publicity pertaining to distribution of the
+  software without specific, written prior permission.
+
+  The author disclaim all warranties with regard to this
+  software, including all implied warranties of merchantability
+  and fitness.  In no event shall the author be liable for any
+  special, indirect or consequential damages or any damages
+  whatsoever resulting from loss of use, data or profits, whether
+  in an action of contract, negligence or other tortious action,
+  arising out of or in connection with the use or performance of
+  this software.
+*/
+
+/** \file
+ *
+ *  Simple HTTP Webserver Application. When connected to the uIP stack,
+ *  this will serve out files to HTTP clients on port 80.
+ */
+
+#define  INCLUDE_FROM_HTTPSERVERAPP_C
+#include "HTTPServerApp.h"
+
+/** HTTP server response header, for transmission before the page contents. This indicates to the host that a page exists at the
+ *  given location, and gives extra connection information.
+ */
+const char PROGMEM HTTP200Header[] = "HTTP/1.1 200 OK\r\n"
+                                     "Server: LUFA " LUFA_VERSION_STRING "\r\n"
+                                     "Connection: close\r\n"
+                                     "MIME-version: 1.0\r\n"
+                                     "Content-Type: ";
+
+/** HTTP server response header, for transmission before a resource not found error. This indicates to the host that the given
+ *  URL is invalid, and gives extra error information.
+ */
+const char PROGMEM HTTP404Header[] = "HTTP/1.1 404 Not Found\r\n"
+                                     "Server: LUFA " LUFA_VERSION_STRING "\r\n"
+                                     "Connection: close\r\n"
+                                     "MIME-version: 1.0\r\n"
+                                     "Content-Type: text/plain\r\n\r\n"
+                                     "Error 404: File Not Found: /";
+
+/** Default filename to fetch when a directory is requested */
+const char PROGMEM DefaultDirFileName[] = "index.htm";
+
+/** Default MIME type sent if no other MIME type can be determined. */
+const char PROGMEM DefaultMIMEType[] = "text/plain";
+
+/** List of MIME types for each supported file extension. */
+const MIME_Type_t MIMETypes[] =
+       {
+               {.Extension = "htm", .MIMEType = "text/html"},
+               {.Extension = "jpg", .MIMEType = "image/jpeg"},
+               {.Extension = "gif", .MIMEType = "image/gif"},
+               {.Extension = "bmp", .MIMEType = "image/bmp"},
+               {.Extension = "png", .MIMEType = "image/png"},
+               {.Extension = "ico", .MIMEType = "image/x-icon"},
+               {.Extension = "exe", .MIMEType = "application/octet-stream"},
+               {.Extension = "gz",  .MIMEType = "application/x-gzip"},
+               {.Extension = "zip", .MIMEType = "application/zip"},
+               {.Extension = "pdf", .MIMEType = "application/pdf"},
+       };
+
+/** FATFs structure to hold the internal state of the FAT driver for the Dataflash contents. */
+FATFS DiskFATState;
+
+
+/** Initialization function for the simple HTTP webserver. */
+void HTTPServerApp_Init(void)
+{
+       /* Listen on port 80 for HTTP connections from hosts */
+       uip_listen(HTONS(HTTP_SERVER_PORT));
+
+       /* Mount the Dataflash disk via FatFS */
+       f_mount(0, &DiskFATState);
+}
+
+/** uIP stack application callback for the simple HTTP webserver. This function must be called each time the
+ *  TCP/IP stack needs a TCP packet to be processed.
+ */
+void HTTPServerApp_Callback(void)
+{
+       uip_tcp_appstate_t* const AppState = &uip_conn->appstate;
+
+       if (uip_aborted() || uip_timedout() || uip_closed())
+       {
+               /* Lock to the closed state so that no further processing will occur on the connection */
+               AppState->HTTPServer.CurrentState  = WEBSERVER_STATE_Closing;
+               AppState->HTTPServer.NextState     = WEBSERVER_STATE_Closing;
+       }
+
+       if (uip_connected())
+       {
+               /* New connection - initialize connection state values */
+               AppState->HTTPServer.CurrentState  = WEBSERVER_STATE_OpenRequestedFile;
+               AppState->HTTPServer.NextState     = WEBSERVER_STATE_OpenRequestedFile;
+               AppState->HTTPServer.FileOpen      = false;
+               AppState->HTTPServer.ACKedFilePos  = 0;
+               AppState->HTTPServer.SentChunkSize = 0;
+       }
+
+       if (uip_acked())
+       {
+               /* Add the amount of ACKed file data to the total sent file bytes counter */
+               AppState->HTTPServer.ACKedFilePos += AppState->HTTPServer.SentChunkSize;
+
+               /* Progress to the next state once the current state's data has been ACKed */
+               AppState->HTTPServer.CurrentState = AppState->HTTPServer.NextState;
+       }
+
+       if (uip_rexmit())
+       {
+               /* Return file pointer to the last ACKed position */
+               f_lseek(&AppState->HTTPServer.FileHandle, AppState->HTTPServer.ACKedFilePos);
+       }
+
+       if (uip_rexmit() || uip_acked() || uip_newdata() || uip_connected() || uip_poll())
+       {
+               switch (AppState->HTTPServer.CurrentState)
+               {
+                       case WEBSERVER_STATE_OpenRequestedFile:
+                               HTTPServerApp_OpenRequestedFile();
+                               break;
+                       case WEBSERVER_STATE_SendResponseHeader:
+                               HTTPServerApp_SendResponseHeader();
+                               break;
+                       case WEBSERVER_STATE_SendData:
+                               HTTPServerApp_SendData();
+                               break;
+                       case WEBSERVER_STATE_Closing:
+                               /* Connection is being terminated for some reason - close file handle */
+                               f_close(&AppState->HTTPServer.FileHandle);
+                               AppState->HTTPServer.FileOpen = false;
+
+                               /* If connection is not already closed, close it */
+                               uip_close();
+
+                               AppState->HTTPServer.CurrentState = WEBSERVER_STATE_Closed;
+                               AppState->HTTPServer.NextState    = WEBSERVER_STATE_Closed;
+                               break;
+               }
+       }
+}
+
+/** HTTP Server State handler for the Request Process state. This state manages the processing of incoming HTTP
+ *  GET requests to the server from the receiving HTTP client.
+ */
+static void HTTPServerApp_OpenRequestedFile(void)
+{
+       uip_tcp_appstate_t* const AppState    = &uip_conn->appstate;
+       char*               const AppData     = (char*)uip_appdata;
+
+       /* No HTTP header received from the client, abort processing */
+       if (!(uip_newdata()))
+         return;
+
+       char* RequestToken      = strtok(AppData, " ");
+       char* RequestedFileName = strtok(NULL, " ");
+
+       /* Must be a GET request, abort otherwise */
+       if (strcmp_P(RequestToken, PSTR("GET")) != 0)
+       {
+               uip_abort();
+               return;
+       }
+
+       /* Copy over the requested filename */
+       strlcpy(AppState->HTTPServer.FileName, &RequestedFileName[1], sizeof(AppState->HTTPServer.FileName));
+
+       /* Determine the length of the URI so that it can be checked to see if it is a directory */
+       uint8_t FileNameLen = strlen(AppState->HTTPServer.FileName);
+
+       /* If the URI is a directory, append the default filename */
+       if ((AppState->HTTPServer.FileName[FileNameLen - 1] == '/') || !(FileNameLen))
+       {
+               strlcpy_P(&AppState->HTTPServer.FileName[FileNameLen], DefaultDirFileName,
+                         (sizeof(AppState->HTTPServer.FileName) - FileNameLen));
+       }
+
+       /* Try to open the file from the Dataflash disk */
+       AppState->HTTPServer.FileOpen     = (f_open(&AppState->HTTPServer.FileHandle, AppState->HTTPServer.FileName,
+                                                   (FA_OPEN_EXISTING | FA_READ)) == FR_OK);
+
+       /* Lock to the SendResponseHeader state until connection terminated */
+       AppState->HTTPServer.CurrentState = WEBSERVER_STATE_SendResponseHeader;
+       AppState->HTTPServer.NextState    = WEBSERVER_STATE_SendResponseHeader;
+}
+
+/** HTTP Server State handler for the HTTP Response Header Send state. This state manages the transmission of
+ *  the HTTP response header to the receiving HTTP client.
+ */
+static void HTTPServerApp_SendResponseHeader(void)
+{
+       uip_tcp_appstate_t* const AppState    = &uip_conn->appstate;
+       char*               const AppData     = (char*)uip_appdata;
+
+       char* Extension     = strpbrk(AppState->HTTPServer.FileName, ".");
+       bool  FoundMIMEType = false;
+
+       /* If the file isn't already open, it wasn't found - send back a 404 error response and abort */
+       if (!(AppState->HTTPServer.FileOpen))
+       {
+               /* Copy over the HTTP 404 response header and send it to the receiving client */
+               strcpy_P(AppData, HTTP404Header);
+               strcat(AppData, AppState->HTTPServer.FileName);
+               uip_send(AppData, strlen(AppData));
+
+               AppState->HTTPServer.NextState = WEBSERVER_STATE_Closing;
+               return;
+       }
+
+       /* Copy over the HTTP 200 response header and send it to the receiving client */
+       strcpy_P(AppData, HTTP200Header);
+
+       /* Check to see if a MIME type for the requested file's extension was found */
+       if (Extension != NULL)
+       {
+               /* Look through the MIME type list, copy over the required MIME type if found */
+               for (uint8_t i = 0; i < (sizeof(MIMETypes) / sizeof(MIMETypes[0])); i++)
+               {
+                       if (strcmp(&Extension[1], MIMETypes[i].Extension) == 0)
+                       {
+                               strcat(AppData, MIMETypes[i].MIMEType);
+                               FoundMIMEType = true;
+                               break;
+                       }
+               }
+       }
+
+       /* Check if a MIME type was found and copied to the output buffer */
+       if (!(FoundMIMEType))
+       {
+               /* MIME type not found - copy over the default MIME type */
+               strcat_P(AppData, DefaultMIMEType);
+       }
+
+       /* Add the end-of-line terminator and end-of-headers terminator after the MIME type */
+       strcat_P(AppData, PSTR("\r\n\r\n"));
+
+       /* Send the MIME header to the receiving client */
+       uip_send(AppData, strlen(AppData));
+
+       /* When the MIME header is ACKed, progress to the data send stage */
+       AppState->HTTPServer.NextState = WEBSERVER_STATE_SendData;
+}
+
+/** HTTP Server State handler for the Data Send state. This state manages the transmission of file chunks
+ *  to the receiving HTTP client.
+ */
+static void HTTPServerApp_SendData(void)
+{
+       uip_tcp_appstate_t* const AppState    = &uip_conn->appstate;
+       char*               const AppData     = (char*)uip_appdata;
+
+       /* Get the maximum segment size for the current packet */
+       uint16_t MaxChunkSize = uip_mss();
+
+       /* Read the next chunk of data from the open file */
+       f_read(&AppState->HTTPServer.FileHandle, AppData, MaxChunkSize, &AppState->HTTPServer.SentChunkSize);
+
+       /* Send the next file chunk to the receiving client */
+       uip_send(AppData, AppState->HTTPServer.SentChunkSize);
+
+       /* Check if we are at the last chunk of the file, if so next ACK should close the connection */
+       if (MaxChunkSize != AppState->HTTPServer.SentChunkSize)
+         AppState->HTTPServer.NextState = WEBSERVER_STATE_Closing;
+}
+