Source code

Revision control

Copy as Markdown

Other Tools

use crate::ast_emitter::AstEmitter;
use crate::emitter::EmitError;
use crate::emitter_scope::NameLocation;
use ast::source_atom_set::SourceAtomSetIndex;
use stencil::env_coord::{EnvironmentHops, EnvironmentSlot};
use stencil::frame_slot::FrameSlot;
use stencil::gcthings::GCThingIndex;
use stencil::scope::BindingKind;
#[derive(Debug, PartialEq)]
enum AssignmentReferenceKind {
GlobalVar(GCThingIndex),
GlobalLexical(GCThingIndex),
FrameSlotLexical(FrameSlot),
FrameSlotNonLexical(FrameSlot),
EnvironmentCoordLexical(EnvironmentHops, EnvironmentSlot),
EnvironmentCoordNonLexical(EnvironmentHops, EnvironmentSlot),
Dynamic(GCThingIndex),
#[allow(dead_code)]
Prop(GCThingIndex),
#[allow(dead_code)]
Elem,
}
// See AssignmentReferenceEmitter.
// This uses struct to hide the details from the consumer.
#[derive(Debug)]
#[must_use]
pub struct AssignmentReference {
kind: AssignmentReferenceKind,
}
impl AssignmentReference {
fn new(kind: AssignmentReferenceKind) -> Self {
Self { kind }
}
fn stack_slots(&self) -> usize {
match self.kind {
AssignmentReferenceKind::GlobalVar(_) => 1,
AssignmentReferenceKind::GlobalLexical(_) => 1,
AssignmentReferenceKind::FrameSlotLexical(_) => 0,
AssignmentReferenceKind::FrameSlotNonLexical(_) => 0,
AssignmentReferenceKind::EnvironmentCoordLexical(_, _) => 0,
AssignmentReferenceKind::EnvironmentCoordNonLexical(_, _) => 0,
AssignmentReferenceKind::Dynamic(_) => 1,
AssignmentReferenceKind::Prop(_) => 1,
AssignmentReferenceKind::Elem => 2,
}
}
}
#[derive(Debug, PartialEq)]
enum DeclarationReferenceKind {
GlobalVar(GCThingIndex),
GlobalLexical(GCThingIndex),
FrameSlot(FrameSlot),
EnvironmentCoord(EnvironmentHops, EnvironmentSlot),
}
// See DeclarationReferenceEmitter.
// This uses struct to hide the details from the consumer.
#[derive(Debug)]
#[must_use]
pub struct DeclarationReference {
kind: DeclarationReferenceKind,
}
impl DeclarationReference {
fn new(kind: DeclarationReferenceKind) -> Self {
Self { kind }
}
}
#[derive(Debug, PartialEq)]
enum CallKind {
Normal,
// FIXME: Support eval, Function#call, Function#apply etc.
}
#[derive(Debug, PartialEq)]
enum ValueIsOnStack {
No,
Yes,
}
fn check_frame_temporary_dead_zone(
emitter: &mut AstEmitter,
slot: FrameSlot,
is_on_stack: ValueIsOnStack,
) {
// FIXME: Use cache to avoid emitting check_lexical twice or more.
// FIXME: Support aliased lexical.
// [stack] VAL?
if is_on_stack == ValueIsOnStack::No {
emitter.emit.get_local(slot.into());
// [stack] VAL
}
emitter.emit.check_lexical(slot.into());
// [stack] VAL
if is_on_stack == ValueIsOnStack::No {
emitter.emit.pop();
// [stack]
}
// [stack] VAL?
}
fn check_env_temporary_dead_zone(
emitter: &mut AstEmitter,
hops: EnvironmentHops,
slot: EnvironmentSlot,
is_on_stack: ValueIsOnStack,
) {
// FIXME: Use cache to avoid emitting check_lexical twice or more.
// FIXME: Support aliased lexical.
// [stack] VAL?
if is_on_stack == ValueIsOnStack::No {
emitter.emit.get_aliased_var(hops.into(), slot.into());
// [stack] VAL
}
emitter.emit.check_aliased_lexical(hops.into(), slot.into());
// [stack] VAL
if is_on_stack == ValueIsOnStack::No {
emitter.emit.pop();
// [stack]
}
// [stack] VAL?
}
// See *ReferenceEmitter.
// This uses struct to hide the details from the consumer.
#[derive(Debug)]
#[must_use]
pub struct CallReference {
kind: CallKind,
}
impl CallReference {
fn new(kind: CallKind) -> Self {
Self { kind }
}
}
// Struct for emitting bytecode for get `name` operation.
pub struct GetNameEmitter {
pub name: SourceAtomSetIndex,
}
impl GetNameEmitter {
pub fn emit(self, emitter: &mut AstEmitter) {
let name_index = emitter.emit.get_atom_gcthing_index(self.name);
let loc = emitter.lookup_name(self.name);
// [stack]
match loc {
NameLocation::Global(_kind) => {
emitter.emit.get_g_name(name_index);
// [stack] VAL
}
NameLocation::Dynamic => {
emitter.emit.get_name(name_index);
// [stack] VAL
}
NameLocation::FrameSlot(slot, kind) => {
emitter.emit.get_local(slot.into());
// [stack] VAL
if kind == BindingKind::Let || kind == BindingKind::Const {
check_frame_temporary_dead_zone(emitter, slot, ValueIsOnStack::Yes);
// [stack] VAL
}
}
NameLocation::EnvironmentCoord(hops, slot, kind) => {
emitter.emit.get_aliased_var(hops.into(), slot.into());
if kind == BindingKind::Let || kind == BindingKind::Const {
check_env_temporary_dead_zone(emitter, hops, slot, ValueIsOnStack::Yes);
// [stack] VAL
}
}
}
}
}
// Struct for emitting bytecode for get `obj.key` operation.
pub struct GetPropEmitter<F>
where
F: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub obj: F,
pub key: SourceAtomSetIndex,
}
impl<F> GetPropEmitter<F>
where
F: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub fn emit(self, emitter: &mut AstEmitter) -> Result<(), EmitError> {
let key_index = emitter.emit.get_atom_gcthing_index(self.key);
// [stack]
let depth = emitter.emit.stack_depth();
(self.obj)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] OBJ
emitter.emit.get_prop(key_index);
// [stack] VAL
Ok(())
}
}
// Struct for emitting bytecode for get `super.key` operation.
pub struct GetSuperPropEmitter<F>
where
F: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub this: F,
pub key: SourceAtomSetIndex,
}
impl<F> GetSuperPropEmitter<F>
where
F: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub fn emit(self, emitter: &mut AstEmitter) -> Result<(), EmitError> {
let key_index = emitter.emit.get_atom_gcthing_index(self.key);
// [stack]
let depth = emitter.emit.stack_depth();
(self.this)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] THIS
emitter.emit.callee();
// [stack] THIS CALLEE
emitter.emit.super_base();
// [stack] THIS OBJ
emitter.emit.get_prop_super(key_index);
// [stack] VAL
Ok(())
}
}
// Struct for emitting bytecode for get `obj[key]` operation.
pub struct GetElemEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<(), EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub obj: F1,
pub key: F2,
}
impl<F1, F2> GetElemEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<(), EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub fn emit(self, emitter: &mut AstEmitter) -> Result<(), EmitError> {
// [stack]
let depth = emitter.emit.stack_depth();
(self.obj)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] OBJ
let depth = emitter.emit.stack_depth();
(self.key)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] OBJ KEY
emitter.emit.get_elem();
// [stack] VAL
Ok(())
}
}
// Struct for emitting bytecode for get `super[key]` operation.
pub struct GetSuperElemEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<(), EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub this: F1,
pub key: F2,
}
impl<F1, F2> GetSuperElemEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<(), EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub fn emit(self, emitter: &mut AstEmitter) -> Result<(), EmitError> {
// [stack]
let depth = emitter.emit.stack_depth();
(self.this)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] THIS
let depth = emitter.emit.stack_depth();
(self.key)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] THIS KEY
emitter.emit.callee();
// [stack] THIS KEY CALLEE
emitter.emit.super_base();
// [stack] THIS KEY OBJ
emitter.emit.get_elem_super();
// [stack] VAL
Ok(())
}
}
// Struct for emitting bytecode for `name` reference.
pub struct NameReferenceEmitter {
pub name: SourceAtomSetIndex,
}
impl NameReferenceEmitter {
pub fn emit_for_call(self, emitter: &mut AstEmitter) -> CallReference {
let name_index = emitter.emit.get_atom_gcthing_index(self.name);
let loc = emitter.lookup_name(self.name);
// [stack]
match loc {
NameLocation::Global(_kind) => {
emitter.emit.get_g_name(name_index);
// [stack] CALLEE
// NOTE: We don't support non-syntactic scope.
// See NameOpEmitter::emitGet in SpiderMonkey for omitted
// cases.
emitter.emit.undefined();
// [stack] CALLEE THIS
}
NameLocation::Dynamic => {
emitter.emit.get_name(name_index);
// [stack] CALLEE
// NOTE: We don't support non-syntactic scope or with statement.
// See NameOpEmitter::emitGet in SpiderMonkey for omitted
// cases.
emitter.emit.undefined();
// [stack] CALLEE THIS
}
NameLocation::FrameSlot(slot, kind) => {
emitter.emit.get_local(slot.into());
// [stack] CALLEE
if kind == BindingKind::Let || kind == BindingKind::Const {
check_frame_temporary_dead_zone(emitter, slot, ValueIsOnStack::Yes);
// [stack] CALLEE
}
emitter.emit.undefined();
// [stack] CALLEE THIS
}
NameLocation::EnvironmentCoord(hops, slot, kind) => {
emitter.emit.get_aliased_var(hops.into(), slot.into());
// [stack] CALLEE
if kind == BindingKind::Let || kind == BindingKind::Const {
check_env_temporary_dead_zone(emitter, hops, slot, ValueIsOnStack::Yes);
// [stack] CALLEE
}
emitter.emit.undefined();
// [stack] CALLEE THIS
}
}
CallReference::new(CallKind::Normal)
}
pub fn emit_for_assignment_with_loc(
self,
emitter: &mut AstEmitter,
loc: NameLocation,
) -> AssignmentReference {
let name_index = emitter.emit.get_atom_gcthing_index(self.name);
// [stack]
match loc {
NameLocation::Global(kind) => match kind {
BindingKind::Var => {
emitter.emit.bind_g_name(name_index);
// [stack] GLOBAL
AssignmentReference::new(AssignmentReferenceKind::GlobalVar(name_index))
}
BindingKind::Let | BindingKind::Const => {
emitter.emit.bind_g_name(name_index);
// [stack] GLOBAL
AssignmentReference::new(AssignmentReferenceKind::GlobalLexical(name_index))
}
},
NameLocation::Dynamic => {
emitter.emit.bind_name(name_index);
// [stack] ENV
AssignmentReference::new(AssignmentReferenceKind::Dynamic(name_index))
}
NameLocation::FrameSlot(slot, kind) => {
if kind == BindingKind::Let || kind == BindingKind::Const {
AssignmentReference::new(AssignmentReferenceKind::FrameSlotLexical(slot))
} else {
AssignmentReference::new(AssignmentReferenceKind::FrameSlotNonLexical(slot))
}
}
NameLocation::EnvironmentCoord(hops, slot, kind) => {
if kind == BindingKind::Let || kind == BindingKind::Const {
AssignmentReference::new(AssignmentReferenceKind::EnvironmentCoordLexical(
hops, slot,
))
} else {
AssignmentReference::new(AssignmentReferenceKind::EnvironmentCoordNonLexical(
hops, slot,
))
}
}
}
}
pub fn emit_for_assignment(self, emitter: &mut AstEmitter) -> AssignmentReference {
let loc = emitter.lookup_name(self.name);
self.emit_for_assignment_with_loc(emitter, loc)
}
/// Ignore any lexical scope and assign to var scope.
/// Used by Annex B function.
pub fn emit_for_var_assignment(self, emitter: &mut AstEmitter) -> AssignmentReference {
let loc = emitter.lookup_name_in_var(self.name);
self.emit_for_assignment_with_loc(emitter, loc)
}
pub fn emit_for_declaration(self, emitter: &mut AstEmitter) -> DeclarationReference {
let name_index = emitter.emit.get_atom_gcthing_index(self.name);
let loc = emitter.lookup_name(self.name);
// [stack]
match loc {
NameLocation::Global(kind) => match kind {
BindingKind::Var => {
emitter.emit.bind_g_name(name_index);
// [stack] GLOBAL
DeclarationReference::new(DeclarationReferenceKind::GlobalVar(name_index))
}
BindingKind::Let | BindingKind::Const => {
DeclarationReference::new(DeclarationReferenceKind::GlobalLexical(name_index))
}
},
NameLocation::Dynamic => {
panic!("declaration should have non-dynamic location");
}
NameLocation::FrameSlot(slot, _kind) => {
DeclarationReference::new(DeclarationReferenceKind::FrameSlot(slot))
}
NameLocation::EnvironmentCoord(hops, slot, _kind) => {
// FIXME: does this happen????
DeclarationReference::new(DeclarationReferenceKind::EnvironmentCoord(hops, slot))
}
}
}
}
// Struct for emitting bytecode for `obj.key` reference.
pub struct PropReferenceEmitter<F>
where
F: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub obj: F,
pub key: SourceAtomSetIndex,
}
impl<F> PropReferenceEmitter<F>
where
F: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub fn emit_for_call(self, emitter: &mut AstEmitter) -> Result<CallReference, EmitError> {
let key_index = emitter.emit.get_atom_gcthing_index(self.key);
// [stack]
let depth = emitter.emit.stack_depth();
(self.obj)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] THIS
emitter.emit.dup();
// [stack] THIS THIS
// FIXME: Support super.
emitter.emit.get_prop(key_index);
// [stack] THIS CALLEE
emitter.emit.swap();
// [stack] CALLEE THIS
Ok(CallReference::new(CallKind::Normal))
}
#[allow(dead_code)]
pub fn emit_for_assignment(
self,
emitter: &mut AstEmitter,
) -> Result<AssignmentReference, EmitError> {
let key_index = emitter.emit.get_atom_gcthing_index(self.key);
// [stack]
let depth = emitter.emit.stack_depth();
(self.obj)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] OBJ
Ok(AssignmentReference::new(AssignmentReferenceKind::Prop(
key_index,
)))
}
}
// Struct for emitting bytecode for `obj[key]` reference.
pub struct ElemReferenceEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<(), EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub obj: F1,
pub key: F2,
}
impl<F1, F2> ElemReferenceEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<(), EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub fn emit_for_call(self, emitter: &mut AstEmitter) -> Result<CallReference, EmitError> {
// [stack]
let depth = emitter.emit.stack_depth();
(self.obj)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] THIS
emitter.emit.dup();
// [stack] THIS THIS
let depth = emitter.emit.stack_depth();
(self.key)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] THIS THIS KEY
// FIXME: Support super.
emitter.emit.get_elem();
// [stack] THIS CALLEE
emitter.emit.swap();
// [stack] CALLEE THIS
Ok(CallReference::new(CallKind::Normal))
}
#[allow(dead_code)]
pub fn emit_for_assignment(
self,
emitter: &mut AstEmitter,
) -> Result<AssignmentReference, EmitError> {
// [stack]
let depth = emitter.emit.stack_depth();
(self.obj)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] OBJ
let depth = emitter.emit.stack_depth();
(self.key)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] OBJ KEY
Ok(AssignmentReference::new(AssignmentReferenceKind::Elem))
}
}
// Struct for emitting bytecode for call `callee(arguments)` operation.
pub struct CallEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<CallReference, EmitError>,
F2: Fn(&mut AstEmitter) -> Result<usize, EmitError>,
{
pub callee: F1,
pub arguments: F2,
}
impl<F1, F2> CallEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<CallReference, EmitError>,
F2: Fn(&mut AstEmitter) -> Result<usize, EmitError>,
{
pub fn emit(self, emitter: &mut AstEmitter) -> Result<(), EmitError> {
// [stack]
let depth = emitter.emit.stack_depth();
let reference = (self.callee)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 2);
// [stack] CALLEE THIS
// FIXME: Support spread.
let depth = emitter.emit.stack_depth();
let len = (self.arguments)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + len);
// [stack] CALLEE THIS ARGS...
match reference.kind {
CallKind::Normal => {
emitter.emit.call(len as u16);
// [stack] VAL
}
}
Ok(())
}
}
// Struct for emitting bytecode for `new callee(arguments)` operation.
pub struct NewEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<(), EmitError>,
F2: Fn(&mut AstEmitter) -> Result<usize, EmitError>,
{
pub callee: F1,
pub arguments: F2,
}
impl<F1, F2> NewEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<(), EmitError>,
F2: Fn(&mut AstEmitter) -> Result<usize, EmitError>,
{
pub fn emit(self, emitter: &mut AstEmitter) -> Result<(), EmitError> {
// [stack]
let depth = emitter.emit.stack_depth();
(self.callee)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] CALLEE
emitter.emit.is_constructing();
// [stack] CALLEE JS_IS_CONSTRUCTING
// FIXME: Support spread.
let depth = emitter.emit.stack_depth();
let len = (self.arguments)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + len);
// [stack] CALLEE JS_IS_CONSTRUCTING ARGS...
emitter.emit.dup_at(len as u32 + 1);
// [stack] CALLEE JS_IS_CONSTRUCTING ARGS... CALLEE
emitter.emit.new_(len as u16);
// [stack] VAL
Ok(())
}
}
// Struct for emitting bytecode for assignment `lhs = rhs` operation.
pub struct AssignmentEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<AssignmentReference, EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub lhs: F1,
pub rhs: F2,
}
impl<F1, F2> AssignmentEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<AssignmentReference, EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub fn emit(self, emitter: &mut AstEmitter) -> Result<(), EmitError> {
// [stack]
let depth = emitter.emit.stack_depth();
let reference = (self.lhs)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + reference.stack_slots());
// [stack] REF...
let depth = emitter.emit.stack_depth();
(self.rhs)(emitter)?;
debug_assert_eq!(emitter.emit.stack_depth(), depth + 1);
// [stack] REF... VAL
match reference.kind {
AssignmentReferenceKind::GlobalVar(name_index) => {
// [stack] GLOBAL VAL
emitter.emit.set_g_name(name_index);
// [stack] VAL
}
AssignmentReferenceKind::GlobalLexical(name_index) => {
// [stack] VAL
emitter.emit.set_g_name(name_index);
// [stack] VAL
}
AssignmentReferenceKind::Dynamic(name_index) => {
// [stack] ENV VAL
emitter.emit.set_name(name_index);
// [stack] VAL
}
AssignmentReferenceKind::FrameSlotLexical(slot) => {
// [stack] VAL
check_frame_temporary_dead_zone(emitter, slot, ValueIsOnStack::No);
// [stack] VAL
emitter.emit.set_local(slot.into());
// [stack] VAL
}
AssignmentReferenceKind::FrameSlotNonLexical(slot) => {
// [stack] VAL
emitter.emit.set_local(slot.into());
// [stack] VAL
}
AssignmentReferenceKind::EnvironmentCoordLexical(hops, slot) => {
// [stack] VAL
check_env_temporary_dead_zone(emitter, hops, slot, ValueIsOnStack::No);
// [stack] VAL
emitter.emit.set_aliased_var(hops.into(), slot.into());
// [stack] VAL
}
AssignmentReferenceKind::EnvironmentCoordNonLexical(hops, slot) => {
// [stack] VAL
emitter.emit.set_aliased_var(hops.into(), slot.into());
// [stack] VAL
}
AssignmentReferenceKind::Prop(key_index) => {
// [stack] OBJ VAL
// FIXME: Support strict mode and super.
emitter.emit.set_prop(key_index);
// [stack] VAL
}
AssignmentReferenceKind::Elem => {
// [stack] OBJ KEY VAL
// FIXME: Support strict mode and super.
emitter.emit.set_elem();
// [stack] VAL
}
}
Ok(())
}
// FIXME: Support compound assignment
}
// Struct for emitting bytecode for declaration `lhs = rhs` operation.
pub struct DeclarationEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<DeclarationReference, EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub lhs: F1,
pub rhs: F2,
}
impl<F1, F2> DeclarationEmitter<F1, F2>
where
F1: Fn(&mut AstEmitter) -> Result<DeclarationReference, EmitError>,
F2: Fn(&mut AstEmitter) -> Result<(), EmitError>,
{
pub fn emit(self, emitter: &mut AstEmitter) -> Result<(), EmitError> {
let reference = (self.lhs)(emitter)?;
(self.rhs)(emitter)?;
match reference.kind {
DeclarationReferenceKind::GlobalVar(name_index) => {
// [stack] GLOBAL VAL
emitter.emit.set_g_name(name_index);
// [stack] VAL
}
DeclarationReferenceKind::GlobalLexical(name_index) => {
// [stack] VAL
emitter.emit.init_g_lexical(name_index);
// [stack] VAL
}
DeclarationReferenceKind::FrameSlot(slot) => {
// [stack] VAL
emitter.emit.init_lexical(slot.into());
// [stack] VAL
}
DeclarationReferenceKind::EnvironmentCoord(hops, slot) => {
// [stack] VAL
emitter.emit.init_aliased_lexical(hops.into(), slot.into());
// [stack] VAL
}
}
Ok(())
}
}
// FIXME: Add increment