1 | //= loongarch.h - Generic JITLink loongarch edge kinds, utilities -*- C++ -*-=// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | // |
9 | // Generic utilities for graphs representing loongarch objects. |
10 | // |
11 | //===----------------------------------------------------------------------===// |
12 | |
13 | #ifndef LLVM_EXECUTIONENGINE_JITLINK_LOONGARCH_H |
14 | #define LLVM_EXECUTIONENGINE_JITLINK_LOONGARCH_H |
15 | |
16 | #include "TableManager.h" |
17 | #include "llvm/ExecutionEngine/JITLink/JITLink.h" |
18 | #include "llvm/ExecutionEngine/Orc/Shared/MemoryFlags.h" |
19 | |
20 | namespace llvm { |
21 | namespace jitlink { |
22 | namespace loongarch { |
23 | |
24 | /// Represents loongarch fixups. |
25 | enum EdgeKind_loongarch : Edge::Kind { |
26 | /// A plain 64-bit pointer value relocation. |
27 | /// |
28 | /// Fixup expression: |
29 | /// Fixup <- Target + Addend : uint64 |
30 | /// |
31 | Pointer64 = Edge::FirstRelocation, |
32 | |
33 | /// A plain 32-bit pointer value relocation. |
34 | /// |
35 | /// Fixup expression: |
36 | /// Fixup <- Target + Addend : uint32 |
37 | /// |
38 | /// Errors: |
39 | /// - The target must reside in the low 32-bits of the address space, |
40 | /// otherwise an out-of-range error will be returned. |
41 | /// |
42 | Pointer32, |
43 | |
44 | /// A 26-bit PC-relative branch. |
45 | /// |
46 | /// Represents a PC-relative call or branch to a target within +/-128Mb. The |
47 | /// target must be 4-byte aligned. |
48 | /// |
49 | /// Fixup expression: |
50 | /// Fixup <- (Target - Fixup + Addend) >> 2 : int26 |
51 | /// |
52 | /// Notes: |
53 | /// The '26' in the name refers to the number operand bits and follows the |
54 | /// naming convention used by the corresponding ELF relocations. Since the low |
55 | /// two bits must be zero (because of the 4-byte alignment of the target) the |
56 | /// operand is effectively a signed 28-bit number. |
57 | /// |
58 | /// Errors: |
59 | /// - The result of the unshifted part of the fixup expression must be |
60 | /// 4-byte aligned otherwise an alignment error will be returned. |
61 | /// - The result of the fixup expression must fit into an int26 otherwise an |
62 | /// out-of-range error will be returned. |
63 | /// |
64 | Branch26PCRel, |
65 | |
66 | /// A 32-bit delta. |
67 | /// |
68 | /// Delta from the fixup to the target. |
69 | /// |
70 | /// Fixup expression: |
71 | /// Fixup <- Target - Fixup + Addend : int32 |
72 | /// |
73 | /// Errors: |
74 | /// - The result of the fixup expression must fit into an int32, otherwise |
75 | /// an out-of-range error will be returned. |
76 | /// |
77 | Delta32, |
78 | |
79 | /// A 32-bit negative delta. |
80 | /// |
81 | /// Delta from the target back to the fixup. |
82 | /// |
83 | /// Fixup expression: |
84 | /// Fixup <- Fixup - Target + Addend : int32 |
85 | /// |
86 | /// Errors: |
87 | /// - The result of the fixup expression must fit into an int32, otherwise |
88 | /// an out-of-range error will be returned. |
89 | /// |
90 | NegDelta32, |
91 | |
92 | /// A 64-bit delta. |
93 | /// |
94 | /// Delta from the fixup to the target. |
95 | /// |
96 | /// Fixup expression: |
97 | /// Fixup <- Target - Fixup + Addend : int64 |
98 | /// |
99 | Delta64, |
100 | |
101 | /// The signed 20-bit delta from the fixup page to the page containing the |
102 | /// target. |
103 | /// |
104 | /// Fixup expression: |
105 | /// Fixup <- (((Target + Addend + ((Target + Addend) & 0x800)) & ~0xfff) |
106 | // - (Fixup & ~0xfff)) >> 12 : int20 |
107 | /// |
108 | /// Notes: |
109 | /// For PCALAU12I fixups. |
110 | /// |
111 | /// Errors: |
112 | /// - The result of the fixup expression must fit into an int20 otherwise an |
113 | /// out-of-range error will be returned. |
114 | /// |
115 | Page20, |
116 | |
117 | /// The 12-bit offset of the target within its page. |
118 | /// |
119 | /// Typically used to fix up ADDI/LD_W/LD_D immediates. |
120 | /// |
121 | /// Fixup expression: |
122 | /// Fixup <- ((Target + Addend) >> Shift) & 0xfff : int12 |
123 | /// |
124 | PageOffset12, |
125 | |
126 | /// A GOT entry getter/constructor, transformed to Page20 pointing at the GOT |
127 | /// entry for the original target. |
128 | /// |
129 | /// Indicates that this edge should be transformed into a Page20 targeting |
130 | /// the GOT entry for the edge's current target, maintaining the same addend. |
131 | /// A GOT entry for the target should be created if one does not already |
132 | /// exist. |
133 | /// |
134 | /// Edges of this kind are usually handled by a GOT/PLT builder pass inserted |
135 | /// by default. |
136 | /// |
137 | /// Fixup expression: |
138 | /// NONE |
139 | /// |
140 | /// Errors: |
141 | /// - *ASSERTION* Failure to handle edges of this kind prior to the fixup |
142 | /// phase will result in an assert/unreachable during the fixup phase. |
143 | /// |
144 | RequestGOTAndTransformToPage20, |
145 | |
146 | /// A GOT entry getter/constructor, transformed to Pageoffset12 pointing at |
147 | /// the GOT entry for the original target. |
148 | /// |
149 | /// Indicates that this edge should be transformed into a PageOffset12 |
150 | /// targeting the GOT entry for the edge's current target, maintaining the |
151 | /// same addend. A GOT entry for the target should be created if one does not |
152 | /// already exist. |
153 | /// |
154 | /// Edges of this kind are usually handled by a GOT/PLT builder pass inserted |
155 | /// by default. |
156 | /// |
157 | /// Fixup expression: |
158 | /// NONE |
159 | /// |
160 | RequestGOTAndTransformToPageOffset12, |
161 | }; |
162 | |
163 | /// Returns a string name for the given loongarch edge. For debugging purposes |
164 | /// only. |
165 | const char *getEdgeKindName(Edge::Kind K); |
166 | |
167 | // Returns extract bits Val[Hi:Lo]. |
168 | inline uint32_t (uint32_t Val, unsigned Hi, unsigned Lo) { |
169 | return (Val & (((1UL << (Hi + 1)) - 1))) >> Lo; |
170 | } |
171 | |
172 | /// Apply fixup expression for edge to block content. |
173 | inline Error applyFixup(LinkGraph &G, Block &B, const Edge &E) { |
174 | using namespace support; |
175 | |
176 | char *BlockWorkingMem = B.getAlreadyMutableContent().data(); |
177 | char *FixupPtr = BlockWorkingMem + E.getOffset(); |
178 | uint64_t FixupAddress = (B.getAddress() + E.getOffset()).getValue(); |
179 | uint64_t TargetAddress = E.getTarget().getAddress().getValue(); |
180 | int64_t Addend = E.getAddend(); |
181 | |
182 | switch (E.getKind()) { |
183 | case Pointer64: |
184 | *(ulittle64_t *)FixupPtr = TargetAddress + Addend; |
185 | break; |
186 | case Pointer32: { |
187 | uint64_t Value = TargetAddress + Addend; |
188 | if (Value > std::numeric_limits<uint32_t>::max()) |
189 | return makeTargetOutOfRangeError(G, B, E); |
190 | *(ulittle32_t *)FixupPtr = Value; |
191 | break; |
192 | } |
193 | case Branch26PCRel: { |
194 | int64_t Value = TargetAddress - FixupAddress + Addend; |
195 | |
196 | if (!isInt<28>(x: Value)) |
197 | return makeTargetOutOfRangeError(G, B, E); |
198 | |
199 | if (!isShiftedInt<26, 2>(x: Value)) |
200 | return makeAlignmentError(Loc: orc::ExecutorAddr(FixupAddress), Value, N: 4, E); |
201 | |
202 | uint32_t RawInstr = *(little32_t *)FixupPtr; |
203 | uint32_t Imm = static_cast<uint32_t>(Value >> 2); |
204 | uint32_t Imm15_0 = extractBits(Val: Imm, /*Hi=*/Hi: 15, /*Lo=*/Lo: 0) << 10; |
205 | uint32_t Imm25_16 = extractBits(Val: Imm, /*Hi=*/Hi: 25, /*Lo=*/Lo: 16); |
206 | *(little32_t *)FixupPtr = RawInstr | Imm15_0 | Imm25_16; |
207 | break; |
208 | } |
209 | case Delta32: { |
210 | int64_t Value = TargetAddress - FixupAddress + Addend; |
211 | |
212 | if (!isInt<32>(x: Value)) |
213 | return makeTargetOutOfRangeError(G, B, E); |
214 | *(little32_t *)FixupPtr = Value; |
215 | break; |
216 | } |
217 | case NegDelta32: { |
218 | int64_t Value = FixupAddress - TargetAddress + Addend; |
219 | if (!isInt<32>(x: Value)) |
220 | return makeTargetOutOfRangeError(G, B, E); |
221 | *(little32_t *)FixupPtr = Value; |
222 | break; |
223 | } |
224 | case Delta64: |
225 | *(little64_t *)FixupPtr = TargetAddress - FixupAddress + Addend; |
226 | break; |
227 | case Page20: { |
228 | uint64_t Target = TargetAddress + Addend; |
229 | uint64_t TargetPage = |
230 | (Target + (Target & 0x800)) & ~static_cast<uint64_t>(0xfff); |
231 | uint64_t PCPage = FixupAddress & ~static_cast<uint64_t>(0xfff); |
232 | |
233 | int64_t PageDelta = TargetPage - PCPage; |
234 | if (!isInt<32>(x: PageDelta)) |
235 | return makeTargetOutOfRangeError(G, B, E); |
236 | |
237 | uint32_t RawInstr = *(little32_t *)FixupPtr; |
238 | uint32_t Imm31_12 = extractBits(Val: PageDelta, /*Hi=*/Hi: 31, /*Lo=*/Lo: 12) << 5; |
239 | *(little32_t *)FixupPtr = RawInstr | Imm31_12; |
240 | break; |
241 | } |
242 | case PageOffset12: { |
243 | uint64_t TargetOffset = (TargetAddress + Addend) & 0xfff; |
244 | |
245 | uint32_t RawInstr = *(ulittle32_t *)FixupPtr; |
246 | uint32_t Imm11_0 = TargetOffset << 10; |
247 | *(ulittle32_t *)FixupPtr = RawInstr | Imm11_0; |
248 | break; |
249 | } |
250 | default: |
251 | return make_error<JITLinkError>( |
252 | Args: "In graph " + G.getName() + ", section " + B.getSection().getName() + |
253 | " unsupported edge kind " + getEdgeKindName(K: E.getKind())); |
254 | } |
255 | |
256 | return Error::success(); |
257 | } |
258 | |
259 | /// loongarch null pointer content. |
260 | extern const char NullPointerContent[8]; |
261 | inline ArrayRef<char> getGOTEntryBlockContent(LinkGraph &G) { |
262 | return {reinterpret_cast<const char *>(NullPointerContent), |
263 | G.getPointerSize()}; |
264 | } |
265 | |
266 | /// loongarch stub content. |
267 | /// |
268 | /// Contains the instruction sequence for an indirect jump via an in-memory |
269 | /// pointer: |
270 | /// pcalau12i $t8, %page20(ptr) |
271 | /// ld.[w/d] $t8, %pageoff12(ptr) |
272 | /// jr $t8 |
273 | constexpr size_t StubEntrySize = 12; |
274 | extern const uint8_t LA64StubContent[StubEntrySize]; |
275 | extern const uint8_t LA32StubContent[StubEntrySize]; |
276 | inline ArrayRef<char> getStubBlockContent(LinkGraph &G) { |
277 | auto StubContent = |
278 | G.getPointerSize() == 8 ? LA64StubContent : LA32StubContent; |
279 | return {reinterpret_cast<const char *>(StubContent), StubEntrySize}; |
280 | } |
281 | |
282 | /// Creates a new pointer block in the given section and returns an |
283 | /// Anonymous symbol pointing to it. |
284 | /// |
285 | /// If InitialTarget is given then an Pointer64 relocation will be added to the |
286 | /// block pointing at InitialTarget. |
287 | /// |
288 | /// The pointer block will have the following default values: |
289 | /// alignment: PointerSize |
290 | /// alignment-offset: 0 |
291 | inline Symbol &createAnonymousPointer(LinkGraph &G, Section &PointerSection, |
292 | Symbol *InitialTarget = nullptr, |
293 | uint64_t InitialAddend = 0) { |
294 | auto &B = G.createContentBlock(Parent&: PointerSection, Content: getGOTEntryBlockContent(G), |
295 | Address: orc::ExecutorAddr(), Alignment: G.getPointerSize(), AlignmentOffset: 0); |
296 | if (InitialTarget) |
297 | B.addEdge(K: G.getPointerSize() == 8 ? Pointer64 : Pointer32, Offset: 0, |
298 | Target&: *InitialTarget, Addend: InitialAddend); |
299 | return G.addAnonymousSymbol(Content&: B, Offset: 0, Size: G.getPointerSize(), IsCallable: false, IsLive: false); |
300 | } |
301 | |
302 | /// Create a jump stub that jumps via the pointer at the given symbol and |
303 | /// an anonymous symbol pointing to it. Return the anonymous symbol. |
304 | inline Symbol &createAnonymousPointerJumpStub(LinkGraph &G, |
305 | Section &StubSection, |
306 | Symbol &PointerSymbol) { |
307 | Block &StubContentBlock = G.createContentBlock( |
308 | Parent&: StubSection, Content: getStubBlockContent(G), Address: orc::ExecutorAddr(), Alignment: 4, AlignmentOffset: 0); |
309 | StubContentBlock.addEdge(K: Page20, Offset: 0, Target&: PointerSymbol, Addend: 0); |
310 | StubContentBlock.addEdge(K: PageOffset12, Offset: 4, Target&: PointerSymbol, Addend: 0); |
311 | return G.addAnonymousSymbol(Content&: StubContentBlock, Offset: 0, Size: StubEntrySize, IsCallable: true, IsLive: false); |
312 | } |
313 | |
314 | /// Global Offset Table Builder. |
315 | class GOTTableManager : public TableManager<GOTTableManager> { |
316 | public: |
317 | static StringRef getSectionName() { return "$__GOT" ; } |
318 | |
319 | bool visitEdge(LinkGraph &G, Block *B, Edge &E) { |
320 | Edge::Kind KindToSet = Edge::Invalid; |
321 | switch (E.getKind()) { |
322 | case RequestGOTAndTransformToPage20: |
323 | KindToSet = Page20; |
324 | break; |
325 | case RequestGOTAndTransformToPageOffset12: |
326 | KindToSet = PageOffset12; |
327 | break; |
328 | default: |
329 | return false; |
330 | } |
331 | assert(KindToSet != Edge::Invalid && |
332 | "Fell through switch, but no new kind to set" ); |
333 | DEBUG_WITH_TYPE("jitlink" , { |
334 | dbgs() << " Fixing " << G.getEdgeKindName(E.getKind()) << " edge at " |
335 | << B->getFixupAddress(E) << " (" << B->getAddress() << " + " |
336 | << formatv("{0:x}" , E.getOffset()) << ")\n" ; |
337 | }); |
338 | E.setKind(KindToSet); |
339 | E.setTarget(getEntryForTarget(G, Target&: E.getTarget())); |
340 | return true; |
341 | } |
342 | |
343 | Symbol &createEntry(LinkGraph &G, Symbol &Target) { |
344 | return createAnonymousPointer(G, PointerSection&: getGOTSection(G), InitialTarget: &Target); |
345 | } |
346 | |
347 | private: |
348 | Section &getGOTSection(LinkGraph &G) { |
349 | if (!GOTSection) |
350 | GOTSection = &G.createSection(Name: getSectionName(), |
351 | Prot: orc::MemProt::Read | orc::MemProt::Exec); |
352 | return *GOTSection; |
353 | } |
354 | |
355 | Section *GOTSection = nullptr; |
356 | }; |
357 | |
358 | /// Procedure Linkage Table Builder. |
359 | class PLTTableManager : public TableManager<PLTTableManager> { |
360 | public: |
361 | PLTTableManager(GOTTableManager &GOT) : GOT(GOT) {} |
362 | |
363 | static StringRef getSectionName() { return "$__STUBS" ; } |
364 | |
365 | bool visitEdge(LinkGraph &G, Block *B, Edge &E) { |
366 | if (E.getKind() == Branch26PCRel && !E.getTarget().isDefined()) { |
367 | DEBUG_WITH_TYPE("jitlink" , { |
368 | dbgs() << " Fixing " << G.getEdgeKindName(E.getKind()) << " edge at " |
369 | << B->getFixupAddress(E) << " (" << B->getAddress() << " + " |
370 | << formatv("{0:x}" , E.getOffset()) << ")\n" ; |
371 | }); |
372 | E.setTarget(getEntryForTarget(G, Target&: E.getTarget())); |
373 | return true; |
374 | } |
375 | return false; |
376 | } |
377 | |
378 | Symbol &createEntry(LinkGraph &G, Symbol &Target) { |
379 | return createAnonymousPointerJumpStub(G, StubSection&: getStubsSection(G), |
380 | PointerSymbol&: GOT.getEntryForTarget(G, Target)); |
381 | } |
382 | |
383 | public: |
384 | Section &getStubsSection(LinkGraph &G) { |
385 | if (!StubsSection) |
386 | StubsSection = &G.createSection(Name: getSectionName(), |
387 | Prot: orc::MemProt::Read | orc::MemProt::Exec); |
388 | return *StubsSection; |
389 | } |
390 | |
391 | GOTTableManager &GOT; |
392 | Section *StubsSection = nullptr; |
393 | }; |
394 | |
395 | } // namespace loongarch |
396 | } // namespace jitlink |
397 | } // namespace llvm |
398 | |
399 | #endif |
400 | |