xref: /spdk/lib/ftl/mngt/ftl_mngt_startup.c (revision 18c8b52afa69f39481ebb75711b2f30b11693f9d)
1 /*   SPDX-License-Identifier: BSD-3-Clause
2  *   Copyright (c) Intel Corporation.
3  *   All rights reserved.
4  */
5 
6 #include "ftl_core.h"
7 #include "ftl_mngt.h"
8 #include "ftl_mngt_steps.h"
9 
10 static const struct ftl_mngt_process_desc desc_startup;
11 static const struct ftl_mngt_process_desc desc_first_start;
12 static const struct ftl_mngt_process_desc desc_restore;
13 static const struct ftl_mngt_process_desc desc_clean_start;
14 
15 static void
16 ftl_mngt_select_startup_mode(struct spdk_ftl_dev *dev,
17 			     struct ftl_mngt_process *mngt)
18 {
19 	if (dev->conf.mode & SPDK_FTL_MODE_CREATE) {
20 		ftl_mngt_call_process(mngt, &desc_first_start);
21 	} else {
22 		ftl_mngt_call_process(mngt, &desc_restore);
23 	}
24 }
25 
26 static void
27 ftl_mngt_select_restore_mode(struct spdk_ftl_dev *dev,
28 			     struct ftl_mngt_process *mngt)
29 {
30 	if (dev->sb->clean) {
31 		ftl_mngt_call_process(mngt, &desc_clean_start);
32 	} else {
33 		ftl_mngt_recover(dev, mngt);
34 	}
35 }
36 
37 /*
38  * Common startup steps required by FTL in all cases (creation, load, dirty shutdown recovery).
39  * Includes actions like opening the devices, calculating the expected size and version of metadata, etc.
40  */
41 static const struct ftl_mngt_process_desc desc_startup = {
42 	.name = "FTL startup",
43 	.steps = {
44 		{
45 			.name = "Check configuration",
46 			.action = ftl_mngt_check_conf,
47 		},
48 		{
49 			.name = "Open base bdev",
50 			.action = ftl_mngt_open_base_bdev,
51 			.cleanup = ftl_mngt_close_base_bdev
52 		},
53 		{
54 			.name = "Open cache bdev",
55 			.action = ftl_mngt_open_cache_bdev,
56 			.cleanup = ftl_mngt_close_cache_bdev
57 		},
58 #ifdef SPDK_FTL_VSS_EMU
59 		{
60 			.name = "Initialize VSS emu",
61 			.action = ftl_mngt_md_init_vss_emu,
62 			.cleanup = ftl_mngt_md_deinit_vss_emu
63 		},
64 #endif
65 		{
66 			.name = "Initialize superblock",
67 			.action = ftl_mngt_superblock_init,
68 			.cleanup = ftl_mngt_superblock_deinit
69 		},
70 		{
71 			.name = "Initialize memory pools",
72 			.action = ftl_mngt_init_mem_pools,
73 			.cleanup = ftl_mngt_deinit_mem_pools
74 		},
75 		{
76 			.name = "Initialize bands",
77 			.action = ftl_mngt_init_bands,
78 			.cleanup = ftl_mngt_deinit_bands
79 		},
80 		{
81 			.name = "Register IO device",
82 			.action = ftl_mngt_register_io_device,
83 			.cleanup = ftl_mngt_unregister_io_device
84 		},
85 		{
86 			.name = "Initialize core IO channel",
87 			.action = ftl_mngt_init_io_channel,
88 			.cleanup = ftl_mngt_deinit_io_channel
89 		},
90 		{
91 			.name = "Decorate bands",
92 			.action = ftl_mngt_decorate_bands
93 		},
94 		{
95 			.name = "Initialize layout",
96 			.action = ftl_mngt_init_layout
97 		},
98 		{
99 			.name = "Initialize metadata",
100 			.action = ftl_mngt_init_md,
101 			.cleanup = ftl_mngt_deinit_md
102 		},
103 		{
104 			.name = "Initialize NV cache",
105 			.action = ftl_mngt_init_nv_cache,
106 			.cleanup = ftl_mngt_deinit_nv_cache
107 		},
108 		{
109 			.name = "Initialize valid map",
110 			.action = ftl_mngt_init_vld_map,
111 			.cleanup = ftl_mngt_deinit_vld_map
112 		},
113 		{
114 			.name = "Initialize trim map",
115 			.action = ftl_mngt_init_unmap_map,
116 			.cleanup = ftl_mngt_deinit_unmap_map
117 		},
118 		{
119 			.name = "Initialize bands metadata",
120 			.action = ftl_mngt_init_bands_md,
121 			.cleanup = ftl_mngt_deinit_bands_md
122 		},
123 		{
124 			.name = "Initialize reloc",
125 			.action = ftl_mngt_init_reloc,
126 			.cleanup = ftl_mngt_deinit_reloc
127 		},
128 		{
129 			.name = "Select startup mode",
130 			.action = ftl_mngt_select_startup_mode
131 		},
132 		{}
133 	}
134 };
135 
136 /*
137  * Steps executed when creating FTL for the first time - most important being scrubbing
138  * old data/metadata (so it's not leaked during dirty shutdown recovery) and laying out
139  * regions for the new metadata (initializing band states, etc).
140  */
141 static const struct ftl_mngt_process_desc desc_first_start = {
142 	.name = "FTL first start",
143 	.steps = {
144 		{
145 			.name = "Initialize L2P",
146 			.action = ftl_mngt_init_l2p,
147 			.cleanup = ftl_mngt_deinit_l2p
148 		},
149 		{
150 			.name = "Clear L2P",
151 			.action = ftl_mngt_clear_l2p,
152 			.cleanup = ftl_mngt_clear_l2p
153 		},
154 		{
155 			.name = "Scrub NV cache",
156 			.action = ftl_mngt_scrub_nv_cache,
157 		},
158 		{
159 			.name = "Finalize band initialization",
160 			.action = ftl_mngt_finalize_init_bands,
161 		},
162 		{
163 			.name = "Save initial band info metadata",
164 			.action = ftl_mngt_persist_band_info_metadata,
165 		},
166 		{
167 			.name = "Save initial chunk info metadata",
168 			.action = ftl_mngt_persist_nv_cache_metadata,
169 		},
170 		{
171 			.name = "Initialize P2L checkpointing",
172 			.action = ftl_mngt_p2l_init_ckpt,
173 			.cleanup = ftl_mngt_p2l_deinit_ckpt
174 		},
175 		{
176 			.name = "Wipe P2L region",
177 			.action = ftl_mngt_p2l_wipe,
178 		},
179 		{
180 			.name = "Clear trim map",
181 			.action = ftl_mngt_unmap_clear,
182 		},
183 		{
184 			.name = "Free P2L region bufs",
185 			.action = ftl_mngt_p2l_free_bufs,
186 		},
187 		{
188 			.name = "Set FTL dirty state",
189 			.action = ftl_mngt_set_dirty,
190 		},
191 		{
192 			.name = "Start core poller",
193 			.action = ftl_mngt_start_core_poller,
194 			.cleanup = ftl_mngt_stop_core_poller
195 		},
196 		{
197 			.name = "Finalize initialization",
198 			.action = ftl_mngt_finalize_startup,
199 		},
200 		{}
201 	}
202 };
203 
204 /*
205  * Step utilized on loading of an FTL instance - decides on dirty/clean shutdown path.
206  */
207 static const struct ftl_mngt_process_desc desc_restore = {
208 	.name = "FTL restore",
209 	.steps = {
210 		{
211 			.name = "Select recovery mode",
212 			.action = ftl_mngt_select_restore_mode,
213 		},
214 		{}
215 	}
216 };
217 
218 /*
219  * Loading of FTL after clean shutdown.
220  */
221 static const struct ftl_mngt_process_desc desc_clean_start = {
222 	.name = "Clean startup",
223 	.steps = {
224 		{
225 			.name = "Restore metadata",
226 			.action = ftl_mngt_restore_md
227 		},
228 		{
229 			.name = "Initialize P2L checkpointing",
230 			.action = ftl_mngt_p2l_init_ckpt,
231 			.cleanup = ftl_mngt_p2l_deinit_ckpt
232 		},
233 		{
234 			.name = "Restore P2L checkpoints",
235 			.action = ftl_mngt_p2l_restore_ckpt
236 		},
237 		{
238 			.name = "Initialize L2P",
239 			.action = ftl_mngt_init_l2p,
240 			.cleanup = ftl_mngt_deinit_l2p
241 		},
242 		{
243 			.name = "Restore L2P",
244 			.action = ftl_mngt_restore_l2p,
245 		},
246 		{
247 			.name = "Finalize band initialization",
248 			.action = ftl_mngt_finalize_init_bands,
249 		},
250 		{
251 			.name = "Free P2L region bufs",
252 			.action = ftl_mngt_p2l_free_bufs,
253 		},
254 		{
255 			.name = "Start core poller",
256 			.action = ftl_mngt_start_core_poller,
257 			.cleanup = ftl_mngt_stop_core_poller
258 		},
259 		{
260 			.name = "Self test on startup",
261 			.action = ftl_mngt_self_test,
262 		},
263 		{
264 			.name = "Set FTL dirty state",
265 			.action = ftl_mngt_set_dirty,
266 		},
267 		{
268 			.name = "Finalize initialization",
269 			.action = ftl_mngt_finalize_startup,
270 		},
271 		{}
272 	}
273 };
274 
275 int
276 ftl_mngt_call_dev_startup(struct spdk_ftl_dev *dev, ftl_mngt_completion cb, void *cb_cntx)
277 {
278 	return ftl_mngt_process_execute(dev, &desc_startup, cb, cb_cntx);
279 }
280 
281 struct ftl_unmap_ctx {
282 	uint64_t lba;
283 	uint64_t num_blocks;
284 	spdk_ftl_fn cb_fn;
285 	void *cb_arg;
286 };
287 
288 static void
289 ftl_mngt_process_unmap_cb(void *ctx, int status)
290 {
291 	struct ftl_mngt_process *mngt = ctx;
292 
293 	if (status) {
294 		ftl_mngt_fail_step(ctx);
295 	} else {
296 		ftl_mngt_next_step(mngt);
297 	}
298 }
299 
300 static void
301 ftl_mngt_process_unmap(struct spdk_ftl_dev *dev, struct ftl_mngt_process *mngt)
302 {
303 	struct ftl_io *io = ftl_mngt_get_process_ctx(mngt);
304 	struct ftl_unmap_ctx *ctx = ftl_mngt_get_caller_ctx(mngt);
305 	int rc;
306 
307 	if (!dev->ioch) {
308 		ftl_mngt_fail_step(mngt);
309 		return;
310 	}
311 
312 	rc = spdk_ftl_unmap(dev, io, dev->ioch, ctx->lba, ctx->num_blocks, ftl_mngt_process_unmap_cb, mngt);
313 	if (rc == -EAGAIN) {
314 		ftl_mngt_continue_step(mngt);
315 	}
316 }
317 
318 /*
319  * RPC unmap path.
320  */
321 static const struct ftl_mngt_process_desc g_desc_unmap = {
322 	.name = "FTL unmap",
323 	.ctx_size = sizeof(struct ftl_io),
324 	.steps = {
325 		{
326 			.name = "Process unmap",
327 			.action = ftl_mngt_process_unmap,
328 		},
329 		{}
330 	}
331 };
332 
333 static void
334 ftl_mngt_unmap_cb(struct spdk_ftl_dev *dev, void *_ctx, int status)
335 {
336 	struct ftl_unmap_ctx *ctx = _ctx;
337 
338 	ctx->cb_fn(ctx->cb_arg, status);
339 
340 	free(ctx);
341 }
342 
343 int
344 ftl_mngt_unmap(struct spdk_ftl_dev *dev, uint64_t lba, uint64_t num_blocks, spdk_ftl_fn cb,
345 	       void *cb_cntx)
346 {
347 	struct ftl_unmap_ctx *ctx;
348 
349 	ctx = calloc(1, sizeof(*ctx));
350 	if (ctx == NULL) {
351 		return -EAGAIN;
352 	}
353 
354 	ctx->lba = lba;
355 	ctx->num_blocks = num_blocks;
356 	ctx->cb_fn = cb;
357 	ctx->cb_arg = cb_cntx;
358 
359 	return ftl_mngt_process_execute(dev, &g_desc_unmap, ftl_mngt_unmap_cb, ctx);
360 }
361 
362 void
363 ftl_mngt_rollback_device(struct spdk_ftl_dev *dev, struct ftl_mngt_process *mngt)
364 {
365 	ftl_mngt_call_process_rollback(mngt, &desc_startup);
366 }
367