<!DOCTYPE html>
<html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Archetypum | Tomasulo algorithm visualizer</title>
        <meta name="author" content="Martin Brugnara<mb@disi.unitn.it>, Gianluca Bortoli <gianluca.bortoli@unitn.it>">
        <meta name="keyword" content="Tomasulo, Tomasulo algorithm, visualizer, reservation station, ROB">
        <meta name="description" content="Higly parametrized Tomasulo algorithm visualizer">
        <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <style type="text/css">
* {
  box-sizing: border-box;
  vertical-align: middle; /* Otherwise f**king tables does not align */
}

html {
  font-family: Verdana, Geneva, sans-serif;
  font-size: 13px;
}

a, a:link, a:visited, a:hover, a:active {
  text-transform: none;
  text-decoration: none;
  color: #000;
}

a:hover {cursor: pointer}

#main {
  position: relative;
  display: block;
  width: 800px;
  height: 600px;
  border: 1px solid #777;
}

a.tab-link {
  display: block;
  position: relative;
  top: -2rem;
  width: 33.33%;
  height: 2rem;
  line-height: 2rem;
  text-align: center;
  text-transform: uppercase;
}

.tab-link:hover {
  cursor: pointer;
  background-color: #ffec5e;
}

.tab {
  display:block;
  position: absolute;
  width: 100%;
  top: 2rem;
  bottom: 1.5rem;
}

.tab-body {
  margin-top: -2rem;
  padding: 20px;
  line-height: 1em;
}

.tab.active, .tab.active .tab-link {
  z-index: 100;
  background-color: #ffab40;
}
.tab.active .tab-link { color: #fff; }

#main .tab:nth-child(2) .tab-link {left: 33.33%;}
#main .tab:nth-child(3) .tab-link {left: 66.66%;}


#main > footer {
  position: absolute;
  bottom: 0;
  width: 100%;
  height: 1.5rem;
  line-height: 1.5rem;
  text-align: center;
  font-size: 0.8rem;
}

#main > footer a {font-weight: bold;}

#octocat {
  display: inline-block;
  height: 14px;
  width: 14px;
  vertical-align: text-bottom;
  background-size: 100% 100%;
  background-image: url('');
}

/* -------------------------------------------------------------------------- */
a.btn {
    display: inline-block;
    zoom: 1;
    white-space: nowrap;
    vertical-align: middle;
    text-align: center;
    cursor: pointer;
    -webkit-user-drag: none;
    -webkit-user-select: none;
    -moz-user-select: none;
    -ms-user-select: none;
    user-select: none;
    font-size: 1.5em;
    padding: .5em 1em;
    line-height: 1em;
    border-radius: 2px;
    text-transform: uppercase;
}

a.btn.blu {
    background-color: #0078e7;
    color: #fff;
}




/* -------------------------------------------------------------------------- */
#exlist {
  display: inline-block;
  list-style-type: none;
  margin: 0;
  padding: 0;
}
#exlist li {display: inline}
#exlist li:after {content:', '}
#exlist li:last-child:after {content:''}
#exlist a {text-decoration: underline;}

#raw-wrap {
  display: block;
  margin:auto;
  margin-top: 0;
  margin-bottom: 1em;
}

#raw-wrap > * {
  display: inline-block;
  width: 50%;
  vertical-align: top;
}

#raw-src {
  display: block;
  margin: auto;
  font-size: 1.25em;
  font-family: "Lucida Console", Monaco, monospace;
  text-transform: uppercase;
  resize: none;
}

#raw-wrap h4 {
  text-align: center;
}

#help ul {list-style-type: none;}
#help > ul {padding-left: 0;}
#help > ul > li {margin-bottom: 1.2em;}
#help p {margin-bottom: 0.5em;}
#help ul ul {
  font-family: "Lucida Console", Monaco, monospace;
  line-height: 1.2em;
}

#help ul ul .txt {font-family: Verdana, Geneva, sans-serif;}

#tab-config .tab-body {
  padding: 0;
  height: 90%;
}

#ctable {
  height: 496px;
  width: 100%;
  border-spacing: 15px;
}

#ctable td {
  border: 1px solid #fff;
  width: 50%;
  height: 33.33%;
}

#ctable td > div {
  height: 100%;
  width: 100%;
}

#ctable header {
  position: relative;
  top: -0.7em;
  background-color: #ffab40;
  padding-left: .75em;
  padding-right: .75em;
  left: 1em;
  width: auto;
  display: inline-block;
}

.inner-table {
  width: 100%;
  height: 100%;
  margin-top: -0.5em;
  border-spacing: 10px;
  text-align: center;
}

#ctable .inner-table td {
  border: none;
}

#ctable .inner-table tr.row3 td {width: 33.33%}

/* -------------------------------------------------------------------------- */
#control-bar {
  display: block;
  width: 798px;
  height: 2.5rem;
  margin: -10px 0 10px 0;
  background: white;
  /* box-shadow: 0 4px 5px 0 rgba(0,0,0,0.14), 0 1px 10px 0 rgba(0,0,0,0.12), 0 2px 4px -1px rgba(0,0,0,0.3); */
  box-shadow: 0 2px 2px 0 rgba(0,0,0,0.14), 0 1px 5px 0 rgba(0,0,0,0.12), 0 3px 1px -2px rgba(0,0,0,0.2);
  border-collapse: collapse;
}

#control-bar > table {
  width: 720px;
  margin-left: 40px;
}

#control-bar > table td {
  width: 33.33%;
}

#control-bar > table td:nth-child(2) {text-align: center;}
#control-bar > table td:nth-child(3) {text-align: right;}

.icon {
  display: inline-block;
  width: 28px;
  height: 28px;
  background-repeat: no-repeat;
  background-size: 100% 100%;
  background-position: center;
}

.icon.clock {background-image: url('data:image/svg+xml;utf8,<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M0 0h24v24H0z" fill="none"/><path d="M22 5.72l-4.6-3.86-1.29 1.53 4.6 3.86L22 5.72zM7.88 3.39L6.6 1.86 2 5.71l1.29 1.53 4.59-3.85zM12.5 8H11v6l4.75 2.85.75-1.23-4-2.37V8zM12 4c-4.97 0-9 4.03-9 9s4.02 9 9 9c4.97 0 9-4.03 9-9s-4.03-9-9-9zm0 16c-3.87 0-7-3.13-7-7s3.13-7 7-7 7 3.13 7 7-3.13 7-7 7z"/></svg>');}
.icon.reset {background-image: url('data:image/svg+xml;utf8,<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M12 4V1L8 5l4 4V6c3.31 0 6 2.69 6 6 0 1.01-.25 1.97-.7 2.8l1.46 1.46C19.54 15.03 20 13.57 20 12c0-4.42-3.58-8-8-8zm0 14c-3.31 0-6-2.69-6-6 0-1.01.25-1.97.7-2.8L5.24 7.74C4.46 8.97 4 10.43 4 12c0 4.42 3.58 8 8 8v3l4-4-4-4v3z"/><path d="M0 0h24v24H0z" fill="none"/></svg>');}
.icon.pause {background-image: url('data:image/svg+xml;utf8,<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M6 19h4V5H6v14zm8-14v14h4V5h-4z"/><path d="M0 0h24v24H0z" fill="none"/></svg>');}
.icon.play {background-image: url('data:image/svg+xml;utf8,<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M8 5v14l11-7z"/><path d="M0 0h24v24H0z" fill="none"/></svg>');}
.icon.play-circle {background-image: url('data:image/svg+xml;utf8,<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M0 0h24v24H0z" fill="none"/><path d="M10 16.5l6-4.5-6-4.5v9zM12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"/></svg>');}
.icon.play-circle-fill {background-image: url('data:image/svg+xml;utf8,<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M0 0h24v24H0z" fill="none"/><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm-2 14.5v-9l6 4.5-6 4.5z"/></svg>');}
.icon.pc {background-image: url('data:image/svg+xml;utf8,<svg enable-background="new 0 0 24 24" id="Layer_1" version="1.1" viewBox="0 0 24 24" x="0px" xml:space="preserve" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" y="0px"><g id="XMLID_1_"><path d="M0,0h24v24H0V0z" fill="none"/><g id="XMLID_2_"><rect height="2" id="XMLID_3_" width="12" x="4" y="10"/><rect height="2" id="XMLID_4_" width="12" x="4" y="6"/><rect height="2" id="XMLID_5_" width="8" x="4" y="14"/><polygon id="XMLID_6_" points="14,14 14,20 19,17   "/></g></g></svg>');}
.icon.step {background-image: url('data:image/svg+xml;utf8,<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M6 18l8.5-6L6 6v12zM16 6v12h2V6h-2z"/><path d="M0 0h24v24H0z" fill="none"/></svg>');}
.icon.speed {background-image: url('data:image/svg+xml;utf8,<svg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="0 0 87.607 87.607" style="enable-background:new 0 0 87.607 87.607;" xml:space="preserve"><g><g><g><path d="M52.173,57.876c-0.028-0.329-0.122-0.658-0.286-0.971L31.254,17.971c-0.609-1.144-1.983-1.684-3.229-1.267 c-1.248,0.423-1.993,1.674-1.756,2.95l8.041,43.173c0.063,0.343,0.196,0.65,0.377,0.93c0.078,0.369,0.173,0.736,0.3,1.104 c1.295,3.694,4.838,6.184,8.813,6.184c1.027,0,2.048-0.166,3.024-0.497c2.354-0.792,4.253-2.438,5.347-4.636 c1.095-2.198,1.251-4.68,0.444-6.994C52.492,58.555,52.34,58.21,52.173,57.876z M43.803,19.533c-1.471,0-2.662,1.17-2.662,2.613 c0,1.443,1.191,2.614,2.662,2.614c21.218,0,38.476,16.961,38.476,37.806c0,1.442,1.195,2.613,2.664,2.613 c1.473,0,2.664-1.171,2.664-2.613C87.605,38.834,67.956,19.533,43.803,19.533z M16.276,29.087C5.933,37.306,0,49.507,0,62.565 c0,1.443,1.191,2.615,2.663,2.615c1.474,0,2.665-1.172,2.665-2.615c0-11.47,5.211-22.189,14.298-29.409 c1.146-0.908,1.32-2.556,0.397-3.679C19.1,28.355,17.423,28.179,16.276,29.087z"/></g></g></g></svg>');}

body.playing #play {display:none}
body:not(.playing) #pause {display:none}


#clock, #pc {
  font-family: "Lucida Console", Monaco, monospace;
  font-size: 1.5em;
}

#exec-tbl {
  width: 100%;
  height: 490px;
  border-collapse: collapse;
  border-spacing: 0;
}

#exec-tbl table caption,
#exec-tbl table th {
  padding-bottom: 2px;
}
#exec-tbl table td {
  border: 1px solid #000;
  background-color: #dfdfdf;
}

#exec-tbl > tr > td:nth-child(2),
#exec-tbl > tr > td:nth-child(3) {
  width: 50%;
  height: 50%;
}

#exec-tbl td {
  vertical-align: top;
}

#exec-tbl table {
  border-spacing: 0;
  border-collapse: collapse;
  margin: auto;
}

#exec-tbl table td {
  font-size: 14px;
  font-family: "Lucida Console", Monaco, monospace;
  padding: 5px 4px 4px 4px;
  min-width:32px;
  text-align: center;
}

#exec-tbl table caption, #exec-tbl table thead th {
  text-transform: capitalize;
}

#exec-tbl table tbody > tr:nth-child(odd) > td {background-color: #fff;}
/* #exec-tbl table tbody > tr:nth-child(even) > td {background: #efefef;} */

#exec-tbl td table  {
  vertical-align: top
}

#sourcecode tr > td:nth-child(2),
#scexec tr > td:nth-child(2),
#rob tr > td:nth-child(2) {
  width: 111px;
  text-align: left;
}

#tab-exec .tab-body {
  padding: 20px 0 10px 0;
}

.scrollable {
  width: 100%;
  height: 100%;
  margin: 0;
  padding: 0;
  overflow-y: scroll;
  overflow-x: hidden;
}

#cache tr > td:nth-child(2),
#cache tr > td:nth-child(3),
#cache tr > td:nth-child(5),
#cache tr > td:nth-child(6) {
  border-right: transparent;
  border-left: transparent;
}
#cache tr > td:nth-child(4),
#cache tr > td:nth-child(7) {
  border-left: transparent;
}


.tick, .busy {
  background-image: url('data:image/svg+xml;utf8,<svg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" viewBox="0 0 1000 1000" enable-background="new 0 0 1000 1000" xml:space="preserve"><g><path d="M990,190.6l-65.5-63.2L335.2,743.9L73.2,493.2L10,561l327.4,311.6l63.2-67.7l0,0L990,190.6z"/></g></svg>');
  background-repeat: no-repeat;
  background-size: 60% 60%;
  background-position: center;
}

tr.current td { background-color: #ffec5e !important; }

.new-val {
  color: mediumvioletred;
  font-weight: bold;
}

.flushed {text-decoration: line-through}

.act-btn {
  position: absolute;
  bottom: 15px;
  left: 400px;
  -ms-transform: translate(-50%,0);
  -webkit-transform: translate(-50%,0);
  transform: translate(-50%,0);
}

#reg tbody tr td:nth-child(even) {border-left: 0;}
#reg tbody tr td:nth-child(odd) {border-right: 0;}
        </style>
    </head>
    <body>
        <div id="main">
            <section id="tab-source" class="tab">
                <a href="#tab-source" class="tab-link">source code</a>
                <section class="tab-body">
                    <header>
                        <h3>Please type your program or select an example.</h3>
                        <span>Examples: </span><ul id="exlist"></ul>
                    </header>

                    <div id="raw-wrap"><!--
                        --><div>
                            <h4>Source</h4>
                            <textarea class="tech" id="raw-src" cols="25" rows="17"></textarea>
                        </div><!--
                        --><div id="help">
                            <h4>Instruction set</h4>
                            <ul>
                                <li><p>Arithmetic</p>
                                    &nbsp;&nbsp;&nbsp;&nbsp;OP1, OP2 <span class="txt">:= Register or Immediate value.</span><br>
                                    &nbsp;&nbsp;&nbsp;&nbsp;DST <span class="txt">:= Register.</span>
                                    <ul>
                                        <li>ADD OP1, OP2, DST</li>
                                        <li>SUB OP1, OP2, DST</li>
                                        <li>MUL OP1, OP2, DST</li>
                                        <li>DIV OP1, OP2, DST</li>
                                    </ul>
                                </li>

                                <li><p>Flow control</p>
                                    &nbsp;&nbsp;&nbsp;&nbsp;ADDR<span class="txt">:= Instruction address (row #).</span>
                                    <ul>
                                        <li>JMP ADDR</li>
                                        <li>JZ &nbsp;ADDR</li>
                                        <li>JNZ ADDR</li>
                                    </ul>
                                </li>

                                <li><p>Memory interface</p>
                                    &nbsp;&nbsp;&nbsp;&nbsp;<span class="txt">REG := Register</span><br>
                                    &nbsp;&nbsp;&nbsp;&nbsp;<span class="txt">VAL, ADDR := Register or Immediate value.</span>
                                    <ul>
                                        <li>LDR ADDR, REG<br>
                                            &nbsp;&nbsp;&nbsp;&nbsp;<span class="txt">Page size := 1.<br>
                                        </li>
                                        <li>STR VAL, ADDR</li>
                                    </ul>
                                </li>

                                <li><p>Extra</p>
                                    <ul>
                                        <li>; comment</li>
                                    </ul>
                                </li>
                            </ul>
                        </div><!--
                    --></div>
                    <a href="#" id="rdy" class="act-btn btn blu">load</a>
                </section>
            </section>

            <section id="tab-config" class="tab">
                <a href="#tab-config" class="tab-link">configuration</a>
                <section class="tab-body">
                    <table id="ctable">
                        <tr>
                            <td>
                                <div>
                                    <header>Register File</header>
                                    <table class="inner-table">
                                        <tr>
                                            <td>registers</td>
                                            <td># <input type="number" min="0" id="ri" value="8" size="2" max="99"></td>
                                        </tr>
                                    </table>
                                </div>
                            </td>
                            <td>
                                <div>
                                    <header>Functional Units</header>
                                    <table class="inner-table">
                                        <tr class="row3">
                                            <td>adders</td>
                                            <td># <input type="number" min="0" id="iaddr" value=3 size=2 max="99"></td>
                                            <td>latency <input type="number" min="0" id="iaddrd" value=2 size=2 max="99"></td>
                                        </tr>
                                        <tr class="row3">
                                            <td>multipliers</td>
                                            <td># <input type="number" min="0" id="imult" value=2 size=2 max="99"></td>
                                            <td>latency <input type="number" min="0" id="imultd" value=4 size=2 max="99"></td>
                                        </tr>
                                        <tr class="row3">
                                            <td>divider</td>
                                            <td># <input type="number" min="0" id="idiv" value=1 size=2 max="99"></td>
                                            <td>latency <input type="number" min="0" id="idivd" value=6 size=2 max="99"></td>
                                        </tr>
                                    </table>
                                </div>
                            </td>
                        </tr>
                        <tr>
                            <td>
                                <div>
                                    <header>Memory</header>
                                    <table class="inner-table">
                                        <tr><td>read latency</td><td><input type="number" min="0" id="rl" value="10" size="2" max="99"></td></tr>
                                        <tr><td>write latency</td><td><input type="number" min="0" id="wl" value="15" size="2" max="99"></td></tr>
                                    </table>
                                </div>
                            </td>
                            <td>
                                <div>
                                    <header>Cache</header>
                                    <table class="inner-table">
                                        <tr>
                                            <td>algorithm</td>
                                            <td><select id="cache_alg"><option value="no-cache">no cache</option><option value="nwayset_wb" selected>n-way w.back</option><option value="nwayset_wt">n-way w.through</option></select></td>
                                        </tr>
                                        <tr>
                                            <td>N (ways) <input type="number" min="0" id="nways" value=2 size=2 max="99"></td>
                                            <td>size <input type="number" min="0" id="csize" value=4 size=2 max="99"></td>
                                        </tr>
                                        <tr>
                                            <td>read latency <input type="number" min="0" id="crl" value=0 size=2 max="99"></td>
                                            <td>write latency <input type="number" min="0" id="cwl" value=0 size=2 max="99"></td>
                                        </tr>
                                    </table>
                                </div>
                            </td>
                        </tr>
                        <tr>
                            <td>
                                <div>
                                    <header>Pipeline</header>
                                    <table class="inner-table">
                                        <tr><td>issue-exec delay</td><td><input id="ied" type="checkbox" checked></td></tr>
                                        <tr><td>exec-write delay</td><td><input id="ewd" type="checkbox" checked></td></tr>
                                        <tr><td>ROB (<small>0 = disabled</small>)</td><td>size <input type="number" min="0" id="rsize" value=8 size=2 max="99"></td>
                                    </table>
                                </div>
                            </td>
                            <td>
                                <div>
                                    <header>Branch Predictor</header>
                                    <table class="inner-table">
                                        <tr>
                                            <td>kind</td>
                                            <td><select id="bp"><optgroup label="Static predictors"><option value="non">Non speculative</option><option value="at">Always taken</option><option value="ant">Always not taken</option><option value="btfnt">BTFNT</option></optgroup><optgroup label="Dinamic predictors"><option value="nbit" selected>n-bit</option></optgroup></select></td>
                                        </tr>
                                        <tr><td>N (bits)</td><td><input type="number" id="nbit_n" value=2 size=2 min=0 max="99"></td></tr>
                                        <tr><td>K (size)</td><td><input type="number" id="nbit_k" value=4 size=2 min=0 max="99"></td></tr>
                                    </table>
                                </div>
                            </td>
                        </tr>
                    </table>
                </section>

                <input type="hidden" min="0" id="rf" value="0" size="2" max="99">
                <a href="#" id="apply_conf" class="act-btn btn blu">apply</a>
            </section>

            <section id="tab-exec" class="tab active">
                <a href="#tab-exec" class="tab-link">execution</a>
                <section class="tab-body">
                    <header id="control-bar">
                        <table>
                            <tr>
                                <td>
                                    <i class="icon clock"></i><span id="clock">0</span>
                                    &nbsp;
                                    <i class="icon pc"></i><span id="pc">0</span>
                                </td>
                                <td>
                                    <a id="reset"><i class="icon reset"></i></a>
                                    <a id="play"><i class="icon play"></i></a>
                                    <a id="pause"><i class="icon pause"></i></a>
                                    <a id="step"><i class="icon step"></i></a>
                                </td>
                                <td>
                                    <i class="icon speed"></i>
                                    <input id="speed" type="range" min="1" max="20" step="1" value="5" />
                                </td>
                            </tr>
                        </table>
                    </header>

                    <table id="exec-tbl">
                        <tr>
                            <td rowspan="2">
                                <div class="scrollable" id="scroll_source">
                                    <table id="sourcecode_tbl">
                                        <caption>Source Code</caption>
                                        <thead>
                                            <tr>
                                                <th>#</th>
                                                <th>Instruction</th>
                                            </tr>
                                        </thead>
                                        <tbody class="tech" id="sourcecode">
                                        </tbody>
                                    </table>
                                </div>
                            </td>
                            <td>
                                <div class="scrollable" id="scroll_exec">
                                    <table id="exec_tbl">
                                        <caption>Execution</caption>
                                        <thead>
                                            <tr>
                                                <th>#</th>
                                                <th>Instruction</th>
                                                <th>Issue</th>
                                                <th>Exec</th>
                                                <th>Write</th>
                                                <th>Commit</th>
                                            </tr>
                                        </thead>
                                        <tbody class="tech" id="scexec">
                                        </tbody>
                                    </table>
                                </div>
                            </td>
                            <td>
                                <div class="scrollable">
                                    <table id="rob">
                                    </table>
                                </div>
                            </td>
                        </tr>
                        <tr>
                            <td>
                                <div class="scrollable">
                                    <table>
                                        <caption>Reservation Stations</caption>
                                        <thead>
                                            <tr>
                                                <th>Name</th>
                                                <th>Busy</th>
                                                <th>Op</th>
                                                <th>V<sub>j</sub></th>
                                                <th>V<sub>k</sub></th>
                                                <th>Q<sub>j</sub></th>
                                                <th>Q<sub>k</sub></th>
                                                <th>tag</th>
                                                <th>due</th>
                                            </tr>
                                        </thead>
                                        <tbody class="tech" id="rs">
                                        </tbody>
                                    </table>
                                </div>
                            </td>
                            <td>
                                <div class="scrollable">
                                    <div>
                                        <table id="reg">
                                        </table>
                                    </div>

                                    <br>
                                    <br>

                                    <div>
                                        <table id="cache">
                                        </table>
                                    </div>
                                </div>
                            </td>
                        </tr>
                    </table>
                </section>
            </section>
            <footer>
                Made by <a href="mailto:mb@disi.unitn.it">Martin Brugnara</a> and <a href="mailto:gianluca.bortoli@unitn.it">Gianluca Bortoli</a>
                <a href="https://github.com/MartinBrugnara/archetypum" target="_blank"><i id="octocat" alt="octocat"></i></a>
            </footer>
        </div>

        <script type="text/javascript">
"use strict";
class CircularBuffer {
    constructor(size) {
        this.head = 0;
        this.tail = 0;
        this.availableData = 0;
        this.nextTag = () => this.tail;
        this[Symbol.iterator] = function (me) {
            return function* () {
                for (let idx = 0; idx < me.availableData; idx++) {
                    let i = (me.head + idx) % me.buffer.length;
                    yield [i, me.buffer[i]];
                }
            };
        }(this);
        this.reverse = function (me) {
            return function* () {
                for (let idx = me.availableData - 1; idx >= 0; idx--) {
                    let i = (me.head + idx) % me.buffer.length;
                    yield [i, me.buffer[i]];
                }
            };
        }(this);
        this.buffer = new Array(size);
    }
    isEmpty() { return this.availableData === 0; }
    isFull() { return this.availableData === this.buffer.length; }
    push(data) {
        if (this.isFull())
            return -1;
        let idx = this.tail % this.buffer.length;
        this.buffer[idx] = data;
        this.tail = (this.tail + 1) % this.buffer.length;
        this.availableData++;
        return idx;
    }
    pop() {
        if (this.isEmpty())
            return null;
        let data = this.buffer[this.head % this.buffer.length];
        this.head = (this.head + 1) % this.buffer.length;
        this.availableData--;
        return data;
    }
}
class CdbMessage {
    constructor(rsName, result, dst) {
        this.rsName = rsName;
        this.result = result;
        this.dst = dst;
    }
}
class Emulator {
    constructor(fuConf, regConf, robSize, cache, memMgm, IU, program) {
        this.cache = cache;
        this.memMgm = memMgm;
        this.IU = IU;
        this.program = program;
        this.clock = 0;
        this.pc = 0;
        this.uid = 0;
        this.useRob = false;
        this.hist = [];
        this.REG = new Register(regConf);
        this.FUs = FuFactory(fuConf);
        if (robSize) {
            this.ROB = new Rob(robSize, memMgm, IU);
            this.useRob = true;
        }
    }
    step() {
        this.CDB = new Queue();
        this.clock += 1;
        if (this.pc < this.program.length) {
            let rawInst = this.program[this.pc];
            let re = new RobEntry(rawInst, rawInst.dst, this.uid);
            let inst = this.REG.patch(rawInst, this.useRob ? this.ROB.patcher : this.REG.patcher, this.uid);
            if (!this.useRob || !this.ROB.isFull()) {
                if (this.useRob)
                    inst.tag = this.ROB.nextTag();
                let issued = false;
                let name = '';
                if (OpKindMap[inst.op] === FuKind.IU) {
                    if (this.IU.speculative ||
                        (!this.IU.speculative && this.ROB.isEmpty()) ||
                        inst.op === Op.JMP) {
                        this.pc = this.IU.nextPc(rawInst, this.REG.FLAGS);
                        re.ready = this.clock;
                        re.value = this.pc;
                        issued = true;
                    }
                    name = 'IU';
                }
                else {
                    for (let fu of this.FUs) {
                        if (fu.tryIssue(this.clock, inst)) {
                            issued = true;
                            this.pc++;
                            name = fu.name;
                            break;
                        }
                    }
                }
                if (issued) {
                    this.hist.push(clone(rawInst));
                    this.hist[this.uid].issued = this.clock;
                    if (this.useRob) {
                        this.ROB.push(re);
                    }
                    else {
                        this.REG.setProducer(inst, name);
                    }
                    this.uid++;
                }
            }
        }
        for (let fu of this.FUs) {
            let rowid = fu.execute(this.clock);
            if (rowid >= 0)
                this.hist[rowid].executed = this.clock;
        }
        for (let fu of this.FUs) {
            let rowid = fu.writeResult(this.clock, this.CDB);
            if (rowid >= 0)
                this.hist[rowid].written = this.clock;
        }
        for (let fu of this.FUs)
            fu.readCDB(this.CDB);
        if (this.useRob) {
            this.ROB.readCDB(this.clock, this.CDB);
            let res = this.ROB.commit(this.clock, this.REG);
            if (res.uid !== -1)
                this.hist[res.uid].committed = this.clock;
            if (res.flush !== -1) {
                for (let re of this.ROB.cb)
                    this.hist[re[1].uid].flushed = true;
                this.ROB.flush();
                for (let fu of this.FUs)
                    fu.flush();
                this.memMgm.flush();
                this.pc = res.flush;
            }
        }
        else {
            this.REG.readCDB(this.CDB);
        }
        for (let fu of this.FUs)
            if (fu.isBusy())
                return true;
        if (this.useRob && !this.ROB.isEmpty())
            return true;
        return this.pc < this.program.length;
    }
}
var FuKind;
(function (FuKind) {
    FuKind[FuKind["ADDER"] = 0] = "ADDER";
    FuKind[FuKind["MULTIPLIER"] = 1] = "MULTIPLIER";
    FuKind[FuKind["DIVIDER"] = 2] = "DIVIDER";
    FuKind[FuKind["MEMORY"] = 3] = "MEMORY";
    FuKind[FuKind["IU"] = 4] = "IU";
})(FuKind || (FuKind = {}));
let FuMap = {};
class FunctionalUnitBaseClass {
    constructor(kind, name) {
        this.kind = kind;
        this.name = name;
        this.instr = null;
        this.issuedTime = -1;
        this.endTime = -1;
    }
    getDue(clockTime) {
        let due = this.endTime >= clockTime ?
            (this.endTime - clockTime - 1) :
            this.duration;
        return String(due);
    }
    getInstr() {
        return this.instr;
    }
    tryIssue(clockTime, instr) {
        if (this.kind !== instr.kind() || this.isBusy())
            return false;
        this.instr = instr;
        this.issuedTime = clockTime;
        return true;
    }
    execute(clockTime) {
        if (this.isBusy() && this.isReady()
            && (!this.endTime || this.endTime < clockTime)
            && (clockTime >= (this.issuedTime + Number(ISSUE_EXEC_DELAY)))) {
            this.endTime = clockTime + this.duration + Number(EXEC_WRITE_DELAY);
            return this.instr.uid;
        }
        return -1;
    }
    computeValue() {
        throw new Error('Implement in child');
    }
    writeResult(clockTime, cdb) {
        if (this.isBusy && this.endTime === clockTime) {
            this.computeValue();
            cdb.push(new CdbMessage(this.instr.tag || this.name, this.result, this.instr.dst));
            let uid = this.instr.uid;
            this.instr = null;
            return uid;
        }
        return -1;
    }
    isBusy() {
        return !!this.instr;
    }
    isReady() {
        return this.instr.qj === null && this.instr.qk === null;
    }
    readCDB(cdb) {
        if (this.instr === null)
            return;
        if (!this.isBusy || (this.instr.qj === null && this.instr.qk === null))
            return;
        for (let msg of cdb) {
            if (this.instr.qj !== null && this.instr.qj === msg.rsName) {
                this.instr.vj = msg.result;
                this.instr.qj = null;
            }
            if (this.instr.qk !== null && this.instr.qk === msg.rsName) {
                this.instr.vk = msg.result;
                this.instr.qk = null;
            }
        }
    }
    flush() {
        this.instr = null;
        this.issuedTime = -1;
        this.endTime = -1;
    }
}
class Adder extends FunctionalUnitBaseClass {
    constructor(name, kwargs) {
        super(FuKind.ADDER, name);
        this.duration = 2;
        if ('duration' in kwargs)
            this.duration = kwargs.duration;
    }
    computeValue() {
        switch (this.instr.op) {
            case Op.ADD:
                this.result = this.instr.vj + this.instr.vk;
                break;
            case Op.SUB:
                this.result = this.instr.vj - this.instr.vk;
                break;
        }
    }
}
FuMap[FuKind.ADDER] = (name, kwargs) => new Adder(name, kwargs);
class Multiplier extends FunctionalUnitBaseClass {
    constructor(name, kwargs) {
        super(FuKind.MULTIPLIER, name);
        this.duration = 4;
        if ('duration' in kwargs)
            this.duration = kwargs.duration;
    }
    computeValue() {
        this.result = this.instr.vj * this.instr.vk;
    }
}
FuMap[FuKind.MULTIPLIER] = (name, kwargs) => new Multiplier(name, kwargs);
class Divider extends FunctionalUnitBaseClass {
    constructor(name, kwargs) {
        super(FuKind.DIVIDER, name);
        this.duration = 4;
        if ('duration' in kwargs)
            this.duration = kwargs.duration;
    }
    computeValue() {
        this.result = Math.floor(this.instr.vj / this.instr.vk);
    }
}
FuMap[FuKind.DIVIDER] = (name, kwargs) => new Divider(name, kwargs);
function FuFactory(conf) {
    let fus = [];
    for (let fuc of conf) {
        for (let i = 0; i < fuc[2]; i++) {
            fus.push(FuMap[fuc[0]](`${fuc[1]}${i}`, fuc[3]));
        }
    }
    return fus;
}
class MemoryFU extends FunctionalUnitBaseClass {
    constructor(name, kwargs) {
        super(FuKind.MEMORY, name);
        this.waiting = false;
        this.startTime = null;
        this.duration = 0;
        this.memMgm = kwargs['memMgm'];
        this.duration = kwargs['duration'];
    }
    execute(clockTime) {
        let starting = false;
        if (this.isBusy() && this.isReady() &&
            (!this.endTime || this.endTime < clockTime) &&
            (clockTime >= (this.issuedTime + Number(ISSUE_EXEC_DELAY))) &&
            !this.waiting) {
            this.waiting = true;
            starting = true;
        }
        if (this.waiting && (clockTime >= (this.issuedTime + Number(ISSUE_EXEC_DELAY)) +
            (this.instr.op === Op.LOAD && this.instr.vk !== 0 ? this.duration : 0))) {
            let done = false;
            switch (this.instr.op) {
                case Op.LOAD:
                    let value = this.memMgm.read(this.name, clockTime, this.instr.vj + this.instr.vk);
                    if (value !== null) {
                        this.result = value;
                        done = true;
                    }
                    break;
                case Op.STORE:
                    done = this.memMgm.write(this.name, clockTime, this.instr.vj, this.instr.vk);
                    this.instr.dst = String(this.instr.vk);
                    this.result = this.instr.vj;
                    break;
            }
            if (done) {
                this.endTime = clockTime + Number(EXEC_WRITE_DELAY);
                this.waiting = false;
            }
        }
        return starting ? this.instr.uid : -1;
    }
    computeValue() {
    }
    flush() {
        super.flush();
        this.waiting = false;
        this.startTime = null;
        this.memMgm.flush();
    }
}
FuMap[FuKind.MEMORY] = (name, kwargs) => new MemoryFU(name, kwargs);
class Graphics {
    constructor(emu) {
        this.emu = emu;
        this.clk = document.getElementById('clock');
        this.pc = document.getElementById('pc');
        this.src = document.getElementById('sourcecode');
        this.scexec = document.getElementById('scexec');
        this.rs = document.getElementById('rs');
        this.reg = document.getElementById('reg');
        this.cache = document.getElementById('cache');
        this.rob = document.getElementById('rob');
        this.scrl_source = document.getElementById('scroll_source');
        this.scrl_source_tbl = document.getElementById('sourcecode_tbl');
        this.scrl_exec = document.getElementById('scroll_exec');
        this.scrl_exec_tbl = document.getElementById('exec_tbl');
    }
    paint() {
        this.clk.innerHTML = String(this.emu.clock);
        this.pc.innerHTML = String(this.emu.pc);
        this.src.innerHTML = this.renderSrc();
        this.scexec.innerHTML = this.renderExec();
        this.rs.innerHTML = this.renderRS();
        this.reg.innerHTML = this.renderREG();
        this.cache.innerHTML = this.renderCache();
        this.rob.innerHTML = this.renderRob();
        this.scroll();
    }
    scroll() {
        this.scrl_exec.scrollTop = this.scrl_exec_tbl.clientHeight;
        let row = this.scrl_source_tbl.querySelector('.current');
        if (row === undefined || row === null)
            return;
        let innerTop = row.getBoundingClientRect().top;
        let outerTop = this.scrl_source_tbl.getBoundingClientRect().top;
        let paneHeight = this.scrl_source.clientHeight;
        let scroll = (innerTop - outerTop) - ((paneHeight - row.clientHeight) / 2);
        this.scrl_source.scrollTop = scroll;
    }
    renderSrc() {
        let rowid = 0;
        let html = [];
        for (let i of this.emu.program) {
            html.push([
                '<tr',
                (this.emu.pc === rowid ? ' class="current"' : ''),
                '>',
                '<td>', String(rowid++), '</td>',
                '<td>', i.toString(), '</td>',
            ]);
        }
        html.push([
            '<tr',
            (this.emu.pc === this.emu.program.length ? ' class="current"' : ''),
            '>',
            '<td>', String(this.emu.program.length), '</td>',
            '<td>', 'EOF', '</td>',
        ]);
        return Array.prototype.concat.apply([], html).join('');
    }
    renderExec() {
        let rowid = 0;
        let html = [];
        for (let i of this.emu.hist) {
            html.push([
                '<tr',
                i.flushed ? ' class="flushed"' : '',
                '>',
                '<td>', String(rowid++), '</td>',
                '<td>', i.toString(), '</td>',
                '<td', (i.issued === this.emu.clock ? ' class="new-val"' : ''), '>',
                String(i.issued >= 0 ? i.issued : ''), '</td>',
                '<td', (i.executed === this.emu.clock ? ' class="new-val"' : ''), '>',
                String(i.executed >= 0 ? i.executed : ''), '</td>',
                '<td', (i.written === this.emu.clock ? ' class="new-val"' : ''), '>',
                String(i.written >= 0 ? i.written : ''), '</td>',
                '<td', (i.committed === this.emu.clock ? ' class="new-val"' : ''), '>',
                String(i.committed >= 0 ? i.committed : ''), '</td>',
                '</tr>',
            ]);
        }
        return Array.prototype.concat.apply([], html).join('');
    }
    renderRS() {
        let html = [];
        for (let f of this.emu.FUs) {
            let instr = f.getInstr();
            html.push([
                '<tr>',
                '<td>', f.name, '</td>',
                '<td', (f.isBusy() ? ' class="busy"' : ''), '></td>',
            ], (instr !== null ? [
                '<td>', instr.op.toString(), '</td>',
                '<td>', (instr.qj === null ? String(instr.vj) : ''), '</td>',
                '<td>', (instr.qk === null ? String(instr.vk) : ''), '</td>',
                '<td>', (instr.qj !== null ? instr.qj : ''), '</td>',
                '<td>', (instr.qk !== null ? instr.qk : ''), '</td>',
                '<td>', (instr.tag !== null ? instr.tag : ''), '</td>',
                '<td>', f.kind !== FuKind.MEMORY ? f.getDue(this.emu.clock) : '', '</td>'
            ] : [
                '<td></td><td></td><td></td><td></td><td></td><td></td><td></td>',
            ]), ['</tr>']);
        }
        return Array.prototype.concat.apply([], html).join('');
    }
    renderREG() {
        let html = [];
        var body = [];
        let tagMap = {};
        if (this.emu.useRob)
            for (let row of this.emu.ROB.cb)
                tagMap[row[1].dst] = String(row[0]);
        html.push(['<caption>Register Status (Q<sub>i</sub>)</caption>'], ['<thead><tr>']);
        for (let key in this.emu.REG.regs) {
            html.push(['<th colspan=2>', key, '</th>']);
            body.push([
                '<td>',
                (this.emu.REG.qi[key] === null ? String(this.emu.REG.regs[key]) : this.emu.REG.qi[key]),
                '</td>',
                '<td>',
                key in tagMap ? 'T' + tagMap[key] : '-',
                '</td>'
            ]);
            if (!(body.length % 4)) {
                html.push(['</tr></thead><tbody class="tech"><tr>'], Array.prototype.concat.apply([], body), ['</tr></tbody>'], ['<thead><tr>']);
                body = [];
            }
        }
        html.push(['</tr></thead><tbody class="tech"><tr>'], Array.prototype.concat.apply([], body), ['</tr></tbody>']);
        if (!body.length)
            html.splice(-4, 4);
        return Array.prototype.concat.apply([], html).join('');
    }
    renderCache() {
        if (this.emu.cache.size === 0)
            return "";
        let html = [];
        html.push(['<caption>cache</caption><thead><tr><th></th>']);
        for (let j = 0; j < this.emu.cache.n; j++)
            html.push(['<th colspan="3">N:', String(j), '</th>']);
        html.push(['</tr><tr><th></th>']);
        for (let j = 0; j < this.emu.cache.n; j++)
            html.push(['<th>addr</th><th>val</th><th>dirty</th>']);
        html.push(['</tr></thead>']);
        html.push(['<tbody>']);
        for (let i = 0; i < this.emu.cache.size / this.emu.cache.n; i++) {
            html.push(['<tr><td>', String(i), '</td>']);
            for (let j = 0; j < this.emu.cache.n; j++) {
                let entry = this.emu.cache._cache[i][j];
                if (entry[0])
                    html.push([
                        '<td>', String(entry[1]), '</td>',
                        '<td>', String(entry[2]), '</td>',
                        '<td', entry[3] ? ' class="busy">' : '>', '</td>'
                    ]);
                else
                    html.push(['<td></td><td></td><td></td>']);
            }
            html.push(['</tr>']);
        }
        html.push(['</tbody>']);
        html.push([
            '<tfoot><tr><td colspan=7>',
            'hit&nbsp;',
            String(Math.round(this.emu.cache.readHit / (this.emu.cache.readHit + this.emu.cache.readMiss) * 100)),
            '% - evictions ', String(this.emu.cache.evictions),
            '</td></tr><tfoot>'
        ]);
        return Array.prototype.concat.apply([], html).join('');
    }
    renderRob() {
        if (!this.emu.useRob)
            return '';
        let html = [];
        html.push(['<caption>reorder buffer</caption><thead><tr><th>tag</th><th>Instruction</th><th>dst</th><th>val</th><th>rdy</th><th>row</th></tr></thead><tbody>']);
        for (let row of this.emu.ROB.cb) {
            html.push([
                '<tr>',
                '<td>', String(row[0]), '</td>',
                '<td>', String(row[1].instr), '</td>',
                '<td>', row[1].dst, '</td>',
                '<td>', String(row[1].value), '</td>',
                '<td', row[1].ready !== null ? ' class="busy">' : '>', '</td>',
                '<td>', String(row[1].instr.rowid), '</td>',
                '</tr>',
            ]);
        }
        for (let i = 0; i < this.emu.ROB.cb.buffer.length - this.emu.ROB.cb.availableData; i++) {
            let tag = (this.emu.ROB.cb.tail + i) % this.emu.ROB.cb.buffer.length;
            html.push([
                '<tr><td>',
                String(tag),
                '</td><td></td><td></td><td></td><td></td><td></td></tr>',
            ]);
        }
        html.push(['</tbody>']);
        return Array.prototype.concat.apply([], html).join('');
    }
}
class RawInstruction {
    constructor(op, src0, src1, dst, rowid) {
        this.op = op;
        this.src0 = src0;
        this.src1 = src1;
        this.dst = dst;
        this.rowid = rowid;
        this.issued = -1;
        this.executed = -1;
        this.written = -1;
        this.committed = -1;
        this.flushed = false;
    }
    toString() {
        if (this.op === Op.LOAD)
            return `${OpString[this.op]} ${this.src0},${this.dst}`;
        return `${OpString[this.op]} ${this.src0}` +
            `${this.src1 ? ',' : ''}${this.src1}` +
            `${this.dst ? ',' : ''}${this.dst}`;
    }
}
class Instruction {
    constructor(op, dst, pc, uid, vj = 0, vk = 0, qj = null, qk = null, tag = null) {
        this.op = op;
        this.dst = dst;
        this.pc = pc;
        this.uid = uid;
        this.vj = vj;
        this.vk = vk;
        this.qj = qj;
        this.qk = qk;
        this.tag = tag;
    }
    kind() {
        return OpKindMap[this.op];
    }
}
var Op;
(function (Op) {
    Op[Op["ADD"] = 0] = "ADD";
    Op[Op["SUB"] = 1] = "SUB";
    Op[Op["MUL"] = 2] = "MUL";
    Op[Op["DIV"] = 3] = "DIV";
    Op[Op["LOAD"] = 4] = "LOAD";
    Op[Op["STORE"] = 5] = "STORE";
    Op[Op["JMP"] = 6] = "JMP";
    Op[Op["JZ"] = 7] = "JZ";
    Op[Op["JNZ"] = 8] = "JNZ";
})(Op || (Op = {}));
let OpKindMap = {};
OpKindMap[Op.ADD] = FuKind.ADDER;
OpKindMap[Op.SUB] = FuKind.ADDER;
OpKindMap[Op.MUL] = FuKind.MULTIPLIER;
OpKindMap[Op.DIV] = FuKind.DIVIDER;
OpKindMap[Op.LOAD] = FuKind.MEMORY;
OpKindMap[Op.STORE] = FuKind.MEMORY;
OpKindMap[Op.JMP] = FuKind.IU;
OpKindMap[Op.JZ] = FuKind.IU;
OpKindMap[Op.JNZ] = FuKind.IU;
let OpString = {};
OpString[Op.ADD] = "ADD";
OpString[Op.SUB] = "SUB";
OpString[Op.MUL] = "MUL";
OpString[Op.DIV] = "DIV";
OpString[Op.LOAD] = "LDR";
OpString[Op.STORE] = "STR";
OpString[Op.JMP] = "JMP";
OpString[Op.JZ] = "JZ";
OpString[Op.JNZ] = "JNZ";
let StringOp = {};
StringOp['ADD'] = Op.ADD;
StringOp['SUB'] = Op.SUB;
StringOp['MUL'] = Op.MUL;
StringOp['DIV'] = Op.DIV;
StringOp['LDR'] = Op.LOAD;
StringOp['STR'] = Op.STORE;
StringOp['JMP'] = Op.JMP;
StringOp['JZ'] = Op.JZ;
StringOp['JNZ'] = Op.JNZ;
let InstrLen = {};
InstrLen['LDR'] = 2;
InstrLen['STR'] = 2;
InstrLen['JMP'] = 1;
InstrLen['JZ'] = 1;
InstrLen['JNZ'] = 1;
function parse(src, rsize) {
    let prg = [];
    let rowid = 0;
    for (let row of src.split("\n")) {
        let crow = row.trim();
        if (!crow.length || crow.lastIndexOf(';', 0) === 0)
            continue;
        let rawcmd = crow.split(' ', 1)[0];
        let cmd = rawcmd.trim().toUpperCase();
        if (!(cmd in StringOp))
            throw new Error(`[${crow}] ${cmd} is not a valid instruction`);
        let args = crow.substring(rawcmd.length).replace(/\s+/g, '').split(',');
        if (cmd in InstrLen && InstrLen[cmd] !== args.length)
            throw new Error(`[${crow}] ${cmd} expects ${InstrLen[cmd]} operands, got ${args.length}`);
        if (!(cmd in InstrLen) && args.length !== 3)
            throw new Error(`[${crow}] ${cmd} expects 3 operands, got ${args.length}`);
        for (let arg of args) {
            let v = isNaN(parseInt(arg, 10));
            let reg = parseInt(arg.substring(1, arg.length), 10);
            if (!(!v || (arg[0] === 'R' && !isNaN(reg) && reg >= 0 && reg < rsize)))
                throw new Error(`[${crow}] ${arg} is not a valid register`);
        }
        if (StringOp[cmd] === Op.LOAD)
            prg.push(new RawInstruction(StringOp[cmd], args[0], '0', args[1], rowid++));
        else
            prg.push(new RawInstruction(StringOp[cmd], args[0], args.length > 1 ? args[1] : "", args.length === 3 ? args[2] : "", rowid++));
    }
    return prg;
}
class MemConf {
    constructor(readLatency, writeLatency) {
        this.readLatency = readLatency;
        this.writeLatency = writeLatency;
    }
}
class Memory {
    constructor(c) {
        this.mem = {};
        this.currentOpComplete = -1;
        this.readLatency = 0;
        this.writeLatency = 0;
        this.readLatency = c.readLatency;
        this.writeLatency = c.writeLatency;
    }
    flush() {
        this.currentOpComplete = -1;
    }
    isBusy() {
        return this.currentOpComplete !== -1;
    }
    read(clock, loc) {
        if (this.currentOpComplete === -1) {
            this.currentOpComplete = clock + this.readLatency;
        }
        else if (clock >= this.currentOpComplete) {
            this.currentOpComplete = -1;
            return this._read(loc);
        }
        return null;
    }
    write(clock, loc, value) {
        if (this.currentOpComplete === -1) {
            this.currentOpComplete = clock + this.writeLatency;
        }
        else if (clock === this.currentOpComplete) {
            this.currentOpComplete = -1;
            this._write(loc, value);
            return true;
        }
        return false;
    }
    _read(loc) {
        if (this.mem[loc] === undefined) {
            this.mem[loc] = Math.floor(Math.random() * 100);
        }
        return this.mem[loc];
    }
    _write(loc, value) {
        this.mem[loc] = value;
    }
}
let XCacheMap = {};
function XCacheFactory(name, c) {
    if (name in XCacheMap)
        return XCacheMap[name](c);
    else
        return new XCache(c);
}
class XCache {
    constructor(c) {
        this._cache = {};
        this.readMiss = 0;
        this.readHit = 0;
        this.evictions = 0;
        this.working = false;
        if (c['mem'] !== undefined)
            this.mem = c.mem;
    }
    isBusy() {
        return this.working;
    }
    flush() {
        this.working = false;
    }
    read(clock, loc) {
        let value = this.mem.read(clock, loc);
        this.working = value === null;
        return value;
    }
    write(clock, loc, value) {
        let finished = this.mem.write(clock, loc, value);
        this.working = !finished;
        return finished;
    }
}
XCacheMap["no-cache"] = (c) => new XCache(c);
class NWayCache extends XCache {
    constructor(c) {
        super(c);
        this.currentOpComplete = -1;
        this.miss = false;
        this.writing = null;
        this.n = 'n' in c ? c['n'] : 2;
        this.size = 'size' in c ? c['size'] : 4;
        this.isWriteBack = 'iswriteback' in c ? c['iswriteback'] : false;
        this.readLatency = 'readLatency' in c ? c['readLatency'] : 0;
        this.writeLatency = 'writeLatency' in c ? c['readLatency'] : 0;
        if (this.size % this.n !== 0)
            throw new Error("Invalid (N,Size) paraemters combination");
        for (let i = 0; i < this.size / this.n; i++) {
            this._cache[i] = [];
            for (let j = 0; j < this.n; j++)
                this._cache[i].push([false, 0, 0, false]);
        }
    }
    flush() {
        super.flush();
        this.writing = null;
        this.currentOpComplete = -1;
        this.miss = false;
    }
    isBusy() {
        return this.currentOpComplete !== -1;
    }
    findValue(loc) {
        let i = loc % this.n;
        for (let j = 0; j < this.n; j++) {
            let entry = this._cache[i][j];
            if (entry[0] && entry[1] === loc) {
                return entry[2];
            }
        }
        return null;
    }
    setValue(loc, value, dirty = false) {
        let i = loc % this.n;
        for (let j = 0; j < this.n; j++) {
            let entry = this._cache[i][j];
            if (entry[0] && entry[1] === loc) {
                this._cache[i][j] = [true, loc, value, dirty];
                return null;
            }
        }
        for (let j = 0; j < this.n; j++) {
            let entry = this._cache[i][j];
            if (!entry[0]) {
                this._cache[i][j] = [true, loc, value, dirty];
                return null;
            }
        }
        let j = Math.floor(Math.random() * this.n);
        let entry = this._cache[i][j];
        this._cache[i][j] = [true, loc, value, dirty];
        if (entry[3] === true) {
            this.evictions++;
            return entry;
        }
        return null;
    }
    read(clock, loc) {
        if (this.currentOpComplete === -1) {
            this.currentOpComplete = clock + this.readLatency;
        }
        if (clock === this.currentOpComplete) {
            let value = this.findValue(loc);
            if (value !== null) {
                this.currentOpComplete = -1;
                this.readHit++;
                return value;
            }
            else {
                this.readMiss++;
                this.miss = true;
            }
        }
        if (this.miss) {
            let value = this.mem.read(clock, loc);
            if (value !== null) {
                this.miss = false;
                let evicted = this.setValue(loc, value);
                if (evicted !== null && this.isWriteBack) {
                    this.result = value;
                    this.writing = evicted;
                }
                else {
                    this.currentOpComplete = -1;
                    return value;
                }
            }
        }
        if (this.writing !== null) {
            if (this.mem.write(clock, this.writing[1], this.writing[2])) {
                let retval = this.result;
                this.result = null;
                this.writing = null;
                this.currentOpComplete = -1;
                return retval;
            }
        }
        return null;
    }
    write(clock, loc, value) {
        if (this.currentOpComplete === -1)
            this.currentOpComplete = clock + this.writeLatency;
        if (clock < this.currentOpComplete)
            return false;
        if (clock === this.currentOpComplete) {
            let evicted = this.setValue(loc, value, true);
            if (evicted !== null && this.isWriteBack) {
                this.writing = evicted;
            }
            if (!this.isWriteBack) {
                this.writing = [true, loc, value, true];
            }
        }
        if (this.writing !== null) {
            if (this.mem.write(clock, this.writing[1], this.writing[2])) {
                this.writing = null;
            }
        }
        if (this.writing === null) {
            this.currentOpComplete = -1;
            return true;
        }
        else {
            return false;
        }
    }
}
XCacheMap["nwayset"] = (c) => new NWayCache(c);
var MgmIntState;
(function (MgmIntState) {
    MgmIntState[MgmIntState["FREE"] = 0] = "FREE";
    MgmIntState[MgmIntState["READ"] = 1] = "READ";
    MgmIntState[MgmIntState["WRITE"] = 2] = "WRITE";
})(MgmIntState || (MgmIntState = {}));
;
class MemoryMGM {
    constructor(cache, useRob) {
        this.cache = cache;
        this.useRob = useRob;
        this.state = MgmIntState.FREE;
        this.currFU = '';
    }
    flush() {
        this.state = MgmIntState.FREE;
        this.currFU = '';
        this.cache.flush();
    }
    read(funame, clock, loc) {
        if (this.state !== MgmIntState.FREE && funame !== this.currFU)
            return null;
        if (this.useRob) {
            for (let entry of this.rob.cb.reverse()) {
                if (String(loc) === entry[1].dst)
                    return entry[1].value;
            }
        }
        if (!this.cache.isBusy()) {
            this.state = MgmIntState.READ;
            this.currFU = funame;
        }
        if (this.state === MgmIntState.READ) {
            let value = this.cache.read(clock, loc);
            if (value !== null)
                this.state = MgmIntState.FREE;
            return value;
        }
        return null;
    }
    write(funame, clock, loc, value, commit = false) {
        if (this.state !== MgmIntState.FREE && funame !== this.currFU)
            return false;
        if (this.useRob && !commit)
            return true;
        if (!this.cache.isBusy()) {
            this.state = MgmIntState.WRITE;
            this.currFU = funame;
        }
        if (this.state === MgmIntState.WRITE) {
            if (this.cache.write(clock, loc, value)) {
                this.state = MgmIntState.FREE;
            }
            return this.state === MgmIntState.FREE;
        }
        return false;
    }
}
class Queue {
    constructor() {
        this._store = [];
    }
    [Symbol.iterator]() {
        return this._store[Symbol.iterator]();
    }
    push(val) {
        this._store.push(val);
    }
    pop() {
        return this._store.shift();
    }
}
var Flag;
(function (Flag) {
    Flag[Flag["ZF"] = 0] = "ZF";
    Flag[Flag["NFLAGS"] = 1] = "NFLAGS";
})(Flag || (Flag = {}));
class Register {
    constructor(conf) {
        this.regs = {};
        this.qi = {};
        this.FLAGS = [];
        this.patcher = function (me, reg) {
            return [me.regs[reg], me.qi[reg]];
        };
        for (let i = 0; i < conf.ints; i++) {
            this.regs[`R${i}`] = 0;
            this.qi[`R${i}`] = null;
        }
        for (let f = 0; f < conf.floats; f++) {
            this.regs[`F${f}`] = 0;
            this.qi[`F${f}`] = null;
        }
        for (let f = 0; f < Flag.NFLAGS; f++)
            this.FLAGS.push(false);
    }
    patch(ri, qfunc, uid) {
        let ins = new Instruction(ri.op, ri.dst, ri.rowid, uid);
        let value = parseInt(ri.src0, 10);
        if (isNaN(value)) {
            let res = qfunc(this, ri.src0), v = res[0], remote = res[1];
            if (remote === null) {
                value = v;
            }
            else {
                value = 0;
                ins.qj = remote;
            }
        }
        ins.vj = value;
        if (ri.src1.length !== 0) {
            value = parseInt(ri.src1, 10);
            if (isNaN(value)) {
                let res = qfunc(this, ri.src1);
                let v = res[0], remote = res[1];
                if (remote === null) {
                    value = v;
                }
                else {
                    value = 0;
                    ins.qk = remote;
                }
            }
            ins.vk = value;
        }
        return ins;
    }
    setProducer(inst, rs) {
        if (this.regs[inst.dst] === undefined)
            return;
        this.regs[inst.dst] = 0;
        this.qi[inst.dst] = rs;
    }
    readCDB(cdb) {
        for (let msg of cdb) {
            if (this.qi[msg.dst] === msg.rsName) {
                this.regs[msg.dst] = msg.result;
                this.qi[msg.dst] = null;
            }
        }
    }
}
class Rob {
    constructor(size, memMgm, IU) {
        this.size = size;
        this.memMgm = memMgm;
        this.IU = IU;
        this.isFull = () => this.cb.isFull();
        this.isEmpty = () => this.cb.isEmpty();
        this.nextTag = () => String(this.cb.nextTag());
        this.push = (r) => this.cb.push(r);
        this.pop = () => this.cb.pop();
        this.patcher = function (me) {
            return function (registry, reg) {
                for (let item of me.cb.reverse()) {
                    let tag = item[0], entry = item[1];
                    if (reg === entry.dst) {
                        if (entry.ready !== null) {
                            return [entry.value, null];
                        }
                        else {
                            return [0, String(tag)];
                        }
                    }
                }
                return [registry.regs[reg], null];
            };
        }(this);
        this.cb = new CircularBuffer(size);
        memMgm.rob = this;
    }
    flush() {
        this.cb = new CircularBuffer(this.size);
        this.memMgm.rob = this;
    }
    readCDB(clock, cdb) {
        for (let msg of cdb) {
            let tag = Number(msg.rsName);
            this.cb.buffer[tag].dst = msg.dst;
            this.cb.buffer[tag].value = msg.result;
            this.cb.buffer[tag].ready = clock;
        }
    }
    setFlags(entry, reg) {
        reg.FLAGS[Flag.ZF] = entry.value === 0;
    }
    commit(clock, reg) {
        if (this.isEmpty())
            return new CommitResponse();
        let head = this.cb.buffer[this.cb.head];
        if (head.ready === null || head.ready >= clock)
            return new CommitResponse();
        if ([FuKind.ADDER, FuKind.MULTIPLIER].indexOf(OpKindMap[head.instr.op]) !== -1)
            this.setFlags(head, reg);
        if (OpKindMap[head.instr.op] === FuKind.IU)
            return new CommitResponse(this.cb.pop().uid, this.IU.validateChoice(head, reg.FLAGS));
        if (head.dst === '') {
            return new CommitResponse(this.cb.pop().uid);
        }
        else if (head.dst in reg.regs) {
            reg.regs[head.dst] = head.value;
            return new CommitResponse(this.cb.pop().uid);
        }
        else {
            if (this.memMgm.write('ROB', clock, Number(head.dst), head.value, true))
                return new CommitResponse(this.cb.pop().uid);
        }
        return new CommitResponse();
    }
}
class RobEntry {
    constructor(instr, dst, uid = -1, value = 0, ready = null) {
        this.instr = instr;
        this.dst = dst;
        this.uid = uid;
        this.value = value;
        this.ready = ready;
    }
}
class CommitResponse {
    constructor(uid = -1, flush = -1) {
        this.uid = uid;
        this.flush = flush;
    }
}
class BranchPredictor {
    constructor(N = 0, K = 0) {
        this.N = N;
        this.K = K;
        this.speculative = true;
    }
    real(instr, flags) {
        if ((instr.op === Op.JZ && flags[Flag.ZF]) || (instr.op === Op.JNZ && !flags[Flag.ZF]))
            return Number(instr.src0);
        else if (instr.op === Op.JMP)
            return Number(instr.src0);
        else
            return instr.rowid + 1;
    }
    validateChoice(head, flags) {
        let correct = this.real(head.instr, flags);
        return correct !== head.value ? correct : -1;
    }
}
let BpMap = {};
class NonSpeculative extends BranchPredictor {
    constructor() {
        super(...arguments);
        this.speculative = false;
    }
    nextPc(instr, flags) {
        return this.real(instr, flags);
    }
}
BpMap['non'] = (n, k) => new NonSpeculative(n, k);
class AlwaysTaken extends BranchPredictor {
    nextPc(instr, flags) {
        return Number(instr.src0);
    }
}
BpMap['at'] = (n, k) => new AlwaysTaken(n, k);
class AlwaysNotTaken extends BranchPredictor {
    nextPc(instr, flags) {
        return instr.rowid + 1;
    }
}
BpMap['ant'] = (n, k) => new AlwaysNotTaken(n, k);
class BTFNT extends BranchPredictor {
    nextPc(instr, flags) {
        let jaddr = Number(instr.src0);
        return jaddr < instr.rowid ? jaddr : instr.rowid + 1;
    }
}
BpMap['btfnt'] = (n, k) => new BTFNT(n, k);
class NBit extends BranchPredictor {
    constructor(N, K) {
        super(N, K);
        this.N = N;
        this.K = K;
        this.BHT = [];
        this.LastPred = [];
        this.AddrTag = [];
        for (let k = 0; k < K; k++) {
            this.BHT.push(N / 2);
            this.LastPred.push(false);
            this.AddrTag.push(-1);
        }
    }
    reset(idx) {
        this.BHT[idx] = this.N / 2;
        this.LastPred[idx] = false;
        this.AddrTag[idx] = -1;
    }
    nextPc(instr, flags) {
        let idx = instr.rowid % this.K;
        if (this.AddrTag[idx] !== instr.rowid) {
            this.reset(idx);
            this.AddrTag[idx] = instr.rowid;
        }
        let pval = this.BHT[idx];
        this.LastPred[idx] = pval > this.N / 2 || (pval === this.N / 2 && this.LastPred[idx]);
        return this.LastPred[idx] ? Number(instr.src0) : instr.rowid + 1;
    }
    validateChoice(head, flags) {
        let ret = super.validateChoice(head, flags);
        let idx = head.instr.rowid % this.K;
        if (this.AddrTag[idx] === head.instr.rowid) {
            this.BHT[idx] += this.real(head.instr, flags) === Number(head.instr.src0) ? 1 : -1;
            this.BHT[idx] = Math.max(0, Math.min(this.BHT[idx], this.N));
        }
        return ret;
    }
}
BpMap['nbit'] = (n, k) => new NBit(n, k);
function clone(obj) {
    const copy = new obj.constructor();
    Object.assign(copy, obj);
    return copy;
}
let EXAMPLES = {
    'Arithmetic speculation': `MUL 1,5,R0
MUL 2,5,R1
ADD 1,R5,R5
MUL 4,5,R3
SUB R5,5,R6
JNZ 0
`,
    'Cache': `; INIT VALUES
STR 0,0
STR 1,1

; FLUSH CACHE (4LOC)
STR 0,2
STR 0,3
STR 0,4
STR 0,5

; MAIN
LDR 0,R0
LDR 1,R1
ADD R0,R1,R2
ADD R5,1,R5
MUL R0,R1,R3
SUB R5,4,R6
STR R3,1
STR R2,0
JNZ 6
`,
};
function displayExamples() {
    let lst = document.getElementById('exlist');
    let html = [];
    for (let ex in EXAMPLES) {
        html.push([
            '<li><a data-action="load-example" data-value="', ex, '">', ex, '</a></li>'
        ]);
    }
    lst.innerHTML = Array.prototype.concat.apply([], html).join('');
    for (let btn of document.querySelectorAll('[data-action="load-example"]')) {
        btn.addEventListener('click', function (e) {
            e.preventDefault();
            e.stopPropagation();
            let example_id = e.target.dataset.value;
            raw_src.value = EXAMPLES[example_id];
        });
    }
    for (let ex in EXAMPLES) {
        raw_src.value = EXAMPLES[ex];
        break;
    }
}
function getActiveTab() {
    return document.getElementsByClassName('active')[0].id;
}
function setActiveTab(id) {
    let next = document.getElementById(id);
    if (!next)
        return;
    let tabs = document.getElementsByClassName('active');
    if (tabs.length)
        tabs[0].classList.remove('active');
    next.classList.add('active');
}
function initTabs() {
    let tab = window.location.hash.substr(1);
    setActiveTab(tab);
    for (let link of document.querySelectorAll('.tab-link')) {
        link.addEventListener('click', function (e) {
            e.preventDefault();
            e.stopPropagation();
            let tabId = e.target.getAttribute('href').replace('#', '');
            setActiveTab(tabId);
        });
    }
}
var ISSUE_EXEC_DELAY = true;
var EXEC_WRITE_DELAY = true;
let rdy = document.getElementById('rdy');
let apply_conf = document.getElementById('apply_conf');
let raw_src = document.getElementById('raw-src');
let iaddr = document.getElementById('iaddr');
let iaddrd = document.getElementById('iaddrd');
let imult = document.getElementById('imult');
let imultd = document.getElementById('imultd');
let idiv = document.getElementById('idiv');
let idivd = document.getElementById('idivd');
let ireg = document.getElementById('ri');
let freg = document.getElementById('rf');
let ied = document.getElementById('ied');
let ewd = document.getElementById('ewd');
let rl = document.getElementById('rl');
let wl = document.getElementById('wl');
let ca = document.getElementById('cache_alg');
let crl = document.getElementById('crl');
let cwl = document.getElementById('cwl');
let nways = document.getElementById('nways');
let csize = document.getElementById('csize');
let rsize = document.getElementById('rsize');
let bp = document.getElementById('bp');
let nbit_n = document.getElementById('nbit_n');
let nbit_k = document.getElementById('nbit_k');
let rst = document.getElementById('reset');
let conf = document.getElementById('conf');
let play = document.getElementById('play');
let pausebtn = document.getElementById('pause');
let one_step = document.getElementById('step');
let speed = document.getElementById('speed');
function main() {
    displayExamples();
    initTabs();
    let resetFunc = (e) => {
        e.preventDefault();
        e.stopPropagation();
        pause();
        setup();
    };
    let stepFunc = () => {
        pause();
        if (STEP)
            STEP();
        else
            alert('Please load a valid program first.');
    };
    rdy.onclick = apply_conf.onclick = rst.onclick = resetFunc;
    play.onclick = playloop;
    pausebtn.onclick = pause;
    one_step.onclick = stepFunc;
    document.addEventListener('keydown', (e) => {
        if (e.srcElement && e.srcElement.id === 'raw-src')
            return;
        if (getActiveTab() !== 'tab-exec')
            return;
        switch (e.keyCode) {
            case 34:
            case 39:
            case 78:
            case 83:
                stepFunc();
                break;
            case 33:
            case 32:
            case 13:
                if (LOOP !== -1)
                    pause();
                else
                    playloop();
                break;
            case 116:
            case 27:
            case 82:
                resetFunc(e);
                break;
            case 187:
            case 221:
                speed.value = String(parseInt(speed.value, 10) + safeInt(speed.getAttribute('step'), 1));
                break;
            case 189:
            case 219:
                speed.value = String(parseInt(speed.value, 10) - safeInt(speed.getAttribute('step'), 1));
                break;
        }
    });
    setup();
}
function playloop() {
    document.body.classList.add('playing');
    if (STEP) {
        if (STEP())
            LOOP = setTimeout(playloop, (10 / Number(speed.value) * 1000));
    }
    else {
        alert('Please load a valid program first.');
    }
}
function pause() {
    document.body.classList.remove('playing');
    if (LOOP !== -1)
        clearTimeout(LOOP);
    LOOP = -1;
}
let safeInt = (s, fallback = 0) => isNaN(parseInt(s, 10)) ? fallback : parseInt(s, 10);
var STEP = null;
var LOOP = -1;
function setup() {
    STEP = null;
    let program;
    try {
        program = parse(raw_src.value, safeInt(ireg.value, 0));
    }
    catch (err) {
        alert(err.message);
        return;
    }
    document.body.classList.remove('playing');
    ISSUE_EXEC_DELAY = ied.checked;
    EXEC_WRITE_DELAY = ewd.checked;
    let MEM = new Memory(new MemConf(safeInt(rl.value, 1), safeInt(wl.value, 2)));
    let ca_val = ca.options[ca.selectedIndex].value;
    let CACHE = XCacheFactory(ca_val.split('_')[0], {
        'mem': MEM,
        'n': safeInt(nways.value, 2),
        'size': safeInt(csize.value, 4),
        'readLatency': safeInt(crl.value, 0),
        'writeLatency': safeInt(cwl.value, 0),
        'iswriteback': ca_val.split('_')[1] === 'wb',
    });
    let memMgm = new MemoryMGM(CACHE, safeInt(rsize.value, 0) > 0);
    let bp_val = bp.options[bp.selectedIndex].value;
    let emu = new Emulator([
        [FuKind.ADDER, 'ADDR', safeInt(iaddr.value, 3), { duration: safeInt(iaddrd.value, 2) }],
        [FuKind.MULTIPLIER, 'MULT', safeInt(imult.value, 3), { duration: safeInt(imultd.value, 4) }],
        [FuKind.DIVIDER, 'DIV', safeInt(idiv.value, 3), { duration: safeInt(idivd.value, 6) }],
        [FuKind.MEMORY, 'MEM', 1, { 'memMgm': memMgm, duration: safeInt(iaddrd.value, 2) }],
    ], { ints: safeInt(ireg.value), floats: safeInt(freg.value) }, safeInt(rsize.value, 0), CACHE, memMgm, BpMap[bp_val](safeInt(nbit_n.value, 2), safeInt(nbit_k.value, 4)), program);
    let g = new Graphics(emu);
    g.paint();
    STEP = () => {
        var notEof = emu.step();
        g.paint();
        return notEof;
    };
}
main();
//# sourceMappingURL=archetypum.js.map        </script>
    </body>
</html>