/*
   Copyright (C) 2009, 2011 Her Majesty the Queen in Right of Canada
   (Communications Research Center Canada)
   */
/*
   This file is part of ODR-DabMux.

   ODR-DabMux is free software: you can redistribute it and/or modify
   it under the terms of the GNU General Public License as
   published by the Free Software Foundation, either version 3 of the
   License, or (at your option) any later version.

   ODR-DabMux is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.

   You should have received a copy of the GNU General Public License
   along with ODR-DabMux.  If not, see <http://www.gnu.org/licenses/>.
   */

#include "dabInputPacketFile.h"
#include "dabInputFile.h"
#include "dabInputFifo.h"
#include "ReedSolomon.h"

#ifdef HAVE_FORMAT_PACKET
#   ifdef HAVE_INPUT_FILE


#include <string.h>
#include <stdio.h>
#include <errno.h>


#ifdef _WIN32
#   pragma pack(push, 1)
#endif
struct packetHeader {
    unsigned char addressHigh:2;
    unsigned char last:1;
    unsigned char first:1;
    unsigned char continuityIndex:2;
    unsigned char packetLength:2;
    unsigned char addressLow;
    unsigned char dataLength:7;
    unsigned char command;
}
#ifdef _WIN32
#   pragma pack(pop)
#else
__attribute((packed))
#endif
;


struct dabInputOperations dabInputPacketFileOperations = {
    dabInputFileInit,
    dabInputFileOpen,
    dabInputSetbuf,
    dabInputFileRead,
    nullptr,
    nullptr,
    dabInputPacketFileRead,
    dabInputSetbitrate,
    dabInputFileClose,
    dabInputFileClean,
    dabInputFileRewind
};


int dabInputPacketFileRead(dabInputOperations* ops, void* args, void* buffer,
        int size)
{
    dabInputFileData* data = (dabInputFileData*)args;
    unsigned char* dataBuffer = (unsigned char*)buffer;
    int written = 0;
    int length;
    packetHeader* header;
    int indexRow;
    int indexCol;

    while (written < size) {
        if (data->enhancedPacketWaiting > 0) {
            *dataBuffer = 192 - data->enhancedPacketWaiting;
            *dataBuffer /= 22;
            *dataBuffer <<= 2;
            *(dataBuffer++) |= 0x03;
            *(dataBuffer++) = 0xfe;
            indexCol = 188;
            indexCol += (192 - data->enhancedPacketWaiting) / 12;
            indexRow = 0;
            indexRow += (192 - data->enhancedPacketWaiting) % 12;
            for (int j = 0; j < 22; ++j) {
                if (data->enhancedPacketWaiting == 0) {
                    *(dataBuffer++) = 0;
                } else {
                    *(dataBuffer++) = data->enhancedPacketData[indexRow][indexCol];
                    if (++indexRow == 12) {
                        indexRow = 0;
                        ++indexCol;
                    }
                    --data->enhancedPacketWaiting;
                }
            }
            written += 24;
            if (data->enhancedPacketWaiting == 0) {
                data->enhancedPacketLength = 0;
            }
        } else if (data->packetLength != 0) {
            header = (packetHeader*)data->packetData;
            if (written + data->packetLength > (unsigned)size) {
                memset(dataBuffer, 0, 22);
                dataBuffer[22] = 0x60;
                dataBuffer[23] = 0x4b;
                length = 24;
            } else if (data->enhancedPacketData != nullptr) {
                if (data->enhancedPacketLength + data->packetLength
                        > (12 * 188)) {
                    memset(dataBuffer, 0, 22);
                    dataBuffer[22] = 0x60;
                    dataBuffer[23] = 0x4b;
                    length = 24;
                } else {
                    memcpy(dataBuffer, data->packetData, data->packetLength);
                    length = data->packetLength;
                    data->packetLength = 0;
                }
            } else {
                memcpy(dataBuffer, data->packetData, data->packetLength);
                length = data->packetLength;
                data->packetLength = 0;
            }
            if (data->enhancedPacketData != nullptr) {
                indexCol = data->enhancedPacketLength / 12;
                indexRow = data->enhancedPacketLength % 12;     // TODO Check if always 0
                for (int j = 0; j < length; ++j) {
                    data->enhancedPacketData[indexRow][indexCol] = dataBuffer[j];
                    if (++indexRow == 12) {
                        indexRow = 0;
                        ++indexCol;
                    }
                }
                data->enhancedPacketLength += length;
                if (data->enhancedPacketLength >= (12 * 188)) {
                    data->enhancedPacketLength = (12 * 188);
                    ReedSolomon encoder(204, 188);
                    for (int j = 0; j < 12; ++j) {
                        encoder.encode(data->enhancedPacketData[j], 188);
                    }
                    data->enhancedPacketWaiting = 192;
                }
            }
            written += length;
            dataBuffer += length;
        } else {
            int nbBytes = ops->read(args, dataBuffer, 3);
            header = (packetHeader*)dataBuffer;
            if (nbBytes == -1) {
                if (errno == EAGAIN) goto END_PACKET;
                perror("Packet file");
                return -1;
            } else if (nbBytes == 0) {
                if (ops->rewind(args) == -1) {
                    goto END_PACKET;
                }
                continue;
            } else if (nbBytes < 3) {
                etiLog.log(error,
                        "Error while reading file for packet header; "
                        "read %i out of 3 bytes\n", nbBytes);
                break;
            }

            length = header->packetLength * 24 + 24;
            if (written + length > size) {
                memcpy(data->packetData, header, 3);
                ops->read(args, &data->packetData[3], length - 3);
                data->packetLength = length;
                continue;
            }
            if (data->enhancedPacketData != nullptr) {
                if (data->enhancedPacketLength + length > (12 * 188)) {
                    memcpy(data->packetData, header, 3);
                    ops->read(args, &data->packetData[3], length - 3);
                    data->packetLength = length;
                    continue;
                }
            }
            nbBytes = ops->read(args, dataBuffer + 3, length - 3);
            if (nbBytes == -1) {
                perror("Packet file");
                return -1;
            } else if (nbBytes == 0) {
                etiLog.log(info,
                        "Packet header read, but no data!\n");
                if (ops->rewind(args) == -1) {
                    goto END_PACKET;
                }
                continue;
            } else if (nbBytes < length - 3) {
                etiLog.log(error, "Error while reading packet file; "
                        "read %i out of %i bytes\n", nbBytes, length - 3);
                break;
            }
            if (data->enhancedPacketData != nullptr) {
                indexCol = data->enhancedPacketLength / 12;
                indexRow = data->enhancedPacketLength % 12;     // TODO Check if always 0
                for (int j = 0; j < length; ++j) {
                    data->enhancedPacketData[indexRow][indexCol] = dataBuffer[j];
                    if (++indexRow == 12) {
                        indexRow = 0;
                        ++indexCol;
                    }
                }
                data->enhancedPacketLength += length;
                if (data->enhancedPacketLength >= (12 * 188)) {
                    if (data->enhancedPacketLength > (12 * 188)) {
                        etiLog.log(error,
                                "Error, too much enhanced packet data!\n");
                    }
                    ReedSolomon encoder(204, 188);
                    for (int j = 0; j < 12; ++j) {
                        encoder.encode(data->enhancedPacketData[j], 188);
                    }
                    data->enhancedPacketWaiting = 192;
                }
            }
            written += length;
            dataBuffer += length;
        }
    }
END_PACKET:
    if (ops->read == dabInputFifoRead) {
        dabInputFifoData* fifoData = (dabInputFifoData*)args;
        dabInputFifoStats* fifoStats = (dabInputFifoStats*)&fifoData->stats;
        fifoStats->frameRecords[fifoStats->frameCount].curSize = written;
        fifoStats->frameRecords[fifoStats->frameCount].maxSize = size;
        if (++fifoStats->frameCount == NB_RECORDS) {
            etiLog.log(info, "Packet subchannel usage: (%i)",
                    fifoStats->id);
            for (int i = 0; i < fifoStats->frameCount; ++i) {
            etiLog.log(info, " %i/%i",
                    fifoStats->frameRecords[i].curSize,
                    fifoStats->frameRecords[i].maxSize);
            }
            etiLog.log(info, "\n");
            fifoStats->frameCount = 0;
        }
    }
    while (written < size) {
        memset(dataBuffer, 0, 22);
        dataBuffer[22] = 0x60;
        dataBuffer[23] = 0x4b;
        dataBuffer += 24;
        written += 24;
    }
    return written;
}


#   endif
#endif