forked from ericyanush/8Bit-Pic-Websockets
-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathHTTP2.h
executable file
·604 lines (510 loc) · 24.3 KB
/
HTTP2.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
/*********************************************************************
*
* HTTP Headers for Microchip TCP/IP Stack
*
*********************************************************************
* FileName: HTTP2.h
* Dependencies: None
* Processor: PIC18, PIC24F, PIC24H, dsPIC30F, dsPIC33F, PIC32
* Compiler: Microchip C32 v1.05 or higher
* Microchip C30 v3.12 or higher
* Microchip C18 v3.30 or higher
* HI-TECH PICC-18 PRO 9.63PL2 or higher
* Company: Microchip Technology, Inc.
*
* Software License Agreement
*
* Copyright (C) 2002-2009 Microchip Technology Inc. All rights
* reserved.
*
* Microchip licenses to you the right to use, modify, copy, and
* distribute:
* (i) the Software when embedded on a Microchip microcontroller or
* digital signal controller product ("Device") which is
* integrated into Licensee's product; or
* (ii) ONLY the Software driver source files ENC28J60.c, ENC28J60.h,
* ENCX24J600.c and ENCX24J600.h ported to a non-Microchip device
* used in conjunction with a Microchip ethernet controller for
* the sole purpose of interfacing with the ethernet controller.
*
* You should refer to the license agreement accompanying this
* Software for additional information regarding your rights and
* obligations.
*
* THE SOFTWARE AND DOCUMENTATION ARE PROVIDED "AS IS" WITHOUT
* WARRANTY OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT
* LIMITATION, ANY WARRANTY OF MERCHANTABILITY, FITNESS FOR A
* PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT SHALL
* MICROCHIP BE LIABLE FOR ANY INCIDENTAL, SPECIAL, INDIRECT OR
* CONSEQUENTIAL DAMAGES, LOST PROFITS OR LOST DATA, COST OF
* PROCUREMENT OF SUBSTITUTE GOODS, TECHNOLOGY OR SERVICES, ANY CLAIMS
* BY THIRD PARTIES (INCLUDING BUT NOT LIMITED TO ANY DEFENSE
* THEREOF), ANY CLAIMS FOR INDEMNITY OR CONTRIBUTION, OR OTHER
* SIMILAR COSTS, WHETHER ASSERTED ON THE BASIS OF CONTRACT, TORT
* (INCLUDING NEGLIGENCE), BREACH OF WARRANTY, OR OTHERWISE.
*
*
* Author Date Comment
*~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
* Nilesh Rajbharti 8/14/01 Original
* Elliott Wood 6/4/07 Complete rewrite (known as HTTP2)
********************************************************************/
#ifndef __HTTP2_H
#define __HTTP2_H
#include "TCPIP Stack/TCPIP.h"
#if defined(STACK_USE_HTTP2_SERVER)
/****************************************************************************
Section:
Server Configuration Settings
***************************************************************************/
#if !defined(HTTP_PORT)
#define HTTP_PORT (80u) // Listening port for HTTP server
#endif
#if !defined(HTTPS_PORT)
#define HTTPS_PORT (443u) // Listening port for HTTPS server (when SSL enabled)
#endif
#if !defined(HTTP_MAX_DATA_LEN)
#define HTTP_MAX_DATA_LEN (100u)
#endif
#if !defined(HTTP_MIN_CALLBACK_FREE)
#define HTTP_MIN_CALLBACK_FREE (16u)
#endif
#define HTTP_CACHE_LEN ("600") // Max lifetime (sec) of static responses as string
#define HTTP_TIMEOUT (45u) // Max time (sec) to await more data before timing out and disconnecting the socket
// Authentication requires Base64 decoding
#if defined(HTTP_USE_AUTHENTICATION)
#ifndef STACK_USE_BASE64_DECODE
#define STACK_USE_BASE64_DECODE
#endif
#endif
/****************************************************************************
Section:
Commands and Server Responses
***************************************************************************/
//Supported Commands and Server Response Codes
typedef enum
{
HTTP_GET = 0u, // GET command is being processed
#if defined (STACK_USE_WEBSOCKETS)
WEBSOCKET_FRAME, //Websockets
#endif
HTTP_POST, // POST command is being processed
HTTP_BAD_REQUEST, // 400 Bad Request will be returned
HTTP_UNAUTHORIZED, // 401 Unauthorized will be returned
HTTP_NOT_FOUND, // 404 Not Found will be returned
HTTP_OVERFLOW, // 414 Request-URI Too Long will be returned
HTTP_INTERNAL_SERVER_ERROR, // 500 Internal Server Error will be returned
HTTP_NOT_IMPLEMENTED, // 501 Not Implemented (not a GET or POST command)
#if defined(HTTP_MPFS_UPLOAD)
HTTP_MPFS_FORM, // Show the MPFS Upload form
HTTP_MPFS_UP, // An MPFS Upload is being processed
HTTP_MPFS_OK, // An MPFS Upload was successful
HTTP_MPFS_ERROR, // An MPFS Upload was not a valid image
#endif
HTTP_REDIRECT, // 302 Redirect will be returned
HTTP_SSL_REQUIRED // 403 Forbidden is returned, indicating SSL is required
} HTTP_STATUS;
/****************************************************************************
Section:
HTTP State Definitions
***************************************************************************/
// Basic HTTP Connection State Machine
typedef enum
{
SM_HTTP_IDLE = 0u, // Socket is idle
SM_HTTP_PARSE_REQUEST, // Parses the first line for a file name and GET args
SM_HTTP_PARSE_HEADERS, // Reads and parses headers one at a time
SM_HTTP_AUTHENTICATE, // Validates the current authorization state
SM_HTTP_PROCESS_GET, // Invokes user callback for GET args or cookies
#if defined (STACK_USE_WEBSOCKETS)
SM_HTTP_INIT_WEBSOCKET,
SM_HTTP_PROC_WEBSOCKET,
#endif
SM_HTTP_PROCESS_POST, // Invokes user callback for POSTed data
SM_HTTP_PROCESS_REQUEST, // Begins the process of returning data
SM_HTTP_SERVE_HEADERS, // Sends any required headers for the response
SM_HTTP_SERVE_COOKIES, // Adds any cookies to the response
SM_HTTP_SERVE_BODY, // Serves the actual content
SM_HTTP_SEND_FROM_CALLBACK, // Invokes a dynamic variable callback
SM_HTTP_DISCONNECT // Disconnects the server and closes all files
} SM_HTTP2;
// Result states for execution callbacks
typedef enum
{
HTTP_IO_DONE = 0u, // Finished with procedure
HTTP_IO_NEED_DATA, // More data needed to continue, call again later
HTTP_IO_WAITING // Waiting for asynchronous process to complete, call again later
} HTTP_IO_RESULT;
// Result states for HTTPPostReadName and HTTPPostReadValue
typedef enum
{
HTTP_READ_OK = 0u, // Read was successful
HTTP_READ_TRUNCATED, // Buffer overflow prevented by truncating value
HTTP_READ_INCOMPLETE // Entire object is not yet in the buffer. Try again later.
} HTTP_READ_STATUS;
// File type definitions
typedef enum
{
HTTP_TXT = 0u, // File is a text document
HTTP_HTM, // File is HTML (extension .htm)
HTTP_HTML, // File is HTML (extension .html)
HTTP_CGI, // File is HTML (extension .cgi)
HTTP_XML, // File is XML (extension .xml)
HTTP_CSS, // File is stylesheet (extension .css)
HTTP_GIF, // File is GIF image (extension .gif)
HTTP_PNG, // File is PNG image (extension .png)
HTTP_JPG, // File is JPG image (extension .jpg)
HTTP_JAVA, // File is java (extension .java)
HTTP_WAV, // File is audio (extension .wav)
HTTP_UNKNOWN // File type is unknown
} HTTP_FILE_TYPE;
// HTTP Connection Struct
// Stores partial state data for each connection
// Meant for storage in fast access RAM
typedef struct
{
SM_HTTP2 sm; // Current connection state
TCP_SOCKET socket; // Socket being served
} HTTP_STUB;
#define sktHTTP httpStubs[curHTTPID].socket // Access the current socket
// Stores extended state data for each connection
typedef struct
{
DWORD byteCount; // How many bytes have been read so far
DWORD nextCallback; // Byte index of the next callback
DWORD callbackID; // Callback ID to execute, also used as watchdog timer
DWORD callbackPos; // Callback position indicator
BYTE *ptrData; // Points to first free byte in data
BYTE *ptrRead; // Points to current read location
MPFS_HANDLE file; // File pointer for the file being served
MPFS_HANDLE offsets; // File pointer for any offset info being used
BYTE hasArgs; // True if there were get or cookie arguments
BYTE isAuthorized; // 0x00-0x79 on fail, 0x80-0xff on pass
HTTP_STATUS httpStatus; // Request method/status
HTTP_FILE_TYPE fileType; // File type to return with Content-Type
BYTE data[HTTP_MAX_DATA_LEN]; // General purpose data buffer
#if defined STACK_USE_WEBSOCKETS
BYTE webSocketKey[WS_KEY_LENGTH];
#endif
#if defined(HTTP_USE_POST)
BYTE smPost; // POST state machine variable
#endif
} HTTP_CONN;
#if defined(HTTP_SAVE_CONTEXT_IN_PIC_RAM)
#define RESERVED_HTTP_MEMORY 0ul // Macro indicating how much RAM to allocate on an ethernet controller to store HTTP state data.
#else
#define RESERVED_HTTP_MEMORY ((DWORD)MAX_HTTP_CONNECTIONS * (DWORD)sizeof(HTTP_CONN))
#endif
/****************************************************************************
Section:
Global HTTP Variables
***************************************************************************/
#if defined(HTTP_SAVE_CONTEXT_IN_PIC_RAM)
extern HTTP_CONN HTTPControlBlocks[MAX_HTTP_CONNECTIONS];
#define curHTTP HTTPControlBlocks[curHTTPID] // Current HTTP connection state
#else
extern HTTP_CONN curHTTP;
#endif
extern HTTP_STUB httpStubs[MAX_HTTP_CONNECTIONS];
extern BYTE curHTTPID;
/****************************************************************************
Section:
Function Prototypes
***************************************************************************/
void HTTPInit(void);
void HTTPServer(void);
BYTE* HTTPURLDecode(BYTE* cData);
BYTE* HTTPGetArg(BYTE* cData, BYTE* cArg);
void HTTPIncFile(ROM BYTE* cFile);
#if defined(__18CXX)
BYTE* HTTPGetROMArg(BYTE* cData, ROM BYTE* cArg);
#else
// Non-ROM variant for C30 / C32
#define HTTPGetROMArg(a,b) HTTPGetArg(a,(BYTE*)b)
#endif
#if defined(HTTP_USE_POST)
HTTP_READ_STATUS HTTPReadPostName(BYTE* cData, WORD wLen);
HTTP_READ_STATUS HTTPReadPostValue(BYTE* cData, WORD wLen);
#endif
/*****************************************************************************
Function:
HTTP_READ_STATUS HTTPReadPostPair(BYTE* cData, WORD wLen)
Summary:
Reads a name and value pair from a URL encoded string in the TCP buffer.
Description:
Reads a name and value pair from a URL encoded string in the TCP buffer.
This function is meant to be called from an HTTPExecutePost callback to
facilitate easier parsing of incoming data. This function also prevents
buffer overflows by forcing the programmer to indicate how many bytes are
expected. At least 2 extra bytes are needed in cData over the maximum
length of data expected to be read.
This function will read until the next '&' character, which indicates the
end of a value parameter. It assumes that the front of the buffer is
the beginning of the name paramter to be read.
This function properly updates curHTTP.byteCount by decrementing it
by the number of bytes read. It also removes the delimiting '&' from
the buffer.
Once complete, two strings will exist in the cData buffer. The first is
the parameter name that was read, while the second is the associated
value.
Precondition:
Front of TCP buffer is the beginning of a name parameter, and the rest of
the TCP buffer contains a URL-encoded string with a name parameter
terminated by a '=' character and a value parameter terminated by a '&'.
Parameters:
cData - where to store the name and value strings once they are read
wLen - how many bytes can be written to cData
Return Values:
HTTP_READ_OK - name and value were successfully read
HTTP_READ_TRUNCTATED - entire name and value could not fit in the buffer,
so input was truncated and data has been lost
HTTP_READ_INCOMPLETE - entire name and value was not yet in the buffer,
so call this function again later to retrieve
Remarks:
This function is aliased to HTTPReadPostValue, since they effectively
perform the same task. The name is provided only for completeness.
***************************************************************************/
#define HTTPReadPostPair(cData, wLen) HTTPReadPostValue(cData, wLen)
/****************************************************************************
Section:
User-Implemented Callback Function Prototypes
***************************************************************************/
/*****************************************************************************
Function:
HTTP_IO_RESULT HTTPExecuteGet(void)
Summary:
Processes GET form field variables and cookies.
Description:
This function is implemented by the application developer in
CustomHTTPApp.c. Its purpose is to parse the data received from
URL parameters (GET method forms) and cookies and perform any
application-specific tasks in response to these inputs. Any
required authentication has already been validated.
When this function is called, curHTTP.data contains sequential
name/value pairs of strings representing the data received. In this
format, HTTPGetArg and HTTPGetROMArg can be used to search for
specific variables in the input. If data buffer space associated
with this connection is required, curHTTP.data may be overwritten
here once the application is done with the values. Any data placed
there will be available to future callbacks for this connection,
including HTTPExecutePost and any HTTPPrint_varname dynamic
substitutions.
This function may also issue redirections by setting curHTTP.data
to the destination file name or URL, and curHTTP.httpStatus to
HTTP_REDIRECT.
Finally, this function may set cookies. Set curHTTP.data to a series
of name/value string pairs (in the same format in which parameters
arrive) and then set curHTTP.hasArgs equal to the number of cookie
name/value pairs. The cookies will be transmitted to the browser,
and any future requests will have those values available in
curHTTP.data.
Precondition:
None
Parameters:
None
Return Values:
HTTP_IO_DONE - application is done processing
HTTP_IO_NEED_DATA - this value may not be returned because more data
will not become available
HTTP_IO_WAITING - the application is waiting for an asynchronous
process to complete, and this function should be
called again later
Remarks:
This function is only called if variables are received via URL
parameters or Cookie arguments. This function may NOT write to the
TCP buffer.
This function may service multiple HTTP requests simultaneously.
Exercise caution when using global or static variables inside this
routine. Use curHTTP.callbackPos or curHTTP.data for storage associated
with individual requests.
***************************************************************************/
HTTP_IO_RESULT HTTPExecuteGet(void);
/*****************************************************************************
Function:
HTTP_IO_RESULT HTTPExecutePost(void)
Summary:
Processes POST form variables and data.
Description:
This function is implemented by the application developer in
CustomHTTPApp.c. Its purpose is to parse the data received from
POST forms and perform any application-specific tasks in response
to these inputs. Any required authentication has already been
validated before this function is called.
When this function is called, POST data will be waiting in the
TCP buffer. curHTTP.byteCount will indicate the number of bytes
remaining to be received before the browser request is complete.
Since data is still in the TCP buffer, the application must call
TCPGet or TCPGetArray in order to retrieve bytes. When this is done,
curHTTP.byteCount MUST be updated to reflect how many bytes now
remain. The functions TCPFind, TCPFindString, TCPFindROMString,
TCPFindArray, and TCPFindROMArray may be helpful to locate data
in the TCP buffer.
In general, data submitted from web forms via POST is URL encoded.
The HTTPURLDecode function can be used to decode this information
back to a standard string if required. If data buffer space
associated with this connection is required, curHTTP.data may be
overwritten here once the application is done with the values.
Any data placed there will be available to future callbacks for
this connection, including HTTPExecutePost and any
HTTPPrint_varname dynamic substitutions.
Whenever a POST form is processed it is recommended to issue a
redirect back to the browser, either to a status page or to
the same form page that was posted. This prevents accidental
duplicate submissions (by clicking refresh or back/forward) and
avoids browser warnings about "resubmitting form data". Redirects
may be issued to the browser by setting curHTTP.data to the
destination file or URL, and curHTTP.httpStatus to HTTP_REDIRECT.
Finally, this function may set cookies. Set curHTTP.data to a series
of name/value string pairs (in the same format in which parameters
arrive) and then set curHTTP.hasArgs equal to the number of cookie
name/value pairs. The cookies will be transmitted to the browser,
and any future requests will have those values available in
curHTTP.data.
Precondition:
None
Parameters:
None
Return Values:
HTTP_IO_DONE - application is done processing
HTTP_IO_NEED_DATA - more data is needed to continue, and this
function should be called again later
HTTP_IO_WAITING - the application is waiting for an asynchronous
process to complete, and this function should
be called again later
Remarks:
This function is only called when the request method is POST, and is
only used when HTTP_USE_POST is defined. This method may NOT write
to the TCP buffer.
This function may service multiple HTTP requests simultaneously.
Exercise caution when using global or static variables inside this
routine. Use curHTTP.callbackPos or curHTTP.data for storage associated
with individual requests.
***************************************************************************/
#if defined(HTTP_USE_POST)
HTTP_IO_RESULT HTTPExecutePost(void);
#endif
/*****************************************************************************
Function:
BYTE HTTPNeedsAuth(BYTE* cFile)
Summary:
Determines if a given file name requires authentication
Description:
This function is implemented by the application developer in
CustomHTTPApp.c. Its function is to determine if a file being
requested requires authentication to view. The user name and password,
if supplied, will arrive later with the request headers, and will be
processed at that time.
Return values 0x80 - 0xff indicate that authentication is not required,
while values from 0x00 to 0x79 indicate that a user name and password
are required before proceeding. While most applications will only use a
single value to grant access and another to require authorization, the
range allows multiple "realms" or sets of pages to be protected, with
different credential requirements for each.
The return value of this function is saved as curHTTP.isAuthorized, and
will be available to future callbacks, including HTTPCheckAuth and any
of the HTTPExecuteGet, HTTPExecutePost, or HTTPPrint_varname callbacks.
Precondition:
None
Parameters:
cFile - the name of the file being requested
Return Values:
<= 0x79 - valid authentication is required
>= 0x80 - access is granted for this connection
Remarks:
This function may NOT write to the TCP buffer.
***************************************************************************/
#if defined(HTTP_USE_AUTHENTICATION)
BYTE HTTPNeedsAuth(BYTE* cFile);
#endif
/*****************************************************************************
Function:
BYTE HTTPCheckAuth(BYTE* cUser, BYTE* cPass)
Summary:
Performs validation on a specific user name and password.
Description:
This function is implemented by the application developer in
CustomHTTPApp.c. Its function is to determine if the user name and
password supplied by the client are acceptable for this resource.
The value of curHTTP.isAuthorized will be set to the previous return
value of HTTPRequiresAuthorization. This callback function can check
this value to determine if only specific user names or passwords will
be accepted for this resource.
Return values 0x80 - 0xff indicate that the credentials were accepted,
while values from 0x00 to 0x79 indicate that authorization failed.
While most applications will only use a single value to grant access,
flexibility is provided to store multiple values in order to
indicate which user (or user's group) logged in.
The return value of this function is saved as curHTTP.isAuthorized, and
will be available to future callbacks, including any of the
HTTPExecuteGet, HTTPExecutePost, or HTTPPrint_varname callbacks.
Precondition:
None
Parameters:
cUser - the user name supplied by the client
cPass - the password supplied by the client
Return Values:
<= 0x79 - the credentials were rejected
>= 0x80 - access is granted for this connection
Remarks:
This function is only called when an Authorization header is
encountered.
This function may NOT write to the TCP buffer.
***************************************************************************/
#if defined(HTTP_USE_AUTHENTICATION)
BYTE HTTPCheckAuth(BYTE* cUser, BYTE* cPass);
#endif
/*****************************************************************************
Function:
void HTTPPrint_varname(void)
void HTTPPrint_varname(WORD wParam1)
void HTTPPrint_varname(WORD wParam1, WORD wParam2, ...)
Summary:
Inserts dynamic content into a web page
Description:
Functions in this style are implemented by the application developer in
CustomHTTPApp.c. These functions generate dynamic content to be
inserted into web pages and other files returned by the HTTP2 server.
Functions of this type are called when a dynamic variable is located
in a web page. (ie, ~varname~ ) The name between the tilde '~'
characters is appended to the base function name. In this example, the
callback would be named HTTPPrint_varname.
The function prototype is located in your project's HTTPPrint.h, which
is automatically generated by the MPFS2 Utility. The prototype will
have WORD parameters included for each parameter passed in the dynamic
variable. For example, the variable "~myArray(2,6)~" will generate the
prototype "void HTTPPrint_varname(WORD, WORD);".
When called, this function should write its output directly to the TCP
socket using any combination of TCPIsPutReady, TCPPut, TCPPutArray,
TCPPutString, TCPPutROMArray, and TCPPutROMString.
Before calling, the HTTP2 server guarantees that at least
HTTP_MIN_CALLBACK_FREE bytes (defaults to 16 bytes) are free in the
output buffer. If the function is writing less than this amount, it
should simply write the data to the socket and return.
In situations where a function needs to write more this amount, it
must manage its output state using curHTTP.callbackPos. This value
will be set to zero before the function is called. If the function is
managing its output state, it must set this to a non-zero value before
returning. Typically this is used to track how many bytes have been
written, or how many remain to be written. If curHTTP.callbackPos is
non-zero, the function will be called again when more buffer space is
available. Once the callback completes, set this value back to zero
to resume normal servicing of the request.
Precondition:
None
Parameters:
wParam1 - first parameter passed in the dynamic variable (if any)
wParam2 - second parameter passed in the dynamic variable (if any)
... - additional parameters as necessary
Returns:
None
Remarks:
This function may service multiple HTTP requests simultaneously,
especially when managing its output state. Exercise caution when using
global or static variables inside this routine. Use curHTTP.callbackPos
or curHTTP.data for storage associated with individual requests.
***************************************************************************/
#if defined(DOCUMENTATION_ONLY)
void HTTPPrint_varname(WORD wParam1, WORD wParam2, ...);
#endif
#endif
#endif