#!/usr/bin/lua5.3
--[[
 * =======================================================================================
 *
 *      Filename:  likwid-topology.lua
 *
 *      Description:  A application to determine the thread and cache topology
 *                    on x86, ARM and POWER processors. Also the Nvidia GPU topology
 *                    is printed.
 *
 *      Version:   5.5.1
 *      Released:  23.12.2025
 *
 *      Author:   Thomas Gruber (tr), thomas.roehl@gmail.com
 *      Project:  likwid
 *
 *      Copyright (C) 2016 RRZE, University Erlangen-Nuremberg
 *
 *      This program 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.
 *
 *      This program 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
 *      this program.  If not, see <http://www.gnu.org/licenses/>.
 *
 * =======================================================================================
]]
package.path = '/usr/share/lua/?.lua;' .. package.path

local likwid = require("likwid")

print_stdout = print
print_stderr = function(...) for k,v in pairs({...}) do io.stderr:write(v .. "\n") end end

libinfo = likwid.getLibInfo()

function version()
    io.stdout:write(string.format("likwid-topology -- Version %d.%d.%d (commit: %s)\n",
                libinfo["majorVersion"], libinfo["minorVersion"],
                libinfo["bugfixVersion"], likwid.commit))
end

function usage()
    version()
    print()
    print("A tool to print the thread and cache topology on CPUs and GPUs.")
    print()
    print("Options:")
    print("  -h, --help            Help message")
    print("  -v, --version         Version information")
    print("  -V, --verbose <level> Set verbosity")
    print("  -c, --caches          List cache information")
    print("  -C, --clock           Measure processor clock")
    if libinfo["nvidiaSupport"] or libinfo["rocmSupport"] then
        print("  -G, --gpus            List GPU information")
    end
    print("  -O                    CSV output")
    print("  -o, --output <file>   Store output to file. (Optional: Apply text filter)")
    print("  -g                    Graphical output")
end

print_caches = false
print_gpus = false
print_graphical = false
measure_clock = false
outfile = nil
output_csv = {}

for opt,arg in likwid.getopt(arg, {"h","v","c","C","g","o:","V:", "G","O","help","version","verbose:","clock","caches","output:", "gpus"}) do
    if (type(arg) == "string") then
        local s,e = arg:find("-");
        if s == 1 then
            print_stderr(string.format("Argument %s to option -%s starts with invalid character -.", arg, opt))
            print_stderr("Did you forget an argument to an option?")
            os.exit(1)
        end
    end
    if opt == "h" or opt == "help" then
        usage()
        os.exit(0)
    elseif opt == "v" or opt == "version" then
        version()
        os.exit(0)
    elseif opt == "V" or opt == "verbose" then
        if tonumber(arg) >= 0 and tonumber(arg) <=3 then
            likwid.setVerbosity(tonumber(arg))
        else
            print_stderr("Verbosity level not valid. Must be between 0 (only errors) and 3 (developer output)")
        end
    elseif opt == "c" or opt == "caches" then
        print_caches = true
    elseif opt == "C" or opt == "clock" then
        measure_clock = true
    elseif opt == "g" then
        print_graphical = true
    elseif opt == "G" or opt == "gpus" then
        print_gpus = true
    elseif opt == "O" then
        print_csv = true
    elseif opt == "o" or opt == "output" then
        local suffix = ""
        if string.match(arg, "%.") then
            suffix = string.match(arg, ".-[^\\/]-%.?([^%.\\/]*)$")
        end
        if suffix ~= "txt" then
            print_csv = true
        end
        outfile = arg:gsub("%%h", likwid.gethostname())
        io.output(arg..".tmp")
        print = function(...) for k,v in pairs({...}) do io.write(v .. "\n") end end
    elseif opt == "?" then
        print_stderr("Invalid commandline option -"..arg)
        os.exit(1)
    elseif opt == "!" then
        print_stderr("Option requires an argument")
        os.exit(1)
    end
end

likwid.setenv("LIKWID_NO_ACCESS", "1")
local config = likwid.getConfiguration()
local cpuinfo = likwid.getCpuInfo()
local cputopo = likwid.getCpuTopology()
local numainfo = likwid.getNumaInfo()
local affinity = likwid.getAffinityInfo()


table.insert(output_csv, likwid.hline)
local lines = 3
if measure_clock then
    lines = 4
end
table.insert(output_csv, "STRUCT,Info,"..tostring(lines))
table.insert(output_csv, string.format("CPU name:\t%s",cpuinfo["osname"]))
table.insert(output_csv, string.format("CPU type:\t%s",cpuinfo["name"]))
table.insert(output_csv, string.format("CPU stepping:\t%s",cpuinfo["stepping"]))
if (measure_clock) then
    if cpuinfo["clock"] == 0 then
        table.insert(output_csv, string.format("CPU clock:\t%3.2f GHz", likwid.getCpuClock() * 1.E-09))
    else
        table.insert(output_csv, string.format("CPU clock:\t%3.2f GHz", cpuinfo["clock"] * 1.E-09))
    end
end

table.insert(output_csv, likwid.sline)
table.insert(output_csv, "STRUCT,Hardware Thread Topology,3")
table.insert(output_csv, "Hardware Thread Topology")
table.insert(output_csv, likwid.sline)
table.insert(output_csv, string.format("Sockets:\t\t%u",cputopo["numSockets"]))
table.insert(output_csv, string.format("CPU dies:\t\t%u",cputopo["numDies"]))
table.insert(output_csv, string.format("Cores per socket:\t%u",cputopo["numCoresPerSocket"]))
table.insert(output_csv, string.format("Threads per core:\t%u",cputopo["numThreadsPerCore"]))
table.insert(output_csv, likwid.hline)
table.insert(output_csv, "TABLE,Topology,"..tostring(cputopo["numHWThreads"]))
header = {"HWThread", "Thread", "Core", "Die", "Socket", "Available"}
for cntr=0,cputopo["numHWThreads"]-1 do
    local line = {tostring(cntr),
                  tostring(cputopo["threadPool"][cntr]["threadId"]),
                  tostring(cputopo["threadPool"][cntr]["coreId"]),
                  tostring(cputopo["threadPool"][cntr]["dieId"]),
                  tostring(cputopo["threadPool"][cntr]["packageId"])}
    if cputopo["threadPool"][cntr]["inCpuSet"] == 1 then
        table.insert(line, "*")
    else
        table.insert(line, "")
    end
    if print_csv then
        if cntr == 0 then
            table.insert(output_csv, table.concat(header, ","))
        end
        table.insert(output_csv, table.concat(line, ","))
    else
        table.insert(output_csv, likwid.printTextTable(header, line, cntr == 0))
    end
end
table.insert(output_csv, likwid.hline)

table.insert(output_csv, "STRUCT,Sockets,"..tostring(cputopo["numSockets"]))
for socket=0, #cputopo["topologyTree"] do
    csv_str = string.format("Socket %d:\t\t( ",cputopo["topologyTree"][socket]["ID"])
    for core = 0, #cputopo["topologyTree"][socket]["Children"] do
        for thread = 0, #cputopo["topologyTree"][socket]["Children"][core]["Children"] do
            csv_str = csv_str ..tostring(cputopo["topologyTree"][socket]["Children"][core]["Children"][thread]).. ","
        end
    end
    table.insert(output_csv, csv_str:sub(1,#csv_str-1).." )")
end

table.insert(output_csv, likwid.hline)


table.insert(output_csv, likwid.sline)
table.insert(output_csv, "Cache Topology")
table.insert(output_csv, likwid.sline)

for level=1,cputopo["numCacheLevels"] do
    if (cputopo["cacheLevels"][level]["type"] ~= "INSTRUCTIONCACHE") then
        lines = 3
        if print_caches then lines = 9 end
        table.insert(output_csv, string.format("STRUCT,Cache Topology L%d,%d", cputopo["cacheLevels"][level]["level"],lines))
        table.insert(output_csv, string.format("Level:\t\t\t%d",cputopo["cacheLevels"][level]["level"]))
        if (cputopo["cacheLevels"][level]["size"] < 1048576) then
            table.insert(output_csv, string.format("Size:\t\t\t%.0f kB",cputopo["cacheLevels"][level]["size"]/1024))
        else
            local x = cputopo["cacheLevels"][level]["size"]/1048576
            if tonumber(string.format("%.2f", x)) == tonumber(string.format("%.0f", x)) then
                table.insert(output_csv, string.format("Size:\t\t\t%.0f MB", x))
            else
                table.insert(output_csv, string.format("Size:\t\t\t%.2f MB", x))
            end
        end

        if (print_caches) then
            if (cputopo["cacheLevels"][level]["type"] == "DATACACHE") then
                table.insert(output_csv, "Type:\t\t\tData cache")
            elseif (cputopo["cacheLevels"][level]["type"] == "UNIFIEDCACHE") then
                table.insert(output_csv, "Type:\t\t\tUnified cache")
            end

            table.insert(output_csv, string.format("Associativity:\t\t%d",cputopo["cacheLevels"][level]["associativity"]))
            table.insert(output_csv, string.format("Number of sets:\t\t%d",cputopo["cacheLevels"][level]["sets"]))
            table.insert(output_csv, string.format("Cache line size:\t%d",cputopo["cacheLevels"][level]["lineSize"]))

            if (cputopo["cacheLevels"][level]["inclusive"] == 0) then
                table.insert(output_csv, "Cache type:\t\tNon Inclusive")
            else
                table.insert(output_csv, "Cache type:\t\tInclusive")
            end
            table.insert(output_csv, string.format("Shared by threads:\t%d",cputopo["cacheLevels"][level]["threads"]))
        end
        local threads = cputopo["cacheLevels"][level]["threads"]
        str = "Cache groups:\t\t( "
        for socket=0, #cputopo["topologyTree"] do
            for core = 0, #cputopo["topologyTree"][socket]["Children"] do
                for cpu = 0, #cputopo["topologyTree"][socket]["Children"][core]["Children"] do
                    if (threads ~= 0) then
                        str = str .. cputopo["topologyTree"][socket]["Children"][core]["Children"][cpu] .. " "
                        threads = threads - 1
                    else
                        str = str .. string.format(") ( %d ",cputopo["topologyTree"][socket]["Children"][core]["Children"][cpu])
                        threads = cputopo["cacheLevels"][level]["threads"]
                        threads = threads - 1
                    end
                end
            end
        end
        str = str .. ")"
        table.insert(output_csv, str)
        table.insert(output_csv, likwid.hline)
    end
end


table.insert(output_csv, likwid.sline)
table.insert(output_csv, "NUMA Topology")
table.insert(output_csv, likwid.sline)

if (numainfo["numberOfNodes"] == 0) then
    table.insert(output_csv, "No NUMA")
else
    table.insert(output_csv, string.format("NUMA domains:\t\t%d",numainfo["numberOfNodes"]))
    table.insert(output_csv, likwid.hline)
    for node=1,numainfo["numberOfNodes"] do
        table.insert(output_csv, string.format("STRUCT,NUMA Topology %d,5",numainfo["nodes"][node]["id"]))
        table.insert(output_csv, string.format("Domain:\t\t\t%d",numainfo["nodes"][node]["id"]))
        csv_str = "Processors:\t\t( "
        for cpu=1,numainfo["nodes"][node]["numberOfProcessors"] do
            csv_str = csv_str .. numainfo["nodes"][node]["processors"][cpu] .. ","
        end
        table.insert(output_csv, csv_str:sub(1,#csv_str-1).. " )")
        csv_str = "Distances:\t\t"
        for cpu=1,numainfo["nodes"][node]["numberOfDistances"] do
            csv_str = csv_str .. numainfo["nodes"][node]["distances"][cpu][cpu-1] .. ","
        end
        table.insert(output_csv, csv_str:sub(1,#csv_str-1))
        table.insert(output_csv, string.format("Free memory:\t\t%g MB",tonumber(numainfo["nodes"][node]["freeMemory"]/1024.0)))
        table.insert(output_csv, string.format("Total memory:\t\t%g MB",tonumber(numainfo["nodes"][node]["totalMemory"]/1024.0)))
        table.insert(output_csv, likwid.hline)
    end
end

if libinfo["nvidiaSupport"] then
    local cudatopo = likwid.getCudaTopology()
    if cudatopo then
        table.insert(output_csv, likwid.sline)
        table.insert(output_csv, "GPU Topology")
        table.insert(output_csv, likwid.sline)
        table.insert(output_csv, string.format("GPU count:\t\t%d", cudatopo["numDevices"]))
        table.insert(output_csv, likwid.hline)

        for i=1, cudatopo["numDevices"] do
            gpu = cudatopo["devices"][i]
            table.insert(output_csv, string.format("STRUCT,GPU Topology %d,9", gpu["id"]))
            table.insert(output_csv, string.format("ID:\t\t\t%d", gpu["id"]))
            table.insert(output_csv, string.format("Name:\t\t\t%s", gpu["name"]))
            table.insert(output_csv, string.format("Compute capability:\t%d.%d", gpu["ccapMajor"], gpu["ccapMinor"]))
            table.insert(output_csv, string.format("L2 size:\t\t%.2f MB", gpu["l2Size"]/(1024*1024)))
            table.insert(output_csv, string.format("Memory:\t\t\t%.2f GB", gpu["memory"]/(1024*1024*1024)))
            table.insert(output_csv, string.format("SIMD width:\t\t%d", gpu["simdWidth"]))
            table.insert(output_csv, string.format("Clock rate:\t\t%d kHz", gpu["clockRatekHz"]))
            table.insert(output_csv, string.format("Memory clock rate:\t%d kHz", gpu["memClockRatekHz"]))
            table.insert(output_csv, string.format("Attached to NUMA node:\t%d", gpu["numaNode"]))
            if print_gpus then
                table.insert(output_csv, string.format("Number of SPs:\t\t%d", gpu["numMultiProcs"]))
                table.insert(output_csv, string.format("Max. threads per SP:\t%d", gpu["maxThreadPerMultiProc"]))
                table.insert(output_csv, string.format("Max. threads per block:\t%d", gpu["maxThreadsPerBlock"]))
                local s = {}
                for i, data in pairs(gpu["maxThreadsDim"]) do
                    table.insert(s, string.format("%d", data))
                end
                table.insert(output_csv, string.format("Max. thread dimensions:\t%s", table.concat(s, "/")))
                table.insert(output_csv, string.format("Max. regs per block:\t%d", gpu["regsPerBlock"]))
                table.insert(output_csv, string.format("Shared mem per block:\t%d", gpu["sharedMemPerBlock"]))

                table.insert(output_csv, string.format("Memory bus width:\t%d", gpu["memBusWidth"]))
                table.insert(output_csv, string.format("Texture alignment:\t%d", gpu["textureAlign"]))
                table.insert(output_csv, string.format("Surface alignment:\t%d", gpu["surfaceAlign"]))
                if gpu["ecc"] == 0 then
                    table.insert(output_csv, "ECC:\t\t\toff")
                else
                    table.insert(output_csv, "ECC:\t\t\ton")
                end
                if gpu["integrated"] == 0 then
                    table.insert(output_csv, "GPU integrated:\t\tno")
                else
                    table.insert(output_csv, "GPU integrated:\t\tyes")
                end
                s = {}
                for i, data in pairs(gpu["maxGridSize"]) do
                    table.insert(s, string.format("%d", data))
                end
                table.insert(output_csv, string.format("Max. grid sizes:\t%s", table.concat(s, "/")))
                table.insert(output_csv, string.format("PCI bus:\t\t0x%x", gpu["pciBus"]))
                table.insert(output_csv, string.format("PCI domain:\t\t0x%x", gpu["pciDom"]))
                table.insert(output_csv, string.format("PCI device:\t\t0x%x", gpu["pciDev"]))
                table.insert(output_csv, string.format("PCI function:\t\t0x%x", gpu["pciFunc"]))
            end
            table.insert(output_csv, likwid.hline)
        end
    end
    likwid.putCudaTopology()
end

if libinfo["rocmSupport"] then
    local rocmtopo = likwid.getRocmTopology()
    if rocmtopo then
        table.insert(output_csv, likwid.sline)
        table.insert(output_csv, "ROCm GPU Topology")
        table.insert(output_csv, likwid.sline)
        table.insert(output_csv, string.format("GPU count:\t\t%d", rocmtopo["numDevices"]))
        table.insert(output_csv, likwid.hline)

        for i=1, rocmtopo["numDevices"] do
            gpu = rocmtopo["devices"][i]
            table.insert(output_csv, string.format("STRUCT,GPU Topology %d,9", gpu["id"]))
            table.insert(output_csv, string.format("ID:\t\t\t%d", gpu["id"]))
            table.insert(output_csv, string.format("Name:\t\t\t%s", gpu["name"]))
            table.insert(output_csv, string.format("Compute capability:\t%d.%d", gpu["ccapMajor"], gpu["ccapMinor"]))
            table.insert(output_csv, string.format("L2 size:\t\t%.2f MB", gpu["l2Size"]/(1024*1024)))
            table.insert(output_csv, string.format("Memory:\t\t\t%.2f GB", gpu["memory"]/(1024*1024*1024)))
            table.insert(output_csv, string.format("Clock rate:\t\t%d kHz", gpu["clockRatekHz"]))
            table.insert(output_csv, string.format("Memory clock rate:\t%d kHz", gpu["memClockRatekHz"]))
            table.insert(output_csv, string.format("Attached to NUMA node:\t%d", gpu["numaNode"]))
            if print_gpus then
                table.insert(output_csv, string.format("Number of SPs:\t\t%d", gpu["numMultiProcs"]))
                table.insert(output_csv, string.format("Max. threads per SP:\t%d", gpu["maxThreadPerMultiProc"]))
                table.insert(output_csv, string.format("Max. threads per block:\t%d", gpu["maxThreadsPerBlock"]))
                local s = {}
                for i, data in pairs(gpu["maxThreadsDim"]) do
                    table.insert(s, string.format("%d", data))
                end
                table.insert(output_csv, string.format("Max. thread dimensions:\t%s", table.concat(s, "/")))
                table.insert(output_csv, string.format("Max. regs per block:\t%d", gpu["regsPerBlock"]))
                table.insert(output_csv, string.format("Shared mem per block:\t%d", gpu["sharedMemPerBlock"]))

                table.insert(output_csv, string.format("Memory bus width:\t%d", gpu["memBusWidth"]))
                table.insert(output_csv, string.format("Texture alignment:\t%d", gpu["textureAlign"]))
                if gpu["ecc"] == 0 then
                    table.insert(output_csv, "ECC:\t\t\toff")
                else
                    table.insert(output_csv, "ECC:\t\t\ton")
                end
                if gpu["integrated"] == 0 then
                    table.insert(output_csv, "GPU integrated:\t\tno")
                else
                    table.insert(output_csv, "GPU integrated:\t\tyes")
                end
                s = {}
                for i, data in pairs(gpu["maxGridSize"]) do
                    table.insert(s, string.format("%d", data))
                end
                table.insert(output_csv, string.format("Max. grid sizes:\t%s", table.concat(s, "/")))
                table.insert(output_csv, string.format("PCI bus:\t\t0x%x", gpu["pciBus"]))
                table.insert(output_csv, string.format("PCI domain:\t\t0x%x", gpu["pciDom"]))
                table.insert(output_csv, string.format("PCI device:\t\t0x%x", gpu["pciDev"]))
            end
            table.insert(output_csv, likwid.hline)
        end
    end
    likwid.putRocmTopology()
end

if print_csv then
    longest_line = 0
    local tmpList = {}
    for i=#output_csv,1,-1 do
        output_csv[i] = output_csv[i]:gsub("[\t]+",",")
        output_csv[i] = output_csv[i]:gsub("%( ","")
        output_csv[i] = output_csv[i]:gsub(" %)[%s]*",",")
        output_csv[i] = output_csv[i]:gsub(",$","")
        if  output_csv[i]:sub(1,1) == "*" or
            output_csv[i]:sub(1,1) == "-" or
            output_csv[i]:match("^Hardware Thread Topology") or
            output_csv[i]:match("^Cache Topology") or
            output_csv[i]:match("^NUMA Topology") then
            table.remove(output_csv,i)
        end
        tmpList = likwid.stringsplit(output_csv[i],",")
        if #tmpList > longest_line then longest_line = #tmpList end
    end
    for i=1,#output_csv do
        tmpList = likwid.stringsplit(output_csv[i],",")
        if #tmpList < longest_line then
            output_csv[i] = output_csv[i]..string.rep(",",longest_line-#tmpList)
        end
    end
else
    for i=#output_csv,1,-1 do
        output_csv[i] = output_csv[i]:gsub(","," ")
        if output_csv[i]:match("^TABLE") or
           output_csv[i]:match("^STRUCT") then
            table.remove(output_csv,i)
        end
    end
end

for _,line in pairs(output_csv) do print(line) end

if print_graphical and not print_csv then
    print_stdout("\n")
    print_stdout(likwid.sline)
    print_stdout("Graphical Topology")
    print_stdout(likwid.sline)
    for socket=0, #cputopo["topologyTree"] do
        print_stdout(string.format("Socket %d:",cputopo["topologyTree"][socket]["ID"]))
        container = {}
        for core = 0, #cputopo["topologyTree"][socket]["Children"] do
            local tmpString = ""
            for thread = 0, #cputopo["topologyTree"][socket]["Children"][core]["Children"] do
                if thread == 0 then
                    tmpString = tmpString .. tostring(cputopo["topologyTree"][socket]["Children"][core]["Children"][thread])
                else
                    tmpString = tmpString .. " " .. tostring(cputopo["topologyTree"][socket]["Children"][core]["Children"][thread]).. " "
                end
            end
            likwid.addSimpleAsciiBox(container, 1, core+1, tmpString)
        end

        local columnCursor = 1
        local lineCursor = 2
        for cache=1,cputopo["numCacheLevels"] do
            if cputopo["cacheLevels"][cache]["type"] ~= "INSTRUCTIONCACHE" then
                local cachesAtCurLevel = 0
                local sharedCores = cputopo["cacheLevels"][cache]["threads"]/cputopo["numThreadsPerCore"]
                if sharedCores >= cputopo["numCoresPerSocket"] then
                    cachesAtCurLevel = 1
                else
                    cachesAtCurLevel = cputopo["numCoresPerSocket"]/sharedCores
                end
                columnCursor = 1
                for cachesAtLevel=1,cachesAtCurLevel do
                    local tmpString = ""
                    local cacheWidth = 0
                    if cputopo["cacheLevels"][cache]["size"] < 1048576 then
                        tmpString = string.format("%.0f kB", cputopo["cacheLevels"][cache]["size"]/1024)
                    else
                        tmpString = string.format("%.0f MB", cputopo["cacheLevels"][cache]["size"]/1048576)
                    end
                    if sharedCores > 1 then
                        if sharedCores > cputopo["numCoresPerSocket"] then
                            cacheWidth = sharedCores
                        else
                            cacheWidth = sharedCores - 1
                        end
                        likwid.addJoinedAsciiBox(container, lineCursor, columnCursor,columnCursor + cacheWidth, tmpString)
                        columnCursor = columnCursor + cacheWidth
                    else
                        likwid.addSimpleAsciiBox(container, lineCursor, columnCursor, tmpString)
                        columnCursor = columnCursor + 1
                    end
                end
                lineCursor = lineCursor + 1
            end
        end
        likwid.printAsciiBox(container);
    end
end

if outfile then
    local suffix = ""
    if string.match(outfile, ".-[^\\/]-%.?([^%.\\/]*)$") then
        suffix = string.match(outfile, ".-[^\\/]-%.?([^%.\\/]*)$")
    end
    local command = "/usr/share/likwid/filter/" .. suffix
    local tmpfile = outfile..".tmp"
    if suffix == "" then
        os.rename(tmpfile, outfile)
    elseif suffix ~= "txt" and suffix ~= "csv" and not likwid.access(command,"x") then
        print_stderr("Cannot find filter script, save output in CSV format to file "..outfile)
        os.rename(tmpfile, outfile)
    else
        if suffix ~= "txt" and suffix ~= "csv" then
            command = command .." ".. tmpfile .. " topology"
            local f = assert(io.popen(command), "r")
            if f ~= nil then
                local o = f:read("*a")
                if o:len() > 0 then
                    print_stderr(string.format("Failed to executed filter script %s.",command))
                else
                    os.rename(outfile.."."..suffix, outfile)
                    if not likwid.access(tmpfile, "e") then
                        os.remove(tmpfile)
                    end
                end
            else
                print_stderr("Failed to call filter script, save output in CSV format to file "..outfile)
                os.rename(tmpfile, outfile)
                os.remove(tmpfile)
            end
        else
            os.rename(tmpfile, outfile)
            os.remove(tmpfile)
        end
    end
end

likwid.putAffinityInfo()
likwid.putNumaInfo()
likwid.putTopology()
likwid.putConfiguration()
likwid.unsetenv("LIKWID_NO_ACCESS")
os.exit(0)
