1 #include <errno.h>
2 #include <fcntl.h>
3 #include <libsystem/syscall.h>
4 #include <stddef.h>
5 #include <stdio.h>
6 #include <string.h>
7 #include <unistd.h>
8 
9 /**
10  * @brief 关闭文件接口
11  *
12  * @param fd 文件描述符
13  * @return int
14  */
close(int fd)15 int close(int fd)
16 {
17     return syscall_invoke(SYS_CLOSE, fd, 0, 0, 0, 0, 0, 0, 0);
18 }
19 
20 /**
21  * @brief 从文件读取数据的接口
22  *
23  * @param fd 文件描述符
24  * @param buf 缓冲区
25  * @param count 待读取数据的字节数
26  * @return ssize_t 成功读取的字节数
27  */
read(int fd,void * buf,size_t count)28 ssize_t read(int fd, void *buf, size_t count)
29 {
30     return (ssize_t)syscall_invoke(SYS_READ, fd, (uint64_t)buf, count, 0, 0, 0, 0, 0);
31 }
32 
33 /**
34  * @brief 向文件写入数据的接口
35  *
36  * @param fd 文件描述符
37  * @param buf 缓冲区
38  * @param count 待写入数据的字节数
39  * @return ssize_t 成功写入的字节数
40  */
write(int fd,void const * buf,size_t count)41 ssize_t write(int fd, void const *buf, size_t count)
42 {
43     return (ssize_t)syscall_invoke(SYS_WRITE, fd, (uint64_t)buf, count, 0, 0, 0, 0, 0);
44 }
45 
46 /**
47  * @brief 调整文件的访问位置
48  *
49  * @param fd 文件描述符号
50  * @param offset 偏移量
51  * @param whence 调整模式
52  * @return uint64_t 调整结束后的文件访问位置
53  */
lseek(int fd,off_t offset,int whence)54 off_t lseek(int fd, off_t offset, int whence)
55 {
56     return (off_t)syscall_invoke(SYS_LSEEK, fd, offset, whence, 0, 0, 0, 0, 0);
57 }
58 
59 /**
60  * @brief fork当前进程
61  *
62  * @return pid_t
63  */
fork(void)64 pid_t fork(void)
65 {
66     return (pid_t)syscall_invoke(SYS_FORK, 0, 0, 0, 0, 0, 0, 0, 0);
67 }
68 /**
69  * @brief 调用匿名管道
70  *
71  * @return int 如果失败返回负数
72  */
pipe(int fd[2])73 int pipe(int fd[2])
74 {
75     return (int)syscall_invoke(SYS_PIPE, fd, 0, 0, 0, 0, 0, 0, 0);
76 }
77 /**
78  * @brief fork当前进程,但是与父进程共享VM、flags、fd
79  *
80  * @return pid_t
81  */
vfork(void)82 pid_t vfork(void)
83 {
84     return (pid_t)syscall_invoke(SYS_VFORK, 0, 0, 0, 0, 0, 0, 0, 0);
85 }
86 
87 /**
88  * @brief 将堆内存调整为end_brk
89  *
90  * @param end_brk 新的堆区域的结束地址
91  * end_brk=-1  ===> 返回堆区域的起始地址
92  * end_brk=-2  ===> 返回堆区域的结束地址
93  * @return uint64_t 错误码
94  *
95  */
brk(uint64_t end_brk)96 uint64_t brk(uint64_t end_brk)
97 {
98     uint64_t x = (uint64_t)syscall_invoke(SYS_BRK, (uint64_t)end_brk, 0, 0, 0, 0, 0, 0, 0);
99     if (x < end_brk)
100     {
101         errno = -ENOMEM;
102         return -1;
103     }
104     return 0;
105 }
106 
107 /**
108  * @brief 将堆内存空间加上offset(注意,该系统调用只应在普通进程中调用,而不能是内核线程)
109  *
110  * @param increment offset偏移量
111  * @return uint64_t the previous program break
112  */
sbrk(int64_t increment)113 void *sbrk(int64_t increment)
114 {
115     void *retval = (void *)syscall_invoke(SYS_SBRK, (uint64_t)increment, 0, 0, 0, 0, 0, 0, 0);
116     if (retval == (void *)-ENOMEM)
117         return (void *)(-1);
118     else
119     {
120         errno = 0;
121         return (void *)retval;
122     }
123 }
124 
125 /**
126  * @brief 切换当前工作目录
127  *
128  * @param dest_path 目标目录
129  * @return int64_t 成功:0,失败:负值(错误码)
130  */
chdir(char * dest_path)131 int64_t chdir(char *dest_path)
132 {
133     if (dest_path == NULL)
134     {
135         errno = -EFAULT;
136         return -1;
137     }
138     else
139     {
140         return syscall_invoke(SYS_CHDIR, (uint64_t)dest_path, 0, 0, 0, 0, 0, 0, 0);
141     }
142 }
143 
144 /**
145  * @brief 执行新的程序
146  *
147  * @param path 文件路径
148  * @param argv 参数列表
149  * @return int
150  */
execv(const char * path,char * const argv[])151 int execv(const char *path, char *const argv[])
152 {
153     if (path == NULL)
154     {
155         errno = -ENOENT;
156         return -1;
157     }
158     int retval = syscall_invoke(SYS_EXECVE, (uint64_t)path, (uint64_t)argv, 0, 0, 0, 0, 0, 0);
159     if (retval != 0)
160         return -1;
161     else
162         return 0;
163 }
164 
165 /**
166  * @brief 删除文件夹
167  *
168  * @param path 绝对路径
169  * @return int 错误码
170  */
rmdir(const char * path)171 int rmdir(const char *path)
172 {
173     return syscall_invoke(SYS_UNLINK_AT, 0, (uint64_t)path, AT_REMOVEDIR, 0, 0, 0, 0, 0);
174 }
175 
176 /**
177  * @brief 删除文件
178  *
179  * @param path 绝对路径
180  * @return int
181  */
rm(const char * path)182 int rm(const char *path)
183 {
184     return syscall_invoke(SYS_UNLINK_AT, 0, (uint64_t)path, 0, 0, 0, 0, 0, 0);
185 }
186 
187 /**
188  * @brief  交换n字节
189  *  @param src  源地址
190  *  @param dest  目的地址
191  * @param nbytes  交换字节数
192  */
swab(void * restrict src,void * restrict dest,ssize_t nbytes)193 void swab(void *restrict src, void *restrict dest, ssize_t nbytes)
194 {
195     unsigned char buf[32];
196     char *_src = src;
197     char *_dest = dest;
198     uint32_t transfer;
199     for (; nbytes > 0; nbytes -= transfer)
200     {
201         transfer = (nbytes > 32) ? 32 : nbytes;
202         memcpy(buf, _src, transfer);
203         memcpy(_src, _dest, transfer);
204         memcpy(_dest, buf, transfer);
205         _src += transfer;
206         _dest += transfer;
207     }
208 }
209 
210 /**
211  * @brief 获取当前进程的pid(进程标识符)
212  *
213  * @return pid_t 当前进程的pid
214  */
getpid(void)215 pid_t getpid(void)
216 {
217     return syscall_invoke(SYS_GETPID, 0, 0, 0, 0, 0, 0, 0, 0);
218 }
219 
dup(int fd)220 int dup(int fd)
221 {
222     return syscall_invoke(SYS_DUP, fd, 0, 0, 0, 0, 0, 0, 0);
223 }
224 
dup2(int ofd,int nfd)225 int dup2(int ofd, int nfd)
226 {
227     return syscall_invoke(SYS_DUP2, ofd, nfd, 0, 0, 0, 0, 0, 0);
228 }