codegen.cc 7.76 KB
Newer Older
1
// Copyright 2012 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 6
#include "src/v8.h"

7 8 9
#if defined(V8_OS_AIX)
#include <fenv.h>
#endif
10 11 12 13 14
#include "src/bootstrapper.h"
#include "src/codegen.h"
#include "src/compiler.h"
#include "src/cpu-profiler.h"
#include "src/debug.h"
15
#include "src/parser.h"
16 17
#include "src/prettyprinter.h"
#include "src/rewriter.h"
18
#include "src/runtime/runtime.h"
19

20 21
namespace v8 {
namespace internal {
22

23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54

#if defined(_WIN64)
typedef double (*ModuloFunction)(double, double);
static ModuloFunction modulo_function = NULL;
// Defined in codegen-x64.cc.
ModuloFunction CreateModuloFunction();

void init_modulo_function() {
  modulo_function = CreateModuloFunction();
}


double modulo(double x, double y) {
  // Note: here we rely on dependent reads being ordered. This is true
  // on all architectures we currently support.
  return (*modulo_function)(x, y);
}
#elif defined(_WIN32)

double modulo(double x, double y) {
  // Workaround MS fmod bugs. ECMA-262 says:
  // dividend is finite and divisor is an infinity => result equals dividend
  // dividend is a zero and divisor is nonzero finite => result equals dividend
  if (!(std::isfinite(x) && (!std::isfinite(y) && !std::isnan(y))) &&
      !(x == 0 && (y != 0 && std::isfinite(y)))) {
    x = fmod(x, y);
  }
  return x;
}
#else  // POSIX

double modulo(double x, double y) {
55 56 57 58 59 60 61
#if defined(V8_OS_AIX)
  // AIX raises an underflow exception for (Number.MIN_VALUE % Number.MAX_VALUE)
  feclearexcept(FE_ALL_EXCEPT);
  double result = std::fmod(x, y);
  int exception = fetestexcept(FE_UNDERFLOW);
  return (exception ? x : result);
#else
62
  return std::fmod(x, y);
63
#endif
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
}
#endif  // defined(_WIN64)


#define UNARY_MATH_FUNCTION(name, generator)             \
static UnaryMathFunction fast_##name##_function = NULL;  \
void init_fast_##name##_function() {                     \
  fast_##name##_function = generator;                    \
}                                                        \
double fast_##name(double x) {                           \
  return (*fast_##name##_function)(x);                   \
}

UNARY_MATH_FUNCTION(exp, CreateExpFunction())
UNARY_MATH_FUNCTION(sqrt, CreateSqrtFunction())

#undef UNARY_MATH_FUNCTION


void lazily_initialize_fast_exp() {
  if (fast_exp_function == NULL) {
    init_fast_exp_function();
  }
}


90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
#define __ ACCESS_MASM(masm_)

#ifdef DEBUG

Comment::Comment(MacroAssembler* masm, const char* msg)
    : masm_(masm), msg_(msg) {
  __ RecordComment(msg);
}


Comment::~Comment() {
  if (msg_[0] == '[') __ RecordComment("]");
}

#endif  // DEBUG

#undef __

108

109
void CodeGenerator::MakeCodePrologue(CompilationInfo* info, const char* kind) {
110 111 112 113
  bool print_source = false;
  bool print_ast = false;
  const char* ftype;

114
  if (info->isolate()->bootstrapper()->IsActive()) {
115 116 117 118 119 120 121 122 123 124
    print_source = FLAG_print_builtin_source;
    print_ast = FLAG_print_builtin_ast;
    ftype = "builtin";
  } else {
    print_source = FLAG_print_source;
    print_ast = FLAG_print_ast;
    ftype = "user-defined";
  }

  if (FLAG_trace_codegen || print_source || print_ast) {
125
    PrintF("[generating %s code for %s function: ", kind, ftype);
126 127 128 129 130
    if (info->IsStub()) {
      const char* name =
          CodeStub::MajorName(info->code_stub()->MajorKey(), true);
      PrintF("%s", name == NULL ? "<unknown>" : name);
    } else {
131
      AllowDeferredHandleDereference allow_deference_for_trace;
132
      PrintF("%s", info->function()->debug_name()->ToCString().get());
133
    }
134
    PrintF("]\n");
135 136
  }

137
#ifdef DEBUG
138
  if (info->parse_info() && print_source) {
139
    PrintF("--- Source from AST ---\n%s\n",
140 141
           PrettyPrinter(info->isolate(), info->zone())
               .PrintProgram(info->function()));
142 143
  }

144
  if (info->parse_info() && print_ast) {
145 146
    PrintF("--- AST ---\n%s\n", AstPrinter(info->isolate(), info->zone())
                                    .PrintProgram(info->function()));
147 148
  }
#endif  // DEBUG
149
}
150 151


152
Handle<Code> CodeGenerator::MakeCodeEpilogue(MacroAssembler* masm,
153
                                             Code::Flags flags,
154
                                             CompilationInfo* info) {
155 156
  Isolate* isolate = info->isolate();

157
  // Allocate and install the code.
158
  CodeDesc desc;
159 160 161
  bool is_crankshafted =
      Code::ExtractKindFromFlags(flags) == Code::OPTIMIZED_FUNCTION ||
      info->IsStub();
162
  masm->GetCode(&desc);
163
  Handle<Code> code =
164
      isolate->factory()->NewCode(desc, flags, masm->CodeObject(),
165
                                  false, is_crankshafted,
166
                                  info->prologue_offset(),
167
                                  info->is_debug() && !is_crankshafted);
168 169
  isolate->counters()->total_compiled_code_size()->Increment(
      code->instruction_size());
170 171
  isolate->heap()->IncrementCodeGeneratedBytes(is_crankshafted,
      code->instruction_size());
172 173 174 175 176
  return code;
}


void CodeGenerator::PrintCode(Handle<Code> code, CompilationInfo* info) {
177
#ifdef ENABLE_DISASSEMBLER
178
  AllowDeferredHandleDereference allow_deference_for_print_code;
179
  bool print_code = info->isolate()->bootstrapper()->IsActive()
180
      ? FLAG_print_builtin_code
181 182 183
      : (FLAG_print_code ||
         (info->IsStub() && FLAG_print_code_stubs) ||
         (info->IsOptimizing() && FLAG_print_opt_code));
184
  if (print_code) {
185
    const char* debug_name;
186
    SmartArrayPointer<char> debug_name_holder;
187 188 189 190
    if (info->IsStub()) {
      CodeStub::Major major_key = info->code_stub()->MajorKey();
      debug_name = CodeStub::MajorName(major_key, false);
    } else {
191 192 193
      debug_name_holder =
          info->parse_info()->function()->debug_name()->ToCString();
      debug_name = debug_name_holder.get();
194
    }
195 196

    CodeTracer::Scope tracing_scope(info->isolate()->GetCodeTracer());
197
    OFStream os(tracing_scope.file());
198 199 200 201 202 203

    // Print the source code if available.
    FunctionLiteral* function = nullptr;
    bool print_source =
        info->parse_info() && (code->kind() == Code::OPTIMIZED_FUNCTION ||
                               code->kind() == Code::FUNCTION);
204
    if (print_source) {
205
      function = info->function();
206 207
      Handle<Script> script = info->script();
      if (!script->IsUndefined() && !script->source()->IsUndefined()) {
208
        os << "--- Raw source ---\n";
209 210
        StringCharacterStream stream(String::cast(script->source()),
                                     function->start_position());
211 212 213 214 215
        // fun->end_position() points to the last character in the stream. We
        // need to compensate by adding one to calculate the length.
        int source_len =
            function->end_position() - function->start_position() + 1;
        for (int i = 0; i < source_len; i++) {
216
          if (stream.HasMore()) {
217
            os << AsReversiblyEscapedUC16(stream.GetNext());
218
          }
219
        }
220
        os << "\n\n";
221 222
      }
    }
223
    if (info->IsOptimizing()) {
224
      if (FLAG_print_unopt_code && info->parse_info()) {
225
        os << "--- Unoptimized code ---\n";
226
        info->closure()->shared()->code()->Disassemble(debug_name, os);
227
      }
228 229
      os << "--- Optimized code ---\n"
         << "optimization_id = " << info->optimization_id() << "\n";
230
    } else {
231
      os << "--- Code ---\n";
232
    }
233
    if (print_source) {
234
      os << "source_position = " << function->start_position() << "\n";
235
    }
236
    code->Disassemble(debug_name, os);
237
    os << "--- End code ---\n";
238 239 240 241
  }
#endif  // ENABLE_DISASSEMBLER
}

242

243 244 245
bool CodeGenerator::RecordPositions(MacroAssembler* masm,
                                    int pos,
                                    bool right_here) {
246
  if (pos != RelocInfo::kNoPosition) {
247 248
    masm->positions_recorder()->RecordStatementPosition(pos);
    masm->positions_recorder()->RecordPosition(pos);
249
    if (right_here) {
250
      return masm->positions_recorder()->WriteRecordedPositions();
251
    }
252
  }
253
  return false;
254 255
}

256 257
}  // namespace internal
}  // namespace v8