xattr.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * linux/fs/hfsplus/xattr.c
  4. *
  5. * Vyacheslav Dubeyko <slava@dubeyko.com>
  6. *
  7. * Logic of processing extended attributes
  8. */
  9. #include "hfsplus_fs.h"
  10. #include <linux/posix_acl_xattr.h>
  11. #include <linux/nls.h>
  12. #include "xattr.h"
  13. #include "acl.h"
  14. static int hfsplus_removexattr(struct inode *inode, const char *name);
  15. const struct xattr_handler *hfsplus_xattr_handlers[] = {
  16. &hfsplus_xattr_osx_handler,
  17. &hfsplus_xattr_user_handler,
  18. &hfsplus_xattr_trusted_handler,
  19. #ifdef CONFIG_HFSPLUS_FS_POSIX_ACL
  20. &posix_acl_access_xattr_handler,
  21. &posix_acl_default_xattr_handler,
  22. #endif
  23. &hfsplus_xattr_security_handler,
  24. NULL
  25. };
  26. static int strcmp_xattr_finder_info(const char *name)
  27. {
  28. if (name) {
  29. return strncmp(name, HFSPLUS_XATTR_FINDER_INFO_NAME,
  30. sizeof(HFSPLUS_XATTR_FINDER_INFO_NAME));
  31. }
  32. return -1;
  33. }
  34. static int strcmp_xattr_acl(const char *name)
  35. {
  36. if (name) {
  37. return strncmp(name, HFSPLUS_XATTR_ACL_NAME,
  38. sizeof(HFSPLUS_XATTR_ACL_NAME));
  39. }
  40. return -1;
  41. }
  42. static bool is_known_namespace(const char *name)
  43. {
  44. if (strncmp(name, XATTR_SYSTEM_PREFIX, XATTR_SYSTEM_PREFIX_LEN) &&
  45. strncmp(name, XATTR_USER_PREFIX, XATTR_USER_PREFIX_LEN) &&
  46. strncmp(name, XATTR_SECURITY_PREFIX, XATTR_SECURITY_PREFIX_LEN) &&
  47. strncmp(name, XATTR_TRUSTED_PREFIX, XATTR_TRUSTED_PREFIX_LEN))
  48. return false;
  49. return true;
  50. }
  51. static void hfsplus_init_header_node(struct inode *attr_file,
  52. u32 clump_size,
  53. char *buf, u16 node_size)
  54. {
  55. struct hfs_bnode_desc *desc;
  56. struct hfs_btree_header_rec *head;
  57. u16 offset;
  58. __be16 *rec_offsets;
  59. u32 hdr_node_map_rec_bits;
  60. char *bmp;
  61. u32 used_nodes;
  62. u32 used_bmp_bytes;
  63. u64 tmp;
  64. hfs_dbg(ATTR_MOD, "init_hdr_attr_file: clump %u, node_size %u\n",
  65. clump_size, node_size);
  66. /* The end of the node contains list of record offsets */
  67. rec_offsets = (__be16 *)(buf + node_size);
  68. desc = (struct hfs_bnode_desc *)buf;
  69. desc->type = HFS_NODE_HEADER;
  70. desc->num_recs = cpu_to_be16(HFSPLUS_BTREE_HDR_NODE_RECS_COUNT);
  71. offset = sizeof(struct hfs_bnode_desc);
  72. *--rec_offsets = cpu_to_be16(offset);
  73. head = (struct hfs_btree_header_rec *)(buf + offset);
  74. head->node_size = cpu_to_be16(node_size);
  75. tmp = i_size_read(attr_file);
  76. do_div(tmp, node_size);
  77. head->node_count = cpu_to_be32(tmp);
  78. head->free_nodes = cpu_to_be32(be32_to_cpu(head->node_count) - 1);
  79. head->clump_size = cpu_to_be32(clump_size);
  80. head->attributes |= cpu_to_be32(HFS_TREE_BIGKEYS | HFS_TREE_VARIDXKEYS);
  81. head->max_key_len = cpu_to_be16(HFSPLUS_ATTR_KEYLEN - sizeof(u16));
  82. offset += sizeof(struct hfs_btree_header_rec);
  83. *--rec_offsets = cpu_to_be16(offset);
  84. offset += HFSPLUS_BTREE_HDR_USER_BYTES;
  85. *--rec_offsets = cpu_to_be16(offset);
  86. hdr_node_map_rec_bits = 8 * (node_size - offset - (4 * sizeof(u16)));
  87. if (be32_to_cpu(head->node_count) > hdr_node_map_rec_bits) {
  88. u32 map_node_bits;
  89. u32 map_nodes;
  90. desc->next = cpu_to_be32(be32_to_cpu(head->leaf_tail) + 1);
  91. map_node_bits = 8 * (node_size - sizeof(struct hfs_bnode_desc) -
  92. (2 * sizeof(u16)) - 2);
  93. map_nodes = (be32_to_cpu(head->node_count) -
  94. hdr_node_map_rec_bits +
  95. (map_node_bits - 1)) / map_node_bits;
  96. be32_add_cpu(&head->free_nodes, 0 - map_nodes);
  97. }
  98. bmp = buf + offset;
  99. used_nodes =
  100. be32_to_cpu(head->node_count) - be32_to_cpu(head->free_nodes);
  101. used_bmp_bytes = used_nodes / 8;
  102. if (used_bmp_bytes) {
  103. memset(bmp, 0xFF, used_bmp_bytes);
  104. bmp += used_bmp_bytes;
  105. used_nodes %= 8;
  106. }
  107. *bmp = ~(0xFF >> used_nodes);
  108. offset += hdr_node_map_rec_bits / 8;
  109. *--rec_offsets = cpu_to_be16(offset);
  110. }
  111. static int hfsplus_create_attributes_file(struct super_block *sb)
  112. {
  113. int err = 0;
  114. struct hfsplus_sb_info *sbi = HFSPLUS_SB(sb);
  115. struct inode *attr_file;
  116. struct hfsplus_inode_info *hip;
  117. u32 clump_size;
  118. u16 node_size = HFSPLUS_ATTR_TREE_NODE_SIZE;
  119. char *buf;
  120. int index, written;
  121. struct address_space *mapping;
  122. struct page *page;
  123. int old_state = HFSPLUS_EMPTY_ATTR_TREE;
  124. hfs_dbg(ATTR_MOD, "create_attr_file: ino %d\n", HFSPLUS_ATTR_CNID);
  125. check_attr_tree_state_again:
  126. switch (atomic_read(&sbi->attr_tree_state)) {
  127. case HFSPLUS_EMPTY_ATTR_TREE:
  128. if (old_state != atomic_cmpxchg(&sbi->attr_tree_state,
  129. old_state,
  130. HFSPLUS_CREATING_ATTR_TREE))
  131. goto check_attr_tree_state_again;
  132. break;
  133. case HFSPLUS_CREATING_ATTR_TREE:
  134. /*
  135. * This state means that another thread is in process
  136. * of AttributesFile creation. Theoretically, it is
  137. * possible to be here. But really __setxattr() method
  138. * first of all calls hfs_find_init() for lookup in
  139. * B-tree of CatalogFile. This method locks mutex of
  140. * CatalogFile's B-tree. As a result, if some thread
  141. * is inside AttributedFile creation operation then
  142. * another threads will be waiting unlocking of
  143. * CatalogFile's B-tree's mutex. However, if code will
  144. * change then we will return error code (-EAGAIN) from
  145. * here. Really, it means that first try to set of xattr
  146. * fails with error but second attempt will have success.
  147. */
  148. return -EAGAIN;
  149. case HFSPLUS_VALID_ATTR_TREE:
  150. return 0;
  151. case HFSPLUS_FAILED_ATTR_TREE:
  152. return -EOPNOTSUPP;
  153. default:
  154. BUG();
  155. }
  156. attr_file = hfsplus_iget(sb, HFSPLUS_ATTR_CNID);
  157. if (IS_ERR(attr_file)) {
  158. pr_err("failed to load attributes file\n");
  159. return PTR_ERR(attr_file);
  160. }
  161. BUG_ON(i_size_read(attr_file) != 0);
  162. hip = HFSPLUS_I(attr_file);
  163. clump_size = hfsplus_calc_btree_clump_size(sb->s_blocksize,
  164. node_size,
  165. sbi->sect_count,
  166. HFSPLUS_ATTR_CNID);
  167. mutex_lock(&hip->extents_lock);
  168. hip->clump_blocks = clump_size >> sbi->alloc_blksz_shift;
  169. mutex_unlock(&hip->extents_lock);
  170. if (sbi->free_blocks <= (hip->clump_blocks << 1)) {
  171. err = -ENOSPC;
  172. goto end_attr_file_creation;
  173. }
  174. while (hip->alloc_blocks < hip->clump_blocks) {
  175. err = hfsplus_file_extend(attr_file, false);
  176. if (unlikely(err)) {
  177. pr_err("failed to extend attributes file\n");
  178. goto end_attr_file_creation;
  179. }
  180. hip->phys_size = attr_file->i_size =
  181. (loff_t)hip->alloc_blocks << sbi->alloc_blksz_shift;
  182. hip->fs_blocks = hip->alloc_blocks << sbi->fs_shift;
  183. inode_set_bytes(attr_file, attr_file->i_size);
  184. }
  185. buf = kzalloc(node_size, GFP_NOFS);
  186. if (!buf) {
  187. pr_err("failed to allocate memory for header node\n");
  188. err = -ENOMEM;
  189. goto end_attr_file_creation;
  190. }
  191. hfsplus_init_header_node(attr_file, clump_size, buf, node_size);
  192. mapping = attr_file->i_mapping;
  193. index = 0;
  194. written = 0;
  195. for (; written < node_size; index++, written += PAGE_SIZE) {
  196. void *kaddr;
  197. page = read_mapping_page(mapping, index, NULL);
  198. if (IS_ERR(page)) {
  199. err = PTR_ERR(page);
  200. goto failed_header_node_init;
  201. }
  202. kaddr = kmap_atomic(page);
  203. memcpy(kaddr, buf + written,
  204. min_t(size_t, PAGE_SIZE, node_size - written));
  205. kunmap_atomic(kaddr);
  206. set_page_dirty(page);
  207. put_page(page);
  208. }
  209. hfsplus_mark_inode_dirty(attr_file, HFSPLUS_I_ATTR_DIRTY);
  210. sbi->attr_tree = hfs_btree_open(sb, HFSPLUS_ATTR_CNID);
  211. if (!sbi->attr_tree)
  212. pr_err("failed to load attributes file\n");
  213. failed_header_node_init:
  214. kfree(buf);
  215. end_attr_file_creation:
  216. iput(attr_file);
  217. if (!err)
  218. atomic_set(&sbi->attr_tree_state, HFSPLUS_VALID_ATTR_TREE);
  219. else if (err == -ENOSPC)
  220. atomic_set(&sbi->attr_tree_state, HFSPLUS_EMPTY_ATTR_TREE);
  221. else
  222. atomic_set(&sbi->attr_tree_state, HFSPLUS_FAILED_ATTR_TREE);
  223. return err;
  224. }
  225. int __hfsplus_setxattr(struct inode *inode, const char *name,
  226. const void *value, size_t size, int flags)
  227. {
  228. int err = 0;
  229. struct hfs_find_data cat_fd;
  230. hfsplus_cat_entry entry;
  231. u16 cat_entry_flags, cat_entry_type;
  232. u16 folder_finderinfo_len = sizeof(struct DInfo) +
  233. sizeof(struct DXInfo);
  234. u16 file_finderinfo_len = sizeof(struct FInfo) +
  235. sizeof(struct FXInfo);
  236. if ((!S_ISREG(inode->i_mode) &&
  237. !S_ISDIR(inode->i_mode)) ||
  238. HFSPLUS_IS_RSRC(inode))
  239. return -EOPNOTSUPP;
  240. if (value == NULL)
  241. return hfsplus_removexattr(inode, name);
  242. err = hfs_find_init(HFSPLUS_SB(inode->i_sb)->cat_tree, &cat_fd);
  243. if (err) {
  244. pr_err("can't init xattr find struct\n");
  245. return err;
  246. }
  247. err = hfsplus_find_cat(inode->i_sb, inode->i_ino, &cat_fd);
  248. if (err) {
  249. pr_err("catalog searching failed\n");
  250. goto end_setxattr;
  251. }
  252. if (!strcmp_xattr_finder_info(name)) {
  253. if (flags & XATTR_CREATE) {
  254. pr_err("xattr exists yet\n");
  255. err = -EOPNOTSUPP;
  256. goto end_setxattr;
  257. }
  258. hfs_bnode_read(cat_fd.bnode, &entry, cat_fd.entryoffset,
  259. sizeof(hfsplus_cat_entry));
  260. if (be16_to_cpu(entry.type) == HFSPLUS_FOLDER) {
  261. if (size == folder_finderinfo_len) {
  262. memcpy(&entry.folder.user_info, value,
  263. folder_finderinfo_len);
  264. hfs_bnode_write(cat_fd.bnode, &entry,
  265. cat_fd.entryoffset,
  266. sizeof(struct hfsplus_cat_folder));
  267. hfsplus_mark_inode_dirty(inode,
  268. HFSPLUS_I_CAT_DIRTY);
  269. } else {
  270. err = -ERANGE;
  271. goto end_setxattr;
  272. }
  273. } else if (be16_to_cpu(entry.type) == HFSPLUS_FILE) {
  274. if (size == file_finderinfo_len) {
  275. memcpy(&entry.file.user_info, value,
  276. file_finderinfo_len);
  277. hfs_bnode_write(cat_fd.bnode, &entry,
  278. cat_fd.entryoffset,
  279. sizeof(struct hfsplus_cat_file));
  280. hfsplus_mark_inode_dirty(inode,
  281. HFSPLUS_I_CAT_DIRTY);
  282. } else {
  283. err = -ERANGE;
  284. goto end_setxattr;
  285. }
  286. } else {
  287. err = -EOPNOTSUPP;
  288. goto end_setxattr;
  289. }
  290. goto end_setxattr;
  291. }
  292. if (!HFSPLUS_SB(inode->i_sb)->attr_tree) {
  293. err = hfsplus_create_attributes_file(inode->i_sb);
  294. if (unlikely(err))
  295. goto end_setxattr;
  296. }
  297. if (hfsplus_attr_exists(inode, name)) {
  298. if (flags & XATTR_CREATE) {
  299. pr_err("xattr exists yet\n");
  300. err = -EOPNOTSUPP;
  301. goto end_setxattr;
  302. }
  303. err = hfsplus_delete_attr(inode, name);
  304. if (err)
  305. goto end_setxattr;
  306. err = hfsplus_create_attr(inode, name, value, size);
  307. if (err)
  308. goto end_setxattr;
  309. } else {
  310. if (flags & XATTR_REPLACE) {
  311. pr_err("cannot replace xattr\n");
  312. err = -EOPNOTSUPP;
  313. goto end_setxattr;
  314. }
  315. err = hfsplus_create_attr(inode, name, value, size);
  316. if (err)
  317. goto end_setxattr;
  318. }
  319. cat_entry_type = hfs_bnode_read_u16(cat_fd.bnode, cat_fd.entryoffset);
  320. if (cat_entry_type == HFSPLUS_FOLDER) {
  321. cat_entry_flags = hfs_bnode_read_u16(cat_fd.bnode,
  322. cat_fd.entryoffset +
  323. offsetof(struct hfsplus_cat_folder, flags));
  324. cat_entry_flags |= HFSPLUS_XATTR_EXISTS;
  325. if (!strcmp_xattr_acl(name))
  326. cat_entry_flags |= HFSPLUS_ACL_EXISTS;
  327. hfs_bnode_write_u16(cat_fd.bnode, cat_fd.entryoffset +
  328. offsetof(struct hfsplus_cat_folder, flags),
  329. cat_entry_flags);
  330. hfsplus_mark_inode_dirty(inode, HFSPLUS_I_CAT_DIRTY);
  331. } else if (cat_entry_type == HFSPLUS_FILE) {
  332. cat_entry_flags = hfs_bnode_read_u16(cat_fd.bnode,
  333. cat_fd.entryoffset +
  334. offsetof(struct hfsplus_cat_file, flags));
  335. cat_entry_flags |= HFSPLUS_XATTR_EXISTS;
  336. if (!strcmp_xattr_acl(name))
  337. cat_entry_flags |= HFSPLUS_ACL_EXISTS;
  338. hfs_bnode_write_u16(cat_fd.bnode, cat_fd.entryoffset +
  339. offsetof(struct hfsplus_cat_file, flags),
  340. cat_entry_flags);
  341. hfsplus_mark_inode_dirty(inode, HFSPLUS_I_CAT_DIRTY);
  342. } else {
  343. pr_err("invalid catalog entry type\n");
  344. err = -EIO;
  345. goto end_setxattr;
  346. }
  347. end_setxattr:
  348. hfs_find_exit(&cat_fd);
  349. return err;
  350. }
  351. static int name_len(const char *xattr_name, int xattr_name_len)
  352. {
  353. int len = xattr_name_len + 1;
  354. if (!is_known_namespace(xattr_name))
  355. len += XATTR_MAC_OSX_PREFIX_LEN;
  356. return len;
  357. }
  358. static int copy_name(char *buffer, const char *xattr_name, int name_len)
  359. {
  360. int len = name_len;
  361. int offset = 0;
  362. if (!is_known_namespace(xattr_name)) {
  363. strncpy(buffer, XATTR_MAC_OSX_PREFIX, XATTR_MAC_OSX_PREFIX_LEN);
  364. offset += XATTR_MAC_OSX_PREFIX_LEN;
  365. len += XATTR_MAC_OSX_PREFIX_LEN;
  366. }
  367. strncpy(buffer + offset, xattr_name, name_len);
  368. memset(buffer + offset + name_len, 0, 1);
  369. len += 1;
  370. return len;
  371. }
  372. int hfsplus_setxattr(struct inode *inode, const char *name,
  373. const void *value, size_t size, int flags,
  374. const char *prefix, size_t prefixlen)
  375. {
  376. char *xattr_name;
  377. int res;
  378. xattr_name = kmalloc(NLS_MAX_CHARSET_SIZE * HFSPLUS_ATTR_MAX_STRLEN + 1,
  379. GFP_KERNEL);
  380. if (!xattr_name)
  381. return -ENOMEM;
  382. strcpy(xattr_name, prefix);
  383. strcpy(xattr_name + prefixlen, name);
  384. res = __hfsplus_setxattr(inode, xattr_name, value, size, flags);
  385. kfree(xattr_name);
  386. return res;
  387. }
  388. static ssize_t hfsplus_getxattr_finder_info(struct inode *inode,
  389. void *value, size_t size)
  390. {
  391. ssize_t res = 0;
  392. struct hfs_find_data fd;
  393. u16 entry_type;
  394. u16 folder_rec_len = sizeof(struct DInfo) + sizeof(struct DXInfo);
  395. u16 file_rec_len = sizeof(struct FInfo) + sizeof(struct FXInfo);
  396. u16 record_len = max(folder_rec_len, file_rec_len);
  397. u8 folder_finder_info[sizeof(struct DInfo) + sizeof(struct DXInfo)];
  398. u8 file_finder_info[sizeof(struct FInfo) + sizeof(struct FXInfo)];
  399. if (size >= record_len) {
  400. res = hfs_find_init(HFSPLUS_SB(inode->i_sb)->cat_tree, &fd);
  401. if (res) {
  402. pr_err("can't init xattr find struct\n");
  403. return res;
  404. }
  405. res = hfsplus_find_cat(inode->i_sb, inode->i_ino, &fd);
  406. if (res)
  407. goto end_getxattr_finder_info;
  408. entry_type = hfs_bnode_read_u16(fd.bnode, fd.entryoffset);
  409. if (entry_type == HFSPLUS_FOLDER) {
  410. hfs_bnode_read(fd.bnode, folder_finder_info,
  411. fd.entryoffset +
  412. offsetof(struct hfsplus_cat_folder, user_info),
  413. folder_rec_len);
  414. memcpy(value, folder_finder_info, folder_rec_len);
  415. res = folder_rec_len;
  416. } else if (entry_type == HFSPLUS_FILE) {
  417. hfs_bnode_read(fd.bnode, file_finder_info,
  418. fd.entryoffset +
  419. offsetof(struct hfsplus_cat_file, user_info),
  420. file_rec_len);
  421. memcpy(value, file_finder_info, file_rec_len);
  422. res = file_rec_len;
  423. } else {
  424. res = -EOPNOTSUPP;
  425. goto end_getxattr_finder_info;
  426. }
  427. } else
  428. res = size ? -ERANGE : record_len;
  429. end_getxattr_finder_info:
  430. if (size >= record_len)
  431. hfs_find_exit(&fd);
  432. return res;
  433. }
  434. ssize_t __hfsplus_getxattr(struct inode *inode, const char *name,
  435. void *value, size_t size)
  436. {
  437. struct hfs_find_data fd;
  438. hfsplus_attr_entry *entry;
  439. __be32 xattr_record_type;
  440. u32 record_type;
  441. u16 record_length = 0;
  442. ssize_t res = 0;
  443. if ((!S_ISREG(inode->i_mode) &&
  444. !S_ISDIR(inode->i_mode)) ||
  445. HFSPLUS_IS_RSRC(inode))
  446. return -EOPNOTSUPP;
  447. if (!strcmp_xattr_finder_info(name))
  448. return hfsplus_getxattr_finder_info(inode, value, size);
  449. if (!HFSPLUS_SB(inode->i_sb)->attr_tree)
  450. return -EOPNOTSUPP;
  451. entry = hfsplus_alloc_attr_entry();
  452. if (!entry) {
  453. pr_err("can't allocate xattr entry\n");
  454. return -ENOMEM;
  455. }
  456. res = hfs_find_init(HFSPLUS_SB(inode->i_sb)->attr_tree, &fd);
  457. if (res) {
  458. pr_err("can't init xattr find struct\n");
  459. goto failed_getxattr_init;
  460. }
  461. res = hfsplus_find_attr(inode->i_sb, inode->i_ino, name, &fd);
  462. if (res) {
  463. if (res == -ENOENT)
  464. res = -ENODATA;
  465. else
  466. pr_err("xattr searching failed\n");
  467. goto out;
  468. }
  469. hfs_bnode_read(fd.bnode, &xattr_record_type,
  470. fd.entryoffset, sizeof(xattr_record_type));
  471. record_type = be32_to_cpu(xattr_record_type);
  472. if (record_type == HFSPLUS_ATTR_INLINE_DATA) {
  473. record_length = hfs_bnode_read_u16(fd.bnode,
  474. fd.entryoffset +
  475. offsetof(struct hfsplus_attr_inline_data,
  476. length));
  477. if (record_length > HFSPLUS_MAX_INLINE_DATA_SIZE) {
  478. pr_err("invalid xattr record size\n");
  479. res = -EIO;
  480. goto out;
  481. }
  482. } else if (record_type == HFSPLUS_ATTR_FORK_DATA ||
  483. record_type == HFSPLUS_ATTR_EXTENTS) {
  484. pr_err("only inline data xattr are supported\n");
  485. res = -EOPNOTSUPP;
  486. goto out;
  487. } else {
  488. pr_err("invalid xattr record\n");
  489. res = -EIO;
  490. goto out;
  491. }
  492. if (size) {
  493. hfs_bnode_read(fd.bnode, entry, fd.entryoffset,
  494. offsetof(struct hfsplus_attr_inline_data,
  495. raw_bytes) + record_length);
  496. }
  497. if (size >= record_length) {
  498. memcpy(value, entry->inline_data.raw_bytes, record_length);
  499. res = record_length;
  500. } else
  501. res = size ? -ERANGE : record_length;
  502. out:
  503. hfs_find_exit(&fd);
  504. failed_getxattr_init:
  505. hfsplus_destroy_attr_entry(entry);
  506. return res;
  507. }
  508. ssize_t hfsplus_getxattr(struct inode *inode, const char *name,
  509. void *value, size_t size,
  510. const char *prefix, size_t prefixlen)
  511. {
  512. int res;
  513. char *xattr_name;
  514. xattr_name = kmalloc(NLS_MAX_CHARSET_SIZE * HFSPLUS_ATTR_MAX_STRLEN + 1,
  515. GFP_KERNEL);
  516. if (!xattr_name)
  517. return -ENOMEM;
  518. strcpy(xattr_name, prefix);
  519. strcpy(xattr_name + prefixlen, name);
  520. res = __hfsplus_getxattr(inode, xattr_name, value, size);
  521. kfree(xattr_name);
  522. return res;
  523. }
  524. static inline int can_list(const char *xattr_name)
  525. {
  526. if (!xattr_name)
  527. return 0;
  528. return strncmp(xattr_name, XATTR_TRUSTED_PREFIX,
  529. XATTR_TRUSTED_PREFIX_LEN) ||
  530. capable(CAP_SYS_ADMIN);
  531. }
  532. static ssize_t hfsplus_listxattr_finder_info(struct dentry *dentry,
  533. char *buffer, size_t size)
  534. {
  535. ssize_t res = 0;
  536. struct inode *inode = d_inode(dentry);
  537. struct hfs_find_data fd;
  538. u16 entry_type;
  539. u8 folder_finder_info[sizeof(struct DInfo) + sizeof(struct DXInfo)];
  540. u8 file_finder_info[sizeof(struct FInfo) + sizeof(struct FXInfo)];
  541. unsigned long len, found_bit;
  542. int xattr_name_len, symbols_count;
  543. res = hfs_find_init(HFSPLUS_SB(inode->i_sb)->cat_tree, &fd);
  544. if (res) {
  545. pr_err("can't init xattr find struct\n");
  546. return res;
  547. }
  548. res = hfsplus_find_cat(inode->i_sb, inode->i_ino, &fd);
  549. if (res)
  550. goto end_listxattr_finder_info;
  551. entry_type = hfs_bnode_read_u16(fd.bnode, fd.entryoffset);
  552. if (entry_type == HFSPLUS_FOLDER) {
  553. len = sizeof(struct DInfo) + sizeof(struct DXInfo);
  554. hfs_bnode_read(fd.bnode, folder_finder_info,
  555. fd.entryoffset +
  556. offsetof(struct hfsplus_cat_folder, user_info),
  557. len);
  558. found_bit = find_first_bit((void *)folder_finder_info, len*8);
  559. } else if (entry_type == HFSPLUS_FILE) {
  560. len = sizeof(struct FInfo) + sizeof(struct FXInfo);
  561. hfs_bnode_read(fd.bnode, file_finder_info,
  562. fd.entryoffset +
  563. offsetof(struct hfsplus_cat_file, user_info),
  564. len);
  565. found_bit = find_first_bit((void *)file_finder_info, len*8);
  566. } else {
  567. res = -EOPNOTSUPP;
  568. goto end_listxattr_finder_info;
  569. }
  570. if (found_bit >= (len*8))
  571. res = 0;
  572. else {
  573. symbols_count = sizeof(HFSPLUS_XATTR_FINDER_INFO_NAME) - 1;
  574. xattr_name_len =
  575. name_len(HFSPLUS_XATTR_FINDER_INFO_NAME, symbols_count);
  576. if (!buffer || !size) {
  577. if (can_list(HFSPLUS_XATTR_FINDER_INFO_NAME))
  578. res = xattr_name_len;
  579. } else if (can_list(HFSPLUS_XATTR_FINDER_INFO_NAME)) {
  580. if (size < xattr_name_len)
  581. res = -ERANGE;
  582. else {
  583. res = copy_name(buffer,
  584. HFSPLUS_XATTR_FINDER_INFO_NAME,
  585. symbols_count);
  586. }
  587. }
  588. }
  589. end_listxattr_finder_info:
  590. hfs_find_exit(&fd);
  591. return res;
  592. }
  593. ssize_t hfsplus_listxattr(struct dentry *dentry, char *buffer, size_t size)
  594. {
  595. ssize_t err;
  596. ssize_t res = 0;
  597. struct inode *inode = d_inode(dentry);
  598. struct hfs_find_data fd;
  599. u16 key_len = 0;
  600. struct hfsplus_attr_key attr_key;
  601. char *strbuf;
  602. int xattr_name_len;
  603. if ((!S_ISREG(inode->i_mode) &&
  604. !S_ISDIR(inode->i_mode)) ||
  605. HFSPLUS_IS_RSRC(inode))
  606. return -EOPNOTSUPP;
  607. res = hfsplus_listxattr_finder_info(dentry, buffer, size);
  608. if (res < 0)
  609. return res;
  610. else if (!HFSPLUS_SB(inode->i_sb)->attr_tree)
  611. return (res == 0) ? -EOPNOTSUPP : res;
  612. err = hfs_find_init(HFSPLUS_SB(inode->i_sb)->attr_tree, &fd);
  613. if (err) {
  614. pr_err("can't init xattr find struct\n");
  615. return err;
  616. }
  617. strbuf = kmalloc(NLS_MAX_CHARSET_SIZE * HFSPLUS_ATTR_MAX_STRLEN +
  618. XATTR_MAC_OSX_PREFIX_LEN + 1, GFP_KERNEL);
  619. if (!strbuf) {
  620. res = -ENOMEM;
  621. goto out;
  622. }
  623. err = hfsplus_find_attr(inode->i_sb, inode->i_ino, NULL, &fd);
  624. if (err) {
  625. if (err == -ENOENT) {
  626. if (res == 0)
  627. res = -ENODATA;
  628. goto end_listxattr;
  629. } else {
  630. res = err;
  631. goto end_listxattr;
  632. }
  633. }
  634. for (;;) {
  635. key_len = hfs_bnode_read_u16(fd.bnode, fd.keyoffset);
  636. if (key_len == 0 || key_len > fd.tree->max_key_len) {
  637. pr_err("invalid xattr key length: %d\n", key_len);
  638. res = -EIO;
  639. goto end_listxattr;
  640. }
  641. hfs_bnode_read(fd.bnode, &attr_key,
  642. fd.keyoffset, key_len + sizeof(key_len));
  643. if (be32_to_cpu(attr_key.cnid) != inode->i_ino)
  644. goto end_listxattr;
  645. xattr_name_len = NLS_MAX_CHARSET_SIZE * HFSPLUS_ATTR_MAX_STRLEN;
  646. if (hfsplus_uni2asc(inode->i_sb,
  647. (const struct hfsplus_unistr *)&fd.key->attr.key_name,
  648. strbuf, &xattr_name_len)) {
  649. pr_err("unicode conversion failed\n");
  650. res = -EIO;
  651. goto end_listxattr;
  652. }
  653. if (!buffer || !size) {
  654. if (can_list(strbuf))
  655. res += name_len(strbuf, xattr_name_len);
  656. } else if (can_list(strbuf)) {
  657. if (size < (res + name_len(strbuf, xattr_name_len))) {
  658. res = -ERANGE;
  659. goto end_listxattr;
  660. } else
  661. res += copy_name(buffer + res,
  662. strbuf, xattr_name_len);
  663. }
  664. if (hfs_brec_goto(&fd, 1))
  665. goto end_listxattr;
  666. }
  667. end_listxattr:
  668. kfree(strbuf);
  669. out:
  670. hfs_find_exit(&fd);
  671. return res;
  672. }
  673. static int hfsplus_removexattr(struct inode *inode, const char *name)
  674. {
  675. int err = 0;
  676. struct hfs_find_data cat_fd;
  677. u16 flags;
  678. u16 cat_entry_type;
  679. int is_xattr_acl_deleted = 0;
  680. int is_all_xattrs_deleted = 0;
  681. if (!HFSPLUS_SB(inode->i_sb)->attr_tree)
  682. return -EOPNOTSUPP;
  683. if (!strcmp_xattr_finder_info(name))
  684. return -EOPNOTSUPP;
  685. err = hfs_find_init(HFSPLUS_SB(inode->i_sb)->cat_tree, &cat_fd);
  686. if (err) {
  687. pr_err("can't init xattr find struct\n");
  688. return err;
  689. }
  690. err = hfsplus_find_cat(inode->i_sb, inode->i_ino, &cat_fd);
  691. if (err) {
  692. pr_err("catalog searching failed\n");
  693. goto end_removexattr;
  694. }
  695. err = hfsplus_delete_attr(inode, name);
  696. if (err)
  697. goto end_removexattr;
  698. is_xattr_acl_deleted = !strcmp_xattr_acl(name);
  699. is_all_xattrs_deleted = !hfsplus_attr_exists(inode, NULL);
  700. if (!is_xattr_acl_deleted && !is_all_xattrs_deleted)
  701. goto end_removexattr;
  702. cat_entry_type = hfs_bnode_read_u16(cat_fd.bnode, cat_fd.entryoffset);
  703. if (cat_entry_type == HFSPLUS_FOLDER) {
  704. flags = hfs_bnode_read_u16(cat_fd.bnode, cat_fd.entryoffset +
  705. offsetof(struct hfsplus_cat_folder, flags));
  706. if (is_xattr_acl_deleted)
  707. flags &= ~HFSPLUS_ACL_EXISTS;
  708. if (is_all_xattrs_deleted)
  709. flags &= ~HFSPLUS_XATTR_EXISTS;
  710. hfs_bnode_write_u16(cat_fd.bnode, cat_fd.entryoffset +
  711. offsetof(struct hfsplus_cat_folder, flags),
  712. flags);
  713. hfsplus_mark_inode_dirty(inode, HFSPLUS_I_CAT_DIRTY);
  714. } else if (cat_entry_type == HFSPLUS_FILE) {
  715. flags = hfs_bnode_read_u16(cat_fd.bnode, cat_fd.entryoffset +
  716. offsetof(struct hfsplus_cat_file, flags));
  717. if (is_xattr_acl_deleted)
  718. flags &= ~HFSPLUS_ACL_EXISTS;
  719. if (is_all_xattrs_deleted)
  720. flags &= ~HFSPLUS_XATTR_EXISTS;
  721. hfs_bnode_write_u16(cat_fd.bnode, cat_fd.entryoffset +
  722. offsetof(struct hfsplus_cat_file, flags),
  723. flags);
  724. hfsplus_mark_inode_dirty(inode, HFSPLUS_I_CAT_DIRTY);
  725. } else {
  726. pr_err("invalid catalog entry type\n");
  727. err = -EIO;
  728. goto end_removexattr;
  729. }
  730. end_removexattr:
  731. hfs_find_exit(&cat_fd);
  732. return err;
  733. }
  734. static int hfsplus_osx_getxattr(const struct xattr_handler *handler,
  735. struct dentry *unused, struct inode *inode,
  736. const char *name, void *buffer, size_t size)
  737. {
  738. /*
  739. * Don't allow retrieving properly prefixed attributes
  740. * by prepending them with "osx."
  741. */
  742. if (is_known_namespace(name))
  743. return -EOPNOTSUPP;
  744. /*
  745. * osx is the namespace we use to indicate an unprefixed
  746. * attribute on the filesystem (like the ones that OS X
  747. * creates), so we pass the name through unmodified (after
  748. * ensuring it doesn't conflict with another namespace).
  749. */
  750. return __hfsplus_getxattr(inode, name, buffer, size);
  751. }
  752. static int hfsplus_osx_setxattr(const struct xattr_handler *handler,
  753. struct dentry *unused, struct inode *inode,
  754. const char *name, const void *buffer,
  755. size_t size, int flags)
  756. {
  757. /*
  758. * Don't allow setting properly prefixed attributes
  759. * by prepending them with "osx."
  760. */
  761. if (is_known_namespace(name))
  762. return -EOPNOTSUPP;
  763. /*
  764. * osx is the namespace we use to indicate an unprefixed
  765. * attribute on the filesystem (like the ones that OS X
  766. * creates), so we pass the name through unmodified (after
  767. * ensuring it doesn't conflict with another namespace).
  768. */
  769. return __hfsplus_setxattr(inode, name, buffer, size, flags);
  770. }
  771. const struct xattr_handler hfsplus_xattr_osx_handler = {
  772. .prefix = XATTR_MAC_OSX_PREFIX,
  773. .get = hfsplus_osx_getxattr,
  774. .set = hfsplus_osx_setxattr,
  775. };