unistd.c 4.0 KB

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