/*
 * Copyright (C) Igor Sysoev
 * Copyright (C) NGINX, Inc.
 */

#include <nxt_main.h>


#define NXT_FASTCGI_DATA_MIDDLE         0
#define NXT_FASTCGI_DATA_END_ON_BORDER  1
#define NXT_FASTCGI_DATA_END            2


static nxt_int_t nxt_fastcgi_buffer(nxt_fastcgi_parse_t *fp, nxt_buf_t ***tail,
    nxt_buf_t *in);


void
nxt_fastcgi_record_parse(nxt_task_t *task, nxt_fastcgi_parse_t *fp,
    nxt_buf_t *in)
{
    u_char        ch;
    nxt_int_t     ret, stream;
    nxt_buf_t     *b, *nb, **tail[2];
    const char    *msg;
    enum {
        sw_fastcgi_version = 0,
        sw_fastcgi_type,
        sw_fastcgi_request_id_high,
        sw_fastcgi_request_id_low,
        sw_fastcgi_content_length_high,
        sw_fastcgi_content_length_low,
        sw_fastcgi_padding_length,
        sw_fastcgi_reserved,
        sw_fastcgi_data,
        sw_fastcgi_padding,
        sw_fastcgi_end_request,
    } state;

    fp->out[0] = NULL;
    fp->out[1] = NULL;

    tail[0] = &fp->out[0];
    tail[1] = &fp->out[1];

    state = fp->state;

    for (b = in; b != NULL; b = b->next) {

        if (nxt_buf_is_sync(b)) {
            **tail = b;
            *tail = &b->next;
            continue;
        }

        fp->pos = b->mem.pos;

        while (fp->pos < b->mem.free) {
            /*
             * The sw_fastcgi_data state is tested outside the
             * switch to preserve fp->pos and to not touch memory.
             */
            if (state == sw_fastcgi_data) {

                /*
                 * fp->type here can be only NXT_FASTCGI_STDOUT
                 * or NXT_FASTCGI_STDERR.  NXT_FASTCGI_END_REQUEST
                 * is tested in sw_fastcgi_reserved.
                 */
                stream = fp->type - NXT_FASTCGI_STDOUT;

                ret = nxt_fastcgi_buffer(fp, &tail[stream], b);

                if (ret == NXT_FASTCGI_DATA_MIDDLE) {
                    goto next;
                }

                if (nxt_slow_path(ret == NXT_ERROR)) {
                    fp->error = 1;
                    goto done;
                }

                if (fp->padding == 0) {
                    state = sw_fastcgi_version;

                } else {
                    state = sw_fastcgi_padding;
                }

                if (ret == NXT_FASTCGI_DATA_END_ON_BORDER) {
                    goto next;
                }

                /* ret == NXT_FASTCGI_DATA_END */
            }

            ch = *fp->pos++;

            nxt_thread_log_debug("fastcgi record byte: %02Xd", ch);

            switch (state) {

            case sw_fastcgi_version:
                if (nxt_fast_path(ch == 1)) {
                    state = sw_fastcgi_type;
                    continue;
                }

                msg = "unsupported FastCGI protocol version";
                goto fastcgi_error;

            case sw_fastcgi_type:
                switch (ch) {
                case NXT_FASTCGI_STDOUT:
                case NXT_FASTCGI_STDERR:
                case NXT_FASTCGI_END_REQUEST:
                    fp->type = ch;
                    state = sw_fastcgi_request_id_high;
                    continue;
                default:
                    msg = "invalid FastCGI record type";
                    goto fastcgi_error;
                }

            case sw_fastcgi_request_id_high:
                /* FastCGI multiplexing is not supported. */
                if (nxt_fast_path(ch == 0)) {
                    state = sw_fastcgi_request_id_low;
                    continue;
                }

                msg = "unexpected FastCGI request ID high byte";
                goto fastcgi_error;

            case sw_fastcgi_request_id_low:
                if (nxt_fast_path(ch == 1)) {
                    state = sw_fastcgi_content_length_high;
                    continue;
                }

                msg = "unexpected FastCGI request ID low byte";
                goto fastcgi_error;

            case sw_fastcgi_content_length_high:
                fp->length = ch << 8;
                state = sw_fastcgi_content_length_low;
                continue;

            case sw_fastcgi_content_length_low:
                fp->length |= ch;
                state = sw_fastcgi_padding_length;
                continue;

            case sw_fastcgi_padding_length:
                fp->padding = ch;
                state = sw_fastcgi_reserved;
                continue;

            case sw_fastcgi_reserved:
                nxt_thread_log_debug("fastcgi record type:%d "
                                     "length:%uz padding:%d",
                                     fp->type, fp->length, fp->padding);

                if (nxt_fast_path(fp->type != NXT_FASTCGI_END_REQUEST)) {
                    state = sw_fastcgi_data;
                    continue;
                }

                state = sw_fastcgi_end_request;
                continue;

            case sw_fastcgi_data:
                /*
                 * This state is processed before the switch.
                 * It added here just to suppress a warning.
                 */
                continue;

            case sw_fastcgi_padding:
                /*
                 * No special fast processing of padding
                 * because it usually takes just 1-7 bytes.
                 */
                fp->padding--;

                if (fp->padding == 0) {
                    nxt_thread_log_debug("fastcgi record end");
                    state = sw_fastcgi_version;
                }
                continue;

            case sw_fastcgi_end_request:
                /* Just skip 8 bytes of END_REQUEST. */
                fp->length--;

                if (fp->length != 0) {
                    continue;
                }

                fp->done = 1;

                nxt_thread_log_debug("fastcgi end request");

                goto done;
            }
        }

        if (b->retain == 0) {
            /* No record data was found in a buffer. */
            nxt_thread_current_work_queue_add(task->thread,
                                              b->completion_handler,
                                              task, b, b->parent);
        }

    next:

        continue;
    }

    fp->state = state;

    return;

fastcgi_error:

    nxt_thread_log_error(NXT_LOG_ERR, "upstream sent %s: %d", msg, ch);

    fp->fastcgi_error = 1;

done:

    nb = fp->last_buf(fp);

    if (nxt_fast_path(nb != NULL)) {
        *tail[0] = nb;

    } else {
        fp->error = 1;
    }

    // STUB: fp->fastcgi_error = 1;
    // STUB: fp->error = 1;

    return;
}


static nxt_int_t
nxt_fastcgi_buffer(nxt_fastcgi_parse_t *fp, nxt_buf_t ***tail, nxt_buf_t *in)
{
    u_char     *p;
    size_t     size;
    nxt_buf_t  *b;

    if (fp->length == 0) {
        return NXT_FASTCGI_DATA_END;
    }

    p = fp->pos;
    size = in->mem.free - p;

    if (fp->length >= size && in->retain == 0) {
        /*
         * Use original buffer if the buffer is lesser than or equal to
         * FastCGI record size and this is the first record in the buffer.
         */
        in->mem.pos = p;
        **tail = in;
        *tail = &in->next;

    } else {
        b = nxt_buf_mem_alloc(fp->mem_pool, 0, 0);
        if (nxt_slow_path(b == NULL)) {
            return NXT_ERROR;
        }

        **tail = b;
        *tail = &b->next;

        b->parent = in;
        in->retain++;
        b->mem.pos = p;
        b->mem.start = p;

        if (fp->length < size) {
            p += fp->length;
            fp->pos = p;

            b->mem.free = p;
            b->mem.end = p;

            return NXT_FASTCGI_DATA_END;
        }

        b->mem.free = in->mem.free;
        b->mem.end = in->mem.free;
    }

    fp->length -= size;

    if (fp->length == 0) {
        return NXT_FASTCGI_DATA_END_ON_BORDER;
    }

    return NXT_FASTCGI_DATA_MIDDLE;
}