* Copyright (c) 2022 Huawei Device Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef ECMASCRIPT_COMPILER_BUILTINS_CONTAINERS_QUEUE_STUB_BUILDER_H
#define ECMASCRIPT_COMPILER_BUILTINS_CONTAINERS_QUEUE_STUB_BUILDER_H
#include "ecmascript/compiler/builtins/builtins_stubs.h"
#include "ecmascript/js_api/js_api_queue.h"
namespace panda::ecmascript::kungfu {
class ContainersQueueStubBuilder : public BuiltinsStubBuilder {
public:
explicit ContainersQueueStubBuilder(StubBuilder *parent, GateRef globalEnv)
: BuiltinsStubBuilder(parent, globalEnv) {}
~ContainersQueueStubBuilder() override = default;
NO_MOVE_SEMANTIC(ContainersQueueStubBuilder);
NO_COPY_SEMANTIC(ContainersQueueStubBuilder);
void GenerateCircuit() override {}
#define DECLARE_CONTAINERS_QUEUE_STUB_BUILDER(method, ...) \
void method(GateRef glue, GateRef thisValue, GateRef numArgs, Variable *result, Label *exit, Label *slowPath);
BUILTINS_WITH_CONTAINERS_QUEUE_STUB_BUILDER(DECLARE_CONTAINERS_QUEUE_STUB_BUILDER)
#undef DECLARE_CONTAINERS_QUEUE_STUB_BUILDER
GateRef GetArrayLength(GateRef glue, GateRef obj)
{
auto env = GetEnvironment();
Label entry(env);
env->SubCfgEntry(&entry);
Label exit(env);
Label endGreatBeging(env);
Label endNotGreatBeging(env);
DEFVARIABLE(length, VariableType::INT32(), Int32(0));
GateRef begin = LoadPrimitive(VariableType::INT32(), obj, IntPtr(JSAPIQueue::FRONT_OFFSET));
GateRef end = LoadPrimitive(VariableType::INT32(), obj, IntPtr(JSAPIQueue::TAIL_OFFSET));
BRANCH(Int32GreaterThanOrEqual(end, begin), &endGreatBeging, &endNotGreatBeging);
Bind(&endGreatBeging);
{
length = Int32Sub(end, begin);
Jump(&exit);
}
Bind(&endNotGreatBeging);
{
GateRef elementsOffset = IntPtr(JSObject::ELEMENTS_OFFSET);
GateRef elements = Load(VariableType::JS_POINTER(), glue, obj, elementsOffset);
GateRef elementsSize = LoadPrimitive(VariableType::INT32(), elements, IntPtr(TaggedArray::LENGTH_OFFSET));
length = Int32Add(Int32Sub(end, begin), elementsSize);
Jump(&exit);
}
Bind(&exit);
auto ret = *length;
env->SubCfgExit();
return ret;
}
GateRef Get(GateRef glue, GateRef obj, GateRef index)
{
GateRef elementsOffset = IntPtr(JSObject::ELEMENTS_OFFSET);
GateRef elements = Load(VariableType::JS_POINTER(), glue, obj, elementsOffset);
GateRef capacity = LoadPrimitive(VariableType::INT32(), elements, IntPtr(TaggedArray::LENGTH_OFFSET));
GateRef front = LoadPrimitive(VariableType::INT32(), obj, IntPtr(JSAPIQueue::FRONT_OFFSET));
GateRef curIndex = Int32Mod(Int32Add(front, index), capacity);
return GetValueFromTaggedArray(glue, elements, curIndex);
}
GateRef GetNextPosition(GateRef glue, GateRef obj, GateRef index)
{
GateRef elementsOffset = IntPtr(JSObject::ELEMENTS_OFFSET);
GateRef elements = Load(VariableType::JS_POINTER(), glue, obj, elementsOffset);
GateRef elementsSize = LoadPrimitive(VariableType::INT32(), elements, IntPtr(TaggedArray::LENGTH_OFFSET));
return Int32Mod(Int32Add(index, Int32(1)), elementsSize);
}
GateRef GetCurrentFront(GateRef obj)
{
return LoadPrimitive(VariableType::INT32(), obj, IntPtr(JSAPIQueue::FRONT_OFFSET));
}
};
}
#endif