// Copyright 2017 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef V8_OBJECTS_JS_REGEXP_INL_H_ #define V8_OBJECTS_JS_REGEXP_INL_H_ #include "src/objects/js-regexp.h" #include "src/objects/js-array-inl.h" #include "src/objects/objects-inl.h" // Needed for write barriers #include "src/objects/smi.h" #include "src/objects/string.h" // Has to be the last include (doesn't have include guards): #include "src/objects/object-macros.h" namespace v8 { namespace internal { #include "torque-generated/src/objects/js-regexp-tq-inl.inc" TQ_OBJECT_CONSTRUCTORS_IMPL(JSRegExp) TQ_OBJECT_CONSTRUCTORS_IMPL(JSRegExpResult) TQ_OBJECT_CONSTRUCTORS_IMPL(JSRegExpResultIndices) TQ_OBJECT_CONSTRUCTORS_IMPL(JSRegExpResultWithIndices) ACCESSORS(JSRegExp, last_index, Object, kLastIndexOffset) JSRegExp::Type JSRegExp::type_tag() const { Object data = this->data(); if (data.IsUndefined()) return JSRegExp::NOT_COMPILED; Smi smi = Smi::cast(FixedArray::cast(data).get(kTagIndex)); return static_cast(smi.value()); } int JSRegExp::capture_count() const { switch (type_tag()) { case ATOM: return 0; case EXPERIMENTAL: case IRREGEXP: return Smi::ToInt(DataAt(kIrregexpCaptureCountIndex)); default: UNREACHABLE(); } } int JSRegExp::max_register_count() const { CHECK_EQ(type_tag(), IRREGEXP); return Smi::ToInt(DataAt(kIrregexpMaxRegisterCountIndex)); } String JSRegExp::atom_pattern() const { DCHECK_EQ(type_tag(), ATOM); return String::cast(DataAt(JSRegExp::kAtomPatternIndex)); } String JSRegExp::source() const { return String::cast(TorqueGeneratedClass::source()); } JSRegExp::Flags JSRegExp::flags() const { Smi smi = Smi::cast(TorqueGeneratedClass::flags()); return Flags(smi.value()); } String JSRegExp::EscapedPattern() { DCHECK(this->source().IsString()); return String::cast(source()); } Object JSRegExp::capture_name_map() { DCHECK(TypeSupportsCaptures(type_tag())); Object value = DataAt(kIrregexpCaptureNameMapIndex); DCHECK_NE(value, Smi::FromInt(JSRegExp::kUninitializedValue)); return value; } void JSRegExp::set_capture_name_map(Handle capture_name_map) { if (capture_name_map.is_null()) { SetDataAt(JSRegExp::kIrregexpCaptureNameMapIndex, Smi::zero()); } else { SetDataAt(JSRegExp::kIrregexpCaptureNameMapIndex, *capture_name_map); } } Object JSRegExp::DataAt(int index) const { DCHECK(type_tag() != NOT_COMPILED); return FixedArray::cast(data()).get(index); } void JSRegExp::SetDataAt(int index, Object value) { DCHECK(type_tag() != NOT_COMPILED); // Only implementation data can be set this way. DCHECK_GE(index, kFirstTypeSpecificIndex); FixedArray::cast(data()).set(index, value); } bool JSRegExp::HasCompiledCode() const { if (type_tag() != IRREGEXP) return false; Smi uninitialized = Smi::FromInt(kUninitializedValue); #ifdef DEBUG DCHECK(DataAt(kIrregexpLatin1CodeIndex).IsCodeT() || DataAt(kIrregexpLatin1CodeIndex) == uninitialized); DCHECK(DataAt(kIrregexpUC16CodeIndex).IsCodeT() || DataAt(kIrregexpUC16CodeIndex) == uninitialized); DCHECK(DataAt(kIrregexpLatin1BytecodeIndex).IsByteArray() || DataAt(kIrregexpLatin1BytecodeIndex) == uninitialized); DCHECK(DataAt(kIrregexpUC16BytecodeIndex).IsByteArray() || DataAt(kIrregexpUC16BytecodeIndex) == uninitialized); #endif // DEBUG return (DataAt(kIrregexpLatin1CodeIndex) != uninitialized || DataAt(kIrregexpUC16CodeIndex) != uninitialized); } void JSRegExp::DiscardCompiledCodeForSerialization() { DCHECK(HasCompiledCode()); Smi uninitialized = Smi::FromInt(kUninitializedValue); SetDataAt(kIrregexpLatin1CodeIndex, uninitialized); SetDataAt(kIrregexpUC16CodeIndex, uninitialized); SetDataAt(kIrregexpLatin1BytecodeIndex, uninitialized); SetDataAt(kIrregexpUC16BytecodeIndex, uninitialized); } } // namespace internal } // namespace v8 #include "src/objects/object-macros-undef.h" #endif // V8_OBJECTS_JS_REGEXP_INL_H_