lithium-codegen.cc 8.41 KB
Newer Older
1
// Copyright 2013 the V8 project authors. All rights reserved.
2 3
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
4

5
#include "src/lithium-codegen.h"
6

7 8 9 10
#include <sstream>

#include "src/v8.h"

11
#if V8_TARGET_ARCH_IA32
12 13
#include "src/ia32/lithium-ia32.h"  // NOLINT
#include "src/ia32/lithium-codegen-ia32.h"  // NOLINT
14
#elif V8_TARGET_ARCH_X64
15 16
#include "src/x64/lithium-x64.h"  // NOLINT
#include "src/x64/lithium-codegen-x64.h"  // NOLINT
17
#elif V8_TARGET_ARCH_ARM
18 19
#include "src/arm/lithium-arm.h"  // NOLINT
#include "src/arm/lithium-codegen-arm.h"  // NOLINT
20
#elif V8_TARGET_ARCH_ARM64
21 22
#include "src/arm64/lithium-arm64.h"  // NOLINT
#include "src/arm64/lithium-codegen-arm64.h"  // NOLINT
23
#elif V8_TARGET_ARCH_MIPS
24 25
#include "src/mips/lithium-mips.h"  // NOLINT
#include "src/mips/lithium-codegen-mips.h"  // NOLINT
26 27 28
#elif V8_TARGET_ARCH_MIPS64
#include "src/mips64/lithium-mips64.h"  // NOLINT
#include "src/mips64/lithium-codegen-mips64.h"  // NOLINT
danno@chromium.org's avatar
danno@chromium.org committed
29
#elif V8_TARGET_ARCH_X87
30 31
#include "src/x87/lithium-x87.h"  // NOLINT
#include "src/x87/lithium-codegen-x87.h"  // NOLINT
32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
#else
#error Unsupported target architecture.
#endif

namespace v8 {
namespace internal {


HGraph* LCodeGenBase::graph() const {
  return chunk()->graph();
}


LCodeGenBase::LCodeGenBase(LChunk* chunk,
                           MacroAssembler* assembler,
                           CompilationInfo* info)
    : chunk_(static_cast<LPlatformChunk*>(chunk)),
      masm_(assembler),
      info_(info),
      zone_(info->zone()),
      status_(UNUSED),
      current_block_(-1),
      current_instruction_(-1),
      instructions_(chunk->instructions()),
      last_lazy_deopt_pc_(0) {
}


bool LCodeGenBase::GenerateBody() {
61
  DCHECK(is_generating());
62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93
  bool emit_instructions = true;
  LCodeGen* codegen = static_cast<LCodeGen*>(this);
  for (current_instruction_ = 0;
       !is_aborted() && current_instruction_ < instructions_->length();
       current_instruction_++) {
    LInstruction* instr = instructions_->at(current_instruction_);

    // Don't emit code for basic blocks with a replacement.
    if (instr->IsLabel()) {
      emit_instructions = !LLabel::cast(instr)->HasReplacement() &&
          (!FLAG_unreachable_code_elimination ||
           instr->hydrogen_value()->block()->IsReachable());
      if (FLAG_code_comments && !emit_instructions) {
        Comment(
            ";;; <@%d,#%d> -------------------- B%d (unreachable/replaced) "
            "--------------------",
            current_instruction_,
            instr->hydrogen_value()->id(),
            instr->hydrogen_value()->block()->block_id());
      }
    }
    if (!emit_instructions) continue;

    if (FLAG_code_comments && instr->HasInterestingComment(codegen)) {
      Comment(";;; <@%d,#%d> %s",
              current_instruction_,
              instr->hydrogen_value()->id(),
              instr->Mnemonic());
    }

    GenerateBodyInstructionPre(instr);

94
    HValue* value = instr->hydrogen_value();
95 96 97
    if (!value->position().IsUnknown()) {
      RecordAndWritePosition(
        chunk()->graph()->SourcePositionToScriptPosition(value->position()));
98
    }
99 100 101 102 103 104 105 106 107 108 109

    instr->CompileToNative(codegen);

    GenerateBodyInstructionPost(instr);
  }
  EnsureSpaceForLazyDeopt(Deoptimizer::patch_size());
  last_lazy_deopt_pc_ = masm()->pc_offset();
  return !is_aborted();
}


110 111
void LCodeGenBase::CheckEnvironmentUsage() {
#ifdef DEBUG
112
  bool dead_block = false;
113 114
  for (int i = 0; i < instructions_->length(); i++) {
    LInstruction* instr = instructions_->at(i);
115 116 117 118 119 120
    HValue* hval = instr->hydrogen_value();
    if (instr->IsLabel()) dead_block = LLabel::cast(instr)->HasReplacement();
    if (dead_block || !hval->block()->IsReachable()) continue;

    HInstruction* hinstr = HInstruction::cast(hval);
    if (!hinstr->CanDeoptimize() && instr->HasEnvironment()) {
121
      V8_Fatal(__FILE__, __LINE__, "CanDeoptimize is wrong for %s (%s)",
122 123 124 125
               hinstr->Mnemonic(), instr->Mnemonic());
    }

    if (instr->HasEnvironment() && !instr->environment()->has_been_used()) {
126
      V8_Fatal(__FILE__, __LINE__, "unused environment for %s (%s)",
127
               hinstr->Mnemonic(), instr->Mnemonic());
128 129 130 131 132 133
    }
  }
#endif
}


134 135 136
void LCodeGenBase::Comment(const char* format, ...) {
  if (!FLAG_code_comments) return;
  char buffer[4 * KB];
137
  StringBuilder builder(buffer, arraysize(buffer));
138 139 140 141 142 143 144 145
  va_list arguments;
  va_start(arguments, format);
  builder.AddFormattedList(format, arguments);
  va_end(arguments);

  // Copy the string before recording it in the assembler to avoid
  // issues when the stack allocated buffer goes out of scope.
  size_t length = builder.position();
146
  Vector<char> copy = Vector<char>::New(static_cast<int>(length) + 1);
147
  MemCopy(copy.start(), builder.Finalize(), copy.length());
148 149 150 151
  masm()->RecordComment(copy.start());
}


152
void LCodeGenBase::DeoptComment(const Deoptimizer::Reason& reason) {
153
  std::ostringstream os;
154 155 156
  os << ";;; deoptimize at " << HSourcePosition(reason.raw_position) << " "
     << reason.mnemonic;
  if (reason.detail != NULL) os << ": " << reason.detail;
157
  Comment("%s", os.str().c_str());
158 159 160
}


161 162
int LCodeGenBase::GetNextEmittedBlock() const {
  for (int i = current_block_ + 1; i < graph()->blocks()->length(); ++i) {
163
    if (!graph()->blocks()->at(i)->IsReachable()) continue;
164 165 166 167 168 169
    if (!chunk_->GetLabel(i)->HasReplacement()) return i;
  }
  return -1;
}


170 171 172 173 174
static void AddWeakObjectToCodeDependency(Isolate* isolate,
                                          Handle<Object> object,
                                          Handle<Code> code) {
  Heap* heap = isolate->heap();
  heap->EnsureWeakObjectToCodeTable();
175
  Handle<DependentCode> dep(heap->LookupWeakObjectToCodeDependency(object));
176
  dep = DependentCode::Insert(dep, DependentCode::kWeakCodeGroup, code);
177
  heap->AddWeakObjectToCodeDependency(object, dep);
178 179 180
}


181
void LCodeGenBase::RegisterWeakObjectsInOptimizedCode(Handle<Code> code) {
182
  DCHECK(code->is_optimized_code());
183 184
  ZoneList<Handle<Map> > maps(1, zone());
  ZoneList<Handle<JSObject> > objects(1, zone());
185 186 187
  ZoneList<Handle<Cell> > cells(1, zone());
  int mode_mask = RelocInfo::ModeMask(RelocInfo::EMBEDDED_OBJECT) |
                  RelocInfo::ModeMask(RelocInfo::CELL);
188
  for (RelocIterator it(*code, mode_mask); !it.done(); it.next()) {
189 190
    RelocInfo::Mode mode = it.rinfo()->rmode();
    if (mode == RelocInfo::CELL &&
191
        code->IsWeakObjectInOptimizedCode(it.rinfo()->target_cell())) {
192 193 194
      Handle<Cell> cell(it.rinfo()->target_cell());
      cells.Add(cell, zone());
    } else if (mode == RelocInfo::EMBEDDED_OBJECT &&
195
               code->IsWeakObjectInOptimizedCode(it.rinfo()->target_object())) {
196 197 198 199 200 201
      if (it.rinfo()->target_object()->IsMap()) {
        Handle<Map> map(Map::cast(it.rinfo()->target_object()));
        maps.Add(map, zone());
      } else if (it.rinfo()->target_object()->IsJSObject()) {
        Handle<JSObject> object(JSObject::cast(it.rinfo()->target_object()));
        objects.Add(object, zone());
202 203 204
      } else if (it.rinfo()->target_object()->IsCell()) {
        Handle<Cell> cell(Cell::cast(it.rinfo()->target_object()));
        cells.Add(cell, zone());
205 206 207
      }
    }
  }
208 209 210 211
  if (FLAG_enable_ool_constant_pool) {
    code->constant_pool()->set_weak_object_state(
        ConstantPoolArray::WEAK_OBJECTS_IN_OPTIMIZED_CODE);
  }
212 213 214 215 216 217 218
#ifdef VERIFY_HEAP
  // This disables verification of weak embedded objects after full GC.
  // AddDependentCode can cause a GC, which would observe the state where
  // this code is not yet in the depended code lists of the embedded maps.
  NoWeakObjectVerificationScope disable_verification_of_embedded_objects;
#endif
  for (int i = 0; i < maps.length(); i++) {
219
    Map::AddDependentCode(maps.at(i), DependentCode::kWeakCodeGroup, code);
220 221
  }
  for (int i = 0; i < objects.length(); i++) {
222
    AddWeakObjectToCodeDependency(isolate(), objects.at(i), code);
223
  }
224
  for (int i = 0; i < cells.length(); i++) {
225
    AddWeakObjectToCodeDependency(isolate(), cells.at(i), code);
226
  }
227 228 229
}


230
void LCodeGenBase::Abort(BailoutReason reason) {
231 232 233 234 235 236 237
  info()->AbortOptimization(reason);
  status_ = ABORTED;
}


void LCodeGenBase::Retry(BailoutReason reason) {
  info()->RetryOptimization(reason);
238 239 240 241
  status_ = ABORTED;
}


242
void LCodeGenBase::AddDeprecationDependency(Handle<Map> map) {
243
  if (map->is_deprecated()) return Retry(kMapBecameDeprecated);
244 245 246 247
  chunk_->AddDeprecationDependency(map);
}


248
void LCodeGenBase::AddStabilityDependency(Handle<Map> map) {
249
  if (!map->is_stable()) return Retry(kMapBecameUnstable);
250 251 252
  chunk_->AddStabilityDependency(map);
}

253
} }  // namespace v8::internal