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 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
//! Thread Local Storage (TLS) support.
//!
//! ## TLS layout for x86_64
//!
//! ```text
//! aligned --> +-------------------------+- static_tls_offset
//! allocation | | \
//! | .tdata | |
//! | address | | |
//! | grow up + - - - - - - - - - - - - + > Static TLS block
//! v | | | (length: static_tls_size)
//! | .tbss | |
//! | | |
//! +-------------------------+ |
//! | / PADDING / / / / / / / | /
//! +-------------------------+
//! tls_ptr -+-> self pointer (void *) | \
//! (tp_offset) | | |
//! | Custom TCB format | > Thread Control Block (TCB)
//! | (might be used | | (length: TCB_SIZE)
//! | by a libC) | |
//! | | /
//! +-------------------------+- (total length: tls_area_size)
//! ```
//!
//! ## TLS layout for AArch64 and RISC-V
//!
//! ```text
//! +-------------------------+
//! | | \
//! | Custom TCB format | |
//! | (might be used | > Thread Control Block (TCB)
//! | by a libC) | | (length: TCB_SIZE)
//! | | /
//! tls_ptr -+-------------------------+
//! (tp_offset) | GAP_ABOVE_TP |
//! +-------------------------+- static_tls_offset
//! | | \
//! | .tdata | |
//! | | |
//! + - - - - - - - - - - - - + > Static TLS block
//! | | | (length: static_tls_size)
//! | .tbss | |
//! | | /
//! +-------------------------+- (total length: tls_area_size)
//! ```
//!
//! Reference:
//! 1. <https://github.com/unikraft/unikraft/blob/staging/arch/x86/x86_64/tls.c>
//! 2. <https://github.com/unikraft/unikraft/blob/staging/arch/arm/arm64/tls.c>
extern crate alloc;
use memory_addr::align_up;
use core::alloc::Layout;
use core::ptr::NonNull;
const TLS_ALIGN: usize = 0x10;
cfg_if::cfg_if! {
if #[cfg(target_arch = "x86_64")] {
const TCB_SIZE: usize = 8; // to store TLS self pointer
const GAP_ABOVE_TP: usize = 0;
} else if #[cfg(target_arch = "aarch64")] {
const TCB_SIZE: usize = 0;
const GAP_ABOVE_TP: usize = 16;
} else if #[cfg(target_arch = "riscv64")] {
const TCB_SIZE: usize = 0;
const GAP_ABOVE_TP: usize = 0;
}
}
extern "C" {
fn _stdata();
fn _etdata();
fn _etbss();
}
/// The memory region for thread-local storage.
pub struct TlsArea {
base: NonNull<u8>,
layout: Layout,
}
impl Drop for TlsArea {
fn drop(&mut self) {
unsafe {
alloc::alloc::dealloc(self.base.as_ptr(), self.layout);
}
}
}
impl TlsArea {
/// Returns the pointer to the TLS static area.
///
/// One should set the hardware thread pointer register to this value.
pub fn tls_ptr(&self) -> *mut u8 {
unsafe { self.base.as_ptr().add(tp_offset()) }
}
/// Allocates the memory region for TLS, and initializes it.
pub fn alloc() -> Self {
let layout = Layout::from_size_align(tls_area_size(), TLS_ALIGN).unwrap();
let area_base = unsafe { alloc::alloc::alloc_zeroed(layout) };
let tls_load_base = _stdata as *mut u8;
let tls_load_size = _etbss as usize - _stdata as usize;
unsafe {
// copy data from .tbdata section
core::ptr::copy_nonoverlapping(
tls_load_base,
area_base.add(static_tls_offset()),
tls_load_size,
);
// initialize TCB
init_tcb(area_base);
}
Self {
base: NonNull::new(area_base).unwrap(),
layout,
}
}
}
fn static_tls_size() -> usize {
align_up(_etbss as usize - _stdata as usize, TLS_ALIGN)
}
fn static_tls_offset() -> usize {
if cfg!(target_arch = "x86_64") {
0
} else if cfg!(any(target_arch = "aarch64", target_arch = "riscv64")) {
TCB_SIZE + GAP_ABOVE_TP
} else {
unreachable!()
}
}
fn tp_offset() -> usize {
if cfg!(target_arch = "x86_64") {
static_tls_size()
} else if cfg!(any(target_arch = "aarch64", target_arch = "riscv64")) {
TCB_SIZE
} else {
unreachable!()
}
}
fn tls_area_size() -> usize {
if cfg!(target_arch = "x86_64") {
static_tls_size() + TCB_SIZE
} else if cfg!(any(target_arch = "aarch64", target_arch = "riscv64")) {
TCB_SIZE + GAP_ABOVE_TP + static_tls_size()
} else {
unreachable!()
}
}
unsafe fn init_tcb(tls_area: *mut u8) {
if cfg!(target_arch = "x86_64") {
let tp_addr = tls_area.add(tp_offset()).cast::<usize>();
tp_addr.write(tp_addr as usize); // write self pointer
}
}