forked from facebook/hhvm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcode-gen-tls-x64.h
70 lines (60 loc) · 2.8 KB
/
code-gen-tls-x64.h
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
/*
+----------------------------------------------------------------------+
| HipHop for PHP |
+----------------------------------------------------------------------+
| Copyright (c) 2010-present Facebook, Inc. (http://www.facebook.com) |
+----------------------------------------------------------------------+
| This source file is subject to version 3.01 of the PHP license, |
| that is bundled with this package in the file LICENSE, and is |
| available through the world-wide-web at the following url: |
| http://www.php.net/license/3_01.txt |
| If you did not receive a copy of the PHP license and are unable to |
| obtain it through the world-wide-web, please send a note to |
| [email protected] so we can mail you a copy immediately. |
+----------------------------------------------------------------------+
*/
#pragma once
#include "hphp/runtime/vm/jit/vasm-gen.h"
#include "hphp/runtime/vm/jit/vasm-instr.h"
#include "hphp/runtime/vm/jit/vasm-reg.h"
#include "hphp/util/asm-x64.h"
#include "hphp/util/thread-local.h"
namespace HPHP::jit::x64::detail {
///////////////////////////////////////////////////////////////////////////////
/*
* x86 terminology review: "Virtual addresses" are subject to both segmented
* translation and paged translation. "Linear addresses" are post-segmentation
* address, subject only to paging. C and C++ generally only have access to
* bitwise linear addresses.
*
* On Linux, TLS data live at negative virtual addresses off FS: the first
* datum is typically at VA(FS:-sizeof(datum)). Linux's x64 ABI stores the
* linear address of the base of TLS at VA(FS:0). While this is just a
* convention, it is firm: gcc builds binaries that assume it when, e.g.,
* evaluating "&myTlsDatum".
*
* The virtual addresses of TLS data are not exposed to C/C++. To figure it
* out, we take a datum's linear address, and subtract it from the linear
* address where TLS starts.
*/
template <typename T>
Vptr emitTLSAddr(Vout& /*v*/, TLSDatum<T> datum) {
uintptr_t vaddr = uintptr_t(datum.tls) - tlsBase();
return Vptr{baseless(vaddr), Segment::FS};
}
template<typename T>
Vreg emitTLSLea(Vout& v, TLSDatum<T> datum, int offset) {
auto const base = v.makeReg();
auto const addr = v.makeReg();
v << load{Vptr{baseless(0), Segment::FS}, base};
auto const tlsBaseAddr = tlsBase();
auto const datumAddr = uintptr_t(datum.tls) + offset;
if (datumAddr < tlsBaseAddr) {
v << subq{v.cns(tlsBaseAddr - datumAddr), base, addr, v.makeReg()};
} else {
v << addq{v.cns(datumAddr - tlsBaseAddr), base, addr, v.makeReg()};
}
return addr;
}
///////////////////////////////////////////////////////////////////////////////
}