/* Copyright (c) 2017 - 2021 LiteSpeed Technologies Inc.  See LICENSE. */
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#ifndef WIN32
#include <sys/time.h>
#endif
#include <sys/queue.h>

#include "lsquic_types.h"
#include "lsquic.h"
#include "lsquic_int_types.h"
#include "lsquic_packet_common.h"
#include "lsquic_packet_out.h"
#include "lsquic_hash.h"
#include "lsquic_conn.h"
#include "lsquic_parse.h"

struct test {
    /* Inputs. */
    const struct parse_funcs
                   *pf;
    size_t          bufsz;
    uint64_t        cid;    /* Zero means connection ID is not specified */
    const char     *nonce;
    lsquic_packno_t packno;
    enum packno_bits
                    bits;   /* The test has been retrofitted by adding bits parameter.  The test can
                             * be made more complicated by calculating packet number length based on
                             * some other inputs.  However, this is tested elsewhere.
                             */
    union {
        unsigned char   buf[4];
        lsquic_ver_tag_t    val;
    }               ver;

    /* Outputs */
    int             len;            /* Retval */
    char            out[0x100];     /* Contents */
};


static const struct test tests[] = {

    {
        .pf     = select_pf_by_ver(LSQVER_043),
        .bufsz  = GQUIC_MAX_PUBHDR_SZ,
        .cid    = 0x0102030405060708UL,
        .nonce  = NULL,
        .packno = 0x01020304,
        .bits   = GQUIC_PACKNO_LEN_4,
        .len    = 1 + 8 + 0 + 4,
        .out    = {     (0 << 2)                                        /* Nonce present */
                      | 0x08                                            /* Connection ID present */
                      | 0x20                                            /* Packet number length */
                      ,
                      0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01,   /* Connection ID */
                      0x01, 0x02, 0x03, 0x04,                           /* Packet number */
        },
    },

    {
        .pf     = select_pf_by_ver(LSQVER_043),
        .bufsz  = GQUIC_MAX_PUBHDR_SZ,
        .cid    = 0x0102030405060708UL,
        .nonce  = NULL,
        .packno = 0x00,
        .bits   = GQUIC_PACKNO_LEN_1,
        .len    = 1 + 8 + 0 + 1,
        .out    = {     (0 << 2)                                        /* Nonce present */
                      | 0x08                                            /* Connection ID present */
                      | 0x00                                            /* Packet number length */
                      ,
                      0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01,   /* Connection ID */
                      0x00,                                             /* Packet number */
        },
    },

    {
        .pf     = select_pf_by_ver(LSQVER_043),
        .bufsz  = GQUIC_MAX_PUBHDR_SZ,
        .cid    = 0x0102030405060708UL,
        .nonce  = NULL,
        .packno = 0x09,
        .bits   = GQUIC_PACKNO_LEN_1,
        .ver.buf= { 'Q', '0', '4', '3', },
        .len    = 1 + 8 + 4 + 0 + 1,
        .out    = {     (0 << 2)                                        /* Nonce present */
                      | 0x01                                            /* Version present */
                      | 0x08                                            /* Connection ID present */
                      | 0x00                                            /* Packet number length */
                      ,
                      0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01,   /* Connection ID */
                      'Q', '0', '4', '3',
                      0x09,                                             /* Packet number */
        },
    },

#define NONCENSE "0123456789abcdefghijklmnopqrstuv"
#define NONCENSE_BYTES '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v'

    {
        .pf     = select_pf_by_ver(LSQVER_043),
        .bufsz  = GQUIC_MAX_PUBHDR_SZ,
        .cid    = 0x0102030405060708UL,
        .nonce  = NONCENSE,
        .packno = 0x00,
        .bits   = GQUIC_PACKNO_LEN_1,
        .len    = 1 + 8 + 32 + 1,
        .out    = {     (1 << 2)                                        /* Nonce present */
                      | 0x08                                            /* Connection ID present */
                      | 0x00                                            /* Packet number length */
                      ,
                      0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01,   /* Connection ID */
                      NONCENSE_BYTES,
                      0x00,                                             /* Packet number */
        },
    },

    {
        .pf     = select_pf_by_ver(LSQVER_043),
        .bufsz  = GQUIC_MAX_PUBHDR_SZ,
        .cid    = 0,    /* Do not set connection ID */
        .nonce  = NONCENSE,
        .packno = 0x00,
        .bits   = GQUIC_PACKNO_LEN_1,
        .len    = 1 + 0 + 32 + 1,
        .out    = {     (1 << 2)                                        /* Nonce present */
                      | 0x00                                            /* Packet number length */
                      ,
                      NONCENSE_BYTES,
                      0x00,                                             /* Packet number */
        },
    },

    {
        .pf     = select_pf_by_ver(LSQVER_043),
        .bufsz  = GQUIC_MAX_PUBHDR_SZ,
        .cid    = 0x0102030405060708UL,
        .nonce  = NONCENSE,
        .packno = 0x00,
        .bits   = GQUIC_PACKNO_LEN_1,
        .ver.buf= { 'Q', '0', '4', '3', },
        .len    = 1 + 8 + 4 + 32 + 1,
        .out    = {     (1 << 2)                                        /* Nonce present */
                      | 0x01                                            /* Version present */
                      | 0x08                                            /* Connection ID present */
                      | 0x00                                            /* Packet number length */
                      ,
                      0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01,   /* Connection ID */
                      'Q', '0', '4', '3',
                      NONCENSE_BYTES,
                      0x00,                                             /* Packet number */
        },
    },

    {
        .pf     = select_pf_by_ver(LSQVER_043),
        .bufsz  = GQUIC_MAX_PUBHDR_SZ,
        .cid    = 0x0102030405060708UL,
        .nonce  = NONCENSE,
        .packno = 0xA0A1A2A3A4A5A6A7UL,
        .bits   = GQUIC_PACKNO_LEN_6,
        .len    = 1 + 8 + 32 + 6,
        .out    = {     (1 << 2)                                        /* Nonce present */
                      | 0x08                                            /* Connection ID present */
                      | 0x30                                            /* Packet number length */
                      ,
                      0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01,   /* Connection ID */
                      NONCENSE_BYTES,
                      0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7,
        },
    },

    {
        .pf     = select_pf_by_ver(LSQVER_043),
        .bufsz  = GQUIC_MAX_PUBHDR_SZ,
        .cid    = 0x0102030405060708UL,
        .nonce  = NONCENSE,
        .packno = 0xA0A1A2A3A4A5A6A7UL,
        .bits   = GQUIC_PACKNO_LEN_6,
        .len    = 1 + 8 + 32 + 6,
        .out    = {     (1 << 2)                                        /* Nonce present */
                      | 0x08                                            /* Connection ID present */
                      | 0x30                                            /* Packet number length */
                      ,
                      0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01,   /* Connection ID */
                      NONCENSE_BYTES,
                      0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7,
        },
    },

};


static void
run_test (int i)
{
    const struct test *const test = &tests[i];

    struct lsquic_packet_out packet_out =
    {
        .po_flags = (test->cid ? PO_CONN_ID : 0)
                  | (test->ver.val ? PO_VERSION : 0)
                  | (test->nonce ? PO_NONCE: 0)
                  ,
        .po_nonce = (unsigned char *) test->nonce,
        .po_ver_tag = test->ver.val,
        .po_packno = test->packno,
    };
    lsquic_packet_out_set_packno_bits(&packet_out, test->bits);

    lsquic_cid_t cid;
    memset(&cid, 0, sizeof(cid));
    cid.len = sizeof(test->cid);
    memcpy(cid.idbuf, &test->cid, sizeof(test->cid));

    struct lsquic_conn lconn = LSCONN_INITIALIZER_CID(lconn, cid);

    unsigned char out[GQUIC_MAX_PUBHDR_SZ];
    int len = test->pf->pf_gen_reg_pkt_header(&lconn, &packet_out, out,
                                                    sizeof(out), NULL, NULL);

    assert(("Packet length is correct", len == test->len));

    if (test->len > 0)
        assert(("Packet contents are correct",
            0 == memcmp(out, test->out, len)));
}


int
main (void)
{
    unsigned i;
    for (i = 0; i < sizeof(tests) / sizeof(tests[0]); ++i)
        run_test(i);
    return 0;
}