blob: 013814aa5b1528d672894ba83b707fcd041c3c02 [file] [log] [blame]
/*
* Copyright (C) 2016 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
"use strict";
class Code {
constructor()
{
this._blocks = [];
this._stackSlots = [];
this._gpTmps = [];
this._fpTmps = [];
this._callArgAreaSize = 0;
this._frameSize = 0;
}
addBlock(frequency = 1)
{
return addIndexed(this._blocks, BasicBlock, frequency);
}
addStackSlot(byteSize, kind)
{
return addIndexed(this._stackSlots, StackSlot, byteSize, kind);
}
newTmp(type)
{
return addIndexed(this[`_${lowerSymbolName(type)}Tmps`], Tmp, type);
}
get size() { return this._blocks.length; }
at(index) { return this._blocks[index]; }
[Symbol.iterator]()
{
return this._blocks[Symbol.iterator]();
}
get blocks() { return this._blocks; }
get stackSlots() { return this._stackSlots; }
tmps(type) { return this[`_${lowerSymbolName(type)}Tmps`]; }
get callArgAreaSize() { return this._callArgAreaSize; }
requestCallArgAreaSize(size)
{
this._callArgAreaSize = Math.max(this._callArgAreaSize, roundUpToMultipleOf(stackAlignmentBytes, size));
}
get frameSize() { return this._frameSize; }
setFrameSize(frameSize) { this._frameSize = frameSize; }
hash()
{
let result = 0;
for (let block of this) {
result *= 1000001;
result |= 0;
for (let inst of block) {
result *= 97;
result |= 0;
result += inst.hash();
result |= 0;
}
for (let successor of block.successorBlocks) {
result *= 7;
result |= 0;
result += successor.index;
result |= 0;
}
}
for (let slot of this.stackSlots) {
result *= 101;
result |= 0;
result += slot.hash();
result |= 0;
}
return result >>> 0;
}
toString()
{
let result = "";
for (let block of this) {
result += block.toStringDeep();
}
if (this.stackSlots.length) {
result += "Stack slots:\n";
for (let slot of this.stackSlots)
result += ` ${slot}\n`;
}
if (this._frameSize)
result += `Frame size: ${this._frameSize}\n`;
if (this._callArgAreaSize)
result += `Call arg area size: ${this._callArgAreaSize}\n`;
return result;
}
}