2021-04-09 15:06:55 -04:00
|
|
|
local base = string.gsub(@@LUA_SCRIPT_FILENAME@@, "(.*[/\\])(.*)", "%1")
|
|
|
|
|
2021-04-28 04:05:16 -04:00
|
|
|
local util = dofile(base.."/util.lua")
|
2021-04-23 19:04:59 -04:00
|
|
|
local config = dofile(base.."/config.lua")
|
2021-04-23 15:39:11 -04:00
|
|
|
local serpent = dofile(base.."/serpent.lua")
|
2021-04-28 04:05:16 -04:00
|
|
|
local temps = {
|
|
|
|
os.getenv("TMPDIR"),
|
|
|
|
os.getenv("TEMP"),
|
|
|
|
os.getenv("TEMPDIR"),
|
|
|
|
os.getenv("TMP"),
|
|
|
|
}
|
|
|
|
|
|
|
|
local tempDir = "/tmp"
|
|
|
|
for i=1,#temps,1 do
|
|
|
|
local temp = temps[i]
|
|
|
|
if temp ~= nil and temp ~= "" then
|
|
|
|
tempDir = temps[i]
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-04-28 18:54:39 -04:00
|
|
|
local tmpFileName = tempDir.."/donk_runner_"..
|
|
|
|
string.hex(math.floor(random.integer(0, 0xffffffff)))..
|
|
|
|
string.hex(math.floor(random.integer(0, 0xffffffff)))
|
2021-04-09 15:06:55 -04:00
|
|
|
|
|
|
|
local function message(_M, msg, color)
|
|
|
|
if color == nil then
|
|
|
|
color = 0x00009900
|
|
|
|
end
|
|
|
|
|
|
|
|
for i=#_M.onMessageHandler,1,-1 do
|
|
|
|
_M.onMessageHandler[i](msg, color)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-04-09 17:47:08 -04:00
|
|
|
local function save(_M, filename)
|
2021-04-09 15:06:55 -04:00
|
|
|
for i=#_M.onSaveHandler,1,-1 do
|
2021-04-09 17:47:08 -04:00
|
|
|
_M.onSaveHandler[i](filename)
|
2021-04-09 15:06:55 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function onSave(_M, handler)
|
|
|
|
table.insert(_M.onSaveHandler, handler)
|
|
|
|
end
|
|
|
|
|
2021-04-09 17:47:08 -04:00
|
|
|
local function load(_M, filename)
|
2021-04-09 15:06:55 -04:00
|
|
|
for i=#_M.onLoadHandler,1,-1 do
|
2021-04-09 17:47:08 -04:00
|
|
|
_M.onLoadHandler[i](filename)
|
2021-04-09 15:06:55 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function onLoad(_M, handler)
|
|
|
|
table.insert(_M.onLoadHandler, handler)
|
|
|
|
end
|
|
|
|
|
2021-04-09 17:47:08 -04:00
|
|
|
local function onMessage(_M, handler)
|
|
|
|
table.insert(_M.onMessageHandler, handler)
|
|
|
|
end
|
|
|
|
|
2021-04-09 15:06:55 -04:00
|
|
|
return function()
|
|
|
|
local _M = {
|
|
|
|
onMessageHandler = {},
|
2021-04-09 17:47:08 -04:00
|
|
|
onSaveHandler = {},
|
|
|
|
onLoadHandler = {},
|
2021-04-28 18:54:39 -04:00
|
|
|
poppets = {},
|
2021-04-09 15:06:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
_M.onRenderForm = function(handler)
|
|
|
|
end
|
|
|
|
|
|
|
|
_M.onMessage = function(handler)
|
|
|
|
onMessage(_M, handler)
|
|
|
|
end
|
|
|
|
|
2021-04-09 17:47:08 -04:00
|
|
|
_M.message = function(msg, color)
|
|
|
|
message(_M, msg, color)
|
|
|
|
end
|
|
|
|
|
|
|
|
_M.onSave = function(handler)
|
|
|
|
onSave(_M, handler)
|
|
|
|
end
|
|
|
|
|
|
|
|
_M.onLoad = function(handler)
|
|
|
|
onLoad(_M, handler)
|
|
|
|
end
|
|
|
|
|
2021-04-24 01:18:06 -04:00
|
|
|
_M.run = function(species, generationIdx, genomeCallback, finishCallback)
|
2021-04-28 18:54:39 -04:00
|
|
|
local hostProcess = "lsnes"
|
|
|
|
if util.isWin then
|
|
|
|
hostProcess = util.scrapeCmd('*l', 'powershell "(Get-WmiObject Win32_Process -Filter ProcessId=$((Get-WmiObject Win32_Process -Filter ProcessId=$((Get-WmiObject Win32_Process -Filter ProcessId=$PID).ParentProcessId)).ParentProcessId)").ExecutablePath')
|
|
|
|
if hostProcess == nil or hostProcess == "" then
|
|
|
|
hostProcess = "lsnes-bsnes.exe"
|
|
|
|
end
|
|
|
|
else
|
|
|
|
-- FIXME Linux
|
|
|
|
end
|
2021-04-23 18:09:35 -04:00
|
|
|
|
2021-04-28 23:18:26 -04:00
|
|
|
local inputPrefix = tmpFileName..'_input_'
|
|
|
|
local outputPrefix = tmpFileName..'_output_'
|
|
|
|
|
|
|
|
-- Create the input files and output files
|
|
|
|
for i=1,#species,1 do
|
|
|
|
local inputFileName = inputPrefix..i
|
|
|
|
local inputFile = io.open(inputFileName, 'a')
|
|
|
|
inputFile:close()
|
|
|
|
|
|
|
|
local outputFileName = outputPrefix..i
|
|
|
|
local outputFile = io.open(outputFileName, 'a')
|
|
|
|
outputFile:close()
|
|
|
|
end
|
|
|
|
|
2021-04-28 18:54:39 -04:00
|
|
|
while #_M.poppets < #species do
|
|
|
|
local i = #_M.poppets+1
|
2021-04-28 23:18:26 -04:00
|
|
|
local outputFileName = outputPrefix..i
|
|
|
|
local inputFileName = inputPrefix..i
|
2021-04-28 18:54:39 -04:00
|
|
|
|
|
|
|
message(_M, hostProcess)
|
|
|
|
|
2021-04-29 04:56:55 -04:00
|
|
|
local settingsDir = nil
|
|
|
|
if isWin then
|
|
|
|
settingsDir = tempDir.."/donk_runner_settings_"..i
|
|
|
|
util.mkdir(settingsDir)
|
|
|
|
end
|
|
|
|
|
2021-04-28 04:05:16 -04:00
|
|
|
local envs = {
|
2021-04-28 18:54:39 -04:00
|
|
|
RUNNER_INPUT_FILE = inputFileName,
|
|
|
|
RUNNER_OUTPUT_FILE = outputFileName,
|
2021-04-29 04:56:55 -04:00
|
|
|
APPDATA = settingsDir,
|
2021-04-28 04:05:16 -04:00
|
|
|
}
|
|
|
|
|
2021-04-28 18:54:39 -04:00
|
|
|
local waiter = util.waitForChange(outputFileName)
|
|
|
|
|
|
|
|
local cmd = '"'..hostProcess..'" "--rom='..config.ROM..'" --unpause "--lua='..base..'/runner-process.lua"'
|
|
|
|
local poppet = util.popenCmd(cmd, nil, envs)
|
|
|
|
table.insert(_M.poppets, poppet)
|
|
|
|
|
|
|
|
waiter:read("*a")
|
|
|
|
util.closeCmd(waiter)
|
2021-04-09 15:06:55 -04:00
|
|
|
end
|
|
|
|
|
2021-04-28 23:18:26 -04:00
|
|
|
local waiters = {}
|
|
|
|
for i=1,#species,1 do
|
|
|
|
table.insert(waiters, outputPrefix..i)
|
|
|
|
end
|
|
|
|
|
2021-04-29 04:56:55 -04:00
|
|
|
local waiter = util.waitForChange(waiters, nil, tmpFileName.."_output_*")
|
2021-04-28 18:54:39 -04:00
|
|
|
|
|
|
|
message(_M, 'Setting up child processes')
|
|
|
|
|
|
|
|
for i=1,#species,1 do
|
2021-04-28 23:18:26 -04:00
|
|
|
|
2021-04-28 18:54:39 -04:00
|
|
|
local inputFileName = tmpFileName.."_input_"..i
|
|
|
|
local inputFile = io.open(inputFileName, 'w')
|
|
|
|
inputFile:write(serpent.dump({species[i], generationIdx}))
|
|
|
|
inputFile:close()
|
2021-04-09 15:06:55 -04:00
|
|
|
end
|
2021-04-28 18:54:39 -04:00
|
|
|
|
|
|
|
message(_M, 'Waiting for child processes to finish')
|
|
|
|
|
|
|
|
waiter:read("*a")
|
|
|
|
util.closeCmd(waiter)
|
|
|
|
|
|
|
|
message(_M, 'Child processes finished')
|
|
|
|
|
2021-04-23 18:09:35 -04:00
|
|
|
for i=1,#species,1 do
|
2021-04-28 18:54:39 -04:00
|
|
|
message(_M, "Processing output "..i)
|
2021-04-23 18:09:35 -04:00
|
|
|
local outputFileName = tmpFileName..'_output_'..i
|
|
|
|
local outputFile = io.open(outputFileName, "r")
|
|
|
|
local line = ""
|
|
|
|
repeat
|
2021-04-28 04:05:16 -04:00
|
|
|
local ok, obj = serpent.load(line)
|
|
|
|
if not ok then
|
2021-04-23 18:09:35 -04:00
|
|
|
goto continue
|
|
|
|
end
|
|
|
|
|
|
|
|
if obj == nil then
|
|
|
|
goto continue
|
|
|
|
end
|
|
|
|
|
|
|
|
if obj.type == 'onMessage' then
|
|
|
|
message(_M, obj.msg, obj.color)
|
|
|
|
elseif obj.type == 'onLoad' then
|
|
|
|
load(_M, obj.filename)
|
|
|
|
elseif obj.type == 'onSave' then
|
|
|
|
save(_M, obj.filename)
|
|
|
|
elseif obj.type == 'onGenome' then
|
2021-04-24 01:18:06 -04:00
|
|
|
for i=1,#species,1 do
|
|
|
|
local s = species[i]
|
|
|
|
if s.id == obj.speciesId then
|
|
|
|
s.genomes[obj.genomeIndex] = obj.genome
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
2021-04-23 18:09:35 -04:00
|
|
|
genomeCallback(obj.genome, obj.index)
|
|
|
|
elseif obj.type == 'onFinish' then
|
|
|
|
finishCallback()
|
|
|
|
end
|
|
|
|
|
|
|
|
::continue::
|
|
|
|
line = outputFile:read()
|
|
|
|
until(line == "" or line == nil)
|
|
|
|
outputFile:close()
|
2021-04-09 15:06:55 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return _M
|
|
|
|
end
|