1*f5f0964cSRobert Mustacchi /*
2*f5f0964cSRobert Mustacchi * This file and its contents are supplied under the terms of the
3*f5f0964cSRobert Mustacchi * Common Development and Distribution License ("CDDL"), version 1.0.
4*f5f0964cSRobert Mustacchi * You may only use this file in accordance with the terms of version
5*f5f0964cSRobert Mustacchi * 1.0 of the CDDL.
6*f5f0964cSRobert Mustacchi *
7*f5f0964cSRobert Mustacchi * A full copy of the text of the CDDL should have accompanied this
8*f5f0964cSRobert Mustacchi * source. A copy of the CDDL is also available via the Internet at
9*f5f0964cSRobert Mustacchi * http://www.illumos.org/license/CDDL.
10*f5f0964cSRobert Mustacchi */
11*f5f0964cSRobert Mustacchi
12*f5f0964cSRobert Mustacchi /*
13*f5f0964cSRobert Mustacchi * Copyright 2025 Oxide Computer Company
14*f5f0964cSRobert Mustacchi */
15*f5f0964cSRobert Mustacchi
16*f5f0964cSRobert Mustacchi /*
17*f5f0964cSRobert Mustacchi * Reset an NVMe device back to a simple state.
18*f5f0964cSRobert Mustacchi */
19*f5f0964cSRobert Mustacchi
20*f5f0964cSRobert Mustacchi #include <err.h>
21*f5f0964cSRobert Mustacchi #include <stdlib.h>
22*f5f0964cSRobert Mustacchi #include <string.h>
23*f5f0964cSRobert Mustacchi #include "libnvme_test_common.h"
24*f5f0964cSRobert Mustacchi
25*f5f0964cSRobert Mustacchi typedef enum {
26*f5f0964cSRobert Mustacchi RESET_ACT_EMPTY = 0,
27*f5f0964cSRobert Mustacchi RESET_ACT_DEFAULT
28*f5f0964cSRobert Mustacchi } reset_action_t;
29*f5f0964cSRobert Mustacchi
30*f5f0964cSRobert Mustacchi static bool
device_reset_cb(nvme_ctrl_t * ctrl,const nvme_ns_disc_t * disc,void * arg)31*f5f0964cSRobert Mustacchi device_reset_cb(nvme_ctrl_t *ctrl, const nvme_ns_disc_t *disc,
32*f5f0964cSRobert Mustacchi void *arg)
33*f5f0964cSRobert Mustacchi {
34*f5f0964cSRobert Mustacchi const uint32_t nsid = nvme_ns_disc_nsid(disc);
35*f5f0964cSRobert Mustacchi
36*f5f0964cSRobert Mustacchi if (!libnvme_test_setup_ns(ctrl, NVME_NS_DISC_F_ALL, nsid,
37*f5f0964cSRobert Mustacchi UINT32_MAX)) {
38*f5f0964cSRobert Mustacchi exit(EXIT_FAILURE);
39*f5f0964cSRobert Mustacchi }
40*f5f0964cSRobert Mustacchi
41*f5f0964cSRobert Mustacchi return (true);
42*f5f0964cSRobert Mustacchi }
43*f5f0964cSRobert Mustacchi
44*f5f0964cSRobert Mustacchi /*
45*f5f0964cSRobert Mustacchi * Create a single namespace with all of the device capacity. We default to the
46*f5f0964cSRobert Mustacchi * best 4 KiB namespace format we can that has no metadata. If that doesn't
47*f5f0964cSRobert Mustacchi * exist, use the remaining best that we can support.
48*f5f0964cSRobert Mustacchi */
49*f5f0964cSRobert Mustacchi static void
device_reset_create(nvme_ctrl_t * ctrl)50*f5f0964cSRobert Mustacchi device_reset_create(nvme_ctrl_t *ctrl)
51*f5f0964cSRobert Mustacchi {
52*f5f0964cSRobert Mustacchi nvme_ctrl_info_t *info;
53*f5f0964cSRobert Mustacchi uint32_t lba, nsid;
54*f5f0964cSRobert Mustacchi uint64_t size;
55*f5f0964cSRobert Mustacchi nvme_uint128_t cap;
56*f5f0964cSRobert Mustacchi
57*f5f0964cSRobert Mustacchi if (!nvme_ctrl_info_snap(ctrl, &info)) {
58*f5f0964cSRobert Mustacchi libnvme_test_ctrl_fatal(ctrl, "failed to get info snapshot");
59*f5f0964cSRobert Mustacchi }
60*f5f0964cSRobert Mustacchi
61*f5f0964cSRobert Mustacchi if (!libnvme_test_lbaf(info, NVME_TEST_LBA_SIZE, &lba)) {
62*f5f0964cSRobert Mustacchi errx(EXIT_FAILURE, "failed to find 4K LBA format, cannot "
63*f5f0964cSRobert Mustacchi "continue");
64*f5f0964cSRobert Mustacchi }
65*f5f0964cSRobert Mustacchi
66*f5f0964cSRobert Mustacchi if (!nvme_ctrl_info_cap(info, &cap)) {
67*f5f0964cSRobert Mustacchi libnvme_test_ctrl_info_fatal(info, "failed to get device "
68*f5f0964cSRobert Mustacchi "capacity");
69*f5f0964cSRobert Mustacchi }
70*f5f0964cSRobert Mustacchi
71*f5f0964cSRobert Mustacchi /*
72*f5f0964cSRobert Mustacchi * We need to convert the capacity of the device to a number of logical
73*f5f0964cSRobert Mustacchi * blocks. The device's capacity is phrased in bytes. For now, just
74*f5f0964cSRobert Mustacchi * divide the capacity in bytes by the LBA size. If we encounter a
75*f5f0964cSRobert Mustacchi * device with more than a uint64_t worth of bytes in it, for now punt.
76*f5f0964cSRobert Mustacchi */
77*f5f0964cSRobert Mustacchi if (cap.hi != 0) {
78*f5f0964cSRobert Mustacchi errx(EXIT_FAILURE, "encountered device with > uint64_t "
79*f5f0964cSRobert Mustacchi "capacity, this program needs to be updated to deal with "
80*f5f0964cSRobert Mustacchi "that");
81*f5f0964cSRobert Mustacchi }
82*f5f0964cSRobert Mustacchi size = cap.lo / NVME_TEST_LBA_SIZE;
83*f5f0964cSRobert Mustacchi
84*f5f0964cSRobert Mustacchi if (!libnvme_test_ns_create(ctrl, size, lba, &nsid, NULL)) {
85*f5f0964cSRobert Mustacchi exit(EXIT_FAILURE);
86*f5f0964cSRobert Mustacchi }
87*f5f0964cSRobert Mustacchi
88*f5f0964cSRobert Mustacchi if (!libnvme_test_setup_ns(ctrl, NVME_NS_DISC_F_ACTIVE, nsid,
89*f5f0964cSRobert Mustacchi UINT32_MAX)) {
90*f5f0964cSRobert Mustacchi exit(EXIT_FAILURE);
91*f5f0964cSRobert Mustacchi }
92*f5f0964cSRobert Mustacchi
93*f5f0964cSRobert Mustacchi nvme_ctrl_info_free(info);
94*f5f0964cSRobert Mustacchi }
95*f5f0964cSRobert Mustacchi
96*f5f0964cSRobert Mustacchi int
main(void)97*f5f0964cSRobert Mustacchi main(void)
98*f5f0964cSRobert Mustacchi {
99*f5f0964cSRobert Mustacchi nvme_t *nvme;
100*f5f0964cSRobert Mustacchi nvme_ctrl_t *ctrl;
101*f5f0964cSRobert Mustacchi
102*f5f0964cSRobert Mustacchi libnvme_test_init(&nvme, &ctrl);
103*f5f0964cSRobert Mustacchi if (!nvme_ctrl_lock(ctrl, NVME_LOCK_L_WRITE, NVME_LOCK_F_DONT_BLOCK)) {
104*f5f0964cSRobert Mustacchi libnvme_test_ctrl_fatal(ctrl, "failed to obtain write lock");
105*f5f0964cSRobert Mustacchi }
106*f5f0964cSRobert Mustacchi
107*f5f0964cSRobert Mustacchi if (!nvme_ns_discover(ctrl, NVME_NS_DISC_F_ALL,
108*f5f0964cSRobert Mustacchi device_reset_cb, NULL)) {
109*f5f0964cSRobert Mustacchi libnvme_test_ctrl_fatal(ctrl, "failed to iterate namespaces");
110*f5f0964cSRobert Mustacchi }
111*f5f0964cSRobert Mustacchi
112*f5f0964cSRobert Mustacchi device_reset_create(ctrl);
113*f5f0964cSRobert Mustacchi
114*f5f0964cSRobert Mustacchi nvme_ctrl_unlock(ctrl);
115*f5f0964cSRobert Mustacchi nvme_ctrl_fini(ctrl);
116*f5f0964cSRobert Mustacchi nvme_fini(nvme);
117*f5f0964cSRobert Mustacchi
118*f5f0964cSRobert Mustacchi return (EXIT_SUCCESS);
119*f5f0964cSRobert Mustacchi }
120