xattr.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. /* Support for *xattr interfaces on GNU/Hurd.
  2. Copyright (C) 2006-2026 Free Software Foundation, Inc.
  3. This file is part of the GNU C Library.
  4. The GNU C Library is free software; you can redistribute it and/or
  5. modify it under the terms of the GNU Lesser General Public
  6. License as published by the Free Software Foundation; either
  7. version 2.1 of the License, or (at your option) any later version.
  8. The GNU C Library is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  11. Lesser General Public License for more details.
  12. You should have received a copy of the GNU Lesser General Public
  13. License along with the GNU C Library; if not, see
  14. <https://www.gnu.org/licenses/>. */
  15. #include <hurd.h>
  16. #include <hurd/xattr.h>
  17. #include <string.h>
  18. #include <sys/mman.h>
  19. /* Right now we support only a fixed set of xattr names for Hurd features.
  20. There are no RPC interfaces for free-form xattr names and values.
  21. Name Value encoding
  22. ---- ----- --------
  23. gnu.author empty if st_author==st_uid
  24. uid_t giving st_author value
  25. gnu.translator empty if no passive translator
  26. translator and arguments: "/hurd/foo\0arg1\0arg2\0"
  27. */
  28. error_t
  29. _hurd_xattr_get (io_t port, const char *name, void *value, size_t *size)
  30. {
  31. if (strncmp (name, "gnu.", 4))
  32. return EOPNOTSUPP;
  33. name += 4;
  34. if (!strcmp (name, "author"))
  35. {
  36. struct stat64 st;
  37. error_t err = __io_stat (port, &st);
  38. if (err)
  39. return err;
  40. if (st.st_author == st.st_uid)
  41. *size = 0;
  42. else if (value)
  43. {
  44. if (*size < sizeof st.st_author)
  45. {
  46. if (*size > 0)
  47. return ERANGE;
  48. else
  49. {
  50. *size = sizeof st.st_author;
  51. return 0;
  52. }
  53. }
  54. memcpy (value, &st.st_author, sizeof st.st_author);
  55. }
  56. *size = sizeof st.st_author;
  57. return 0;
  58. }
  59. if (!strcmp (name, "translator"))
  60. {
  61. char *buf = value;
  62. mach_msg_type_number_t bufsz = value ? *size : 0;
  63. struct stat64 st;
  64. error_t err;
  65. err = __io_stat (port, &st);
  66. if (err)
  67. return err;
  68. if ((st.st_mode & S_IPTRANS) == 0)
  69. return ENODATA;
  70. err = __file_get_translator (port, &buf, &bufsz);
  71. if (err)
  72. return err;
  73. if (*size < bufsz)
  74. {
  75. if (buf != value)
  76. __munmap (buf, bufsz);
  77. if (*size > 0)
  78. return ERANGE;
  79. else
  80. {
  81. *size = bufsz;
  82. return 0;
  83. }
  84. }
  85. if (buf != value && bufsz > 0)
  86. {
  87. if (value != NULL)
  88. memcpy (value, buf, bufsz);
  89. __munmap (buf, bufsz);
  90. }
  91. *size = bufsz;
  92. return 0;
  93. }
  94. return EOPNOTSUPP;
  95. }
  96. error_t
  97. _hurd_xattr_set (io_t port, const char *name, const void *value, size_t size,
  98. int flags)
  99. {
  100. if (strncmp (name, "gnu.", 4))
  101. return EOPNOTSUPP;
  102. name += 4;
  103. if (!strcmp (name, "author"))
  104. switch (size)
  105. {
  106. default:
  107. return EINVAL;
  108. case 0: /* "Clear" author by setting to st_uid. */
  109. {
  110. struct stat64 st;
  111. error_t err = __io_stat (port, &st);
  112. if (err)
  113. return err;
  114. if (st.st_author == st.st_uid)
  115. {
  116. /* Nothing to do. */
  117. if (flags & XATTR_REPLACE)
  118. return ENODATA;
  119. return 0;
  120. }
  121. if (flags & XATTR_CREATE)
  122. return EEXIST;
  123. return __file_chauthor (port, st.st_uid);
  124. }
  125. case sizeof (uid_t): /* Set the author. */
  126. {
  127. uid_t id;
  128. memcpy (&id, value, sizeof id);
  129. if (flags & (XATTR_CREATE|XATTR_REPLACE))
  130. {
  131. struct stat64 st;
  132. error_t err = __io_stat (port, &st);
  133. if (err)
  134. return err;
  135. if (st.st_author == st.st_uid)
  136. {
  137. if (flags & XATTR_REPLACE)
  138. return ENODATA;
  139. }
  140. else if (flags & XATTR_CREATE)
  141. return EEXIST;
  142. if (st.st_author == id)
  143. /* Nothing to do. */
  144. return 0;
  145. }
  146. return __file_chauthor (port, id);
  147. }
  148. }
  149. if (!strcmp (name, "translator"))
  150. {
  151. if (flags & XATTR_REPLACE)
  152. {
  153. /* Must make sure it's already there. */
  154. char *buf = NULL;
  155. mach_msg_type_number_t bufsz = 0;
  156. error_t err = __file_get_translator (port, &buf, &bufsz);
  157. if (err)
  158. return err;
  159. if (bufsz > 0)
  160. __munmap (buf, bufsz);
  161. else
  162. return ENODATA;
  163. }
  164. return __file_set_translator (port,
  165. FS_TRANS_SET | ((flags & XATTR_CREATE)
  166. ? FS_TRANS_EXCL : 0), 0, 0,
  167. value, size,
  168. MACH_PORT_NULL, MACH_MSG_TYPE_COPY_SEND);
  169. }
  170. return EOPNOTSUPP;
  171. }
  172. error_t
  173. _hurd_xattr_remove (io_t port, const char *name)
  174. {
  175. return _hurd_xattr_set (port, name, NULL, 0, XATTR_REPLACE);
  176. }
  177. error_t
  178. _hurd_xattr_list (io_t port, void *buffer, size_t *size)
  179. {
  180. size_t total = 0;
  181. char *bufp = buffer;
  182. inline void add (const char *name, size_t len)
  183. {
  184. total += len;
  185. if (bufp != NULL && total <= *size)
  186. bufp = __mempcpy (bufp, name, len);
  187. }
  188. #define add(s) add (s, sizeof s)
  189. struct stat64 st;
  190. error_t err = __io_stat (port, &st);
  191. if (err)
  192. return err;
  193. if (st.st_author != st.st_uid)
  194. add ("gnu.author");
  195. if (st.st_mode & S_IPTRANS)
  196. add ("gnu.translator");
  197. if (*size > 0 && total > *size)
  198. return ERANGE;
  199. *size = total;
  200. return 0;
  201. }