diff options
Diffstat (limited to 'drivers/md/dm-exception-store.h')
-rw-r--r-- | drivers/md/dm-exception-store.h | 62 |
1 files changed, 48 insertions, 14 deletions
diff --git a/drivers/md/dm-exception-store.h b/drivers/md/dm-exception-store.h index 8a223a48802c..e8dfa06af3ba 100644 --- a/drivers/md/dm-exception-store.h +++ b/drivers/md/dm-exception-store.h | |||
@@ -26,7 +26,7 @@ typedef sector_t chunk_t; | |||
26 | * of chunks that follow contiguously. Remaining bits hold the number of the | 26 | * of chunks that follow contiguously. Remaining bits hold the number of the |
27 | * chunk within the device. | 27 | * chunk within the device. |
28 | */ | 28 | */ |
29 | struct dm_snap_exception { | 29 | struct dm_exception { |
30 | struct list_head hash_list; | 30 | struct list_head hash_list; |
31 | 31 | ||
32 | chunk_t old_chunk; | 32 | chunk_t old_chunk; |
@@ -64,17 +64,34 @@ struct dm_exception_store_type { | |||
64 | * Find somewhere to store the next exception. | 64 | * Find somewhere to store the next exception. |
65 | */ | 65 | */ |
66 | int (*prepare_exception) (struct dm_exception_store *store, | 66 | int (*prepare_exception) (struct dm_exception_store *store, |
67 | struct dm_snap_exception *e); | 67 | struct dm_exception *e); |
68 | 68 | ||
69 | /* | 69 | /* |
70 | * Update the metadata with this exception. | 70 | * Update the metadata with this exception. |
71 | */ | 71 | */ |
72 | void (*commit_exception) (struct dm_exception_store *store, | 72 | void (*commit_exception) (struct dm_exception_store *store, |
73 | struct dm_snap_exception *e, | 73 | struct dm_exception *e, |
74 | void (*callback) (void *, int success), | 74 | void (*callback) (void *, int success), |
75 | void *callback_context); | 75 | void *callback_context); |
76 | 76 | ||
77 | /* | 77 | /* |
78 | * Returns 0 if the exception store is empty. | ||
79 | * | ||
80 | * If there are exceptions still to be merged, sets | ||
81 | * *last_old_chunk and *last_new_chunk to the most recent | ||
82 | * still-to-be-merged chunk and returns the number of | ||
83 | * consecutive previous ones. | ||
84 | */ | ||
85 | int (*prepare_merge) (struct dm_exception_store *store, | ||
86 | chunk_t *last_old_chunk, chunk_t *last_new_chunk); | ||
87 | |||
88 | /* | ||
89 | * Clear the last n exceptions. | ||
90 | * nr_merged must be <= the value returned by prepare_merge. | ||
91 | */ | ||
92 | int (*commit_merge) (struct dm_exception_store *store, int nr_merged); | ||
93 | |||
94 | /* | ||
78 | * The snapshot is invalid, note this in the metadata. | 95 | * The snapshot is invalid, note this in the metadata. |
79 | */ | 96 | */ |
80 | void (*drop_snapshot) (struct dm_exception_store *store); | 97 | void (*drop_snapshot) (struct dm_exception_store *store); |
@@ -86,19 +103,19 @@ struct dm_exception_store_type { | |||
86 | /* | 103 | /* |
87 | * Return how full the snapshot is. | 104 | * Return how full the snapshot is. |
88 | */ | 105 | */ |
89 | void (*fraction_full) (struct dm_exception_store *store, | 106 | void (*usage) (struct dm_exception_store *store, |
90 | sector_t *numerator, | 107 | sector_t *total_sectors, sector_t *sectors_allocated, |
91 | sector_t *denominator); | 108 | sector_t *metadata_sectors); |
92 | 109 | ||
93 | /* For internal device-mapper use only. */ | 110 | /* For internal device-mapper use only. */ |
94 | struct list_head list; | 111 | struct list_head list; |
95 | }; | 112 | }; |
96 | 113 | ||
114 | struct dm_snapshot; | ||
115 | |||
97 | struct dm_exception_store { | 116 | struct dm_exception_store { |
98 | struct dm_exception_store_type *type; | 117 | struct dm_exception_store_type *type; |
99 | struct dm_target *ti; | 118 | struct dm_snapshot *snap; |
100 | |||
101 | struct dm_dev *cow; | ||
102 | 119 | ||
103 | /* Size of data blocks saved - must be a power of 2 */ | 120 | /* Size of data blocks saved - must be a power of 2 */ |
104 | unsigned chunk_size; | 121 | unsigned chunk_size; |
@@ -109,6 +126,11 @@ struct dm_exception_store { | |||
109 | }; | 126 | }; |
110 | 127 | ||
111 | /* | 128 | /* |
129 | * Obtain the cow device used by a given snapshot. | ||
130 | */ | ||
131 | struct dm_dev *dm_snap_cow(struct dm_snapshot *snap); | ||
132 | |||
133 | /* | ||
112 | * Funtions to manipulate consecutive chunks | 134 | * Funtions to manipulate consecutive chunks |
113 | */ | 135 | */ |
114 | # if defined(CONFIG_LBDAF) || (BITS_PER_LONG == 64) | 136 | # if defined(CONFIG_LBDAF) || (BITS_PER_LONG == 64) |
@@ -120,18 +142,25 @@ static inline chunk_t dm_chunk_number(chunk_t chunk) | |||
120 | return chunk & (chunk_t)((1ULL << DM_CHUNK_NUMBER_BITS) - 1ULL); | 142 | return chunk & (chunk_t)((1ULL << DM_CHUNK_NUMBER_BITS) - 1ULL); |
121 | } | 143 | } |
122 | 144 | ||
123 | static inline unsigned dm_consecutive_chunk_count(struct dm_snap_exception *e) | 145 | static inline unsigned dm_consecutive_chunk_count(struct dm_exception *e) |
124 | { | 146 | { |
125 | return e->new_chunk >> DM_CHUNK_NUMBER_BITS; | 147 | return e->new_chunk >> DM_CHUNK_NUMBER_BITS; |
126 | } | 148 | } |
127 | 149 | ||
128 | static inline void dm_consecutive_chunk_count_inc(struct dm_snap_exception *e) | 150 | static inline void dm_consecutive_chunk_count_inc(struct dm_exception *e) |
129 | { | 151 | { |
130 | e->new_chunk += (1ULL << DM_CHUNK_NUMBER_BITS); | 152 | e->new_chunk += (1ULL << DM_CHUNK_NUMBER_BITS); |
131 | 153 | ||
132 | BUG_ON(!dm_consecutive_chunk_count(e)); | 154 | BUG_ON(!dm_consecutive_chunk_count(e)); |
133 | } | 155 | } |
134 | 156 | ||
157 | static inline void dm_consecutive_chunk_count_dec(struct dm_exception *e) | ||
158 | { | ||
159 | BUG_ON(!dm_consecutive_chunk_count(e)); | ||
160 | |||
161 | e->new_chunk -= (1ULL << DM_CHUNK_NUMBER_BITS); | ||
162 | } | ||
163 | |||
135 | # else | 164 | # else |
136 | # define DM_CHUNK_CONSECUTIVE_BITS 0 | 165 | # define DM_CHUNK_CONSECUTIVE_BITS 0 |
137 | 166 | ||
@@ -140,12 +169,16 @@ static inline chunk_t dm_chunk_number(chunk_t chunk) | |||
140 | return chunk; | 169 | return chunk; |
141 | } | 170 | } |
142 | 171 | ||
143 | static inline unsigned dm_consecutive_chunk_count(struct dm_snap_exception *e) | 172 | static inline unsigned dm_consecutive_chunk_count(struct dm_exception *e) |
144 | { | 173 | { |
145 | return 0; | 174 | return 0; |
146 | } | 175 | } |
147 | 176 | ||
148 | static inline void dm_consecutive_chunk_count_inc(struct dm_snap_exception *e) | 177 | static inline void dm_consecutive_chunk_count_inc(struct dm_exception *e) |
178 | { | ||
179 | } | ||
180 | |||
181 | static inline void dm_consecutive_chunk_count_dec(struct dm_exception *e) | ||
149 | { | 182 | { |
150 | } | 183 | } |
151 | 184 | ||
@@ -162,7 +195,7 @@ static inline sector_t get_dev_size(struct block_device *bdev) | |||
162 | static inline chunk_t sector_to_chunk(struct dm_exception_store *store, | 195 | static inline chunk_t sector_to_chunk(struct dm_exception_store *store, |
163 | sector_t sector) | 196 | sector_t sector) |
164 | { | 197 | { |
165 | return (sector & ~store->chunk_mask) >> store->chunk_shift; | 198 | return sector >> store->chunk_shift; |
166 | } | 199 | } |
167 | 200 | ||
168 | int dm_exception_store_type_register(struct dm_exception_store_type *type); | 201 | int dm_exception_store_type_register(struct dm_exception_store_type *type); |
@@ -173,6 +206,7 @@ int dm_exception_store_set_chunk_size(struct dm_exception_store *store, | |||
173 | char **error); | 206 | char **error); |
174 | 207 | ||
175 | int dm_exception_store_create(struct dm_target *ti, int argc, char **argv, | 208 | int dm_exception_store_create(struct dm_target *ti, int argc, char **argv, |
209 | struct dm_snapshot *snap, | ||
176 | unsigned *args_used, | 210 | unsigned *args_used, |
177 | struct dm_exception_store **store); | 211 | struct dm_exception_store **store); |
178 | void dm_exception_store_destroy(struct dm_exception_store *store); | 212 | void dm_exception_store_destroy(struct dm_exception_store *store); |