Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | /* |
Nathan Scott | 7b71876 | 2005-11-02 14:58:39 +1100 | [diff] [blame] | 2 | * Copyright (c) 2000,2005 Silicon Graphics, Inc. |
| 3 | * All Rights Reserved. |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 4 | * |
Nathan Scott | 7b71876 | 2005-11-02 14:58:39 +1100 | [diff] [blame] | 5 | * This program is free software; you can redistribute it and/or |
| 6 | * modify it under the terms of the GNU General Public License as |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 7 | * published by the Free Software Foundation. |
| 8 | * |
Nathan Scott | 7b71876 | 2005-11-02 14:58:39 +1100 | [diff] [blame] | 9 | * This program is distributed in the hope that it would be useful, |
| 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 12 | * GNU General Public License for more details. |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 13 | * |
Nathan Scott | 7b71876 | 2005-11-02 14:58:39 +1100 | [diff] [blame] | 14 | * You should have received a copy of the GNU General Public License |
| 15 | * along with this program; if not, write the Free Software Foundation, |
| 16 | * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 17 | */ |
| 18 | #ifndef __XFS_EXTFREE_ITEM_H__ |
| 19 | #define __XFS_EXTFREE_ITEM_H__ |
| 20 | |
| 21 | struct xfs_mount; |
| 22 | struct kmem_zone; |
| 23 | |
| 24 | typedef struct xfs_extent { |
| 25 | xfs_dfsbno_t ext_start; |
| 26 | xfs_extlen_t ext_len; |
| 27 | } xfs_extent_t; |
| 28 | |
| 29 | /* |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 30 | * Since an xfs_extent_t has types (start:64, len: 32) |
| 31 | * there are different alignments on 32 bit and 64 bit kernels. |
| 32 | * So we provide the different variants for use by a |
| 33 | * conversion routine. |
| 34 | */ |
| 35 | |
| 36 | typedef struct xfs_extent_32 { |
Tim Shimmin | 128dabc | 2006-09-28 10:55:43 +1000 | [diff] [blame] | 37 | __uint64_t ext_start; |
| 38 | __uint32_t ext_len; |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 39 | } __attribute__((packed)) xfs_extent_32_t; |
| 40 | |
| 41 | typedef struct xfs_extent_64 { |
Tim Shimmin | 128dabc | 2006-09-28 10:55:43 +1000 | [diff] [blame] | 42 | __uint64_t ext_start; |
| 43 | __uint32_t ext_len; |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 44 | __uint32_t ext_pad; |
| 45 | } xfs_extent_64_t; |
| 46 | |
| 47 | /* |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 48 | * This is the structure used to lay out an efi log item in the |
| 49 | * log. The efi_extents field is a variable size array whose |
| 50 | * size is given by efi_nextents. |
| 51 | */ |
| 52 | typedef struct xfs_efi_log_format { |
Tim Shimmin | 128dabc | 2006-09-28 10:55:43 +1000 | [diff] [blame] | 53 | __uint16_t efi_type; /* efi log item type */ |
| 54 | __uint16_t efi_size; /* size of this item */ |
| 55 | __uint32_t efi_nextents; /* # extents to free */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 56 | __uint64_t efi_id; /* efi identifier */ |
| 57 | xfs_extent_t efi_extents[1]; /* array of extents to free */ |
| 58 | } xfs_efi_log_format_t; |
| 59 | |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 60 | typedef struct xfs_efi_log_format_32 { |
Tim Shimmin | 128dabc | 2006-09-28 10:55:43 +1000 | [diff] [blame] | 61 | __uint16_t efi_type; /* efi log item type */ |
| 62 | __uint16_t efi_size; /* size of this item */ |
| 63 | __uint32_t efi_nextents; /* # extents to free */ |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 64 | __uint64_t efi_id; /* efi identifier */ |
| 65 | xfs_extent_32_t efi_extents[1]; /* array of extents to free */ |
| 66 | } __attribute__((packed)) xfs_efi_log_format_32_t; |
| 67 | |
| 68 | typedef struct xfs_efi_log_format_64 { |
Tim Shimmin | 128dabc | 2006-09-28 10:55:43 +1000 | [diff] [blame] | 69 | __uint16_t efi_type; /* efi log item type */ |
| 70 | __uint16_t efi_size; /* size of this item */ |
| 71 | __uint32_t efi_nextents; /* # extents to free */ |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 72 | __uint64_t efi_id; /* efi identifier */ |
| 73 | xfs_extent_64_t efi_extents[1]; /* array of extents to free */ |
| 74 | } xfs_efi_log_format_64_t; |
| 75 | |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 76 | /* |
| 77 | * This is the structure used to lay out an efd log item in the |
| 78 | * log. The efd_extents array is a variable size array whose |
| 79 | * size is given by efd_nextents; |
| 80 | */ |
| 81 | typedef struct xfs_efd_log_format { |
Tim Shimmin | 128dabc | 2006-09-28 10:55:43 +1000 | [diff] [blame] | 82 | __uint16_t efd_type; /* efd log item type */ |
| 83 | __uint16_t efd_size; /* size of this item */ |
| 84 | __uint32_t efd_nextents; /* # of extents freed */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 85 | __uint64_t efd_efi_id; /* id of corresponding efi */ |
| 86 | xfs_extent_t efd_extents[1]; /* array of extents freed */ |
| 87 | } xfs_efd_log_format_t; |
| 88 | |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 89 | typedef struct xfs_efd_log_format_32 { |
Tim Shimmin | 128dabc | 2006-09-28 10:55:43 +1000 | [diff] [blame] | 90 | __uint16_t efd_type; /* efd log item type */ |
| 91 | __uint16_t efd_size; /* size of this item */ |
| 92 | __uint32_t efd_nextents; /* # of extents freed */ |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 93 | __uint64_t efd_efi_id; /* id of corresponding efi */ |
| 94 | xfs_extent_32_t efd_extents[1]; /* array of extents freed */ |
| 95 | } __attribute__((packed)) xfs_efd_log_format_32_t; |
| 96 | |
| 97 | typedef struct xfs_efd_log_format_64 { |
Tim Shimmin | 128dabc | 2006-09-28 10:55:43 +1000 | [diff] [blame] | 98 | __uint16_t efd_type; /* efd log item type */ |
| 99 | __uint16_t efd_size; /* size of this item */ |
| 100 | __uint32_t efd_nextents; /* # of extents freed */ |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 101 | __uint64_t efd_efi_id; /* id of corresponding efi */ |
| 102 | xfs_extent_64_t efd_extents[1]; /* array of extents freed */ |
| 103 | } xfs_efd_log_format_64_t; |
| 104 | |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 105 | |
| 106 | #ifdef __KERNEL__ |
| 107 | |
| 108 | /* |
| 109 | * Max number of extents in fast allocation path. |
| 110 | */ |
| 111 | #define XFS_EFI_MAX_FAST_EXTENTS 16 |
| 112 | |
| 113 | /* |
Dave Chinner | b199c8a | 2010-12-20 11:59:49 +1100 | [diff] [blame] | 114 | * Define EFI flag bits. Manipulated by set/clear/test_bit operators. |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 115 | */ |
Dave Chinner | b199c8a | 2010-12-20 11:59:49 +1100 | [diff] [blame] | 116 | #define XFS_EFI_RECOVERED 1 |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 117 | |
| 118 | /* |
Dave Chinner | 666d644 | 2013-04-03 14:09:21 +1100 | [diff] [blame] | 119 | * This is the "extent free intention" log item. It is used to log the fact |
| 120 | * that some extents need to be free. It is used in conjunction with the |
| 121 | * "extent free done" log item described below. |
| 122 | * |
| 123 | * The EFI is reference counted so that it is not freed prior to both the EFI |
| 124 | * and EFD being committed and unpinned. This ensures that when the last |
| 125 | * reference goes away the EFI will always be in the AIL as it has been |
| 126 | * unpinned, regardless of whether the EFD is processed before or after the EFI. |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 127 | */ |
| 128 | typedef struct xfs_efi_log_item { |
| 129 | xfs_log_item_t efi_item; |
Dave Chinner | 666d644 | 2013-04-03 14:09:21 +1100 | [diff] [blame] | 130 | atomic_t efi_refcount; |
Dave Chinner | b199c8a | 2010-12-20 11:59:49 +1100 | [diff] [blame] | 131 | atomic_t efi_next_extent; |
| 132 | unsigned long efi_flags; /* misc flags */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 133 | xfs_efi_log_format_t efi_format; |
| 134 | } xfs_efi_log_item_t; |
| 135 | |
| 136 | /* |
| 137 | * This is the "extent free done" log item. It is used to log |
| 138 | * the fact that some extents earlier mentioned in an efi item |
| 139 | * have been freed. |
| 140 | */ |
| 141 | typedef struct xfs_efd_log_item { |
| 142 | xfs_log_item_t efd_item; |
| 143 | xfs_efi_log_item_t *efd_efip; |
| 144 | uint efd_next_extent; |
| 145 | xfs_efd_log_format_t efd_format; |
| 146 | } xfs_efd_log_item_t; |
| 147 | |
| 148 | /* |
| 149 | * Max number of extents in fast allocation path. |
| 150 | */ |
| 151 | #define XFS_EFD_MAX_FAST_EXTENTS 16 |
| 152 | |
| 153 | extern struct kmem_zone *xfs_efi_zone; |
| 154 | extern struct kmem_zone *xfs_efd_zone; |
| 155 | |
| 156 | xfs_efi_log_item_t *xfs_efi_init(struct xfs_mount *, uint); |
| 157 | xfs_efd_log_item_t *xfs_efd_init(struct xfs_mount *, xfs_efi_log_item_t *, |
| 158 | uint); |
Tim Shimmin | 6d192a9 | 2006-06-09 14:55:38 +1000 | [diff] [blame] | 159 | int xfs_efi_copy_format(xfs_log_iovec_t *buf, |
| 160 | xfs_efi_log_format_t *dst_efi_fmt); |
Christoph Hellwig | 7d795ca | 2005-06-21 15:41:19 +1000 | [diff] [blame] | 161 | void xfs_efi_item_free(xfs_efi_log_item_t *); |
| 162 | |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 163 | #endif /* __KERNEL__ */ |
| 164 | |
| 165 | #endif /* __XFS_EXTFREE_ITEM_H__ */ |