2 * Copyright (c) 2016 Netflix, Inc.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 #include <sys/cdefs.h>
27 __FBSDID("$FreeBSD$");
29 #include <sys/param.h>
30 #include <sys/ioccom.h>
42 #include "nvmecontrol.h"
44 _Static_assert(sizeof(struct nvme_power_state) == 256 / NBBY,
45 "nvme_power_state size wrong");
47 #define POWER_NONE 0xffffffffu
49 static struct options {
62 power_list_one(int i, struct nvme_power_state *nps)
64 int mpower, apower, ipower;
65 uint8_t mps, nops, aps, apw;
67 mps = (nps->mps_nops >> NVME_PWR_ST_MPS_SHIFT) &
69 nops = (nps->mps_nops >> NVME_PWR_ST_NOPS_SHIFT) &
70 NVME_PWR_ST_NOPS_MASK;
71 apw = (nps->apw_aps >> NVME_PWR_ST_APW_SHIFT) &
73 aps = (nps->apw_aps >> NVME_PWR_ST_APS_SHIFT) &
85 printf("%2d: %2d.%04dW%c %3d.%03dms %3d.%03dms %2d %2d %2d %2d %2d.%04dW %2d.%04dW %d\n",
86 i, mpower / 10000, mpower % 10000,
87 nops ? '*' : ' ', nps->enlat / 1000, nps->enlat % 1000,
88 nps->exlat / 1000, nps->exlat % 1000, nps->rrt, nps->rrl,
89 nps->rwt, nps->rwl, ipower / 10000, ipower % 10000,
90 apower / 10000, apower % 10000, apw);
94 power_list(struct nvme_controller_data *cdata)
98 printf("\nPower States Supported: %d\n\n", cdata->npss + 1);
99 printf(" # Max pwr Enter Lat Exit Lat RT RL WT WL Idle Pwr Act Pwr Workloadd\n");
100 printf("-- -------- --------- --------- -- -- -- -- -------- -------- --\n");
101 for (i = 0; i <= cdata->npss; i++)
102 power_list_one(i, &cdata->power_state[i]);
106 power_set(int fd, int power_val, int workload, int perm)
108 struct nvme_pt_command pt;
111 p = perm ? (1u << 31) : 0;
112 memset(&pt, 0, sizeof(pt));
113 pt.cmd.opc = NVME_OPC_SET_FEATURES;
114 pt.cmd.cdw10 = htole32(NVME_FEAT_POWER_MANAGEMENT | p);
115 pt.cmd.cdw11 = htole32(power_val | (workload << 5));
117 if (ioctl(fd, NVME_PASSTHROUGH_CMD, &pt) < 0)
118 err(1, "set feature power mgmt request failed");
120 if (nvme_completion_is_error(&pt.cpl))
121 errx(1, "set feature power mgmt request returned error");
127 struct nvme_pt_command pt;
129 memset(&pt, 0, sizeof(pt));
130 pt.cmd.opc = NVME_OPC_GET_FEATURES;
131 pt.cmd.cdw10 = htole32(NVME_FEAT_POWER_MANAGEMENT);
133 if (ioctl(fd, NVME_PASSTHROUGH_CMD, &pt) < 0)
134 err(1, "set feature power mgmt request failed");
136 if (nvme_completion_is_error(&pt.cpl))
137 errx(1, "set feature power mgmt request returned error");
139 printf("Current Power Mode is %d\n", pt.cpl.cdw0);
143 power(const struct cmd *f, int argc, char *argv[])
145 struct nvme_controller_data cdata;
148 arg_parse(argc, argv, f);
150 if (opt.list && opt.power != POWER_NONE) {
151 fprintf(stderr, "Can't set power and list power states\n");
152 arg_help(argc, argv, f);
155 open_dev(opt.dev, &fd, 1, 1);
158 read_controller_data(fd, &cdata);
163 if (opt.power != POWER_NONE) {
164 power_set(fd, opt.power, opt.workload, 0);
174 static const struct opts power_opts[] = {
175 #define OPT(l, s, t, opt, addr, desc) { l, s, t, &opt.addr, desc }
176 OPT("list", 'l', arg_none, opt, list,
177 "List the valid power states"),
178 OPT("power", 'p', arg_uint32, opt, power,
179 "Set the power state"),
180 OPT("workload", 'w', arg_uint32, opt, workload,
182 { NULL, 0, arg_none, NULL, NULL }
186 static const struct args power_args[] = {
187 { arg_string, &opt.dev, "controller-id" },
188 { arg_none, NULL, NULL },
191 static struct cmd power_cmd = {
194 .descr = "Manage power states for the drive",
195 .ctx_size = sizeof(opt),
200 CMD_COMMAND(power_cmd);