123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351 |
- #include "curl_setup.h"
- #ifndef CURL_DISABLE_HTTP
- #include "urldata.h"
- #include "sendf.h"
- #include "content_encoding.h"
- #include "http.h"
- #include "non-ascii.h"
- #include "strtoofft.h"
- #include "warnless.h"
- #include "curl_memory.h"
- #include "memdebug.h"
- #ifdef CURL_DOES_CONVERSIONS
- static bool Curl_isxdigit_ascii(char digit)
- {
- return (digit >= 0x30 && digit <= 0x39)
- || (digit >= 0x41 && digit <= 0x46)
- || (digit >= 0x61 && digit <= 0x66);
- }
- #else
- #define Curl_isxdigit_ascii(x) Curl_isxdigit(x)
- #endif
- void Curl_httpchunk_init(struct connectdata *conn)
- {
- struct Curl_chunker *chunk = &conn->chunk;
- chunk->hexindex = 0;
- chunk->dataleft = 0;
- chunk->state = CHUNK_HEX;
- }
- CHUNKcode Curl_httpchunk_read(struct connectdata *conn,
- char *datap,
- ssize_t datalen,
- ssize_t *wrotep)
- {
- CURLcode result = CURLE_OK;
- struct Curl_easy *data = conn->data;
- struct Curl_chunker *ch = &conn->chunk;
- struct SingleRequest *k = &data->req;
- size_t piece;
- curl_off_t length = (curl_off_t)datalen;
- size_t *wrote = (size_t *)wrotep;
- *wrote = 0;
-
- if(data->set.http_te_skip && !k->ignorebody) {
- result = Curl_client_write(conn, CLIENTWRITE_BODY, datap, datalen);
- if(result)
- return CHUNKE_WRITE_ERROR;
- }
- while(length) {
- switch(ch->state) {
- case CHUNK_HEX:
- if(Curl_isxdigit_ascii(*datap)) {
- if(ch->hexindex < MAXNUM_SIZE) {
- ch->hexbuffer[ch->hexindex] = *datap;
- datap++;
- length--;
- ch->hexindex++;
- }
- else {
- return CHUNKE_TOO_LONG_HEX;
- }
- }
- else {
- char *endptr;
- if(0 == ch->hexindex)
-
- return CHUNKE_ILLEGAL_HEX;
-
- ch->hexbuffer[ch->hexindex] = 0;
-
- result = Curl_convert_from_network(conn->data, ch->hexbuffer,
- ch->hexindex);
- if(result) {
-
-
- return CHUNKE_ILLEGAL_HEX;
- }
- if(curlx_strtoofft(ch->hexbuffer, &endptr, 16, &ch->datasize))
- return CHUNKE_ILLEGAL_HEX;
- ch->state = CHUNK_LF;
- }
- break;
- case CHUNK_LF:
-
- if(*datap == 0x0a) {
-
- if(0 == ch->datasize) {
- ch->state = CHUNK_TRAILER;
- conn->trlPos = 0;
- }
- else
- ch->state = CHUNK_DATA;
- }
- datap++;
- length--;
- break;
- case CHUNK_DATA:
-
- piece = curlx_sotouz((ch->datasize >= length)?length:ch->datasize);
-
- if(!conn->data->set.http_te_skip && !k->ignorebody) {
- if(!conn->data->set.http_ce_skip && k->writer_stack)
- result = Curl_unencode_write(conn, k->writer_stack, datap, piece);
- else
- result = Curl_client_write(conn, CLIENTWRITE_BODY, datap, piece);
- if(result)
- return CHUNKE_WRITE_ERROR;
- }
- *wrote += piece;
- ch->datasize -= piece;
- datap += piece;
- length -= piece;
- if(0 == ch->datasize)
-
- ch->state = CHUNK_POSTLF;
- break;
- case CHUNK_POSTLF:
- if(*datap == 0x0a) {
-
- Curl_httpchunk_init(conn);
- }
- else if(*datap != 0x0d)
- return CHUNKE_BAD_CHUNK;
- datap++;
- length--;
- break;
- case CHUNK_TRAILER:
- if((*datap == 0x0d) || (*datap == 0x0a)) {
-
- if(conn->trlPos) {
-
- conn->trailer[conn->trlPos++] = 0x0d;
- conn->trailer[conn->trlPos++] = 0x0a;
- conn->trailer[conn->trlPos] = 0;
-
- result = Curl_convert_from_network(conn->data, conn->trailer,
- conn->trlPos);
- if(result)
-
-
- return CHUNKE_BAD_CHUNK;
- if(!data->set.http_te_skip) {
- result = Curl_client_write(conn, CLIENTWRITE_HEADER,
- conn->trailer, conn->trlPos);
- if(result)
- return CHUNKE_WRITE_ERROR;
- }
- conn->trlPos = 0;
- ch->state = CHUNK_TRAILER_CR;
- if(*datap == 0x0a)
-
- break;
- }
- else {
-
- ch->state = CHUNK_TRAILER_POSTCR;
- break;
- }
- }
- else {
-
- if(conn->trlPos >= conn->trlMax) {
-
- char *ptr;
- if(conn->trlMax) {
- conn->trlMax *= 2;
- ptr = realloc(conn->trailer, conn->trlMax + 3);
- }
- else {
- conn->trlMax = 128;
- ptr = malloc(conn->trlMax + 3);
- }
- if(!ptr)
- return CHUNKE_OUT_OF_MEMORY;
- conn->trailer = ptr;
- }
- conn->trailer[conn->trlPos++]=*datap;
- }
- datap++;
- length--;
- break;
- case CHUNK_TRAILER_CR:
- if(*datap == 0x0a) {
- ch->state = CHUNK_TRAILER_POSTCR;
- datap++;
- length--;
- }
- else
- return CHUNKE_BAD_CHUNK;
- break;
- case CHUNK_TRAILER_POSTCR:
-
- if((*datap != 0x0d) && (*datap != 0x0a)) {
-
- ch->state = CHUNK_TRAILER;
- break;
- }
- if(*datap == 0x0d) {
-
- datap++;
- length--;
- }
-
- ch->state = CHUNK_STOP;
- break;
- case CHUNK_STOP:
- if(*datap == 0x0a) {
- length--;
-
- ch->dataleft = curlx_sotouz(length);
- return CHUNKE_STOP;
- }
- else
- return CHUNKE_BAD_CHUNK;
- }
- }
- return CHUNKE_OK;
- }
- const char *Curl_chunked_strerror(CHUNKcode code)
- {
- switch(code) {
- default:
- return "OK";
- case CHUNKE_TOO_LONG_HEX:
- return "Too long hexadecimal number";
- case CHUNKE_ILLEGAL_HEX:
- return "Illegal or missing hexadecimal sequence";
- case CHUNKE_BAD_CHUNK:
- return "Malformed encoding found";
- case CHUNKE_WRITE_ERROR:
- return "Write error";
- case CHUNKE_BAD_ENCODING:
- return "Bad content-encoding found";
- case CHUNKE_OUT_OF_MEMORY:
- return "Out of memory";
- }
- }
- #endif
|