cueOS  2.4
cueOS - Universal Show Control OS for ARM
http_server.c
1/***============================================================================================================================
2 * Depedencies inclusion
3 * Necessary dependencies should be declared here. Header file should contain as little dependecies declarations as possible
4 *=============================================================================================================================*/
5
6#include "http_server.h"
7#include "altcp.h"
8#include "cmsis_os.h"
9#include <string.h>
10
11
12/***============================================================================================================================
13 * Private functions definitions
14 * These functions are only accessible from within the file's scope
15 *=============================================================================================================================*/
16
24static void _http_close(struct altcp_pcb *pcb, http_request_s *req){
25
26 altcp_arg(pcb, NULL);
27 altcp_recv(pcb, NULL);
28 altcp_sent(pcb, NULL);
29 altcp_poll(pcb, NULL, 0);
30 altcp_err(pcb, NULL);
31
32 if(req != NULL){
34 }
35
36 altcp_close(pcb);
37
38}
39
48static err_t _http_send(struct altcp_pcb *pcb, http_request_s *req){
49
50 err_t err = ERR_OK;
51
52 uint16_t max_len, chunk_len = LWIP_MIN(http_response_get_bytes_left(req->res), 0xffff);
53
54 if(chunk_len <= 0){
55 return ERR_OK;
56 }
57
58 max_len = altcp_sndbuf(pcb);
59 if (max_len < chunk_len) {
60 chunk_len = max_len;
61 }
62
63 do {
64 err = altcp_write(pcb, req->res->data_ptr + req->res->data_ptr_index, chunk_len, 0);
65 if (err == ERR_MEM) {
66 if ((altcp_sndbuf(pcb) == 0) || (altcp_sndqueuelen(pcb) >= TCP_SND_QUEUELEN)) {
67 chunk_len = 1;
68 } else {
69 chunk_len /= 2;
70 }
71 }
72 } while ((err == ERR_MEM) && (chunk_len > 1));
73
74 if (err == ERR_OK) {
75 req->res->data_ptr_index += chunk_len;
76 }
77
79 _http_close(pcb, req);
80 }
81
82 return err;
83
84}
85
86
97static err_t _http_receive(void *arg, struct altcp_pcb *pcb,struct pbuf *p, err_t err){
98
99 http_request_s *req = arg;
100
101 if(err != ERR_OK || p == NULL || req == NULL){
102 if(p != NULL){ altcp_recved(pcb, p->tot_len); pbuf_free(p); }
103 _http_close(pcb, req);
104 }else{
105 altcp_recved(pcb, p->tot_len);
106 if(http_request_parse(req, p)){
107 pbuf_free(p);
108 req->router(req);
109 _http_send(pcb, req);
110 }else{
111 pbuf_free(p);
112 _http_close(pcb, req);
113 }
114 }
115
116 return ERR_OK;
117
118}
119
129static err_t _http_sent(void *arg, struct altcp_pcb *pcb, uint16_t len){
130 http_request_s *req = arg;
131 if(req != NULL){
132 _http_send(pcb, req);
133 }
134 return ERR_OK;
135}
136
146static err_t _http_poll(void *arg, struct altcp_pcb *pcb){
147
148 http_request_s *req = arg;
149
150 if (req == NULL) {
151 _http_close(pcb, NULL);
152 return ERR_OK;
153 } else {
154 req->retry_count++;
155 if (req->retry_count == 10) {
156 _http_close(pcb, req);
157 return ERR_OK;
158 }
159 if (req->res->data_ptr != NULL) {
160 if (_http_send(pcb, req)) {
161 altcp_output(pcb);
162 }
163 }
164 }
165
166 return ERR_OK;
167
168}
169
177static void _http_error(void *arg, err_t err){
178 http_request_s *req = arg;
179 if (req != NULL) {
181 }
182}
183
193static err_t _http_accept(void *arg, struct altcp_pcb *pcb, err_t err){
194
195 http_server_s *server = arg;
196
197 if(err != ERR_OK || pcb == NULL){
198 err = ERR_VAL;
199 }else{
200
201 http_request_s *req = http_request_new(server->router);
202
203 if(req == NULL){
204 return ERR_MEM;
205 }
206
207 altcp_arg(pcb, req);
208 altcp_recv(pcb, _http_receive);
209 altcp_sent(pcb, _http_sent);
210 altcp_poll(pcb, _http_poll, 4);
211 altcp_err(pcb, _http_error);
212
213 }
214
215 return err;
216
217}
218
224static void _http_prepare_pcb(http_server_s *server){
225
226 struct altcp_pcb *pcb;
227
228 pcb = altcp_tcp_new_ip_type(IPADDR_TYPE_ANY);
229
230 altcp_bind(pcb, IP_ANY_TYPE, server->port);
231
232 pcb = altcp_listen(pcb);
233
234 server->pcb = pcb;
235
236 altcp_setprio(pcb, 1);
237 altcp_arg(pcb, server);
238 altcp_accept(pcb, _http_accept);
239
240}
241
242
243/***============================================================================================================================
244 * Public functions definitions
245 * These functions can be accessed outside of the file's scope
246 * @see HTTP_server.h for declarations
247 *=============================================================================================================================*/
248
257
258 http_server_s *server = pvPortMalloc(sizeof(http_server_s));
259
260 server->router = router;
261 server->port = port;
262
263 _http_prepare_pcb(server);
264
265 return server;
266
267}
void http_request_free(http_request_s *req)
Safely frees an HTTP request instance.
Definition: http_request.c:247
http_request_s * http_request_new(router_fn router)
Creates a new HTTP request instance.
Definition: http_request.c:224
void(* router_fn)(http_request_s *req)
Definition: http_request.h:23
uint8_t http_request_parse(http_request_s *req, struct pbuf *p)
Extracts and parses HTTP method headers and content from received data and calls the router function ...
Definition: http_request.c:293
uint32_t http_response_get_bytes_left(http_response_s *res)
Returns the amount of bytes left to be processed.
Definition: http_response.c:84
http_server_s * http_server_init(uint16_t port, router_fn router)
Initialises a new HTTP server instance.
Definition: http_server.c:256
HTTP request object.
Definition: http_request.h:44
http_response_s * res
Definition: http_request.h:56
router_fn router
Definition: http_request.h:54
uint8_t retry_count
Definition: http_request.h:52
uint32_t data_ptr_index
Definition: http_response.h:37
HTTP server structure object.
Definition: http_server.h:25
struct altcp_pcb * pcb
Definition: http_server.h:26
uint16_t port
Definition: http_server.h:27
router_fn router
Definition: http_server.h:28