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!(); }