diff options
author | caiyinyu <caiyinyu@loongson.cn> | 2022-07-19 09:20:50 +0800 |
---|---|---|
committer | Adhemerval Zanella <adhemerval.zanella@linaro.org> | 2022-07-26 12:35:12 -0300 |
commit | 45955fe61844d94f7faa660eda9e515a6571e8c3 (patch) | |
tree | 604ccc92c973e5a8b59e90d322e91f588422b391 /sysdeps/unix/sysv/linux/loongarch/clone3.S | |
parent | 327588226198995d2daa62e127ef45af9bcf182d (diff) | |
download | glibc-45955fe61844d94f7faa660eda9e515a6571e8c3.zip glibc-45955fe61844d94f7faa660eda9e515a6571e8c3.tar.gz glibc-45955fe61844d94f7faa660eda9e515a6571e8c3.tar.bz2 |
LoongArch: Linux Syscall Interface
Diffstat (limited to 'sysdeps/unix/sysv/linux/loongarch/clone3.S')
-rw-r--r-- | sysdeps/unix/sysv/linux/loongarch/clone3.S | 83 |
1 files changed, 83 insertions, 0 deletions
diff --git a/sysdeps/unix/sysv/linux/loongarch/clone3.S b/sysdeps/unix/sysv/linux/loongarch/clone3.S new file mode 100644 index 0000000..38be4c8 --- /dev/null +++ b/sysdeps/unix/sysv/linux/loongarch/clone3.S @@ -0,0 +1,83 @@ +/* The clone3 syscall wrapper. + Copyright (C) 2022 Free Software Foundation, Inc. + + This file is part of the GNU C Library. + + The GNU C Library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + The GNU C Library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with the GNU C Library. If not, see + <https://www.gnu.org/licenses/>. */ + +/* clone3() is even more special than fork() as it mucks with stacks + and invokes a function in the right context after its all over. */ + +#include <sys/asm.h> +#include <sysdep.h> +#define _ERRNO_H 1 +#include <bits/errno.h> +#include <tls.h> +#include "tcb-offsets.h" + +/* int clone3(struct clone_args *cl_args, size_t size, + int (*func)(void *arg), void *arg); */ + +ENTRY (__clone3) + + /* Sanity check arguments. */ + beqz a0, L (invalid) /* No NULL cl_args pointer. */ + beqz a2, L (invalid) /* No NULL function pointer. */ + + /* Do the system call. */ + LI a7, __NR_clone3 + syscall 0 + + blt a0, zero ,L (error) + beqz a0, L (thread_start3) + + /* Successful return from the parent. */ + ret + +L (invalid): + LI a0, -EINVAL + + /* Something bad happened -- no child created. */ +L (error): + b __syscall_error + +END (__clone3) + +/* Load up the arguments to the function. Put this block of code in + its own function so that we can terminate the stack trace with our + debug info. */ +ENTRY (__thread_start3) +L (thread_start3): + +/* Terminate call stack by noting ra is undefined. Use a dummy + .cfi_label to force starting the FDE. */ + .cfi_label .Ldummy + cfi_undefined (1) + + /* Align stack to 16. */ + BSTRINS sp, zero, 3, 0 + + /* Set up arguments for the function call. */ + move a0, a3 /* Argument. */ + jirl ra, a2, 0 /* Call function. */ + + /* Call exit with the function's return value. */ + LI a7, __NR_exit + syscall 0 + + END (__thread_start3) + +libc_hidden_def (__clone3) +weak_alias (__clone3, clone3) |