/* Copyright 2013-2021 MultiMC Contributors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#include <fstream>
#include <string>

#include "PrintInstanceInfo.h"
#include <launch/LaunchTask.h>

#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
namespace {
#if defined(Q_OS_LINUX)
void probeProcCpuinfo(QStringList &log)
{
    std::ifstream cpuin("/proc/cpuinfo");
    for (std::string line; std::getline(cpuin, line);)
    {
        if (strncmp(line.c_str(), "model name", 10) == 0)
        {
            log << QString::fromStdString(line.substr(13, std::string::npos));
            break;
        }
    }
}

void runLspci(QStringList &log)
{
    // FIXME: fixed size buffers...
    char buff[512];
    int gpuline = -1;
    int cline = 0;
    FILE * lspci = popen("lspci -k", "r");

    if (!lspci)
        return;

    while (fgets(buff, 512, lspci) != NULL)
    {
        std::string str(buff);
        if (str.length() < 9)
            continue;
        if (str.substr(8, 3) == "VGA")
        {
            gpuline = cline;
            log << QString::fromStdString(str.substr(35, std::string::npos));
        }
        if (gpuline > -1 && gpuline != cline)
        {
            if (cline - gpuline < 3)
            {
                log << QString::fromStdString(str.substr(1, std::string::npos));
            }
        }
        cline++;
    }
    pclose(lspci);
}
#elif defined(Q_OS_FREEBSD)
void runSysctlHwModel(QStringList &log)
{
    char buff[512];
    FILE *hwmodel = popen("sysctl hw.model", "r");
    while (fgets(buff, 512, hwmodel) != NULL)
    {
	log << QString::fromUtf8(buff);
	break;
    }
    pclose(hwmodel);
}

void runPciconf(QStringList &log)
{
    char buff[512];
    std::string strcard;
    FILE *pciconf = popen("pciconf -lv -a vgapci0", "r");
    while (fgets(buff, 512, pciconf) != NULL)
    {
	if (strncmp(buff, "    vendor", 10) == 0)
	{
	    std::string str(buff);
	    strcard.append(str.substr(str.find_first_of("'") + 1, str.find_last_not_of("'") - (str.find_first_of("'") + 2)));
	    strcard.append(" ");
	}
	else if (strncmp(buff, "    device", 10) == 0)
	{
	    std::string str2(buff);
	    strcard.append(str2.substr(str2.find_first_of("'") + 1, str2.find_last_not_of("'") - (str2.find_first_of("'") + 2)));
	}
	log << QString::fromStdString(strcard);
	break;
    }
    pclose(pciconf);
}
#endif
void runGlxinfo(QStringList & log)
{
    // FIXME: fixed size buffers...
    char buff[512];
    FILE *glxinfo = popen("glxinfo", "r");
    if (!glxinfo)
        return;

    while (fgets(buff, 512, glxinfo) != NULL)
    {
        if (strncmp(buff, "OpenGL version string:", 22) == 0)
        {
            log << QString::fromUtf8(buff);
            break;
        }
    }
    pclose(glxinfo);
}

}
#endif

void PrintInstanceInfo::executeTask()
{
    auto instance = m_parent->instance();
    QStringList log;

#if defined(Q_OS_LINUX)
    ::probeProcCpuinfo(log);
    ::runLspci(log);
    ::runGlxinfo(log);
#elif defined(Q_OS_FREEBSD)
    ::runSysctlHwModel(log);
    ::runPciconf(log);
    ::runGlxinfo(log);
#endif

    logLines(log, MessageLevel::Launcher);
    logLines(instance->verboseDescription(m_session, m_serverToJoin), MessageLevel::Launcher);
    emitSucceeded();
}