unistd.c 4.7 KB

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