2 * xattrs.c --- Modify extended attributes via debugfs.
4 * Copyright (C) 2014 Oracle. This file may be redistributed
5 * under the terms of the GNU Public License.
17 #include "support/cstring.h"
20 #include "ext2fs/lfsck.h"
22 #define PRINT_XATTR_HEX 0x01
23 #define PRINT_XATTR_RAW 0x02
24 #define PRINT_XATTR_C 0x04
25 #define PRINT_XATTR_STATFMT 0x08
26 #define PRINT_XATTR_NOQUOTES 0x10
28 extern const char *debug_prog_name;
30 /* Dump extended attributes */
31 static void print_xattr_hex(FILE *f, const char *str, int len)
35 for (i = 0; i < len; i++)
36 fprintf(f, "%02x ", (unsigned char)str[i]);
39 /* Dump extended attributes */
40 static void print_xattr_string(FILE *f, const char *str, int len, int flags)
45 if (flags & PRINT_XATTR_RAW) {
46 fwrite(str, len, 1, f);
50 if ((flags & PRINT_XATTR_C) == 0) {
51 /* check: is string "printable enough?" */
52 for (i = 0; i < len; i++)
56 if (printable <= len*7/8)
57 flags |= PRINT_XATTR_HEX;
60 if (flags & PRINT_XATTR_HEX) {
61 print_xattr_hex(f, str, len);
63 if ((flags & PRINT_XATTR_NOQUOTES) == 0)
65 print_c_string(f, str, len);
66 if ((flags & PRINT_XATTR_NOQUOTES) == 0)
71 static void print_xattr(FILE *f, char *name, char *value, size_t value_len,
74 print_xattr_string(f, name, strlen(name), PRINT_XATTR_NOQUOTES);
75 fprintf(f, " (%zu)", value_len);
76 if ((print_flags & PRINT_XATTR_STATFMT) &&
77 (strcmp(name, "system.data") == 0))
80 (!(print_flags & PRINT_XATTR_STATFMT) || (value_len < 40))) {
82 print_xattr_string(f, value, value_len, print_flags);
87 static void print_fidstr(FILE *f, void *name, void *value,
88 ext2_ino_t inode_num, size_t value_len)
90 struct filter_fid_old *ff = value;
93 /* Since Lustre 2.4 only the parent FID is stored in filter_fid,
94 * and the self fid is stored in the LMA and is printed below. */
95 if (value_len < sizeof(ff->ff_parent)) {
96 fprintf(stderr, "%s: error: filter_fid for inode %u smaller "
97 "than expected (%lu bytes).\n",
98 debug_prog_name, inode_num, value_len);
101 fid_le_to_cpu(&ff->ff_parent, &ff->ff_parent);
102 stripe = fid_ver(&ff->ff_parent); /* stripe index is stored in f_ver */
103 ff->ff_parent.f_ver = 0;
105 fprintf(f, " fid: ");
106 /* Old larger filter_fid should only ever be used with seq = 0.
107 * FID-on-OST should use LMA for FID_SEQ_NORMAL OST objects. */
108 if (value_len == sizeof(*ff))
109 fprintf(f, "objid=%llu seq=%llu ",
110 ext2fs_le64_to_cpu(ff->ff_objid),
111 ext2fs_le64_to_cpu(ff->ff_seq));
113 fprintf(f, "parent="DFID" stripe=%u", PFID(&ff->ff_parent), stripe);
114 if (value_len > sizeof(*ff)) {
115 struct filter_fid *ff_new = value;
117 fprintf(f, " stripe_size=%u stripe_count=%u",
118 ext2fs_le32_to_cpu(ff_new->ff_stripe_size),
119 ext2fs_le32_to_cpu(ff_new->ff_stripe_count));
120 if (ff_new->ff_pfl_id != 0)
121 fprintf(f, " component_id=%u component_start=%llu "
122 "component_end=%llu",
123 ext2fs_le32_to_cpu(ff_new->ff_pfl_id),
124 ext2fs_le64_to_cpu(ff_new->ff_pfl_start),
125 ext2fs_le64_to_cpu(ff_new->ff_pfl_end));
130 static void print_lmastr(FILE *f, void *name, void *value,
131 ext2_ino_t inode_num, size_t value_len)
133 struct lustre_mdt_attrs *lma = value;
134 struct lustre_ost_attrs *loa = value;
136 if (value_len < offsetof(typeof(*lma), lma_self_fid) +
137 sizeof(lma->lma_self_fid)) {
138 fprintf(stderr, "%s: error: LMA for inode %u smaller than "
139 "expected (%lu bytes).\n",
140 debug_prog_name, inode_num, value_len);
143 fid_le_to_cpu(&lma->lma_self_fid, &lma->lma_self_fid);
144 fprintf(f, " lma: fid="DFID" compat=%x incompat=%x\n",
145 PFID(&lma->lma_self_fid), ext2fs_le32_to_cpu(lma->lma_compat),
146 ext2fs_le32_to_cpu(lma->lma_incompat));
147 if (value_len >= offsetof(typeof(*loa), loa_pfl_end) +
148 sizeof(loa->loa_pfl_end)) {
152 fid_le_to_cpu(&loa->loa_parent_fid, &loa->loa_parent_fid);
153 idx = loa->loa_parent_fid.f_ver & PFID_STRIPE_COUNT_MASK;
154 cnt = loa->loa_parent_fid.f_ver >> PFID_STRIPE_IDX_BITS;
155 loa->loa_parent_fid.f_ver = 0;
157 fprintf(f, " fid: parent="DFID" stripe=%u stripe_size=%u "
158 "stripe_count=%u", PFID(&loa->loa_parent_fid), idx,
159 ext2fs_le32_to_cpu(loa->loa_stripe_size), cnt);
160 if (loa->loa_pfl_id != 0)
161 fprintf(f, " component_id=%u component_start=%llu "
162 "component_end=%llu",
163 ext2fs_le32_to_cpu(loa->loa_pfl_id),
164 ext2fs_le64_to_cpu(loa->loa_pfl_start),
165 ext2fs_le64_to_cpu(loa->loa_pfl_end));
170 static int dump_attr(char *name, char *value, size_t value_len,
171 ext2_ino_t inode_num, void *data)
176 if (EXT2_HAS_INCOMPAT_FEATURE(current_fs->super,
177 EXT4_FEATURE_INCOMPAT_EA_INODE) &&
179 fprintf(out, "inode <%u> ", inode_num);
181 print_xattr(out, name, value, value_len, PRINT_XATTR_STATFMT);
182 if (!strncmp(name, EXT2_ATTR_INDEX_TRUSTED_PREFIX LUSTRE_XATTR_OST_FID,
184 !strncmp(name, EXT2_ATTR_INDEX_LUSTRE_PREFIX LUSTRE_XATTR_OST_FID,
186 print_fidstr(out, name, value, inode_num, value_len);
187 if (!strncmp(name, EXT2_ATTR_INDEX_TRUSTED_PREFIX LUSTRE_XATTR_MDT_LMA,
189 !strncmp(name, EXT2_ATTR_INDEX_LUSTRE_PREFIX LUSTRE_XATTR_MDT_LMA,
191 print_lmastr(out, name, value, inode_num, value_len);
195 void dump_inode_attributes(FILE *out, ext2_ino_t ino)
197 struct ext2_xattr_handle *h;
201 err = ext2fs_xattrs_open(current_fs, ino, &h);
205 err = ext2fs_xattrs_read(h);
209 err = ext2fs_xattrs_count(h, &sz);
213 fprintf(out, "Extended attributes:\n");
214 err = ext2fs_xattrs_iterate(h, dump_attr, out);
219 err = ext2fs_xattrs_close(&h);
222 void do_list_xattr(int argc, char **argv, int sci_idx EXT2FS_ATTR((unused)),
223 void *infop EXT2FS_ATTR((unused)))
228 printf("%s: Usage: %s <file>\n", argv[0],
233 if (check_fs_open(argv[0]))
236 ino = string_to_inode(argv[1]);
240 dump_inode_attributes(stdout, ino);
243 void do_get_xattr(int argc, char **argv, int sci_idx EXT2FS_ATTR((unused)),
244 void *infop EXT2FS_ATTR((unused)))
247 struct ext2_xattr_handle *h;
253 unsigned int handle_flags = 0;
257 while ((i = getopt(argc, argv, "Cf:rxV")) != -1) {
262 fp = fopen(optarg, "w");
269 handle_flags |= XATTR_HANDLE_FLAG_RAW;
272 print_flags |= PRINT_XATTR_HEX;
275 print_flags |= PRINT_XATTR_RAW|
276 PRINT_XATTR_NOQUOTES;
279 print_flags |= PRINT_XATTR_C;
286 if (optind != argc - 2) {
288 printf("%s: Usage: %s [-f outfile]|[-xVC] [-r] <file> <attr>\n",
294 if (check_fs_open(argv[0]))
297 ino = string_to_inode(argv[optind]);
301 err = ext2fs_xattrs_open(current_fs, ino, &h);
305 err = ext2fs_xattrs_flags(h, &handle_flags, NULL);
309 err = ext2fs_xattrs_read(h);
313 err = ext2fs_xattr_get(h, argv[optind + 1], (void **)&buf, &buflen);
318 fwrite(buf, buflen, 1, fp);
320 if (print_flags & PRINT_XATTR_RAW) {
321 if (print_flags & (PRINT_XATTR_HEX|PRINT_XATTR_C))
322 print_flags &= ~PRINT_XATTR_RAW;
323 print_xattr_string(stdout, buf, buflen, print_flags);
325 print_xattr(stdout, argv[optind + 1],
326 buf, buflen, print_flags);
331 ext2fs_free_mem(&buf);
333 ext2fs_xattrs_close(&h);
335 com_err(argv[0], err, "while getting extended attribute");
341 void do_set_xattr(int argc, char **argv, int sci_idx EXT2FS_ATTR((unused)),
342 void *infop EXT2FS_ATTR((unused)))
345 struct ext2_xattr_handle *h;
349 unsigned int handle_flags = 0;
354 while ((i = getopt(argc, argv, "f:r")) != -1) {
359 fp = fopen(optarg, "r");
366 handle_flags |= XATTR_HANDLE_FLAG_RAW;
373 if (!(fp && optind == argc - 2) && !(!fp && optind == argc - 3)) {
375 printf("Usage:\t%s [-r] <file> <attr> <value>\n", argv[0]);
376 printf("\t%s -f <value_file> [-r] <file> <attr>\n", argv[0]);
380 if (check_fs_open(argv[0]))
382 if (check_fs_read_write(argv[0]))
384 if (check_fs_bitmaps(argv[0]))
387 ino = string_to_inode(argv[optind]);
391 err = ext2fs_xattrs_open(current_fs, ino, &h);
395 err = ext2fs_xattrs_flags(h, &handle_flags, NULL);
399 err = ext2fs_xattrs_read(h);
404 err = ext2fs_get_mem(current_fs->blocksize, &buf);
407 buflen = fread(buf, 1, current_fs->blocksize, fp);
409 buf = argv[optind + 2];
410 buflen = parse_c_string(buf);
413 err = ext2fs_xattr_set(h, argv[optind + 1], buf, buflen);
415 ext2fs_xattrs_close(&h);
417 com_err(argv[0], err, "while setting extended attribute");
421 ext2fs_free_mem(&buf);
425 void do_rm_xattr(int argc, char **argv, int sci_idx EXT2FS_ATTR((unused)),
426 void *infop EXT2FS_ATTR((unused)))
429 struct ext2_xattr_handle *h;
434 printf("%s: Usage: %s <file> <attrs>...\n", argv[0], argv[0]);
438 if (check_fs_open(argv[0]))
440 if (check_fs_read_write(argv[0]))
442 if (check_fs_bitmaps(argv[0]))
445 ino = string_to_inode(argv[1]);
449 err = ext2fs_xattrs_open(current_fs, ino, &h);
453 err = ext2fs_xattrs_read(h);
457 for (i = 2; i < argc; i++) {
458 err = ext2fs_xattr_remove(h, argv[i]);
463 ext2fs_xattrs_close(&h);
465 com_err(argv[0], err, "while removing extended attribute");
469 * Return non-zero if the string has a minimal number of non-printable
472 static int is_mostly_printable(const char *cp, int len)
480 if (!isprint(*cp++)) {
489 static void safe_print(FILE *f, const char *cp, int len)
502 if ((ch < 32) || (ch == 0x7f)) {
504 ch ^= 0x40; /* ^@, ^A, ^B; ^? for DEL */
510 static void dump_xattr_raw_entries(FILE *f, unsigned char *buf,
511 unsigned int start, unsigned int len,
512 unsigned value_start)
514 struct ext2_ext_attr_entry ent;
515 unsigned int off = start;
519 if ((*(__u16 *) (buf + off)) == 0) {
520 fprintf(f, "last entry found at offset %u (%04o)\n",
524 if ((off + sizeof(struct ext2_ext_attr_entry)) >= len) {
525 fprintf(f, "xattr buffer overrun at %u (len = %u)\n",
530 ext2fs_swap_ext_attr_entry(&ent,
531 (struct ext2_ext_attr_entry *) (buf + off));
533 ent = *((struct ext2_ext_attr_entry *) (buf + off));
535 fprintf(f, "offset = %d (%04o), name_len = %u, "
537 off, off, ent.e_name_len, ent.e_name_index);
538 vstart = value_start + ent.e_value_offs;
539 fprintf(f, "value_offset = %d (%04o), value_inum = %u, "
540 "value_size = %u\n", ent.e_value_offs,
541 vstart, ent.e_value_inum, ent.e_value_size);
542 off += sizeof(struct ext2_ext_attr_entry);
543 fprintf(f, "name = ");
544 if ((off + ent.e_name_len) >= len)
545 fprintf(f, "<runs off end>");
547 safe_print(f, (char *)(buf + off), ent.e_name_len);
549 if (ent.e_value_size == 0)
551 fprintf(f, "value = ");
552 if (ent.e_value_inum)
553 fprintf(f, "<ino %u>", ent.e_value_inum);
554 else if (ent.e_value_offs >= len ||
555 (vstart + ent.e_value_size) > len)
556 fprintf(f, "<runs off end>");
557 else if (is_mostly_printable((char *)(buf + vstart),
559 safe_print(f, (char *)(buf + vstart),
562 fprintf(f, "<hexdump>\n");
563 do_byte_hexdump(f, (unsigned char *)(buf + vstart),
569 off += (ent.e_name_len + 3) & ~3;
573 void raw_inode_xattr_dump(FILE *f, unsigned char *buf, unsigned int len)
575 __u32 magic = ext2fs_le32_to_cpu(*((__le32 *) buf));
577 fprintf(f, "magic = %08x, length = %u, value_start =4 \n\n",
579 if (magic == EXT2_EXT_ATTR_MAGIC)
580 dump_xattr_raw_entries(f, buf, 4, len, 4);
583 void block_xattr_dump(FILE *f, unsigned char *buf, unsigned int len)
585 struct ext2_ext_attr_header header;
587 #ifdef WORDS_BIGENDIAN
588 ext2fs_swap_ext_attr_header(&header,
589 (struct ext2_ext_attr_header *) buf);
591 header = *((struct ext2_ext_attr_header *) buf);
593 fprintf(f, "magic = %08x, length = %u\n", header.h_magic, len);
594 if (header.h_magic != EXT2_EXT_ATTR_MAGIC)
596 fprintf(f, "refcount = %u, blocks = %u\n", header.h_refcount,
598 fprintf(f, "hash = %08x, checksum = %08x\n", header.h_hash,
600 fprintf(f, "reserved: %08x %08x %08x\n\n", header.h_reserved[0],
601 header.h_reserved[1], header.h_reserved[2]);
603 dump_xattr_raw_entries(f, buf,
604 sizeof(struct ext2_ext_attr_header), len, 0);