#include <stdlib.h>
#include <stdint.h>
#include <getopt.h>
#include <stdio.h>
#include <string.h>
#include <inttypes.h>
#include <sys/stat.h>
#include <errno.h>

#define VTXDIS_VER "0.2"

#define SWAP16(x) (((x & 0xFF00) >> 8) | ((x & 0x00FF) << 8))

typedef struct {
    int16_t pos[3];     /* 0x00 */
    int16_t flag;       /* 0x06 */
    int16_t tpos[2];    /* 0x08 */
    uint8_t cn[4];      /* 0x0C */
} Vtx;                  /* 0x10 */

static char *filename = NULL;
static char *data = NULL;
static int32_t offset = 0;
static int32_t data_len = 0;
static int16_t count = 0;

static const struct option cmdline_opts[] = {
    { "offset", required_argument, NULL, 'o', },
    { "length", required_argument, NULL, 'l', },
    { "file" , required_argument, NULL, 'f', },
    { "version", no_argument, NULL, '~', },
    { "help", no_argument, NULL, '?', },
    { "count", required_argument, NULL, 'c', },
    { 0, 0, 0, 0 },
};

static uint32_t parse_int(const char *num){
    uint32_t ret;
    char outnum[21];

    if(strlen(num) > 2 && num[0] == '0' && (num[1] == 'x' || num[1] == 'X')) {
        strncpy(outnum, &num[2], 20);
        sscanf(outnum, "%"SCNx32, &ret);
    } else if(strlen(num) == 0){
        ret =  -1;
    } else {
        strncpy(outnum, num, 20);
        sscanf(outnum, "%"SCNu32, &ret);
    }
    return ret;

}

static inline void print_usage(void)
{
    puts("vtxdis version " VTXDIS_VER "\n"
    "Usage:\n"
    "  vtxdis -f/--file FILE [options]\n"
    "  vtxdis -?/--help\n"
    "  vtxdis --version\n"
    "Options:\n"
    "  -f, --file       The file path to extract vertex data from.\n"
    "  -c, --count      The number of vertices to extract.\n"
    "  -l, --length     The amount of data to extract vertices from.\n"
    "  -o, --offset     The offset into file to start reading vertex data.\n"
    "  -?, --help       Prints this help message\n"
    "  --version        Prints the current version\n"
    );
}

static inline void print_version(void){
    puts("Version: " VTXDIS_VER);
}

static void print_vtx_data(Vtx *vtx, int vtx_cnt)
{
    puts("{");
    for(int i = 0; i < vtx_cnt; i++)
    {
        Vtx *v = &vtx[i];
        printf("    VTX(%d, %d, %d, %d, %d, %d, %d, %d, %d),\n", v->pos[0], v->pos[1], v->pos[2], v->tpos[0], v->tpos[1], v->cn[0], v->cn[1], v->cn[2], v->cn[3]);

    }
    puts("}");
}

static void parse_file(void)
{
    unsigned int alloc_size = 0;
    struct stat sbuffer;
    stat(filename, &sbuffer);
    if(errno != 0){
        perror("Count not stat file.");
        exit(1);
    }

    /* sanity checks */
    if(count > 0)
    {
        alloc_size = sizeof(Vtx) * count;
        if((offset > 0 && (offset + alloc_size) > sbuffer.st_size) || alloc_size > sbuffer.st_size)
        {
            puts("Requested data is beyond file boundaries.");
            exit(1);
        }
    }
    else if(data_len > 0)
    {
        alloc_size = data_len;

        if((offset > 0 && (offset + alloc_size) > sbuffer.st_size) || alloc_size > sbuffer.st_size)
        {
            puts("Requested data is beyond file boundaries.");
            exit(1);
        }
    }
    else
    {
        if (offset > 0)
        {
            alloc_size = sbuffer.st_size - offset;
        }
        else
        {
            alloc_size = sbuffer.st_size;
        }
    }

    if(alloc_size % sizeof(Vtx) != 0)
    {
        printf("Requested data size is not a multiple of sizeof(Vtx).  Requested size is %8x", alloc_size);
        exit(1);
    }

    FILE *file = fopen(filename, "rb");
    if(!file){
        perror("Could not open file");
        exit(1);
    }

    if(offset > 0){
        if(fseek(file, offset, SEEK_SET)){
            perror("Could not seek file");
            fclose(file);
            exit(1);
        }
    }

    Vtx *data = malloc(alloc_size);
    if(!data){
        fclose(file);
        perror("Could not allocate vtx data");
        exit(1);
    }

    if(!fread(data, alloc_size, 1, file)){
        perror("Could not read from file");
        fclose(file);
        free(data);
        exit(1);
    }

    fclose(file);

    unsigned int vtx_cnt = alloc_size / sizeof(Vtx);
    for(unsigned int i = 0; i < vtx_cnt; i++){
        Vtx *v = &data[i];

        v->pos[0] = SWAP16(v->pos[0]);
        v->pos[1] = SWAP16(v->pos[1]);
        v->pos[2] = SWAP16(v->pos[2]);
        v->flag = SWAP16(v->flag);
        v->tpos[0] = SWAP16(v->tpos[0]);
        v->tpos[1] = SWAP16(v->tpos[1]);
    }

    print_vtx_data(data, vtx_cnt);
    free(data);
}

int main(int argc, char **argv)
{
    int opt;

    while(1){
        opt = getopt_long(argc, argv, "o:l:f:c:v?", cmdline_opts, NULL);
        if(opt == -1){
            break;
        }
        switch(opt){
            case 'd':
                data = optarg;
                break;
            case '~':
                print_version();
                return 0;
            case '?':
                print_usage();
                return 0;
            case 'l':
                data_len = parse_int(optarg);
                break;
            case 'o':
                offset = parse_int(optarg);
                break;
            case 'f':
                filename = optarg;
                break;
            case 'c':
                count = parse_int(optarg);
                break;
        }
    }

    if (filename == NULL && data == NULL)
    {
        puts("Must specify -f or -d");
        print_usage();
        exit(1);
    }

    if(data_len < 0)
    {
        puts("Invalid -l/--length parameter passed.");
        print_usage();
        exit(1);
    }

    if(offset < 0)
    {
        puts("Invalid -o/--offset parameter passed.");
        print_usage();
        exit(1);
    }

    if(count < 0)
    {
        puts("Invalid -c/--count parameter passed.");
        print_usage();
        exit(1);
    }

    if(count > 0 && data_len > 0)
    {
        puts("Cannot specify both -c/--count and -l/--length.");
        print_usage();
        exit(1);
    }

    if(filename != NULL)
    {
        parse_file();
    }

    return 0;
}