123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296 |
- use alloc::{boxed::Box, vec::Vec};
- use core::{intrinsics, ptr, fmt::Debug};
- use crate::{
- header::{libgen, stdio, stdlib},
- ld_so::{self, linker::Linker},
- platform::{self, get_auxvs, new_mspace, types::*, Pal, Sys},
- sync::mutex::Mutex,
- ALLOCATOR,
- };
- #[repr(C)]
- pub struct Stack {
- pub argc: isize,
- pub argv0: *const c_char,
- }
- impl Stack {
- pub fn argv(&self) -> *const *const c_char {
- &self.argv0 as *const _
- }
- pub fn envp(&self) -> *const *const c_char {
- unsafe { self.argv().offset(self.argc + 1) }
- }
- pub fn auxv(&self) -> *const (usize, usize) {
- unsafe {
- let mut envp = self.envp();
- while !(*envp).is_null() {
- envp = envp.add(1);
- }//timer_init
- envp.add(1) as *const (usize, usize)
- }
- }
- }
- impl Debug for Stack{
- fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
- f.debug_struct("Stack")
- .field("argc", &self.argc)
- .field("argv0", &self.argv0)
- .finish()
- }
- }
- unsafe fn copy_string_array(array: *const *const c_char, len: usize) -> Vec<*mut c_char> {
- println!("copy_string_array: array: {:p}, len: {}", array, len);
-
- let mut vec = Vec::with_capacity(len + 1);
-
- let x = vec.into_raw_parts();
- println!("{:#018x?}",x.0);
- let mut vec=alloc::vec::Vec::from_raw_parts(x.0, x.1, x.2);
- println!("new vec ok");
- for i in 0..len {
- let item = *array.add(i);
- let mut len = 0;
- while *item.add(len) != 0 {
- len += 1;
- }
- let buf = platform::alloc(len + 1) as *mut c_char;
- for i in 0..=len {
- *buf.add(i) = *item.add(i);
- }
- vec.push(buf);
- }
- vec.push(ptr::null_mut());
- vec
- }
- // Since Redox and Linux are so similar, it is easy to accidentally run a binary from one on the
- // other. This will test that the current system is compatible with the current binary
- #[no_mangle]
- pub unsafe fn relibc_verify_host() {
- if !Sys::verify() {
- intrinsics::abort();
- }
- }
- #[link_section = ".init_array"]
- #[used]
- static INIT_ARRAY: [extern "C" fn(); 1] = [init_array];
- static mut init_complete: bool = false;
- #[used]
- #[no_mangle]
- static mut __relibc_init_environ: *mut *mut c_char = ptr::null_mut();
- fn alloc_init() {
- unsafe {
- if init_complete {
- return;
- }
- }
- unsafe {
- dbg!("in alloc init");
- if let Some(tcb) = ld_so::tcb::Tcb::current() {
- println!("tcb.mspace {}",tcb.mspace);
- if tcb.mspace != 0 {
- ALLOCATOR.set_book_keeper(tcb.mspace);
- } else if ALLOCATOR.get_book_keeper() == 0 {
- ALLOCATOR.set_book_keeper(new_mspace());
- }
- } else if ALLOCATOR.get_book_keeper() == 0 {
- dbg!("TRY");
- ALLOCATOR.set_book_keeper(new_mspace());
- dbg!("ALLOCATOR OWARI DAWA");
- }
- }
- }
- extern "C" fn init_array() {
- // The thing is that we cannot guarantee if
- // init_array runs first or if relibc_start runs first
- // Still whoever gets to run first must initialize rust
- // memory allocator before doing anything else.
- unsafe {
- if init_complete {
- return;
- }
- }
- alloc_init();
- io_init();
- unsafe {
- if platform::environ.is_null() {
- platform::environ = __relibc_init_environ;
- }
- }
- extern "C" {
- fn pthread_init();
- }
- unsafe {
- pthread_init();
- init_complete = true
- }
- }
- fn io_init() {
- unsafe {
- // Initialize stdin/stdout/stderr,
- // see https://github.com/rust-lang/rust/issues/51718
- stdio::stdin = stdio::default_stdin.get();
- stdio::stdout = stdio::default_stdout.get();
- stdio::stderr = stdio::default_stderr.get();
- }
- }
- #[cfg(target_os = "redox")]
- fn setup_sigstack() {
- use syscall::{Map, MapFlags};
- const SIGSTACK_SIZE: usize = 1024 * 256;
- let sigstack = unsafe {
- syscall::fmap(
- !0,
- &Map {
- address: 0,
- offset: 0,
- flags: MapFlags::MAP_PRIVATE | MapFlags::PROT_READ | MapFlags::PROT_WRITE,
- size: SIGSTACK_SIZE,
- },
- )
- }
- .expect("failed to allocate sigstack")
- + SIGSTACK_SIZE;
- let fd = syscall::open(
- "thisproc:current/sigstack",
- syscall::O_WRONLY | syscall::O_CLOEXEC,
- )
- .expect("failed to open thisproc:current/sigstack");
- syscall::write(fd, &usize::to_ne_bytes(sigstack))
- .expect("failed to write to thisproc:current/sigstack");
- let _ = syscall::close(fd);
- }
- #[inline(never)]
- #[no_mangle]
- pub unsafe extern "C" fn relibc_start(sp: &'static Stack) -> ! {
- extern "C" {
- static __preinit_array_start: extern "C" fn();
- static __preinit_array_end: extern "C" fn();
- static __init_array_start: extern "C" fn();
- static __init_array_end: extern "C" fn();
- fn _init();
- fn main(argc: isize, argv: *mut *mut c_char, envp: *mut *mut c_char) -> c_int;
- }
- println!("relibc_start: sp={:?}", sp);
- // Ensure correct host system before executing more system calls
- relibc_verify_host();
- use core::arch::asm;
-
- // Initialize TLS, if necessary
- ld_so::init(sp);
- println!("alloc init");
- //println!("alloc init ok_________________");
-
- // Set up the right allocator...
- // if any memory rust based memory allocation happen before this step .. we are doomed.
- alloc_init();
- println!("alloc init ok");
- if let Some(tcb) = ld_so::tcb::Tcb::current() {
- // Update TCB mspace
- tcb.mspace = ALLOCATOR.get_book_keeper();
- // Set linker pointer if necessary
- if tcb.linker_ptr.is_null() {
- //TODO: get ld path
- let linker = Linker::new(None);
- //TODO: load root object
- tcb.linker_ptr = Box::into_raw(Box::new(Mutex::new(linker)));
- }
- }
- println!("to copy args");
- // Set up argc and argv
- let argc = sp.argc;
- let argv = sp.argv();
- platform::inner_argv = copy_string_array(argv, argc as usize);
- println!("copy args ok");
- platform::argv = platform::inner_argv.as_mut_ptr();
- // Special code for program_invocation_name and program_invocation_short_name
- if let Some(arg) = platform::inner_argv.get(0) {
- platform::program_invocation_name = *arg;
- platform::program_invocation_short_name = libgen::basename(*arg);
- }
- println!("to check environ");
- // We check for NULL here since ld.so might already have initialized it for us, and we don't
- // want to overwrite it if constructors in .init_array of dependency libraries have called
- // setenv.
- if platform::environ.is_null() {
- // Set up envp
- let envp = sp.envp();
- let mut len = 0;
- while !(*envp.add(len)).is_null() {
- len += 1;
- }
- platform::OUR_ENVIRON = copy_string_array(envp, len);
- platform::environ = platform::OUR_ENVIRON.as_mut_ptr();
- }
- println!("to get auxvs");
- let auxvs = get_auxvs(sp.auxv().cast());
- println!("to init platform");
- crate::platform::init(auxvs);
- // Setup signal stack, otherwise we cannot handle any signals besides SIG_IGN/SIG_DFL behavior.
- #[cfg(target_os = "redox")]
- setup_sigstack();
- println!("before init_array()");
- init_array();
- println!("init_array() ok");
- // Run preinit array
- {
- let mut f = &__preinit_array_start as *const _;
- #[allow(clippy::op_ref)]
- while f < &__preinit_array_end {
- (*f)();
- f = f.offset(1);
- }
- }
- println!("before _init()");
- // Call init section
- _init();
- println!("after _init()");
- // Run init array
- {
- let mut f = &__init_array_start as *const _;
- #[allow(clippy::op_ref)]
- while f < &__init_array_end {
- (*f)();
- f = f.offset(1);
- }
- }
- println!("to run main()");
- // not argv or envp, because programs like bash try to modify this *const* pointer :|
- stdlib::exit(main(argc, platform::argv, platform::environ));
- unreachable!();
- }
|