Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(302)

Side by Side Diff: src/assembler.h

Issue 476323004: Start adding an integrated assembler. (Closed) Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
Patch Set: lit test to check encodings, swap complexi8 Created 6 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
2 // for details. All rights reserved. Use of this source code is governed by a
3 // BSD-style license that can be found in the LICENSE file.
4 //
5 // Modified by the Subzero authors.
6 //
7 //===- subzero/src/assembler.h - Integrated assembler -----------*- C++ -*-===//
8 //
9 // The Subzero Code Generator
10 //
11 // This file is distributed under the University of Illinois Open Source
12 // License. See LICENSE.TXT for details.
13 //
14 //===----------------------------------------------------------------------===//
15 //
16 // This file declares the Assembler base class. Instructions are assembled
17 // by architecture-specific assemblers that derive from this base class.
18 // This base class manages buffers and fixups for emitting code, etc.
19 //
20 //===----------------------------------------------------------------------===//
21
22 #ifndef SUBZERO_SRC_ASSEMBLER_H
23 #define SUBZERO_SRC_ASSEMBLER_H
24
25 #include "IceDefs.h"
26
27 #include "llvm/Support/Allocator.h"
28
29 namespace Ice {
30
31 // Forward declarations.
32 class Assembler;
33 class AssemblerFixup;
34 class AssemblerBuffer;
35 class MemoryRegion;
36
37 // TODO(jvoung): Do we care if uword was uintptr_t or not?
38
39 // Assembler fixups are positions in generated code that hold relocation
40 // information that needs to be processed before finalizing the code
41 // into executable memory.
42 class AssemblerFixup {
43 public:
44 virtual void Process(const MemoryRegion &region, intptr_t position) = 0;
45
46 // It would be ideal if the destructor method could be made private,
47 // but the g++ compiler complains when this is subclassed.
48 virtual ~AssemblerFixup() { llvm_unreachable("~AssemblerFixup used"); }
49
50 private:
51 AssemblerFixup *previous_;
52 intptr_t position_;
53
54 AssemblerFixup *previous() const { return previous_; }
55 void set_previous(AssemblerFixup *previous) { previous_ = previous; }
56
57 intptr_t position() const { return position_; }
58 void set_position(intptr_t position) { position_ = position; }
59
60 friend class AssemblerBuffer;
61 };
62
63 // Assembler buffers are used to emit binary code. They grow on demand.
64 class AssemblerBuffer {
65 public:
66 AssemblerBuffer(Assembler &);
67 ~AssemblerBuffer();
68
69 // Basic support for emitting, loading, and storing.
70 template <typename T> void Emit(T value) {
71 assert(HasEnsuredCapacity());
72 *reinterpret_cast<T *>(cursor_) = value;
73 cursor_ += sizeof(T);
74 }
75
76 template <typename T> T Load(intptr_t position) const {
77 assert(position >= 0 &&
78 position <= (Size() - static_cast<intptr_t>(sizeof(T))));
79 return *reinterpret_cast<T *>(contents_ + position);
80 }
81
82 template <typename T> void Store(intptr_t position, T value) {
83 assert(position >= 0 &&
84 position <= (Size() - static_cast<intptr_t>(sizeof(T))));
85 *reinterpret_cast<T *>(contents_ + position) = value;
86 }
87
88 // Emit a fixup at the current location.
89 void EmitFixup(AssemblerFixup *fixup) {
90 fixup->set_previous(fixup_);
91 fixup->set_position(Size());
92 fixup_ = fixup;
93 }
94
95 // Get the size of the emitted code.
96 intptr_t Size() const { return cursor_ - contents_; }
97 uintptr_t contents() const { return contents_; }
98
99 // Copy the assembled instructions into the specified memory block
100 // and apply all fixups.
101 void FinalizeInstructions(const MemoryRegion &region);
102
103 // To emit an instruction to the assembler buffer, the EnsureCapacity helper
104 // must be used to guarantee that the underlying data area is big enough to
105 // hold the emitted instruction. Usage:
106 //
107 // AssemblerBuffer buffer;
108 // AssemblerBuffer::EnsureCapacity ensured(&buffer);
109 // ... emit bytes for single instruction ...
110
111 #if defined(DEBUG)
112 class EnsureCapacity {
113 public:
114 explicit EnsureCapacity(AssemblerBuffer *buffer);
115 ~EnsureCapacity();
116
117 private:
118 AssemblerBuffer *buffer_;
119 intptr_t gap_;
120
121 intptr_t ComputeGap() { return buffer_->Capacity() - buffer_->Size(); }
122 };
123
124 bool has_ensured_capacity_;
125 bool HasEnsuredCapacity() const { return has_ensured_capacity_; }
126 #else
127 class EnsureCapacity {
128 public:
129 explicit EnsureCapacity(AssemblerBuffer *buffer) {
130 if (buffer->cursor() >= buffer->limit())
131 buffer->ExtendCapacity();
132 }
133 };
134
135 // When building the C++ tests, assertion code is enabled. To allow
136 // asserting that the user of the assembler buffer has ensured the
137 // capacity needed for emitting, we add a dummy method in non-debug mode.
138 bool HasEnsuredCapacity() const { return true; }
139 #endif
140
141 // Returns the position in the instruction stream.
142 intptr_t GetPosition() const { return cursor_ - contents_; }
143
144 private:
145 // The limit is set to kMinimumGap bytes before the end of the data area.
146 // This leaves enough space for the longest possible instruction and allows
147 // for a single, fast space check per instruction.
148 static const intptr_t kMinimumGap = 32;
149
150 uintptr_t contents_;
151 uintptr_t cursor_;
152 uintptr_t limit_;
153 Assembler &assembler_;
154 AssemblerFixup *fixup_;
155 #if defined(DEBUG)
156 bool fixups_processed_;
157 #endif
158
159 uintptr_t cursor() const { return cursor_; }
160 uintptr_t limit() const { return limit_; }
161 intptr_t Capacity() const {
162 assert(limit_ >= contents_);
163 return (limit_ - contents_) + kMinimumGap;
164 }
165
166 // Process the fixup chain.
167 void ProcessFixups(const MemoryRegion &region);
168
169 // Compute the limit based on the data area and the capacity. See
170 // description of kMinimumGap for the reasoning behind the value.
171 static uintptr_t ComputeLimit(uintptr_t data, intptr_t capacity) {
172 return data + capacity - kMinimumGap;
173 }
174
175 void ExtendCapacity();
176
177 friend class AssemblerFixup;
178 };
179
180 class Assembler {
181 public:
182 Assembler() {}
183 ~Assembler() {}
184
185 // Allocate a chunk of bytes using the per-Assembler allocator.
186 uintptr_t AllocateBytes(size_t bytes) {
187 // re: Alignment -- We just need to be able to write bytes to this,
188 // so there's not a real requirement (e.g., not related to NaCl bundle
189 // alignment). When code is finalized and copied out to the .o file's
190 // text section, then we will care about alignment.
191 const size_t Alignment = 16;
192 return reinterpret_cast<uintptr_t>(Allocator.Allocate(bytes, Alignment));
193 }
194
195 // Allocate data of type T using the per-Assembler allocator.
196 template <typename T> T *Allocate() { return Allocator.Allocate<T>(); }
197
198 private:
199 llvm::BumpPtrAllocator Allocator;
200
201 Assembler(const Assembler &) LLVM_DELETED_FUNCTION;
202 Assembler &operator=(const Assembler &) LLVM_DELETED_FUNCTION;
203 };
204
205 } // end of namespace Ice
206
207 #endif // SUBZERO_SRC_ASSEMBLER_H_
OLDNEW
« src/IceUtils.h ('K') | « src/IceUtils.h ('k') | src/assembler.cpp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698