1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
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
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
* vim: set ts=8 sts=2 et sw=2 tw=80:
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include "jit/Linker.h"
#include "gc/GC.h"
#include "jit/JitZone.h"
#include "util/Memory.h"
#include "gc/StoreBuffer-inl.h"
namespace js {
namespace jit {
JitCode* Linker::newCode(JSContext* cx, CodeKind kind) {
JS::AutoAssertNoGC nogc(cx);
if (masm.oom()) {
return fail(cx);
}
static const size_t ExecutableAllocatorAlignment = sizeof(void*);
static_assert(CodeAlignment >= ExecutableAllocatorAlignment,
"Unexpected alignment requirements");
// We require enough bytes for the code, header, and worst-case alignment
// padding.
size_t bytesNeeded = masm.bytesNeeded() + sizeof(JitCodeHeader) +
(CodeAlignment - ExecutableAllocatorAlignment);
if (bytesNeeded >= MAX_BUFFER_SIZE) {
return fail(cx);
}
// ExecutableAllocator requires bytesNeeded to be aligned.
bytesNeeded = AlignBytes(bytesNeeded, ExecutableAllocatorAlignment);
JitZone* jitZone = cx->zone()->getJitZone(cx);
if (!jitZone) {
// Note: don't call fail(cx) here, getJitZone reports OOM.
return nullptr;
}
ExecutablePool* pool;
uint8_t* result =
(uint8_t*)jitZone->execAlloc().alloc(cx, bytesNeeded, &pool, kind);
if (!result) {
return fail(cx);
}
// The JitCodeHeader will be stored right before the code buffer.
uint8_t* codeStart = result + sizeof(JitCodeHeader);
// Bump the code up to a nice alignment.
codeStart = (uint8_t*)AlignBytes((uintptr_t)codeStart, CodeAlignment);
MOZ_ASSERT(codeStart + masm.bytesNeeded() <= result + bytesNeeded);
uint32_t headerSize = codeStart - result;
JitCode* code =
JitCode::New<NoGC>(cx, codeStart, bytesNeeded, headerSize, pool, kind);
if (!code) {
return fail(cx);
}
if (masm.oom()) {
return fail(cx);
}
awjcf.emplace(result, bytesNeeded);
if (!awjcf->makeWritable()) {
return fail(cx);
}
code->copyFrom(masm);
masm.link(code);
if (masm.embedsNurseryPointers()) {
cx->runtime()->gc.storeBuffer().putWholeCell(code);
}
return code;
}
} // namespace jit
} // namespace js
|