Advertisement
SethBling

NearestEvolve.lua

Mar 20th, 2015
2,826
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 12.40 KB | None | 0 0
  1. if gameinfo.getromname() == "Super Mario World (USA)" then
  2. filename = "DP1.state"
  3. buttonNames = {
  4. "A",
  5. "B",
  6. "X",
  7. "Y",
  8. "Up",
  9. "Down",
  10. "Left",
  11. "Right",
  12. }
  13. elseif gameinfo.getromname() == "Super Mario Bros." then
  14. filename = "SMB1-1.state"
  15. buttonNames = {
  16. "A",
  17. "B",
  18. "Up",
  19. "Down",
  20. "Left",
  21. "Right",
  22. }
  23. end
  24.  
  25. boxRadius = 6
  26. inputSize = (boxRadius*2+1)*(boxRadius*2+1)
  27. snapshotSize = inputSize + #buttonNames
  28.  
  29. function getPositions()
  30. if gameinfo.getromname() == "Super Mario World (USA)" then
  31. marioX = memory.read_s16_le(0x94)
  32. marioY = memory.read_s16_le(0x96)
  33.  
  34. local layer1x = memory.read_s16_le(0x1A);
  35. local layer1y = memory.read_s16_le(0x1C);
  36.  
  37. screenX = marioX-layer1x
  38. screenY = marioY-layer1y
  39. elseif gameinfo.getromname() == "Super Mario Bros." then
  40. marioX = memory.readbyte(0x6D) * 0x100 + memory.readbyte(0x86)
  41. marioY = memory.readbyte(0x03B8)+16
  42.  
  43. screenX = memory.readbyte(0x03AD)
  44. screenY = memory.readbyte(0x03B8)
  45. end
  46. end
  47.  
  48. function getTile(dx, dy)
  49. if gameinfo.getromname() == "Super Mario World (USA)" then
  50. x = math.floor((marioX+dx+8)/16)
  51. y = math.floor((marioY+dy)/16)
  52.  
  53. return memory.readbyte(0x1C800 + math.floor(x/0x10)*0x1B0 + y*0x10 + x%0x10)
  54. elseif gameinfo.getromname() == "Super Mario Bros." then
  55. local x = marioX + dx + 8
  56. local y = marioY + dy - 16
  57. local page = math.floor(x/256)%2
  58.  
  59. local subx = math.floor((x%256)/16)
  60. local suby = math.floor((y - 32)/16)
  61. local addr = 0x500 + page*13*16+suby*16+subx
  62.  
  63. if suby >= 13 or suby < 0 then
  64. return 0
  65. end
  66.  
  67. if memory.readbyte(addr) ~= 0 then
  68. return 1
  69. else
  70. return 0
  71. end
  72. end
  73. end
  74.  
  75. function getSprites()
  76. if gameinfo.getromname() == "Super Mario World (USA)" then
  77. local sprites = {}
  78. for slot=0,11 do
  79. local status = memory.readbyte(0x14C8+slot)
  80. if status ~= 0 then
  81. spritex = memory.readbyte(0xE4+slot) + memory.readbyte(0x14E0+slot)*256
  82. spritey = memory.readbyte(0xD8+slot) + memory.readbyte(0x14D4+slot)*256
  83. sprites[#sprites+1] = {["x"]=spritex, ["y"]=spritey}
  84. end
  85. end
  86.  
  87. return sprites
  88. elseif gameinfo.getromname() == "Super Mario Bros." then
  89. local sprites = {}
  90. for slot=0,4 do
  91. local enemy = memory.readbyte(0xF+slot)
  92. if enemy ~= 0 then
  93. local ex = memory.readbyte(0x6E + slot)*0x100 + memory.readbyte(0x87+slot)
  94. local ey = memory.readbyte(0xCF + slot)+24
  95. sprites[#sprites+1] = {["x"]=ex,["y"]=ey}
  96. end
  97. end
  98.  
  99. return sprites
  100. end
  101. end
  102.  
  103. function getExtendedSprites()
  104. if gameinfo.getromname() == "Super Mario World (USA)" then
  105. local extended = {}
  106. for slot=0,11 do
  107. local number = memory.readbyte(0x170B+slot)
  108. if number ~= 0 then
  109. spritex = memory.readbyte(0x171F+slot) + memory.readbyte(0x1733+slot)*256
  110. spritey = memory.readbyte(0x1715+slot) + memory.readbyte(0x1729+slot)*256
  111. extended[#extended+1] = {["x"]=spritex, ["y"]=spritey}
  112. end
  113. end
  114.  
  115. return extended
  116. elseif gameinfo.getromname() == "Super Mario Bros." then
  117. return {}
  118. end
  119. end
  120.  
  121. function getInputs()
  122. getPositions()
  123.  
  124. sprites = getSprites()
  125. extended = getExtendedSprites()
  126.  
  127. local inputs = {}
  128.  
  129. for dy=-boxRadius*16,boxRadius*16,16 do
  130. for dx=-boxRadius*16,boxRadius*16,16 do
  131. inputs[#inputs+1] = 0
  132.  
  133. tile = getTile(dx, dy)
  134. if tile == 1 and marioY+dy < 0x1B0 then
  135. inputs[#inputs] = 1
  136. end
  137.  
  138. for i = 1,#sprites do
  139. distx = math.abs(sprites[i]["x"] - (marioX+dx))
  140. disty = math.abs(sprites[i]["y"] - (marioY+dy))
  141. if distx <= 8 and disty <= 8 then
  142. inputs[#inputs] = -1
  143. end
  144. end
  145.  
  146. for i = 1,#extended do
  147. distx = math.abs(extended[i]["x"] - (marioX+dx))
  148. disty = math.abs(extended[i]["y"] - (marioY+dy))
  149. if distx < 8 and disty < 8 then
  150. inputs[#inputs] = -1
  151. end
  152. end
  153. end
  154. end
  155.  
  156. --mariovx = memory.read_s8(0x7B)
  157. --mariovy = memory.read_s8(0x7D)
  158.  
  159. return inputs
  160. end
  161.  
  162.  
  163. function evaluate(inputs, chromosome)
  164. local outputs = {}
  165. local minDist = 4*inputSize
  166.  
  167. for i=0,(math.floor(#chromosome/snapshotSize)-1) do
  168. local total = 0
  169. for j=1,inputSize do
  170. d = inputs[j] - chromosome[i*snapshotSize + j]
  171. total = total + d*d
  172. end
  173. if total < minDist then
  174. minDist = total
  175. for j=1,#buttonNames do
  176. outputs[j] = chromosome[i*snapshotSize+inputSize+j]
  177. end
  178. nearest = i*snapshotSize
  179. end
  180. end
  181.  
  182. return outputs
  183. end
  184.  
  185. function randomChromosome()
  186. local c = {}
  187.  
  188. for i=1,snapshotSize*100 do
  189. c[i] = math.random(-1, 1)
  190. end
  191.  
  192. return c
  193. end
  194.  
  195. function initializeRun()
  196. savestate.load(filename);
  197. rightmost = 0
  198. frame = 0
  199. timeout = 20
  200. clearJoypad()
  201. end
  202.  
  203. function crossover(c1, c2)
  204. local c = {["chromosome"] = {}, ["fitness"] = 0}
  205. local pick = true
  206. for i=1,#c1["chromosome"] do
  207. if math.random(#c1["chromosome"]/2) == 1 then
  208. pick = not pick
  209. end
  210. if pick then
  211. c["chromosome"][i] = c1["chromosome"][i]
  212. else
  213. c["chromosome"][i] = c2["chromosome"][i]
  214. end
  215. end
  216.  
  217. return c
  218. end
  219.  
  220. function mutate(c, rate)
  221. for i=1,#c["chromosome"] do
  222. if math.random(rate) == 1 then
  223. c["chromosome"][i] = math.random(-1, 1)
  224. end
  225. end
  226. end
  227.  
  228. function difference(c1, c2)
  229. total = 0
  230. for i=1,#c1 do
  231. d = c1[i] - c2[i]
  232. total = total + d*d
  233. end
  234.  
  235. return total / (#c1*4)
  236. end
  237.  
  238. function clone(c)
  239. cl = {["chromosome"] = {}, ["fitness"] = 0}
  240. for i=1,#c["chromosome"] do
  241. cl["chromosome"][i] = c["chromosome"][i]
  242. end
  243. return cl
  244. end
  245.  
  246. function createNewGeneration()
  247. local shock = (generation % 60 == 59)
  248.  
  249. local top = pool[1]["fitness"]
  250. if not shock then
  251. pool[1]["fitness"] = 1000000
  252. end
  253.  
  254. table.sort(pool, function (a,b)
  255. return (a["fitness"] > b["fitness"])
  256. end)
  257.  
  258. pool[1]["fitness"] = top
  259.  
  260. if shock then
  261. for i=2,(#pool) do
  262. pool[i] = clone(pool[1])
  263. mutate(pool[i], 20)
  264. end
  265. else
  266. for i=((#pool)/2+1),(#pool) do
  267. c1 = pool[math.random(2, #pool/2)]
  268. c2 = pool[math.random(2, #pool/2)]
  269. pool[i] = crossover(c1, c2)
  270. mutate(pool[i], 400)
  271. end
  272. forms.settext(localFitness, "Local Fitness: " .. math.floor(pool[2]["fitness"]))
  273. end
  274.  
  275. generation = generation + 1
  276. end
  277.  
  278. function clearJoypad()
  279. controller = {}
  280. for b = 1,#buttonNames do
  281. controller["P1 " .. buttonNames[b]] = false
  282. end
  283. joypad.set(controller)
  284. end
  285.  
  286. function showTop()
  287. clearJoypad()
  288. currentChromosome = 0
  289. initializeRun()
  290. end
  291.  
  292. function onExit()
  293. forms.destroy(form)
  294. end
  295. event.onexit(onExit)
  296.  
  297. function connectionCost(chromosome)
  298. local total = 0
  299. for i=1,#chromosome["chromosome"] do
  300. c = chromosome["chromosome"][i]
  301. total = total + c*c
  302. end
  303.  
  304. return total
  305. end
  306.  
  307. function outputChromosomes()
  308. local filename = forms.gettext(saveLoadFile)
  309. local file = io.open(filename, "w")
  310. file:write(generation .. "\n")
  311. file:write(maxfitness .. "\n")
  312. file:write(#pool .. "\n")
  313. for c=1,#pool do
  314. local size = #pool[c]["chromosome"]
  315. file:write(size .. "\n")
  316. for n=1,size do
  317. file:write(pool[c]["chromosome"][n] .. "\n")
  318. end
  319. end
  320. file:close()
  321. end
  322.  
  323. function inputChromosomes()
  324. initializeSimulation()
  325. currentChromosome=0
  326. local filename = forms.gettext(saveLoadFile)
  327. local file = io.open(filename, "r")
  328. generation = tonumber(file:read())
  329. maxfitness = tonumber(file:read())
  330. forms.settext(maxFitnessLabel, "Top Fitness: " .. math.floor(maxfitness))
  331. local poolsize = tonumber(file:read())
  332. for c=1,poolsize do
  333. local size = tonumber(file:read())
  334. for n=1,size do
  335. pool[c]["chromosome"][n] = tonumber(file:read())
  336. end
  337. end
  338. file:close()
  339. end
  340.  
  341. function initializeSimulation()
  342. pool = {}
  343. for i=1,20 do
  344. pool[i] = {["chromosome"] = randomChromosome(), ["fitness"] = 0}
  345. end
  346. currentChromosome = 1
  347. generation = 0
  348. maxfitness = -1000000
  349. initializeRun()
  350. end
  351.  
  352. form = forms.newform(200, 275, "Fitness")
  353. maxFitnessLabel = forms.label(form, "Top Fitness: ", 5, 8)
  354. goButton = forms.button(form, "Show Top", showTop, 5, 30)
  355. goButton = forms.button(form, "Restart", initializeSimulation, 80, 30)
  356. localFitness = forms.label(form, "Local Fitness: ", 5, 55)
  357. showUI = forms.checkbox(form, "Show Inputs", 5, 74)
  358. inputsLabel = forms.label(form, "Inputs", 5, 96)
  359. showChromosomes = forms.checkbox(form, "Show Map", 5, 118)
  360. saveButton = forms.button(form, "Save", outputChromosomes, 5, 142)
  361. loadButton = forms.button(form, "Load", inputChromosomes, 80, 142)
  362. saveLoadFile = forms.textbox(form, "nearest_chromosomes.txt", 170, 25, nil, 5, 188)
  363. saveLoadLabel = forms.label(form, "Save/Load:", 5, 169)
  364. showNearest = forms.checkbox(form, "Show Nearest", 5, 215)
  365.  
  366. if pool == nil then
  367. initializeSimulation()
  368. else
  369. forms.settext(maxFitnessLabel, "Top Fitness: " .. math.floor(maxfitness))
  370. end
  371.  
  372. while true do
  373. getPositions()
  374.  
  375. timeoutBonus = frame / 4
  376. if timeout + timeoutBonus <= 0 then
  377. local fitness = 0
  378. fitness = rightmost - frame / 2
  379.  
  380. if currentChromosome >= 1 then
  381. pool[currentChromosome]["fitness"] = fitness
  382. end
  383.  
  384. if fitness > maxfitness then
  385. forms.settext(maxFitnessLabel, "Top Fitness: " .. math.floor(fitness))
  386. maxfitness = fitness
  387. end
  388.  
  389. console.writeline("Generation " .. generation .. " chromosome " .. currentChromosome .. " fitness: " .. math.floor(fitness))
  390. if currentChromosome == #pool then
  391. createNewGeneration()
  392. currentChromosome = #pool/2+1
  393. else
  394. if currentChromosome == 2 and pool[currentChromosome+1]["fitness"] ~= 0 then
  395. currentChromosome = (#pool)/2+1
  396. else
  397. currentChromosome = currentChromosome + 1
  398. end
  399. end
  400. initializeRun()
  401. end
  402.  
  403. if timeout + timeoutBonus > 2 and frame % 5 == 0 and currentChromosome >= 1 then
  404. inputs = getInputs()
  405. outputs = evaluate(inputs, pool[currentChromosome]["chromosome"])
  406.  
  407. controller = {}
  408. inputsString = ""
  409. for n = 1,#buttonNames do
  410. if outputs[n] > 0 then
  411. controller["P1 " .. buttonNames[n]] = true
  412. inputsString = inputsString .. buttonNames[n]
  413.  
  414. else
  415. controller["P1 " .. buttonNames[n]] = false
  416. end
  417. if controller["P1 Left"] and controller["P1 Right"] then
  418. controller["P1 Left"] = false
  419. controller["P1 Right"] = false
  420. end
  421. if controller["P1 Up"] and controller["P1 Down"] then
  422. controller["P1 Up"] = false
  423. controller["P1 Down"] = false
  424. end
  425. end
  426.  
  427. forms.settext(inputsLabel, inputsString)
  428. end
  429. joypad.set(controller)
  430.  
  431. if timeout + timeoutBonus <= 2 then
  432. clearJoypad()
  433. end
  434.  
  435. if marioX > rightmost then
  436. timeout = 20
  437. rightmost = marioX
  438. end
  439.  
  440. timeout = timeout - 1
  441. frame = frame + 1
  442.  
  443.  
  444. if forms.ischecked(showUI) and inputs ~= nil then
  445. for dy = 0,boxRadius*2 do
  446. for dx = 0,boxRadius*2 do
  447. input = inputs[dy*(boxRadius*2+1)+dx+1]
  448. local x = screenX+(dx-boxRadius)*16
  449. local y = screenY+(dy-boxRadius)*16
  450. if input == -1 then
  451. gui.drawBox(x, y, x+16, y+16, 0xFFFF0000, 0xA0FF0000)
  452. elseif input == 1 then
  453. gui.drawBox(x, y, x+16, y+16, 0xFF00FF00, 0xA000FF00)
  454. else
  455. gui.drawBox(x, y, x+16, y+16, 0xFFFFFF00, 0xA0FFFF00)
  456. end
  457. --gui.drawText(,,string.format("%i", input),0x80FFFFFF, 11)
  458. end
  459. end
  460.  
  461. gui.drawBox(screenX, screenY, screenX+16, screenY+32, 0xFF000000)
  462. end
  463.  
  464. if forms.ischecked(showNearest) and nearest ~= nil and currentChromosome >= 1 then
  465.  
  466. for dy = 0,boxRadius*2 do
  467. for dx = 0,boxRadius*2 do
  468. input = pool[currentChromosome]["chromosome"][nearest+dy*(boxRadius*2+1)+dx+1]
  469. local x = screenX+(dx-boxRadius)*16
  470. local y = screenY+(dy-boxRadius)*16
  471. if input == -1 then
  472. gui.drawBox(x, y, x+16, y+16, 0xFFFF0000, 0xA0FF0000)
  473. elseif input == 1 then
  474. gui.drawBox(x, y, x+16, y+16, 0xFF00FF00, 0xA000FF00)
  475. else
  476. gui.drawBox(x, y, x+16, y+16, 0xFFFFFF00, 0xA0FFFF00)
  477. end
  478. end
  479. end
  480.  
  481. gui.drawBox(screenX, screenY, screenX+16, screenY+32, 0xFF000000)
  482. end
  483.  
  484. if forms.ischecked(showChromosomes) then
  485. gui.drawBox(0, 3, 201, 3+#pool*3, 0xFFFFFFFF, 0xFFFFFFFF)
  486. for c=1,#pool do
  487. local y = 1+c*3
  488. local size = #pool[c]["chromosome"]
  489. for n=1,size do
  490. if n%math.floor(size/200) == 0 then
  491. local x = 1+n*200/#pool[c]["chromosome"]
  492. v = pool[c]["chromosome"][n]
  493. r = (1-v)/2
  494. g = (1+v)/2
  495. gui.drawLine(x, y, x, y+1, 0xFF000000 + math.floor(r*0xFF)*0x10000 + math.floor(g*0xFF)*0x100)
  496. end
  497. end
  498. end
  499. end
  500.  
  501. emu.frameadvance();
  502. end
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement