unistd.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. #include <libc/unistd.h>
  2. #include <libsystem/syscall.h>
  3. #include <libc/errno.h>
  4. #include <libc/stdio.h>
  5. #include <libc/stddef.h>
  6. #include <libc/string.h>
  7. #include <libc/fcntl.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 fork当前进程,但是与父进程共享VM、flags、fd
  65. *
  66. * @return pid_t
  67. */
  68. pid_t vfork(void)
  69. {
  70. return (pid_t)syscall_invoke(SYS_VFORK, 0, 0, 0, 0, 0, 0, 0, 0);
  71. }
  72. /**
  73. * @brief 将堆内存调整为end_brk
  74. *
  75. * @param end_brk 新的堆区域的结束地址
  76. * end_brk=-1 ===> 返回堆区域的起始地址
  77. * end_brk=-2 ===> 返回堆区域的结束地址
  78. * @return uint64_t 错误码
  79. *
  80. */
  81. uint64_t brk(uint64_t end_brk)
  82. {
  83. uint64_t x = (uint64_t)syscall_invoke(SYS_BRK, (uint64_t)end_brk, 0, 0, 0, 0, 0, 0, 0);
  84. // printf("brk(): end_brk=%#018lx x=%#018lx", (uint64_t)end_brk, x);
  85. return x;
  86. }
  87. /**
  88. * @brief 将堆内存空间加上offset(注意,该系统调用只应在普通进程中调用,而不能是内核线程)
  89. *
  90. * @param increment offset偏移量
  91. * @return uint64_t the previous program break
  92. */
  93. void *sbrk(int64_t increment)
  94. {
  95. void *retval = (void *)syscall_invoke(SYS_SBRK, (uint64_t)increment, 0, 0, 0, 0, 0, 0, 0);
  96. if (retval == (void *)-ENOMEM)
  97. return (void *)(-1);
  98. else
  99. {
  100. errno = 0;
  101. return (void *)retval;
  102. }
  103. }
  104. /**
  105. * @brief 切换当前工作目录
  106. *
  107. * @param dest_path 目标目录
  108. * @return int64_t 成功:0,失败:负值(错误码)
  109. */
  110. int64_t chdir(char *dest_path)
  111. {
  112. if (dest_path == NULL)
  113. {
  114. errno = -EFAULT;
  115. return -1;
  116. }
  117. else
  118. {
  119. return syscall_invoke(SYS_CHDIR, (uint64_t)dest_path, 0, 0, 0, 0, 0, 0, 0);
  120. }
  121. }
  122. /**
  123. * @brief 执行新的程序
  124. *
  125. * @param path 文件路径
  126. * @param argv 参数列表
  127. * @return int
  128. */
  129. int execv(const char *path, char *const argv[])
  130. {
  131. if (path == NULL)
  132. {
  133. errno = -ENOENT;
  134. return -1;
  135. }
  136. int retval = syscall_invoke(SYS_EXECVE, (uint64_t)path, (uint64_t)argv, 0, 0, 0, 0, 0, 0);
  137. if (retval != 0)
  138. return -1;
  139. else
  140. return 0;
  141. }
  142. /**
  143. * @brief 删除文件夹
  144. *
  145. * @param path 绝对路径
  146. * @return int 错误码
  147. */
  148. int rmdir(const char *path)
  149. {
  150. return syscall_invoke(SYS_UNLINK_AT, 0, (uint64_t)path, AT_REMOVEDIR, 0, 0, 0, 0, 0);
  151. }
  152. /**
  153. * @brief 删除文件
  154. *
  155. * @param path 绝对路径
  156. * @return int
  157. */
  158. int rm(const char * path)
  159. {
  160. return syscall_invoke(SYS_UNLINK_AT, 0, (uint64_t)path, 0, 0, 0, 0, 0, 0);
  161. }
  162. /**
  163. * @brief 交换n字节
  164. * @param src 源地址
  165. * @param dest 目的地址
  166. * @param nbytes 交换字节数
  167. */
  168. void swab(void *restrict src, void *restrict dest, ssize_t nbytes)
  169. {
  170. unsigned char buf[32];
  171. char *_src = src;
  172. char *_dest = dest;
  173. uint32_t transfer;
  174. for (; nbytes > 0; nbytes -= transfer)
  175. {
  176. transfer = (nbytes > 32) ? 32 : nbytes;
  177. memcpy(buf, _src, transfer);
  178. memcpy(_src, _dest, transfer);
  179. memcpy(_dest, buf, transfer);
  180. _src += transfer;
  181. _dest += transfer;
  182. }
  183. }