Source code

Revision control

Copy as Markdown

Other Tools

/*
* 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 StackSlot {
constructor(index, byteSize, kind)
{
this._index = index;
this._byteSize = byteSize;
this._kind = kind;
}
get byteSize() { return this._byteSize; }
get kind() { return this._kind; }
get isLocked() { return this._kind == Locked; }
get isSpill() { return this._kind == Spill; }
get index() { return this._index; }
ensureSize(size)
{
if (this._offsetFromFP)
throw new Error("Stack slot already allocated");
this._byteSize = Math.max(this._byteSize, size);
}
get alignment()
{
if (this._byteSize <= 1)
return 1;
if (this._byteSize <= 2)
return 2;
if (this._byteSize <= 4)
return 4;
return 8;
}
get offsetFromFP() { return this._offsetFromFP; }
setOffsetFromFP(value) { this._offsetFromFP = value; }
hash()
{
return ((this._kind == Spill ? 1 : 0) + this._byteSize * 3 + (this._offsetFromFP ? this._offsetFromFP * 7 : 0)) >>> 0;
}
toString()
{
return "" + (this.isSpill ? "spill" : "stack") + this._index + "<" + this._byteSize +
(this._offsetFromFP ? ", offset = " + this._offsetFromFP : "") + ">";
}
static extract(arg)
{
if (arg.isStack)
return arg.stackSlot;
return null;
}
static forEachFast(arg, func)
{
if (!arg.isStack)
return;
let replacement;
if (replacement = func(arg.stackSlot))
return Arg.createStack(replacement, this._offset);
}
static forEach(arg, role, type, width, func)
{
if (!arg.isStack)
return;
let replacement;
if (replacement = func(arg.stackSlot, role, type, width))
return Arg.createStack(replacement, this._offset);
}
}