2018-06-27 12:18:17 +00:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
|
|
|
** Copyright (C) 2018 The Qt Company Ltd.
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
|
|
|
**
|
|
|
|
** This file is part of the QtQml module of the Qt Toolkit.
|
|
|
|
**
|
|
|
|
** $QT_BEGIN_LICENSE:LGPL$
|
|
|
|
** Commercial License Usage
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
|
|
**
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
** Foundation and appearing in the file LICENSE.LGPL3 included in the
|
|
|
|
** packaging of this file. Please review the following information to
|
|
|
|
** ensure the GNU Lesser General Public License version 3 requirements
|
|
|
|
** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
|
|
|
|
**
|
|
|
|
** GNU General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
** General Public License version 2.0 or (at your option) the GNU General
|
|
|
|
** Public license version 3 or any later version approved by the KDE Free
|
|
|
|
** Qt Foundation. The licenses are as published by the Free Software
|
|
|
|
** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-2.0.html and
|
|
|
|
** https://www.gnu.org/licenses/gpl-3.0.html.
|
|
|
|
**
|
|
|
|
** $QT_END_LICENSE$
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
#ifndef QV4STACKFRAME_H
|
|
|
|
#define QV4STACKFRAME_H
|
|
|
|
|
2018-07-31 16:22:09 +00:00
|
|
|
//
|
|
|
|
// W A R N I N G
|
|
|
|
// -------------
|
|
|
|
//
|
|
|
|
// This file is not part of the Qt API. It exists purely as an
|
|
|
|
// implementation detail. This header file may change from version to
|
|
|
|
// version without notice, or even be removed.
|
|
|
|
//
|
|
|
|
// We mean it.
|
|
|
|
//
|
|
|
|
|
2018-06-27 12:18:17 +00:00
|
|
|
#include <private/qv4context_p.h>
|
2018-06-27 15:23:39 +00:00
|
|
|
#include <private/qv4enginebase_p.h>
|
2018-06-27 12:18:17 +00:00
|
|
|
#ifndef V4_BOOTSTRAP
|
|
|
|
#include <private/qv4function_p.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
|
|
|
|
namespace QV4 {
|
|
|
|
|
|
|
|
struct CallData
|
|
|
|
{
|
|
|
|
enum Offsets {
|
|
|
|
Function = 0,
|
|
|
|
Context = 1,
|
|
|
|
Accumulator = 2,
|
|
|
|
This = 3,
|
|
|
|
NewTarget = 4,
|
2018-07-13 10:18:52 +00:00
|
|
|
Argc = 5,
|
|
|
|
|
|
|
|
LastOffset = Argc,
|
|
|
|
OffsetCount = LastOffset + 1
|
2018-06-27 12:18:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Value function;
|
|
|
|
Value context;
|
|
|
|
Value accumulator;
|
|
|
|
Value thisObject;
|
|
|
|
Value newTarget;
|
|
|
|
Value _argc;
|
|
|
|
|
|
|
|
int argc() const {
|
|
|
|
Q_ASSERT(_argc.isInteger());
|
|
|
|
return _argc.int_32();
|
|
|
|
}
|
|
|
|
|
|
|
|
void setArgc(int argc) {
|
|
|
|
Q_ASSERT(argc >= 0);
|
|
|
|
_argc.setInt_32(argc);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline ReturnedValue argument(int i) const {
|
2018-09-11 09:07:32 +00:00
|
|
|
return i < argc() ? args[i].asReturnedValue() : Value::undefinedValue().asReturnedValue();
|
2018-06-27 12:18:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Value args[1];
|
|
|
|
|
|
|
|
static Q_DECL_CONSTEXPR int HeaderSize() { return offsetof(CallData, args) / sizeof(QV4::Value); }
|
|
|
|
};
|
|
|
|
|
|
|
|
Q_STATIC_ASSERT(std::is_standard_layout<CallData>::value);
|
2018-07-13 10:18:52 +00:00
|
|
|
Q_STATIC_ASSERT(offsetof(CallData, function ) == CallData::Function * sizeof(Value));
|
|
|
|
Q_STATIC_ASSERT(offsetof(CallData, context ) == CallData::Context * sizeof(Value));
|
|
|
|
Q_STATIC_ASSERT(offsetof(CallData, accumulator) == CallData::Accumulator * sizeof(Value));
|
|
|
|
Q_STATIC_ASSERT(offsetof(CallData, thisObject ) == CallData::This * sizeof(Value));
|
|
|
|
Q_STATIC_ASSERT(offsetof(CallData, newTarget ) == CallData::NewTarget * sizeof(Value));
|
|
|
|
Q_STATIC_ASSERT(offsetof(CallData, _argc ) == CallData::Argc * sizeof(Value));
|
|
|
|
Q_STATIC_ASSERT(offsetof(CallData, args ) == 6 * sizeof(Value));
|
2018-06-27 12:18:17 +00:00
|
|
|
|
|
|
|
struct Q_QML_EXPORT CppStackFrame {
|
2018-06-27 15:23:39 +00:00
|
|
|
EngineBase *engine;
|
|
|
|
Value *savedStackTop;
|
2018-06-27 12:18:17 +00:00
|
|
|
CppStackFrame *parent;
|
|
|
|
Function *v4Function;
|
|
|
|
CallData *jsFrame;
|
|
|
|
const Value *originalArguments;
|
|
|
|
int originalArgumentsCount;
|
|
|
|
int instructionPointer;
|
|
|
|
const char *yield;
|
|
|
|
const char *unwindHandler;
|
|
|
|
const char *unwindLabel;
|
|
|
|
int unwindLevel;
|
2018-05-31 14:06:34 +00:00
|
|
|
bool yieldIsIterator;
|
2018-09-04 10:19:10 +00:00
|
|
|
bool callerCanHandleTailCall;
|
|
|
|
bool pendingTailCall;
|
2019-01-14 13:21:21 +00:00
|
|
|
bool isTailCalling;
|
2018-06-27 12:18:17 +00:00
|
|
|
|
2018-09-04 10:19:10 +00:00
|
|
|
void init(EngineBase *engine, Function *v4Function, const Value *argv, int argc, bool callerCanHandleTailCall = false) {
|
2018-06-27 15:23:39 +00:00
|
|
|
this->engine = engine;
|
|
|
|
|
|
|
|
this->v4Function = v4Function;
|
|
|
|
originalArguments = argv;
|
|
|
|
originalArgumentsCount = argc;
|
|
|
|
instructionPointer = 0;
|
|
|
|
yield = nullptr;
|
|
|
|
unwindHandler = nullptr;
|
|
|
|
unwindLabel = nullptr;
|
|
|
|
unwindLevel = 0;
|
2018-05-31 14:06:34 +00:00
|
|
|
yieldIsIterator = false;
|
2018-09-04 10:19:10 +00:00
|
|
|
this->callerCanHandleTailCall = callerCanHandleTailCall;
|
|
|
|
pendingTailCall = false;
|
2019-01-14 13:21:21 +00:00
|
|
|
isTailCalling = false;
|
2018-06-27 15:23:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void push() {
|
|
|
|
parent = engine->currentStackFrame;
|
|
|
|
engine->currentStackFrame = this;
|
|
|
|
savedStackTop = engine->jsStackTop;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pop() {
|
|
|
|
engine->currentStackFrame = parent;
|
|
|
|
engine->jsStackTop = savedStackTop;
|
|
|
|
}
|
|
|
|
|
2018-06-27 12:18:17 +00:00
|
|
|
#ifndef V4_BOOTSTRAP
|
2018-06-30 17:19:18 +00:00
|
|
|
static uint requiredJSStackFrameSize(uint nRegisters) {
|
|
|
|
return CallData::HeaderSize() + nRegisters;
|
|
|
|
}
|
2018-06-27 15:23:39 +00:00
|
|
|
static uint requiredJSStackFrameSize(Function *v4Function) {
|
2018-06-27 12:18:17 +00:00
|
|
|
return CallData::HeaderSize() + v4Function->compiledFunction->nRegisters;
|
|
|
|
}
|
2018-06-27 15:23:39 +00:00
|
|
|
uint requiredJSStackFrameSize() const {
|
|
|
|
return requiredJSStackFrameSize(v4Function);
|
|
|
|
}
|
|
|
|
void setupJSFrame(Value *stackSpace, const Value &function, const Heap::ExecutionContext *scope,
|
2018-09-11 09:07:32 +00:00
|
|
|
const Value &thisObject, const Value &newTarget = Value::undefinedValue()) {
|
2018-06-30 17:19:18 +00:00
|
|
|
setupJSFrame(stackSpace, function, scope, thisObject, newTarget,
|
|
|
|
v4Function->nFormals, v4Function->compiledFunction->nRegisters);
|
|
|
|
}
|
|
|
|
void setupJSFrame(Value *stackSpace, const Value &function, const Heap::ExecutionContext *scope,
|
|
|
|
const Value &thisObject, const Value &newTarget, uint nFormals, uint nRegisters)
|
2018-06-27 15:23:39 +00:00
|
|
|
{
|
|
|
|
jsFrame = reinterpret_cast<CallData *>(stackSpace);
|
|
|
|
jsFrame->function = function;
|
|
|
|
jsFrame->context = scope->asReturnedValue();
|
|
|
|
jsFrame->accumulator = Encode::undefined();
|
|
|
|
jsFrame->thisObject = thisObject;
|
|
|
|
jsFrame->newTarget = newTarget;
|
|
|
|
|
|
|
|
uint argc = uint(originalArgumentsCount);
|
2018-06-30 17:19:18 +00:00
|
|
|
if (argc > nFormals)
|
|
|
|
argc = nFormals;
|
2018-06-27 15:23:39 +00:00
|
|
|
jsFrame->setArgc(argc);
|
|
|
|
|
|
|
|
memcpy(jsFrame->args, originalArguments, argc*sizeof(Value));
|
2018-09-11 08:27:32 +00:00
|
|
|
Q_STATIC_ASSERT(Encode::undefined() == 0);
|
|
|
|
memset(jsFrame->args + argc, 0, (nRegisters - argc)*sizeof(Value));
|
2018-08-21 14:51:17 +00:00
|
|
|
|
|
|
|
if (v4Function && v4Function->compiledFunction) {
|
|
|
|
const int firstDeadZoneRegister = v4Function->compiledFunction->firstTemporalDeadZoneRegister;
|
|
|
|
const int registerDeadZoneSize = v4Function->compiledFunction->sizeOfRegisterTemporalDeadZone;
|
|
|
|
|
|
|
|
const Value * tdzEnd = stackSpace + firstDeadZoneRegister + registerDeadZoneSize;
|
|
|
|
for (Value *v = stackSpace + firstDeadZoneRegister; v < tdzEnd; ++v)
|
2018-09-11 09:07:32 +00:00
|
|
|
*v = Value::emptyValue().asReturnedValue();
|
2018-08-21 14:51:17 +00:00
|
|
|
}
|
2018-06-27 15:23:39 +00:00
|
|
|
}
|
2018-06-27 12:18:17 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
QString source() const;
|
|
|
|
QString function() const;
|
|
|
|
inline QV4::ExecutionContext *context() const {
|
|
|
|
return static_cast<ExecutionContext *>(&jsFrame->context);
|
|
|
|
}
|
|
|
|
int lineNumber() const;
|
|
|
|
|
|
|
|
inline QV4::Heap::CallContext *callContext() const {
|
|
|
|
Heap::ExecutionContext *ctx = static_cast<ExecutionContext &>(jsFrame->context).d();\
|
|
|
|
while (ctx->type != Heap::ExecutionContext::Type_CallContext)
|
|
|
|
ctx = ctx->outer;
|
|
|
|
return static_cast<Heap::CallContext *>(ctx);
|
|
|
|
}
|
|
|
|
ReturnedValue thisObject() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
QT_END_NAMESPACE
|
|
|
|
|
|
|
|
#endif
|