Source code

Revision control

Other Tools

/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
* vim: set ts=8 sts=2 et sw=2 tw=80:
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#ifndef frontend_BytecodeControlStructures_h
#define frontend_BytecodeControlStructures_h
#include "mozilla/Assertions.h" // MOZ_ASSERT
#include "mozilla/Attributes.h" // MOZ_MUST_USE
#include "mozilla/Maybe.h" // mozilla::Maybe
#include <stdint.h> // int32_t, uint32_t
#include "ds/Nestable.h" // Nestable
#include "frontend/BytecodeSection.h" // BytecodeOffset
#include "frontend/JumpList.h" // JumpList, JumpTarget
#include "frontend/SharedContext.h" // StatementKind, StatementKindIsLoop, StatementKindIsUnlabeledBreakTarget
#include "frontend/TDZCheckCache.h" // TDZCheckCache
#include "gc/Rooting.h" // RootedAtom, HandleAtom
#include "vm/StencilEnums.h" // TryNoteKind
#include "vm/StringType.h" // JSAtom
namespace js {
namespace frontend {
struct BytecodeEmitter;
class EmitterScope;
class NestableControl : public Nestable<NestableControl> {
StatementKind kind_;
// The innermost scope when this was pushed.
EmitterScope* emitterScope_;
protected:
NestableControl(BytecodeEmitter* bce, StatementKind kind);
public:
using Nestable<NestableControl>::enclosing;
using Nestable<NestableControl>::findNearest;
StatementKind kind() const { return kind_; }
EmitterScope* emitterScope() const { return emitterScope_; }
template <typename T>
bool is() const;
template <typename T>
T& as() {
MOZ_ASSERT(this->is<T>());
return static_cast<T&>(*this);
}
};
class BreakableControl : public NestableControl {
public:
// Offset of the last break.
JumpList breaks;
BreakableControl(BytecodeEmitter* bce, StatementKind kind);
MOZ_MUST_USE bool patchBreaks(BytecodeEmitter* bce);
};
template <>
inline bool NestableControl::is<BreakableControl>() const {
return StatementKindIsUnlabeledBreakTarget(kind_) ||
kind_ == StatementKind::Label;
}
class LabelControl : public BreakableControl {
RootedAtom label_;
// The code offset when this was pushed. Used for effectfulness checking.
BytecodeOffset startOffset_;
public:
LabelControl(BytecodeEmitter* bce, JSAtom* label, BytecodeOffset startOffset);
HandleAtom label() const { return label_; }
BytecodeOffset startOffset() const { return startOffset_; }
};
template <>
inline bool NestableControl::is<LabelControl>() const {
return kind_ == StatementKind::Label;
}
class LoopControl : public BreakableControl {
// Loops' children are emitted in dominance order, so they can always
// have a TDZCheckCache.
TDZCheckCache tdzCache_;
// Here's the basic structure of a loop:
//
// head:
// JSOp::LoopHead
// {loop condition/body}
//
// continueTarget:
// {loop update if present}
//
// # Loop end, backward jump
// JSOp::Goto/JSOp::IfNe head
//
// breakTarget:
// The bytecode offset of JSOp::LoopHead.
JumpTarget head_;
// Stack depth when this loop was pushed on the control stack.
int32_t stackDepth_;
// The loop nesting depth. Used as a hint to Ion.
uint32_t loopDepth_;
public:
// Offset of the last continue in the loop.
JumpList continues;
LoopControl(BytecodeEmitter* bce, StatementKind loopKind);
BytecodeOffset headOffset() const { return head_.offset; }
MOZ_MUST_USE bool emitContinueTarget(BytecodeEmitter* bce);
// `nextPos` is the offset in the source code for the character that
// corresponds to the next instruction after JSOp::LoopHead.
// Can be Nothing() if not available.
MOZ_MUST_USE bool emitLoopHead(BytecodeEmitter* bce,
const mozilla::Maybe<uint32_t>& nextPos);
MOZ_MUST_USE bool emitLoopEnd(BytecodeEmitter* bce, JSOp op,
TryNoteKind tryNoteKind);
};
template <>
inline bool NestableControl::is<LoopControl>() const {
return StatementKindIsLoop(kind_);
}
class TryFinallyControl : public NestableControl {
bool emittingSubroutine_;
public:
// The subroutine when emitting a finally block.
JumpList gosubs;
TryFinallyControl(BytecodeEmitter* bce, StatementKind kind);
void setEmittingSubroutine() { emittingSubroutine_ = true; }
bool emittingSubroutine() const { return emittingSubroutine_; }
};
template <>
inline bool NestableControl::is<TryFinallyControl>() const {
return kind_ == StatementKind::Try || kind_ == StatementKind::Finally;
}
} /* namespace frontend */
} /* namespace js */
#endif /* frontend_BytecodeControlStructures_h */