| "use strict"; |
| |
| (function() |
| { |
| function dump_file(ab, name) |
| { |
| var blob = new Blob([ab]); |
| |
| var a = document.createElement("a"); |
| a["download"] = name; |
| a.href = window.URL.createObjectURL(blob), |
| a.dataset["downloadurl"] = ["application/octet-stream", a["download"], a.href].join(":"); |
| |
| if(document.createEvent) |
| { |
| var ev = document.createEvent("MouseEvent"); |
| ev.initMouseEvent("click", true, true, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null); |
| a.dispatchEvent(ev); |
| } |
| else |
| { |
| a.click(); |
| } |
| } |
| |
| /** |
| * @return {Object.<string, string>} |
| */ |
| function get_query_arguments() |
| { |
| var query = location.search.substr(1).split("&"); |
| var parameters = {}; |
| |
| for(var i = 0; i < query.length; i++) |
| { |
| var param = query[i].split("="); |
| parameters[param[0]] = decodeURIComponent(param[1]); |
| } |
| |
| return parameters; |
| } |
| |
| function set_title(text) |
| { |
| document.title = text + " - Virtual x86" + (DEBUG ? " - debug" : ""); |
| } |
| |
| function time2str(time) |
| { |
| if(time < 60) |
| { |
| return time + "s"; |
| } |
| else if(time < 3600) |
| { |
| return (time / 60 | 0) + "m " + v86util.pad0(time % 60, 2) + "s"; |
| } |
| else |
| { |
| return (time / 3600 | 0) + "h " + |
| v86util.pad0((time / 60 | 0) % 60, 2) + "m " + |
| v86util.pad0(time % 60, 2) + "s"; |
| } |
| } |
| |
| function chr_repeat(chr, count) |
| { |
| var result = ""; |
| |
| while(count-- > 0) |
| { |
| result += chr; |
| } |
| |
| return result; |
| } |
| |
| var progress_ticks = 0; |
| |
| function show_progress(message, e) |
| { |
| var el = $("loading"); |
| el.style.display = "block"; |
| |
| var line = message + " "; |
| |
| if(typeof e.file_index === "number" && e.file_count) |
| { |
| line += "[" + (e.file_index + 1) + "/" + e.file_count + "] "; |
| } |
| |
| if(e.total && typeof e.loaded === "number") |
| { |
| var per100 = Math.floor(e.loaded / e.total * 100); |
| per100 = Math.min(100, Math.max(0, per100)); |
| |
| var per50 = Math.floor(per100 / 2); |
| |
| line += per100 + "% ["; |
| line += chr_repeat("#", per50); |
| line += chr_repeat(" ", 50 - per50) + "]"; |
| } |
| else |
| { |
| line += chr_repeat(".", progress_ticks++ % 50); |
| } |
| |
| el.textContent = line; |
| } |
| |
| function $(id) |
| { |
| var el = document.getElementById(id); |
| |
| if(!el) |
| { |
| console.log("Element with id `" + id + "` not found"); |
| } |
| |
| return el; |
| } |
| |
| function onload() |
| { |
| if(!("responseType" in new XMLHttpRequest)) |
| { |
| alert("Your browser is not supported because it doesn't have XMLHttpRequest.responseType"); |
| return; |
| } |
| |
| var settings = {}; |
| |
| $("start_emulation").onclick = function() |
| { |
| $("boot_options").style.display = "none"; |
| set_profile("custom"); |
| |
| var images = []; |
| var last_file; |
| |
| var floppy_file = $("floppy_image").files[0]; |
| if(floppy_file) |
| { |
| last_file = floppy_file; |
| settings.fda = { buffer: floppy_file }; |
| } |
| |
| var cd_file = $("cd_image").files[0]; |
| if(cd_file) |
| { |
| last_file = cd_file; |
| settings.cdrom = { buffer: cd_file }; |
| } |
| |
| var hd_file = $("hd_image").files[0]; |
| if(hd_file) |
| { |
| last_file = hd_file; |
| settings.hda = { buffer: hd_file }; |
| } |
| |
| if(last_file) |
| { |
| set_title(last_file.name); |
| } |
| |
| start_emulation({ |
| settings: settings, |
| done: function(emulator) { |
| emulator.run(); |
| }, |
| }); |
| }; |
| |
| if(DEBUG) |
| { |
| debug_onload(settings); |
| } |
| |
| var oses = [ |
| { |
| id: "archlinux", |
| state: { |
| //url: "http://localhost/v86-images/v86state.bin", |
| url: "http://104.131.53.7:8086/v86state.bin", |
| size: 69283634, |
| }, |
| //size: 137 * 1024 * 1024, |
| name: "Arch Linux", |
| memory_size: 64 * 1024 * 1024, |
| vga_memory_size: 2 * 1024 * 1024, |
| |
| async_hda: { |
| url: "https://dl.dropboxusercontent.com/u/61029208/arch3.img", |
| //url: "http://104.131.53.7:8086/arch3.img", |
| //url: "http://localhost/v86-images/arch3.img", |
| size: 8 * 1024 * 1024 * 1024, |
| }, |
| |
| filesystem: { |
| //basefs: "http://localhost/v86-images/fs.json", |
| //baseurl: "http://localhost/v86-images/arch/", |
| basefs: "http://104.131.53.7:8086/fs.json", |
| baseurl: "http://104.131.53.7:8086/arch/", |
| }, |
| }, |
| { |
| id: "freedos", |
| fda: { |
| url: "images/freedos722.img", |
| size: 737280, |
| }, |
| name: "FreeDOS", |
| }, |
| { |
| id: "windows1", |
| fda: { |
| url: "images/windows101.img", |
| size: 1474560, |
| }, |
| name: "Windows", |
| }, |
| { |
| id: "linux26", |
| cdrom: { |
| url: "images/linux.iso", |
| size: 5666816, |
| }, |
| name: "Linux", |
| }, |
| { |
| id: "kolibrios", |
| fda: { |
| url: "images/kolibri.img", |
| size: 1474560, |
| }, |
| name: "KolibriOS", |
| }, |
| { |
| id: "openbsd", |
| fda: { |
| url: "images/openbsd.img", |
| size: 1474560, |
| }, |
| name: "OpenBSD", |
| }, |
| { |
| id: "solos", |
| fda: { |
| url: "images/os8.dsk", |
| size: 1474560, |
| }, |
| name: "Sol OS", |
| }, |
| ]; |
| |
| var query_args = get_query_arguments(); |
| var profile = query_args["profile"]; |
| |
| for(var i = 0; i < oses.length; i++) |
| { |
| var infos = oses[i]; |
| var dom_id = "start_" + infos.id; |
| |
| $(dom_id).onclick = function(infos, dom_id) |
| { |
| set_profile(infos.id); |
| $(dom_id).blur(); |
| |
| start_profile(infos); |
| }.bind(this, infos, dom_id); |
| |
| if(profile === infos.id) |
| { |
| start_profile(infos); |
| return; |
| } |
| } |
| |
| function start_profile(infos) |
| { |
| $("boot_options").style.display = "none"; |
| set_title(infos.name); |
| |
| settings.filesystem = infos.filesystem; |
| |
| if(infos.state) |
| { |
| $("reset").style.display = "none"; |
| settings.initial_state = infos.state; |
| } |
| |
| settings.fda = infos.fda; |
| settings.cdrom = infos.cdrom; |
| |
| if(infos.hda) |
| { |
| settings.hda = infos.hda |
| } |
| else if(infos.async_hda) |
| { |
| settings.hda = { |
| url: infos.async_hda.url, |
| size: infos.async_hda.size, |
| async: true, |
| }; |
| } |
| |
| settings.memory_size = infos.memory_size; |
| settings.vga_memory_size = infos.vga_memory_size; |
| |
| start_emulation({ |
| settings: settings, |
| done: function(emulator) |
| { |
| emulator.run(); |
| |
| if(query_args["c"]) |
| { |
| emulator.serial0_send(query_args["c"] + "\n"); |
| } |
| } |
| }); |
| } |
| } |
| |
| function debug_onload(settings) |
| { |
| // called on window.onload, in debug mode |
| |
| //settings.filesystem = { |
| // baseurl: "http://localhost/v86-images/arch/", |
| // basefs: "http://localhost/v86-images/fs.json", |
| //}; |
| |
| $("restore_state").onchange = function() |
| { |
| }; |
| |
| $("start_test").onclick = function() |
| { |
| }; |
| |
| var log_levels = document.getElementById("log_levels"); |
| |
| for(var i = 0; i < LOG_NAMES.length; i++) |
| { |
| var mask = LOG_NAMES[i][0]; |
| |
| if(mask === 1) |
| continue; |
| |
| var name = LOG_NAMES[i][1].toLowerCase(), |
| input = document.createElement("input"), |
| label = document.createElement("label"); |
| |
| input.type = "checkbox"; |
| |
| label.htmlFor = input.id = "log_" + name; |
| |
| if(LOG_LEVEL & mask) |
| { |
| input.checked = true; |
| } |
| input.mask = mask; |
| |
| label.appendChild(input); |
| label.appendChild(document.createTextNode(name + " ")); |
| log_levels.appendChild(label); |
| } |
| |
| log_levels.onchange = function(e) |
| { |
| var target = e.target, |
| mask = target.mask; |
| |
| if(target.checked) |
| { |
| LOG_LEVEL |= mask; |
| } |
| else |
| { |
| LOG_LEVEL &= ~mask; |
| } |
| }; |
| } |
| |
| window.addEventListener("load", onload, false); |
| |
| // old webkit fires popstate on every load, fuck webkit |
| // https://code.google.com/p/chromium/issues/detail?id=63040 |
| window.addEventListener("load", function() |
| { |
| setTimeout(function() |
| { |
| window.addEventListener("popstate", onpopstate); |
| }, 0); |
| }); |
| |
| // works in firefox and chromium |
| if(document.readyState === "complete") |
| { |
| onload(); |
| } |
| |
| function start_emulation(result) |
| { |
| /** @const */ |
| var MB = 1024 * 1024; |
| |
| var settings = result.settings; |
| var memory_size = settings.memory_size; |
| |
| if(!memory_size) |
| { |
| memory_size = parseInt($("memory_size").value, 10) * MB; |
| |
| if(memory_size < 16 * MB || memory_size >= 2048 * MB) |
| { |
| alert("Invalid memory size - ignored."); |
| memory_size = 32 * MB; |
| } |
| } |
| |
| var vga_memory_size = settings.vga_memory_size; |
| |
| if(!vga_memory_size) |
| { |
| vga_memory_size = parseInt($("video_memory_size").value, 10) * MB; |
| |
| if(vga_memory_size <= 64 * 1024 || vga_memory_size >= 2048 * MB) |
| { |
| alert("Invalid video memory size - ignored."); |
| vga_memory_size = 8 * MB; |
| } |
| } |
| |
| var BIOSPATH = "bios/"; |
| var biosfile = DEBUG ? "seabios-debug.bin" : "seabios.bin"; |
| var vgabiosfile = DEBUG ? "vgabios-0.7a.debug.bin" : "bochs-vgabios-0.7a.bin"; |
| |
| |
| var emulator = new V86Starter({ |
| "memory_size": memory_size, |
| "vga_memory_size": vga_memory_size, |
| |
| "screen_container": $("screen_container"), |
| "serial_container": $("serial"), |
| |
| "boot_order": parseInt($("boot_order").value, 16) || 0, |
| |
| "network_relay_url": "ws://relay.widgetry.org/", |
| //"network_relay_url": "ws://localhost:8001/", |
| |
| "bios": { |
| "url": BIOSPATH + biosfile, |
| }, |
| "vga_bios": { |
| "url": BIOSPATH + vgabiosfile, |
| }, |
| |
| "fda": settings.fda, |
| "hda": settings.hda, |
| "cdrom": settings.cdrom, |
| |
| "initial_state": settings.initial_state, |
| "filesystem": settings.filesystem, |
| }); |
| |
| emulator.add_listener("emulator-ready", function() |
| { |
| if(DEBUG) |
| { |
| debug_start(emulator); |
| } |
| |
| init_ui({}, emulator); |
| |
| result.done(emulator); |
| }); |
| |
| emulator.add_listener("download-progress", function(e) |
| { |
| show_progress("Downloading images", e); |
| }); |
| }; |
| |
| function init_ui(settings, emulator) |
| { |
| $("boot_options").style.display = "none"; |
| $("loading").style.display = "none"; |
| $("runtime_options").style.display = "block"; |
| $("runtime_infos").style.display = "block"; |
| document.getElementsByClassName("phone_keyboard")[0].style.display = "block"; |
| |
| var news_element = $("news"); |
| if(news_element) |
| { |
| news_element.style.display = "none"; |
| } |
| |
| $("run").onclick = function() |
| { |
| if(emulator.is_running()) |
| { |
| running_time += Date.now() - last_tick; |
| $("run").value = "Run"; |
| emulator.stop(); |
| } |
| else |
| { |
| $("run").value = "Pause"; |
| emulator.run(); |
| last_tick = Date.now(); |
| } |
| |
| $("run").blur(); |
| }; |
| |
| $("exit").onclick = function() |
| { |
| emulator.stop(); |
| location.href = location.pathname; |
| }; |
| |
| $("lock_mouse").onclick = function() |
| { |
| if(!mouse_is_enabled) |
| { |
| $("toggle_mouse").onclick(); |
| } |
| |
| emulator.lock_mouse(); |
| $("lock_mouse").blur(); |
| }; |
| |
| var mouse_is_enabled = true; |
| |
| $("toggle_mouse").onclick = function() |
| { |
| mouse_is_enabled = !mouse_is_enabled; |
| |
| emulator.mouse_set_status(mouse_is_enabled); |
| $("toggle_mouse").value = (mouse_is_enabled ? "Dis" : "En") + "able mouse"; |
| $("toggle_mouse").blur(); |
| }; |
| |
| |
| var last_tick = Date.now(); |
| var running_time = 0; |
| var summed_ips = 0; |
| var last_instr_counter = 0; |
| |
| function update_info() |
| { |
| if(!emulator.is_running()) |
| { |
| return; |
| } |
| |
| var instruction_counter = emulator.get_instruction_counter(); |
| var last_ips = instruction_counter - last_instr_counter; |
| |
| summed_ips += last_ips; |
| last_instr_counter = instruction_counter; |
| |
| var now = Date.now(); |
| |
| running_time += now - last_tick; |
| last_tick = now; |
| |
| $("speed").textContent = last_ips / 1000 | 0; |
| $("avg_speed").textContent = summed_ips / running_time | 0; |
| $("running_time").textContent = time2str(running_time / 1000 | 0); |
| } |
| |
| setInterval(update_info, 1000); |
| |
| var stats_9p = { |
| read: 0, |
| write: 0, |
| }; |
| |
| emulator.add_listener("9p-read-start", function() |
| { |
| $("info_filesystem").style.display = "block"; |
| $("info_filesystem_status").textContent = "Loading ..."; |
| }); |
| emulator.add_listener("9p-read-end", function(args) |
| { |
| stats_9p.read += args[1]; |
| |
| $("info_filesystem_status").textContent = "Idle"; |
| $("info_filesystem_last_file").textContent = args[0] |
| $("info_filesystem_bytes_read").textContent = stats_9p.read; |
| }); |
| emulator.add_listener("9p-write-end", function(args) |
| { |
| stats_9p.write += args[1]; |
| |
| $("info_filesystem_last_file").textContent = args[0] |
| $("info_filesystem_bytes_written").textContent = stats_9p.write; |
| }); |
| |
| var stats_storage = { |
| read: 0, |
| read_sectors: 0, |
| write: 0, |
| write_sectors: 0, |
| }; |
| |
| emulator.add_listener("ide-read-start", function() |
| { |
| $("info_storage").style.display = "block"; |
| $("info_storage_status").textContent = "Loading ..."; |
| }); |
| emulator.add_listener("ide-read-end", function(args) |
| { |
| stats_storage.read += args[1]; |
| stats_storage.read_sectors += args[2]; |
| |
| $("info_storage_status").textContent = "Idle"; |
| $("info_storage_bytes_read").textContent = stats_storage.read; |
| $("info_storage_sectors_read").textContent = stats_storage.read_sectors; |
| }); |
| emulator.add_listener("ide-write-end", function(args) |
| { |
| stats_storage.write += args[1]; |
| stats_storage.write_sectors += args[2]; |
| |
| $("info_storage_bytes_written").textContent = stats_storage.write; |
| $("info_storage_sectors_written").textContent = stats_storage.write_sectors; |
| }); |
| |
| |
| emulator.add_listener("mouse-enable", function(is_enabled) |
| { |
| $("info_mouse_enabled").textContent = is_enabled ? "Yes" : "No"; |
| }); |
| |
| emulator.add_listener("screen-set-mode", function(is_graphical) |
| { |
| if(is_graphical) |
| { |
| $("info_vga_mode").textContent = "Graphical"; |
| } |
| else |
| { |
| $("info_vga_mode").textContent = "Text"; |
| $("info_res").textContent = "-"; |
| $("info_bpp").textContent = "-"; |
| } |
| }); |
| emulator.add_listener("screen-set-size-graphical", function(args) |
| { |
| $("info_res").textContent = args[0] + "x" + args[1]; |
| $("info_bpp").textContent = args[2]; |
| }); |
| |
| |
| $("reset").onclick = function() |
| { |
| emulator.restart(); |
| $("reset").blur(); |
| }; |
| |
| // writable image types |
| var image_types = ["hda", "hdb", "fda", "fdb"]; |
| |
| for(var i = 0; i < image_types.length; i++) |
| { |
| var elem = $("get_" + image_types[i] + "_image"); |
| var obj = settings[image_types[i]]; |
| |
| if(obj && obj.byteLength < 16 * 1024 * 1024) |
| { |
| elem.onclick = (function(type) |
| { |
| obj.get_buffer(function(b) |
| { |
| dump_file(b, type + ".img"); |
| }); |
| |
| this.blur(); |
| |
| }).bind(elem, image_types[i]); |
| } |
| else |
| { |
| elem.style.display = "none"; |
| } |
| } |
| |
| $("ctrlaltdel").onclick = function() |
| { |
| emulator.keyboard_send_scancodes([ |
| 0x1D, // ctrl |
| 0x38, // alt |
| 0x53, // delete |
| |
| // break codes |
| 0x1D | 0x80, |
| 0x38 | 0x80, |
| 0x53 | 0x80, |
| ]); |
| |
| $("ctrlaltdel").blur(); |
| }; |
| |
| $("alttab").onclick = function() |
| { |
| emulator.keyboard_send_scancodes([ |
| 0x38, // alt |
| 0x0F, // tab |
| ]); |
| |
| setTimeout(function() |
| { |
| emulator.keyboard_send_scancodes([ |
| 0x38 | 0x80, |
| 0x0F | 0x80, |
| ]); |
| }, 100); |
| |
| $("alttab").blur(); |
| }; |
| |
| $("scale").onchange = function() |
| { |
| var n = parseFloat(this.value); |
| |
| if(n || n > 0) |
| { |
| emulator.screen_set_scale(n, n); |
| } |
| }; |
| |
| $("fullscreen").onclick = function() |
| { |
| emulator.screen_go_fullscreen(); |
| }; |
| |
| $("screen_container").onclick = function() |
| { |
| // allow text selection |
| if(window.getSelection().isCollapsed) |
| { |
| document.getElementsByClassName("phone_keyboard")[0].focus(); |
| } |
| }; |
| |
| $("take_screenshot").onclick = function() |
| { |
| emulator.screen_make_screenshot(); |
| |
| $("take_screenshot").blur(); |
| }; |
| |
| $("serial").style.display = "block"; |
| |
| window.addEventListener("keydown", ctrl_w_rescue, false); |
| window.addEventListener("keyup", ctrl_w_rescue, false); |
| window.addEventListener("blur", ctrl_w_rescue, false); |
| |
| function ctrl_w_rescue(e) |
| { |
| if(e.ctrlKey) |
| { |
| window.onbeforeunload = function() |
| { |
| window.onbeforeunload = null; |
| return "CTRL-W cannot be sent to the emulator."; |
| } |
| } |
| else |
| { |
| window.onbeforeunload = null; |
| } |
| } |
| } |
| |
| function debug_start(emulator) |
| { |
| // called as soon as soon as emulation is started, in debug mode |
| var debug = emulator.v86.cpu.debug; |
| |
| $("step").onclick = debug.step.bind(debug); |
| $("run_until").onclick = debug.run_until.bind(debug); |
| $("debugger").onclick = debug.debugger.bind(debug); |
| $("dump_gdt").onclick = debug.dump_gdt_ldt.bind(debug); |
| $("dump_idt").onclick = debug.dump_idt.bind(debug); |
| $("dump_regs").onclick = debug.dump_regs.bind(debug); |
| $("dump_pt").onclick = debug.dump_page_directory.bind(debug); |
| $("dump_instructions").onclick = debug.dump_instructions.bind(debug); |
| |
| $("dump_instructions_file").onclick = function() |
| { |
| var ins = debug.get_instructions(); |
| |
| if(ins) |
| { |
| dump_file(ins, "trace.txt"); |
| } |
| }; |
| |
| $("memory_dump").onclick = function() |
| { |
| dump_file(debug.get_memory_dump(), "v86-memory.bin"); |
| $("memory_dump").blur(); |
| }; |
| |
| $("save_state").onclick = function() |
| { |
| emulator.save_state(function(error, result) |
| { |
| if(error) |
| { |
| console.log("Couldn't save state: ", error); |
| } |
| else |
| { |
| dump_file(result, "v86state.bin"); |
| } |
| }); |
| |
| $("save_state").blur(); |
| }; |
| |
| // helps debugging |
| window.emulator = emulator; |
| window.cpu = emulator.v86.cpu; |
| } |
| |
| function onpopstate(e) |
| { |
| location.reload(); |
| } |
| |
| function set_profile(prof) |
| { |
| if(window.history.pushState) |
| { |
| window.history.pushState({ profile: prof }, "", "?profile=" + prof); |
| } |
| |
| } |
| |
| })(); |