Advertisement
Andy73

vBenchmark

Feb 11th, 2015
254
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 24.64 KB | None | 0 0
  1. --[[
  2.        ______________________________________
  3.       / | \            |                 |  
  4. _    /  |_/  _       _ |_         _      |  
  5.  \  /   | \ /_\ +-+ /  | | +-+-+ '_\ ,_  |/
  6.   \/    |_/ \_  | | \_ | | | | | (_| | ' |\
  7.  
  8.  
  9. vBenchmark
  10. ==========
  11. An advanced benchmarking
  12. program for CC, good for
  13. comparison of various CC
  14. emulators.
  15.  
  16. version 0.2.3 (or Version (table))
  17. (build number @ /buildn)
  18.  
  19.  
  20. Contact me on the
  21. [CC Forums!](http://computercraft.info/forums2)
  22.  
  23. Made by viluon a.k.a. Andy73
  24. (c) 2015
  25.  
  26. Dev Notes:
  27. ==========
  28. TODO:
  29. --graphs
  30. --DONE:CLI
  31. --Settings
  32. --DONE:organized output (a-z)
  33. --all the tests listed in comments
  34. --DONE:+version in goodbye
  35.  
  36. +50
  37.  
  38. ]]
  39.  
  40. local Version={
  41.   App="0.2.3",
  42.   DebugID="REL",
  43. }
  44.  
  45. local SUICIDAL=false
  46.  
  47. term.redirect(term.native())
  48. local w,h=term.getSize()
  49.  
  50. -----startup
  51. term.setBackgroundColor(colors.black)
  52. term.clear()
  53. term.setCursorPos(1,1)
  54.  
  55. local function apicheck(n,p,fex) --name, paste ID, File EXtension fix TODO: file extension fix (Test.lua),
  56.   if not fs.exists("/"..(fex and n..".lua" or n)) and not _G[n] then
  57.     printError("No "..n.." API found, downloading one...")
  58.     shell.run("pastebin get "..p.." /"..(fex and n..".lua" or n))
  59.     error()
  60.   end
  61.   return true
  62. end
  63.  
  64. apicheck("sha256","P3KBjR3M")
  65. apicheck("aes","DMx8M0LP")
  66. apicheck("crc32","x52gQYWp")
  67. apicheck("json","jK22eUfn")
  68. apicheck("Test","wZbpA1Gy",true)
  69.  
  70. --[[if not fs.exists"/Test.lua" and not _G["Test"] then
  71.   printError"The 'Test' library was not found"
  72.   print"Downloading one..."
  73.   return shell.run"pastebin get wZbpA1Gy /Test.lua"
  74. end]]
  75.  
  76. local libs={
  77.   ["Test.lua"]=dofile,
  78.   ["sha256"]=os.loadAPI,
  79.   ["aes"]=os.loadAPI,
  80.   ["crc32"]=os.loadAPI,
  81.   ["json"]=os.loadAPI,
  82. }
  83. for k,v in pairs(libs) do
  84.   local ok,err=pcall(v,"/"..k)
  85.   if not ok then print(err) sleep(2) end
  86. end
  87.  
  88. pcall(fs.delete,"/.vBenchmarkUpdater.lua")
  89.  
  90. Version.Library=Test.Version
  91.  
  92. --------GENERAL FUNCTIONS AND CLASSES
  93.  
  94. local function s()
  95.   local t=tostring({})
  96.   os.queueEvent(t)
  97.   coroutine.yield(t)
  98. end
  99.  
  100. local function pairsByKeys (t, f)
  101.   local a = {}
  102.   for n in pairs(t) do table.insert(a, n) end
  103.   table.sort(a, f)
  104.   local i = 0      -- iterator variable
  105.   local iter = function ()   -- iterator function
  106.     i = i + 1
  107.     if a[i] == nil then return nil
  108.     else return a[i], t[a[i]]
  109.     end
  110.   end
  111.   return iter
  112. end
  113.  
  114. local RESULT
  115. local names={
  116. ["ow"]="Optimized write test",
  117. ["uw"]="Unoptimized write test",
  118. ["tb"]="term.blit write test",
  119. ["ct"]="Color swap/rainbow test",
  120. ["lc"]="Line clear speed test",
  121. ["cs"]="Clear speed test",
  122. ["scr"]="Buffer scroll test",
  123. ["aa"]="Anti-aliasing test",
  124. ["hsh"]="SHA256 hash test",
  125. ["aese"]="AES Encryption perf. test",
  126. ["aesd"]="AES Decryption perf. test",
  127. ["tec"]="Table entry creation",
  128. ["ted"]="Table entry deletion",
  129. ["tea"]="Table entry access",
  130. ["dhke"]="Diffie-Hellman key exchange",
  131. ["crc"]="CRC32 hash test",
  132. ["cmpl1"]="Simple compilation",
  133. ["cmpl2"]="Medium compilation",
  134. }
  135. local TaskList={}
  136. for k,v in pairs(names)do
  137.   TaskList[k]=true
  138. end
  139.  
  140. local outputPath="/benchmark.txt"
  141. local maxTime=2 --maximum b. test time in seconds
  142. local maxCalls=10000 --maximum b. test iterations (number of calls)
  143.  
  144.  
  145.  
  146.  
  147. --WARN IMPORTANT ATTENTION
  148. --implement CLI here!!!!!!!!!!!!!!!!!!!!!
  149.  
  150. local function listToTable( str )
  151.   if type(str)~="string" then error("(1)Expected string, got "..type(str),2) end
  152.   if str:sub(#str,#str)~="," then str=str.."," end
  153.   local out={}
  154.   for entry in str:gmatch("(%a+),") do
  155.     table.insert(out,entry)
  156.   end
  157.   return out
  158. end
  159.  
  160. local function help(c)
  161.   local txt=[[
  162.  
  163. Welcome to vBenchmark!
  164.      
  165.   vBenchmark is a complex benchmarking program for ComputerCraft 1.6+, made by viluon (a.k.a. Andy73)
  166.  
  167.   [Usage]
  168.     vBenchmark -option value --flag -option2 list,of,values
  169.    
  170.    
  171.  
  172.     NOTE: Options and flags are executed in the order they're supplied in
  173.      
  174.    <required value> [optional value] (name)
  175.  
  176.    Available options:
  177.    =================
  178.      update (update) updates the program
  179.      =====
  180.      help OR ? OR /? ['light'] (help/usage) displays this page, optionally using black text on white background (this option stops the execution automatically)
  181.      =====
  182.      -o <path> (output) sets the output file to <path>
  183.      =====
  184.      -list [path] (list) lists the names and corresponding codes of available benchmark functions, optionally saving them to [path]
  185.      =====
  186.      -maxtime <seconds> (maximum time limit) sets the time limit per ONE benchmark function to <seconds> - default: 2
  187.      =====
  188.      -maxcalls <calls> (maximum function calls) sets the maximum amount of function calls per ONE benchmark function to <calls> - default: 10 000
  189.      =====
  190.      -exclude <tests> (exclude) excludes a list of tests from the benchmark, <tests> is a list of test codes
  191.      =====
  192.      -testonly <tests> (test only) sets the test list for the benchmark to <tests>
  193.      =====
  194.  
  195.    Available flags:
  196.    ===============
  197.      --test (test) prints some debug text on the screen
  198.      =====
  199.      --stop (stop) stops the execution of the benchmark
  200.      =====
  201.      --version (version info) displays the version information of the program and used libraries
  202.      =====
  203.      --SUICIDE (suicidal mode) Enables the suicidal mode:
  204.          Suicidal mode is intended for debugging, evaluating and precision testing of CC emulators or SINGLEPLAYER CC. Do NOT use the suicidal benchmark in multiplayer!!! Suicidal benchmark does not yield to produce the most relevant results. As such, it should NOT be used for regular benchmarking. The suicidal mode can corrupt files and do IRREVERSIBLE HARM!!! Use at your own risk.
  205.      =====
  206.      --safe (safe mode) enables safety features:
  207.          -output file backup
  208.          -visible error reports - launches the benchmark in a temporary handler (/.vBenchmarkSafe) that pauses after crashing => errors visible
  209.          -does not intersect with suicidal mode
  210.      ]]
  211.      --TODO -include <paths> include list of custom tests loaded from <paths>
  212.  term.setBackgroundColor(c and colors.white or colors.black)
  213.  term.setTextColor(c and colors.black or colors.white)
  214.  term.clear()
  215.  term.setCursorPos(1,1)
  216.  textutils.pagedPrint(txt)
  217. end
  218.  
  219. local Args={...}
  220. local args={}
  221. for k,v in pairs(Args) do
  222.  args[k]=v
  223. end
  224. local flags={
  225.  ["test"]=function()
  226.    print"This is a test"
  227.    sleep(3)
  228.  end,
  229.  ["stop"]=function()
  230.    error()
  231.  end,
  232.  ["version"]=function()
  233.    print(textutils.serialize(Version))
  234.  end,
  235.  ["suicide"]=function()
  236.    SUICIDAL=true
  237.  end,
  238.  ["safe"]=function()
  239.    if fs.exists(outputPath) then
  240.      local bkpPath,i=".bkp1",2
  241.      while fs.exists(outputPath..bkpPath) do
  242.        bkpPath=".bkp"..i
  243.        i=i+1
  244.      end
  245.      local ok,err=pcall(fs.copy,outputPath,outputPath..bkpPath)
  246.      if not ok then
  247.        print"[ERROR] Failed to create backup"
  248.        print(err)
  249.        print"Press ENTER to exit"
  250.        read" "
  251.      end
  252.    end
  253.    local f=fs.open("/.vBenchmarkSafe","w")
  254.    if not f then print"[ERROR] Failed to open file for writing: /.vBenchmarkSafe" print(e) sleep(5) error() end
  255. f.write([[
  256. print"Launching vBenchmark in safe mode"
  257. sleep(2)
  258. local path,args="]]..shell.getRunningProgram()..[[",{...}
  259. local f=fs.open(path,"r")
  260. local fn=f.readAll()
  261. f.close()
  262. fn=loadstring(fn,"vBenchmark [SAFE MODE]")
  263. if not fn then error("Failed to execute safe mode: file is corrupt!",0) end
  264. setfenv(fn,setmetatable({shell=setmetatable({getRunningProgram=function()return path end},{__index=_G})},{__index=_G}))
  265. local ok,err=pcall(fn,unpack(args)) --line 10
  266. if not ok then
  267. printError"vBenchmark has crashed."
  268. print""
  269. print(err)
  270. print"Enter to continue..."
  271. read" "
  272. end
  273. return true
  274.  
  275. --line 20
  276. ]])
  277.    f.close()
  278.    local n=Args
  279.    for k,v in pairs(n) do
  280.      if v:lower()=="--safe" then table.remove(n,k) end
  281.      if #n<1 or #n==k then break end
  282.    end
  283.    shell.run("/.vBenchmarkSafe",unpack(n))
  284.    error()
  285.  end,
  286. }
  287. local properties={
  288.  ["o"]={
  289.    function(path)
  290.      if type(path)~="string" then error("Expected string as output path, got "..type(path),0) end
  291.      outputPath=path
  292.    end,
  293.    1, --max arguments
  294.    1, --min arguments
  295.  },
  296.  ["list"]={
  297.    function(path)
  298.      for k,v in pairs(names) do
  299.        print(v.."="..k)
  300.      end
  301.      if type(path)=="string" then
  302.        if fs.exists(path) then
  303.          local f=fs.open(path,"w")
  304.          f.write(textutils.serialize(names))
  305.          f.close()
  306.        end
  307.      end
  308.    end,
  309.    1,
  310.    0,
  311.  },
  312.  ["maxcalls"]={
  313.    function(c)
  314.      if not tonumber(c) then error("Expected number for the maximum amount of calls, got "..type(c),0) end
  315.      maxCalls=tonumber(c)
  316.    end,
  317.    1,
  318.    1,
  319.  },
  320.  ["maxtime"]={
  321.    function(c)
  322.      if not tonumber(c) then error("Expected number for the maximum time, got "..type(c),0) end
  323.      maxTime=tonumber(c)
  324.    end,
  325.    1,
  326.    1,
  327.  },
  328.  ["testonly"]={
  329.    function(list)
  330.      if type(list)~="string" then error("Expected string as a list of tests, got "..type(list),0) end
  331.      TaskList={}
  332.      for k,v in pairs(listToTable(list)) do
  333.        TaskList[v]=true
  334.      end
  335.    end,
  336.    1,
  337.    1,
  338.  },
  339.  ["exclude"]={
  340.    function(list)
  341.      for k,v in pairs(listToTable(list)) do
  342.        TaskList[v]=false
  343.      end
  344.    end,
  345.    1,
  346.    1,
  347.  },
  348. }
  349. for i,v in pairs(args) do
  350.  if v:sub(1,2)=="--" then
  351.    if not flags[v:sub(3,#v):lower()] then print("Hint: use '"..shell.getRunningProgram().." help' to see the proper usage")error("Unknown flag: "..v:sub(3,#v),0) end
  352.    flags[v:sub(3,#v):lower()]()
  353.  
  354.  elseif v:sub(1,1)=="-" then
  355.    if not properties[v:sub(2,#v):lower()] then print("Hint: use '"..shell.getRunningProgram().." help' to see the proper usage")error("Unknown option: "..v:sub(2,#v),0) end
  356.    local arg={}
  357.  
  358.    for ii=i+1,i+properties[v:sub(2,#v):lower()][2] do
  359.      if not args[ii] then break end
  360.      if args[ii]:sub(1,1)=="-" then break end --if the argument starts with "-" then break end --and ii-i>properties[v:sub(2,#v):lower()][3] and minimum number of arguments has already been reached
  361.      arg[#arg+1]=args[ii]
  362.      args[ii]=nil
  363.    end
  364.  
  365.    properties[v:sub(2,#v):lower()][1](unpack(arg))
  366.  
  367.  elseif v:lower()=="help" or v=="?" or v=="/?" then
  368.    if args[i+1]=="light" then return help(true) end
  369.    return help()
  370.  
  371.  elseif v:lower()=="update" then
  372.    local u=fs.open("/.vBenchmarkUpdater.lua","w")
  373.    u.write("shell.run'rm "..shell.getRunningProgram().."'shell.run'pastebin get nCEk57MH "..shell.getRunningProgram().."'print'Update complete'sleep(3)return shell.run'"..shell.getRunningProgram().." --stop'")
  374.     u.close()
  375.     return shell.run"/.vBenchmarkUpdater.lua"
  376.   end
  377. end
  378.  
  379. Test.SetTaskList(TaskList)
  380.  
  381. ---------BENCHMARK FUNCTIONS
  382. local function display() --graphical operations (both term and graphics related calculations)
  383.   local w,h=term.getSize()
  384.   ----text
  385.  --------Optimized write
  386.   local function fill(c,bc,ch)
  387.     term.setBackgroundColor(bc or colors.black)
  388.     term.setTextColor(c or colors.white)
  389.     term.clear()
  390.     for y=1,h do
  391.       term.setCursorPos(1,y)
  392.       term.write(string.rep(ch or "X",w))
  393.     end
  394.   end
  395.   local ow=Test.New("ow",nil,maxCalls,maxTime,true,fill) --ow=optimized write, no group set (nil)->graphics, maxCalls max calls, maxTime maximum total benchmark time, true->yield, 'fill' function to benchmark
  396.   ow.Run()
  397.   print(unpack(unpack({ow.GetResults()}))) --DEBUG
  398.  
  399.  --------Unoptimized write
  400.   local function fillUnoptimized(c,bc,ch)
  401.     term.setBackgroundColor(bc or colors.black)
  402.     term.setTextColor(c or colors.white)
  403.     term.clear()
  404.     for y=1,h do
  405.       for x=1,w do
  406.         term.setCursorPos(x,y)
  407.         term.write(ch or "X")
  408.       end
  409.     end
  410.   end
  411.   local uw=Test.New("uw",nil,maxCalls,maxTime,true,fillUnoptimized)
  412.   uw.Run()
  413.  
  414.   local function termBlitTest()
  415.     for y=1,h do
  416.       term.setCursorPos(1,y)
  417.       term.blit("asd")
  418.     end
  419.   end
  420.  
  421.  --------Color test
  422.   local function testColor()
  423.     for c=0,15 do
  424.       term.setBackgroundColor(2^c)
  425.       term.clear()
  426.     end
  427.   end
  428.   local ct=Test.New("ct",nil,maxCalls,maxTime,true,testColor)
  429.   ct.Run()
  430.  --------Line clear
  431.   local function lineClear(c)
  432.    term.setBackgroundColor(c or colors.black)
  433.     for y=1,h do
  434.      term.setCursorPos(1,y)
  435.      term.clearLine()
  436.     end
  437.   end
  438.   local lc=Test.New("lc",nil,maxCalls,maxTime,true,lineClear)
  439.   lc.Run()
  440.  
  441.  --------Clear
  442.   term.setBackgroundColor(colors.black)
  443.   local clear=term.clear
  444.   local cs=Test.New("cs",nil,maxCalls,maxTime,true,clear) --TODO/WARN: supply just the term.clear() function?
  445.   cs.Run()
  446.  
  447.  --------Scroll
  448.   local function scroll()
  449.     fill() --TODO any arguments?
  450.     for y=1,h do
  451.      term.scroll(1)
  452.     end
  453.   end
  454.   local scr=Test.New("scr",nil,maxCalls,maxTime,true,scroll)
  455.   scr.Run()
  456.  
  457.   ----2D graphics
  458.  -------Anti-Aliasing
  459.   local pixelWithAlpha = function(x,y,a)  -- Pseudo Alpha channel. Only works with black background (for obvious reasons)
  460.    
  461.     alphaStep = 255 / 4
  462.    
  463.     if a > 255 - alphaStep then
  464.       term.setBackgroundColor(colors.white)
  465.     elseif a > 255 - alphaStep * 2 then
  466.       term.setBackgroundColor(colors.lightGray)
  467.     elseif a > 255 - alphaStep * 3 then
  468.       term.setBackgroundColor(colors.gray)
  469.     else
  470.       term.setBackgroundColor(colors.black)
  471.       return
  472.     end
  473.    
  474.     term.setCursorPos(x,y)
  475.     term.write(" ")
  476.   end
  477.   local aaLine = function(x0, y0, x1, y1)
  478.  
  479.     dx = math.abs(x1-x0)
  480.     if x0<x1 then sx = 1 else sx = -1 end
  481.     dy = math.abs(y1-y0)
  482.     if y0<y1 then sy = 1 else sy = -1 end
  483.    
  484.     err = dx-dy        
  485.    
  486.     if dx + dy == 0 then
  487.       ed = 1
  488.     else
  489.       ed = math.sqrt(dx*dx+dy*dy)
  490.     end
  491.    
  492.     while true do                  
  493.       pixelWithAlpha(x0,y0, 255 - (255 * math.abs(err-dx+dy)/ed));
  494.       e2 = err
  495.       x2 = x0
  496.       if (2*e2 >= -dx) then            
  497.         if (x0 == x1) then break end
  498.         if (e2+dy < ed) then pixelWithAlpha(x0,y0+sy, 255 - (255*(e2+dy)/ed)) end
  499.         err = err - dy;
  500.         x0 = x0 + sx;
  501.       end
  502.       if (2*e2 <= dy) then                  
  503.         if (y0 == y1) then break end
  504.         if (dx-e2 < ed) then pixelWithAlpha(x2+sx,y0, 255 - (255*(dx-e2)/ed)) end
  505.         err  = err + dx;
  506.         y0 = y0 + sy;
  507.       end
  508.     end
  509.   end
  510.  
  511.   local w1,h1=math.floor(w/4),math.floor(h/2)
  512.  
  513.   local function aaTest()
  514.     local minX,minY=1,1
  515.     local maxX,maxY=15,15
  516.     local x1,y1,x2,y2=minX,minY,minX+1,minY+1
  517.     return function()
  518.       aaLine(x1,y1,x2,y2)
  519.       if x2>maxX then x2=minX y2=y2+1
  520.       elseif y2>maxY then --[[y2=minY y1=y1+1  end?]]
  521.       elseif x1>maxX then x1=minX y1=y1+1
  522.       elseif y1>maxY then y1=minY x2=x2+1
  523.       else x1=x1+1 end
  524.     end
  525.   end
  526.  
  527.   local aa = Test.New("aa",nil,maxCalls,maxTime,true,aaTest())
  528.   aa.Run()
  529.  --aa.Start() --TODO/WARN ATTENTION IMPORTANT what about these? :?
  530.  
  531.   ----3D graphics (TODO)
  532.  
  533.  
  534.  
  535. end
  536.  
  537.  
  538. local function mth() --mathematical operations
  539.  -------SHA256 Hashing
  540.  local l=256
  541.  local str=""
  542.  for i=1,l do
  543.   str=str..
  544.   string.char(math.random(128))
  545.  end
  546.  local sha=sha256.sha256
  547.  local hsh=Test.New("hsh","Math",maxCalls,maxTime,true,sha,str)
  548.  hsh.Run()
  549.  -------CRC32 Hashing
  550.  
  551.  local crch=crc32.Hash
  552.  --local crc=Test.New("crc","Math",maxCalls,maxTime,true,crch,str)
  553.  --crc.Run()
  554.  
  555.  -------AES Encryption
  556.  local enc,ey,didFinish=false
  557.  local function aesEncrypt(k,l)
  558.   local key=""
  559.   local str=""
  560.   for i=1,k do
  561.     key=key.."b"
  562.   end
  563.   for i=1,l do
  564.     str=str..string.char(math.random(128))
  565.   end
  566.   enc=aes.encrypt(key,str)
  567.   ey=key
  568.   didFinish=true
  569.  end
  570.  local aese=Test.New("aese","Math",maxCalls,maxTime,true,aesEncrypt,32,2048)
  571.  aese.Run()
  572.  -------AES Decryption
  573.  local aesDecrypt=aes.decrypt
  574.  if didFinish then
  575.    local aesd=Test.New("aesd","Math",maxCalls,maxTime,true,aesDecrypt,ey,enc)
  576.    aesd.Run()
  577.  end
  578.  -------Diffie-Hellman Key Exchange (by Anavrins)
  579.  local function dfhlKeyEx()
  580.    local function modexp(remainder,exponent,modulo)
  581.      for i=1,exponent-1 do
  582.        remainder=remainder*remainder
  583.        if remainder>=modulo then
  584.         remainder=remainder%modulo
  585.        end
  586.      end
  587.      return remainder
  588.    end
  589.    local public_base,public_primeMod=11,625210769
  590.    local alice_secret=math.random(100000,999999)
  591.    local bob_secret=math.random(100000,999999)
  592.  
  593.    local alice_public=modexp(public_base,alice_secret,public_primeMod)
  594.    local bob_public=modexp(public_base,bob_secret,public_primeMod)
  595.  
  596.    modexp(bob_public,alice_secret,public_primeMod)
  597.    modexp(alice_public,bob_secret,public_primeMod)
  598.  end
  599.  local dhke=Test.New("dhke","Math",maxCalls,maxTime,true,dfhlKeyEx)
  600.  dhke.Run()
  601.  -------Compilation test 1
  602.  local function cmpl1()
  603.   local code=[[
  604.   do end
  605.   ]]
  606.   loadstring(code)
  607.  end
  608.  local cmpl1=Test.New("cmpl1","Math",maxCalls,maxTime,true,cmpl1)
  609.  cmpl1.Run()
  610.  -------Compilation test 2
  611.  local function cmpl2()
  612.   local code=[[
  613.   local function asd()
  614.     while true do end
  615.   end
  616.   for i=1,10 do
  617.     print(i)
  618.   end
  619.   return _G.x or asd
  620.   ]]
  621.   loadstring(code)
  622.  end
  623.  local cmpl2=Test.New("cmpl2","Math",maxCalls,maxTime,true,cmpl2)
  624.  cmpl2.Run()
  625.  
  626.  
  627.  -------Goniometry
  628.  local function gnm()
  629.  
  630.  end
  631. end
  632.  
  633. local function prph() --peripheral benchmark (whatever there is available is benchmarked :D)
  634.  
  635. end
  636.  
  637.  
  638. local function flsys() --filesystem related tests
  639.  
  640. end
  641.  
  642.  
  643. local function ascan() --scan for anomalies (like LOVE2D Lua mathematical bugs, bit lib etc)
  644.  
  645. end
  646.  
  647. local function memal() --memory allocation
  648.  local function newEntry()
  649.    local t={} --table for allocation entries
  650.    for i=1,40000 do
  651.     t[#t+1]=true
  652.    end
  653.    t=nil
  654.  end
  655.  local tec=Test.New("tec","Memory Allocation",maxCalls,maxTime,true,newEntry)
  656.  tec.Run()
  657.  local function delEntry()
  658.    local t={}
  659.    for i=1,40000 do
  660.      t[#t+1]=true
  661.    end
  662.    for i,v in ipairs(t) do
  663.      t[i]=nil
  664.    end
  665.    t=nil
  666.  end
  667.  local ted=Test.New("ted","Memory Allocation",maxCalls,maxTime,true,delEntry)
  668.  ted.Run()
  669.  local function accEntry()
  670.    local t={true}
  671.    local _=false
  672.    for i=1,40000 do
  673.      _=t[1]
  674.    end
  675.  end
  676.  local tea=Test.New("tea","Memory Allocation",maxCalls,maxTime,true,accEntry)
  677.  tea.Run()
  678. end
  679.  
  680. ----------------------
  681. local wins,index={},1 --wins=pages
  682. wins[1]=window.create(term.native(),1,1,w,h-5,true)
  683. local function hideWins()
  684.   for i,v in pairs(wins) do
  685.     wins[i].setVisible(false)
  686.   end
  687. end
  688.  
  689. local function res() --display results
  690.   RESULT=Test.GetResults()
  691.   local last
  692.   if fs.exists(outputPath) then
  693.     local f=fs.open(outputPath,"r")
  694.     last=f.readAll()
  695.     f.close()
  696.     last=textutils.unserialize(last)
  697.     if not last then
  698.       local pth=outputPath.."-unsupported.txt"
  699.       local i=1
  700.       while fs.exists(pth) do
  701.         pth=outputPath.."-unsupported"..i..".txt"
  702.         i=i+1
  703.       end
  704.       fs.move(outputPath,pth)
  705.     end
  706.     RESULT["Last"]=last
  707.   end
  708.  
  709.   term.setBackgroundColor(colors.black)
  710.   term.setTextColor(colors.white)
  711.   term.clear()
  712.   term.setCursorPos(1,1)
  713.   local groups={}
  714.   for k,v in pairs(RESULT) do
  715.     if k~="Last" and k~="Version" and k~="Median" then --median for future upgrade
  716.       if not groups[v["g"]] then groups[v["g"]]={} end
  717.       table.insert(groups[v["g"]],k)
  718.     end
  719.   end
  720.  
  721.   for g,n in pairs(groups)do
  722.     term.setTextColor(colors.yellow)
  723.     print(g)
  724.     --using pairs by KEYS not values => swap n so that keys are actual names and values are the original RESULT subtables
  725.     local list={}
  726.     for k,v in pairs(n) do
  727.       list[names[v]]=v
  728.     end
  729.     for name,v in pairsByKeys(list)do
  730.       if TaskList[v] then
  731.         term.setTextColor(colors.white)
  732.         term.write("  "..name..": ")
  733.         term.setTextColor(colors.lightGray)
  734.         term.write((RESULT[v]["mark"]).."")
  735.  
  736.         if last and last[v] and type(last[v]["mark"])=="number" and (last.Version and (last.Version.Library and (Test.Version and (last.Version.Library.Matrix==Test.Version.Matrix)))) then -- safe check (no nil access attempts hopefully) for compatible Matrix version
  737.           term.setTextColor((last[v]["mark"]>RESULT[v]["mark"] and colors.red or colors.lime))
  738.           term.write(" "..(last[v]["mark"]>RESULT[v]["mark"] and "v" or "^"))
  739.           term.setTextColor(colors.lightGray)
  740.           print(" ("..(last[v]["mark"])..")")
  741.         else
  742.           print()
  743.         end
  744.         local _x,_y=term.getCursorPos()
  745.         if _y>h-9 then --10 for safe end (no scrolling)
  746.           wins[#wins+1]=window.create(term.native(),1,1,w,h-5,false)
  747.           term.redirect(wins[#wins])
  748.         end
  749.       end
  750.     end
  751.   end
  752.   term.redirect(term.native())
  753. end
  754.  
  755. local function render()
  756.   hideWins()
  757.   wins[index].setVisible(true)
  758.   wins[index].redraw()
  759.   local _x,_y=term.getCursorPos()
  760.  
  761.   term.setBackgroundColor(colors.black)
  762.   term.setTextColor(colors.gray)
  763.   term.setCursorPos(w-10,h-5)
  764.  
  765.   term.write("page ")
  766.   term.setTextColor(colors.lightGray)
  767.   term.write(index.."/"..#wins.."    ")--spaces to clean if #wins>10
  768.   term.setCursorPos(_x,_y)
  769. end
  770.  
  771. ----MAIN CODE
  772.  
  773. local function run()
  774.  display()
  775.  mth()
  776.  prph()
  777.  flsys()
  778.  ascan()
  779.  memal()
  780.  --TODO: CC API benchmark (parallel, textutils, etc)
  781. end
  782.  
  783.  
  784. run()
  785. term.setBackgroundColor(colors.black)
  786. term.clear()
  787.  
  788. term.redirect(wins[1])
  789. res()
  790. term.setCursorPos(1,h-5)
  791.  
  792. local _x,_y=term.getCursorPos()
  793. term.setCursorPos(w/2-5,_y+2)
  794. term.setTextColor(colors.gray)
  795. term.write"Loading..."
  796. term.setTextColor(colors.white)
  797. term.setCursorPos(_x,_y)
  798.  
  799. local selfie=fs.open(shell.getRunningProgram(),"r")
  800. local ip="''"
  801.  
  802. if http then
  803.   local h=http.get"http://acos.bluefile.cz/Utilities/Online/ip.php"
  804.   if h then
  805.     ip=h.readAll()
  806.     h.close()
  807.   end
  808. end
  809.  
  810. ip=textutils.unserialize(ip)
  811.  
  812. local c=selfie.readAll()
  813. selfie.close()
  814. Version.Hash=sha256.sha256(c)
  815.  
  816. Version.Fingerprint=sha256.sha256(Version.Hash.."Fingerprint"..os.getComputerID()..ip)
  817. RESULT["Version"]=Version
  818.  
  819. local save=fs.open(outputPath,"w")
  820. save.write("--viluon's Ultimate Benchmark output, os.time()="..os.time()..", os.day()="..os.day().."\n\n")
  821. save.write(textutils.serialize(RESULT))
  822. s""
  823. save.close()
  824. local r,l=0,RESULT.Last
  825. while true do
  826.   r=r+1
  827.   if l==nil then break end
  828.   l=(l and l.Last or nil) --first run fix
  829. end
  830.  
  831. print""
  832. print("Total runs:"..r)
  833. print("Results were logged (see "..outputPath.."). Would you like to re-run the benchmark now?")
  834. print""
  835. term.setBackgroundColor(colors.green)
  836. term.setTextColor(colors.white)
  837. term.write(string.rep(" ",((w/2)/2)-4).."[Re-run]"..string.rep(" ",((w/2)/2)-3))
  838. term.setBackgroundColor(colors.red)
  839. term.setTextColor(colors.black)
  840. term.write(string.rep(" ",((w/2)/2)-3).."[Exit]"..string.rep(" ",((w/2)/2)-1))
  841.  
  842. local rerun=false
  843.  
  844. while true do
  845.   render()
  846.   local ev={coroutine.yield()}
  847.   if ev[1]=="mouse_scroll" then
  848.     index=index+ev[2]
  849.   elseif ev[1]=="mouse_click" and ev[4]==h then
  850.     if ev[3]<w/2 then   --rerun
  851.       rerun=true break
  852.     else                --exit
  853.       local lines={
  854.         "       ______________________________________",
  855.         "      / | \\            |                 |  ",
  856.         "_    /  |_/  _       _ |_         _      |  ",
  857.         " \\  /   | \\ /_\\ +-+ /  | | +-+-+ '_\\ ,_  |/ ",
  858.         "  \\/    |_/ \\_  | | \\_ | | | | | (_| | ' |\\ "
  859.       }
  860.  
  861.       local w,h=term.getSize()
  862.       local X=w/2
  863.       local win=window.create(term.native(),X,1,#lines[1],5,true)
  864.  
  865.       term.setBackgroundColor(colors.black)
  866.       term.clear()
  867.       term.setCursorPos(2,6)
  868.       term.write"(c) "
  869.       term.setTextColor(colors.yellow)
  870.       term.write"viluon"
  871.       term.setTextColor(colors.white)
  872.       term.write" 2015"
  873.      
  874.  
  875.       term.redirect(win)
  876.  
  877.       term.setTextColor(colors.yellow)
  878.  
  879.  
  880.       for i,v in ipairs(lines) do
  881.         term.setCursorPos(1,i)
  882.         term.write(v:sub(1,6))
  883.       end
  884.  
  885.       term.setTextColor(colors.white)
  886.       term.setCursorPos(1,3)
  887.       term.write"_"
  888.  
  889.       for i=6,#lines[1] do
  890.         for k,v in ipairs(lines) do
  891.           term.setCursorPos(i,k)
  892.           term.write(v:sub(i,i))
  893.         end
  894.         X=X-((w/2-2)/(#lines[1]-6))
  895.         win.reposition(X,1)
  896.         sleep(0)
  897.       end
  898.       term.redirect(term.native())
  899.       term.setCursorPos(w-#Version.App-(w-#lines[1]),6)
  900.       term.write("v"..Version.App)
  901.  
  902.       sleep(1)
  903.       break
  904.     end
  905.   elseif ev[1]=="key" then
  906.     if ev[2]==keys.right or ev[2]==keys.pageDown then
  907.       index=index+1
  908.     elseif ev[2]==keys.left or ev[2]==keys.pageUp then
  909.       index=index-1
  910.     end
  911.   end
  912.   if index<1 then index=1
  913.   elseif index>#wins then index=#wins end
  914. end
  915.  
  916. if rerun then term.setTextColor(colors.white) shell.run"clear" return shell.run(shell.getRunningProgram(),unpack(Args)) end
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement