// Copyright(c) 2016 YamaArashi
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cctype>
#include <cassert>
#include <string>
#include <set>
#include "main.h"
#include "error.h"
#include "midi.h"
#include "agb.h"

FILE* g_inputFile = nullptr;
FILE* g_outputFile = nullptr;

std::string g_asmLabel;
int g_masterVolume = 127;
int g_voiceGroup = 0;
int g_priority = 0;
int g_reverb = -1;
int g_clocksPerBeat = 1;
bool g_exactGateTime = false;
bool g_compressionEnabled = true;

[[noreturn]] static void PrintUsage()
{
    std::printf(
        "Usage: MID2AGB name [options]\n"
        "\n"
        "    input_file  filename(.mid) of MIDI file\n"
        "   output_file  filename(.s) for AGB file (default:input_file)\n"
        "\n"
        "options  -L???  label for assembler (default:output_file)\n"
        "         -V???  master volume (default:127)\n"
        "         -G???  voice group number (default:0)\n"
        "         -P???  priority (default:0)\n"
        "         -R???  reverb (default:off)\n"
        "            -X  48 clocks/beat (default:24 clocks/beat)\n"
        "            -E  exact gate-time\n"
        "            -N  no compression\n"
    );
    std::exit(1);
}

static std::string StripExtension(std::string s)
{
    std::size_t pos = s.find_last_of('.');

    if (pos > 0 && pos != std::string::npos)
    {
        s = s.substr(0, pos);
    }

    return s;
}

static std::string GetExtension(std::string s)
{
    std::size_t pos = s.find_last_of('.');

    if (pos > 0 && pos != std::string::npos)
    {
        return s.substr(pos + 1);
    }

    return "";
}

static std::string BaseName(std::string s)
{
    std::size_t posAfterSlash = s.find_last_of("/\\");

    if (posAfterSlash == std::string::npos)
        posAfterSlash = 0;
    else
        posAfterSlash++;

    std::size_t dotPos = s.find_first_of('.', posAfterSlash);
    if (dotPos > posAfterSlash && dotPos != std::string::npos)
        s = s.substr(posAfterSlash, dotPos - posAfterSlash);

    return s;
}

static const char *GetArgument(int argc, char **argv, int& index)
{
    assert(index >= 0 && index < argc);

    const char *option = argv[index];

    assert(option != nullptr);
    assert(option[0] == '-');

    // If there is text following the letter, return that.
    if (std::strlen(option) >= 3)
        return option + 2;

    // Otherwise, try to get the next arg.
    if (index + 1 < argc)
    {
        index++;
        return argv[index];
    }
    else
    {
        return nullptr;
    }
}

int main(int argc, char** argv)
{
    std::string inputFilename;
    std::string outputFilename;

    for (int i = 1; i < argc; i++)
    {
        const char *option = argv[i];

        if (option[0] == '-' && option[1] != '\0')
        {
            const char *arg;

            switch (std::toupper(option[1]))
            {
            case 'E':
                g_exactGateTime = true;
                break;
            case 'G':
                arg = GetArgument(argc, argv, i);
                if (arg == nullptr)
                    PrintUsage();
                g_voiceGroup = std::stoi(arg);
                break;
            case 'L':
                arg = GetArgument(argc, argv, i);
                if (arg == nullptr)
                    PrintUsage();
                g_asmLabel = arg;
                break;
            case 'N':
                g_compressionEnabled = false;
                break;
            case 'P':
                arg = GetArgument(argc, argv, i);
                if (arg == nullptr)
                    PrintUsage();
                g_priority = std::stoi(arg);
                break;
            case 'R':
                arg = GetArgument(argc, argv, i);
                if (arg == nullptr)
                    PrintUsage();
                g_reverb = std::stoi(arg);
                break;
            case 'V':
                arg = GetArgument(argc, argv, i);
                if (arg == nullptr)
                    PrintUsage();
                g_masterVolume = std::stoi(arg);
                break;
            case 'X':
                g_clocksPerBeat = 2;
                break;
            default:
                PrintUsage();
            }
        }
        else
        {
            if (inputFilename.empty())
                inputFilename = argv[i];
            else if (outputFilename.empty())
                outputFilename = argv[i];
            else
                PrintUsage();
        }
    }

    if (inputFilename.empty())
        PrintUsage();

    if (GetExtension(inputFilename) != "mid")
        RaiseError("input filename extension is not \"mid\"");

    if (outputFilename.empty())
        outputFilename = StripExtension(inputFilename) + ".s";

    if (GetExtension(outputFilename) != "s")
        RaiseError("output filename extension is not \"s\"");

    if (g_asmLabel.empty())
        g_asmLabel = BaseName(outputFilename);

    g_inputFile = std::fopen(inputFilename.c_str(), "rb");

    if (g_inputFile == nullptr)
        RaiseError("failed to open \"%s\" for reading", inputFilename.c_str());

    g_outputFile = std::fopen(outputFilename.c_str(), "w");

    if (g_outputFile == nullptr)
        RaiseError("failed to open \"%s\" for writing", outputFilename.c_str());

    ReadMidiFileHeader();
    PrintAgbHeader();
    ReadMidiTracks();
    PrintAgbFooter();

    std::fclose(g_inputFile);
    std::fclose(g_outputFile);

    return 0;
}