40152db5d3
Most people know SAS attached SATA devices by the name SAT or SATL (with the latter being a little more common). Change the device type ATA_BEHIND_SCSI to SATL since it's more specific and meaningful. Suggested by: scottl@
858 lines
22 KiB
C
858 lines
22 KiB
C
/*-
|
|
* Copyright (c) 2016 Spectra Logic Corporation
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions, and the following disclaimer,
|
|
* without modification.
|
|
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
|
|
* substantially similar to the "NO WARRANTY" disclaimer below
|
|
* ("Disclaimer") and any redistribution must be conditioned upon
|
|
* including a substantially similar Disclaimer requirement for further
|
|
* binary redistribution.
|
|
*
|
|
* NO WARRANTY
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
|
|
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGES.
|
|
*
|
|
* Authors: Ken Merry (Spectra Logic Corporation)
|
|
*/
|
|
/*
|
|
* ATA Extended Power Conditions (EPC) support
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
#include <sys/ioctl.h>
|
|
#include <sys/stdint.h>
|
|
#include <sys/types.h>
|
|
#include <sys/endian.h>
|
|
#include <sys/sbuf.h>
|
|
#include <sys/queue.h>
|
|
#include <sys/ata.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <inttypes.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <strings.h>
|
|
#include <fcntl.h>
|
|
#include <ctype.h>
|
|
#include <limits.h>
|
|
#include <err.h>
|
|
#include <locale.h>
|
|
|
|
#include <cam/cam.h>
|
|
#include <cam/cam_debug.h>
|
|
#include <cam/cam_ccb.h>
|
|
#include <cam/scsi/scsi_all.h>
|
|
#include <cam/scsi/scsi_da.h>
|
|
#include <cam/scsi/scsi_pass.h>
|
|
#include <cam/scsi/scsi_message.h>
|
|
#include <camlib.h>
|
|
#include "camcontrol.h"
|
|
|
|
typedef enum {
|
|
EPC_ACTION_NONE = 0x00,
|
|
EPC_ACTION_LIST = 0x01,
|
|
EPC_ACTION_TIMER_SET = 0x02,
|
|
EPC_ACTION_IMMEDIATE = 0x03,
|
|
EPC_ACTION_GETMODE = 0x04
|
|
} epc_action;
|
|
|
|
static struct scsi_nv epc_flags[] = {
|
|
{ "Supported", ATA_PCL_COND_SUPPORTED },
|
|
{ "Saveable", ATA_PCL_COND_SUPPORTED },
|
|
{ "Changeable", ATA_PCL_COND_CHANGEABLE },
|
|
{ "Default Timer Enabled", ATA_PCL_DEFAULT_TIMER_EN },
|
|
{ "Saved Timer Enabled", ATA_PCL_SAVED_TIMER_EN },
|
|
{ "Current Timer Enabled", ATA_PCL_CURRENT_TIMER_EN },
|
|
{ "Hold Power Condition Not Supported", ATA_PCL_HOLD_PC_NOT_SUP }
|
|
};
|
|
|
|
static struct scsi_nv epc_power_cond_map[] = {
|
|
{ "Standby_z", ATA_EPC_STANDBY_Z },
|
|
{ "z", ATA_EPC_STANDBY_Z },
|
|
{ "Standby_y", ATA_EPC_STANDBY_Y },
|
|
{ "y", ATA_EPC_STANDBY_Y },
|
|
{ "Idle_a", ATA_EPC_IDLE_A },
|
|
{ "a", ATA_EPC_IDLE_A },
|
|
{ "Idle_b", ATA_EPC_IDLE_B },
|
|
{ "b", ATA_EPC_IDLE_B },
|
|
{ "Idle_c", ATA_EPC_IDLE_C },
|
|
{ "c", ATA_EPC_IDLE_C }
|
|
};
|
|
|
|
static struct scsi_nv epc_rst_val[] = {
|
|
{ "default", ATA_SF_EPC_RST_DFLT },
|
|
{ "saved", 0}
|
|
};
|
|
|
|
static struct scsi_nv epc_ps_map[] = {
|
|
{ "unknown", ATA_SF_EPC_SRC_UNKNOWN },
|
|
{ "battery", ATA_SF_EPC_SRC_BAT },
|
|
{ "notbattery", ATA_SF_EPC_SRC_NOT_BAT }
|
|
};
|
|
|
|
/*
|
|
* These aren't subcommands of the EPC SET FEATURES subcommand, but rather
|
|
* commands that determine the current capabilities and status of the drive.
|
|
* The EPC subcommands are limited to 4 bits, so we won't collide with any
|
|
* future values.
|
|
*/
|
|
#define CCTL_EPC_GET_STATUS 0x8001
|
|
#define CCTL_EPC_LIST 0x8002
|
|
|
|
static struct scsi_nv epc_cmd_map[] = {
|
|
{ "restore", ATA_SF_EPC_RESTORE },
|
|
{ "goto", ATA_SF_EPC_GOTO },
|
|
{ "timer", ATA_SF_EPC_SET_TIMER },
|
|
{ "state", ATA_SF_EPC_SET_STATE },
|
|
{ "enable", ATA_SF_EPC_ENABLE },
|
|
{ "disable", ATA_SF_EPC_DISABLE },
|
|
{ "source", ATA_SF_EPC_SET_SOURCE },
|
|
{ "status", CCTL_EPC_GET_STATUS },
|
|
{ "list", CCTL_EPC_LIST }
|
|
};
|
|
|
|
static int epc_list(struct cam_device *device, camcontrol_devtype devtype,
|
|
union ccb *ccb, int retry_count, int timeout);
|
|
static void epc_print_pcl_desc(struct ata_power_cond_log_desc *desc,
|
|
const char *prefix);
|
|
static int epc_getmode(struct cam_device *device, camcontrol_devtype devtype,
|
|
union ccb *ccb, int retry_count, int timeout,
|
|
int power_only);
|
|
static int epc_set_features(struct cam_device *device,
|
|
camcontrol_devtype devtype, union ccb *ccb,
|
|
int retry_count, int timeout, int action,
|
|
int power_cond, int timer, int enable, int save,
|
|
int delayed_entry, int hold, int power_src,
|
|
int restore_src);
|
|
|
|
static void
|
|
epc_print_pcl_desc(struct ata_power_cond_log_desc *desc, const char *prefix)
|
|
{
|
|
int first;
|
|
unsigned int i, num_printed, max_chars;
|
|
|
|
first = 1;
|
|
max_chars = 75;
|
|
|
|
num_printed = printf("%sFlags: ", prefix);
|
|
for (i = 0; i < (sizeof(epc_flags) / sizeof(epc_flags[0])); i++) {
|
|
if ((desc->flags & epc_flags[i].value) == 0)
|
|
continue;
|
|
if (first == 0) {
|
|
num_printed += printf(", ");
|
|
}
|
|
if ((num_printed + strlen(epc_flags[i].name)) > max_chars) {
|
|
printf("\n");
|
|
num_printed = printf("%s ", prefix);
|
|
}
|
|
num_printed += printf("%s", epc_flags[i].name);
|
|
first = 0;
|
|
}
|
|
if (first != 0)
|
|
printf("None");
|
|
printf("\n");
|
|
|
|
printf("%sDefault timer setting: %.1f sec\n", prefix,
|
|
(double)(le32dec(desc->default_timer) / 10));
|
|
printf("%sSaved timer setting: %.1f sec\n", prefix,
|
|
(double)(le32dec(desc->saved_timer) / 10));
|
|
printf("%sCurrent timer setting: %.1f sec\n", prefix,
|
|
(double)(le32dec(desc->current_timer) / 10));
|
|
printf("%sNominal time to active: %.1f sec\n", prefix,
|
|
(double)(le32dec(desc->nom_time_to_active) / 10));
|
|
printf("%sMinimum timer: %.1f sec\n", prefix,
|
|
(double)(le32dec(desc->min_timer) / 10));
|
|
printf("%sMaximum timer: %.1f sec\n", prefix,
|
|
(double)(le32dec(desc->max_timer) / 10));
|
|
printf("%sNumber of transitions to power condition: %u\n", prefix,
|
|
le32dec(desc->num_transitions_to_pc));
|
|
printf("%sHours in power condition: %u\n", prefix,
|
|
le32dec(desc->hours_in_pc));
|
|
}
|
|
|
|
static int
|
|
epc_list(struct cam_device *device, camcontrol_devtype devtype, union ccb *ccb,
|
|
int retry_count, int timeout)
|
|
{
|
|
struct ata_power_cond_log_idle *idle_log;
|
|
struct ata_power_cond_log_standby *standby_log;
|
|
uint8_t log_buf[sizeof(*idle_log) + sizeof(*standby_log)];
|
|
uint16_t log_addr = ATA_POWER_COND_LOG;
|
|
uint16_t page_number = ATA_PCL_IDLE;
|
|
uint64_t lba;
|
|
int error = 0;
|
|
|
|
lba = (((uint64_t)page_number & 0xff00) << 32) |
|
|
((page_number & 0x00ff) << 8) |
|
|
(log_addr & 0xff);
|
|
|
|
error = build_ata_cmd(ccb,
|
|
/*retry_count*/ retry_count,
|
|
/*flags*/ CAM_DIR_IN | CAM_DEV_QFRZDIS,
|
|
/*tag_action*/ MSG_SIMPLE_Q_TAG,
|
|
/*protocol*/ AP_PROTO_DMA | AP_EXTEND,
|
|
/*ata_flags*/ AP_FLAG_BYT_BLOK_BLOCKS |
|
|
AP_FLAG_TLEN_SECT_CNT |
|
|
AP_FLAG_TDIR_FROM_DEV,
|
|
/*features*/ 0,
|
|
/*sector_count*/ 2,
|
|
/*lba*/ lba,
|
|
/*command*/ ATA_READ_LOG_DMA_EXT,
|
|
/*auxiliary*/ 0,
|
|
/*data_ptr*/ log_buf,
|
|
/*dxfer_len*/ sizeof(log_buf),
|
|
/*cdb_storage*/ NULL,
|
|
/*cdb_storage_len*/ 0,
|
|
/*sense_len*/ SSD_FULL_SIZE,
|
|
/*timeout*/ timeout ? timeout : 60000,
|
|
/*is48bit*/ 1,
|
|
/*devtype*/ devtype);
|
|
|
|
if (error != 0) {
|
|
warnx("%s: build_ata_cmd() failed, likely programmer error",
|
|
__func__);
|
|
goto bailout;
|
|
}
|
|
|
|
if (retry_count > 0)
|
|
ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER;
|
|
|
|
error = cam_send_ccb(device, ccb);
|
|
if (error != 0) {
|
|
warn("error sending ATA READ LOG EXT CCB");
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) {
|
|
cam_error_print(device, ccb, CAM_ESF_ALL, CAM_EPF_ALL,stderr);
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
idle_log = (struct ata_power_cond_log_idle *)log_buf;
|
|
standby_log =
|
|
(struct ata_power_cond_log_standby *)&log_buf[sizeof(*idle_log)];
|
|
|
|
printf("ATA Power Conditions Log:\n");
|
|
printf(" Idle power conditions page:\n");
|
|
printf(" Idle A condition:\n");
|
|
epc_print_pcl_desc(&idle_log->idle_a_desc, " ");
|
|
printf(" Idle B condition:\n");
|
|
epc_print_pcl_desc(&idle_log->idle_b_desc, " ");
|
|
printf(" Idle C condition:\n");
|
|
epc_print_pcl_desc(&idle_log->idle_c_desc, " ");
|
|
printf(" Standby power conditions page:\n");
|
|
printf(" Standby Y condition:\n");
|
|
epc_print_pcl_desc(&standby_log->standby_y_desc, " ");
|
|
printf(" Standby Z condition:\n");
|
|
epc_print_pcl_desc(&standby_log->standby_z_desc, " ");
|
|
bailout:
|
|
return (error);
|
|
}
|
|
|
|
static int
|
|
epc_getmode(struct cam_device *device, camcontrol_devtype devtype,
|
|
union ccb *ccb, int retry_count, int timeout, int power_only)
|
|
{
|
|
struct ata_params *ident = NULL;
|
|
struct ata_identify_log_sup_cap sup_cap;
|
|
const char *mode_name = NULL;
|
|
uint8_t error = 0, ata_device = 0, status = 0;
|
|
uint16_t count = 0;
|
|
uint64_t lba = 0;
|
|
uint32_t page_number, log_address;
|
|
uint64_t caps = 0;
|
|
int avail_bytes = 0;
|
|
int res_available = 0;
|
|
int retval;
|
|
|
|
retval = 0;
|
|
|
|
if (power_only != 0)
|
|
goto check_power_mode;
|
|
|
|
/*
|
|
* Get standard ATA Identify data.
|
|
*/
|
|
retval = ata_do_identify(device, retry_count, timeout, ccb, &ident);
|
|
if (retval != 0) {
|
|
warnx("Couldn't get identify data");
|
|
goto bailout;
|
|
}
|
|
|
|
/*
|
|
* Get the ATA Identify Data Log (0x30),
|
|
* Supported Capabilities Page (0x03).
|
|
*/
|
|
log_address = ATA_IDENTIFY_DATA_LOG;
|
|
page_number = ATA_IDL_SUP_CAP;
|
|
lba = (((uint64_t)page_number & 0xff00) << 32) |
|
|
((page_number & 0x00ff) << 8) |
|
|
(log_address & 0xff);
|
|
|
|
bzero(&sup_cap, sizeof(sup_cap));
|
|
/*
|
|
* XXX KDM check the supported protocol.
|
|
*/
|
|
retval = build_ata_cmd(ccb,
|
|
/*retry_count*/ retry_count,
|
|
/*flags*/ CAM_DIR_IN | CAM_DEV_QFRZDIS,
|
|
/*tag_action*/ MSG_SIMPLE_Q_TAG,
|
|
/*protocol*/ AP_PROTO_DMA |
|
|
AP_EXTEND,
|
|
/*ata_flags*/ AP_FLAG_BYT_BLOK_BLOCKS |
|
|
AP_FLAG_TLEN_SECT_CNT |
|
|
AP_FLAG_TDIR_FROM_DEV,
|
|
/*features*/ 0,
|
|
/*sector_count*/ 1,
|
|
/*lba*/ lba,
|
|
/*command*/ ATA_READ_LOG_DMA_EXT,
|
|
/*auxiliary*/ 0,
|
|
/*data_ptr*/ (uint8_t *)&sup_cap,
|
|
/*dxfer_len*/ sizeof(sup_cap),
|
|
/*cdb_storage*/ NULL,
|
|
/*cdb_storage_len*/ 0,
|
|
/*sense_len*/ SSD_FULL_SIZE,
|
|
/*timeout*/ timeout ? timeout : 60000,
|
|
/*is48bit*/ 1,
|
|
/*devtype*/ devtype);
|
|
|
|
if (retval != 0) {
|
|
warnx("%s: build_ata_cmd() failed, likely a programmer error",
|
|
__func__);
|
|
goto bailout;
|
|
}
|
|
|
|
if (retry_count > 0)
|
|
ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER;
|
|
|
|
retval = cam_send_ccb(device, ccb);
|
|
if (retval != 0) {
|
|
warn("error sending ATA READ LOG CCB");
|
|
retval = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) {
|
|
cam_error_print(device, ccb, CAM_ESF_ALL, CAM_EPF_ALL,stderr);
|
|
retval = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
if (ccb->ccb_h.func_code == XPT_SCSI_IO) {
|
|
avail_bytes = ccb->csio.dxfer_len - ccb->csio.resid;
|
|
} else {
|
|
avail_bytes = ccb->ataio.dxfer_len - ccb->ataio.resid;
|
|
}
|
|
if (avail_bytes < (int)sizeof(sup_cap)) {
|
|
warnx("Couldn't get enough of the ATA Supported "
|
|
"Capabilities log, %d bytes returned", avail_bytes);
|
|
retval = 1;
|
|
goto bailout;
|
|
}
|
|
caps = le64dec(sup_cap.sup_cap);
|
|
if ((caps & ATA_SUP_CAP_VALID) == 0) {
|
|
warnx("Supported capabilities bits are not valid");
|
|
retval = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
printf("APM: %sSupported, %sEnabled\n",
|
|
(ident->support.command2 & ATA_SUPPORT_APM) ? "" : "NOT ",
|
|
(ident->enabled.command2 & ATA_SUPPORT_APM) ? "" : "NOT ");
|
|
printf("EPC: %sSupported, %sEnabled\n",
|
|
(ident->support2 & ATA_SUPPORT_EPC) ? "" : "NOT ",
|
|
(ident->enabled2 & ATA_ENABLED_EPC) ? "" : "NOT ");
|
|
printf("Low Power Standby %sSupported\n",
|
|
(caps & ATA_SC_LP_STANDBY_SUP) ? "" : "NOT ");
|
|
printf("Set EPC Power Source %sSupported\n",
|
|
(caps & ATA_SC_SET_EPC_PS_SUP) ? "" : "NOT ");
|
|
|
|
|
|
check_power_mode:
|
|
|
|
retval = build_ata_cmd(ccb,
|
|
/*retry_count*/ retry_count,
|
|
/*flags*/ CAM_DIR_NONE | CAM_DEV_QFRZDIS,
|
|
/*tag_action*/ MSG_SIMPLE_Q_TAG,
|
|
/*protocol*/ AP_PROTO_NON_DATA |
|
|
AP_EXTEND,
|
|
/*ata_flags*/ AP_FLAG_BYT_BLOK_BLOCKS |
|
|
AP_FLAG_TLEN_NO_DATA |
|
|
AP_FLAG_CHK_COND,
|
|
/*features*/ ATA_SF_EPC,
|
|
/*sector_count*/ 0,
|
|
/*lba*/ 0,
|
|
/*command*/ ATA_CHECK_POWER_MODE,
|
|
/*auxiliary*/ 0,
|
|
/*data_ptr*/ NULL,
|
|
/*dxfer_len*/ 0,
|
|
/*cdb_storage*/ NULL,
|
|
/*cdb_storage_len*/ 0,
|
|
/*sense_len*/ SSD_FULL_SIZE,
|
|
/*timeout*/ timeout ? timeout : 60000,
|
|
/*is48bit*/ 0,
|
|
/*devtype*/ devtype);
|
|
|
|
if (retval != 0) {
|
|
warnx("%s: build_ata_cmd() failed, likely a programmer error",
|
|
__func__);
|
|
goto bailout;
|
|
}
|
|
|
|
if (retry_count > 0)
|
|
ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER;
|
|
|
|
retval = cam_send_ccb(device, ccb);
|
|
if (retval != 0) {
|
|
warn("error sending ATA CHECK POWER MODE CCB");
|
|
retval = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
/*
|
|
* Check to see whether we got the requested ATA result if this
|
|
* is an SCSI ATA PASS-THROUGH command.
|
|
*/
|
|
if (((ccb->ccb_h.status & CAM_STATUS_MASK) == CAM_SCSI_STATUS_ERROR)
|
|
&& (ccb->csio.scsi_status == SCSI_STATUS_CHECK_COND)) {
|
|
int error_code, sense_key, asc, ascq;
|
|
|
|
retval = scsi_extract_sense_ccb(ccb, &error_code,
|
|
&sense_key, &asc, &ascq);
|
|
if (retval == 0) {
|
|
cam_error_print(device, ccb, CAM_ESF_ALL, CAM_EPF_ALL,
|
|
stderr);
|
|
retval = 1;
|
|
goto bailout;
|
|
}
|
|
if ((sense_key == SSD_KEY_RECOVERED_ERROR)
|
|
&& (asc == 0x00)
|
|
&& (ascq == 0x1d)) {
|
|
res_available = 1;
|
|
}
|
|
|
|
}
|
|
if (((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)
|
|
&& (res_available == 0)) {
|
|
cam_error_print(device, ccb, CAM_ESF_ALL, CAM_EPF_ALL,stderr);
|
|
retval = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
retval = get_ata_status(device, ccb, &error, &count, &lba, &ata_device,
|
|
&status);
|
|
if (retval != 0) {
|
|
warnx("Unable to get ATA CHECK POWER MODE result");
|
|
retval = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
mode_name = scsi_nv_to_str(epc_power_cond_map,
|
|
sizeof(epc_power_cond_map) / sizeof(epc_power_cond_map[0]), count);
|
|
printf("Current power state: ");
|
|
/* Note: ident can be null in power_only mode */
|
|
if ((ident == NULL)
|
|
|| (ident->enabled2 & ATA_ENABLED_EPC)) {
|
|
if (mode_name != NULL)
|
|
printf("%s", mode_name);
|
|
else if (count == 0xff) {
|
|
printf("PM0:Active or PM1:Idle");
|
|
}
|
|
} else {
|
|
switch (count) {
|
|
case 0x00:
|
|
printf("PM2:Standby");
|
|
break;
|
|
case 0x80:
|
|
printf("PM1:Idle");
|
|
break;
|
|
case 0xff:
|
|
printf("PM0:Active or PM1:Idle");
|
|
break;
|
|
}
|
|
}
|
|
printf("(0x%02x)\n", count);
|
|
|
|
if (power_only != 0)
|
|
goto bailout;
|
|
|
|
if (caps & ATA_SC_LP_STANDBY_SUP) {
|
|
uint32_t wait_mode;
|
|
|
|
wait_mode = (lba >> 20) & 0xff;
|
|
if (wait_mode == 0xff) {
|
|
printf("Device not waiting to enter lower power "
|
|
"condition");
|
|
} else {
|
|
mode_name = scsi_nv_to_str(epc_power_cond_map,
|
|
sizeof(epc_power_cond_map) /
|
|
sizeof(epc_power_cond_map[0]), wait_mode);
|
|
printf("Device waiting to enter mode %s (0x%02x)\n",
|
|
(mode_name != NULL) ? mode_name : "Unknown",
|
|
wait_mode);
|
|
}
|
|
printf("Device is %sheld in the current power condition\n",
|
|
(lba & 0x80000) ? "" : "NOT ");
|
|
}
|
|
bailout:
|
|
return (retval);
|
|
|
|
}
|
|
|
|
static int
|
|
epc_set_features(struct cam_device *device, camcontrol_devtype devtype,
|
|
union ccb *ccb, int retry_count, int timeout, int action,
|
|
int power_cond, int timer, int enable, int save,
|
|
int delayed_entry, int hold, int power_src, int restore_src)
|
|
{
|
|
uint64_t lba;
|
|
uint16_t count = 0;
|
|
int error;
|
|
|
|
error = 0;
|
|
|
|
lba = action;
|
|
|
|
switch (action) {
|
|
case ATA_SF_EPC_SET_TIMER:
|
|
lba |= ((timer << ATA_SF_EPC_TIMER_SHIFT) &
|
|
ATA_SF_EPC_TIMER_MASK);
|
|
/* FALLTHROUGH */
|
|
case ATA_SF_EPC_SET_STATE:
|
|
lba |= (enable ? ATA_SF_EPC_TIMER_EN : 0) |
|
|
(save ? ATA_SF_EPC_TIMER_SAVE : 0);
|
|
count = power_cond;
|
|
break;
|
|
case ATA_SF_EPC_GOTO:
|
|
count = power_cond;
|
|
lba |= (delayed_entry ? ATA_SF_EPC_GOTO_DELAY : 0) |
|
|
(hold ? ATA_SF_EPC_GOTO_HOLD : 0);
|
|
break;
|
|
case ATA_SF_EPC_RESTORE:
|
|
lba |= restore_src |
|
|
(save ? ATA_SF_EPC_RST_SAVE : 0);
|
|
break;
|
|
case ATA_SF_EPC_ENABLE:
|
|
case ATA_SF_EPC_DISABLE:
|
|
break;
|
|
case ATA_SF_EPC_SET_SOURCE:
|
|
count = power_src;
|
|
break;
|
|
}
|
|
|
|
error = build_ata_cmd(ccb,
|
|
/*retry_count*/ retry_count,
|
|
/*flags*/ CAM_DIR_NONE | CAM_DEV_QFRZDIS,
|
|
/*tag_action*/ MSG_SIMPLE_Q_TAG,
|
|
/*protocol*/ AP_PROTO_NON_DATA | AP_EXTEND,
|
|
/*ata_flags*/ AP_FLAG_BYT_BLOK_BLOCKS |
|
|
AP_FLAG_TLEN_NO_DATA |
|
|
AP_FLAG_TDIR_FROM_DEV,
|
|
/*features*/ ATA_SF_EPC,
|
|
/*sector_count*/ count,
|
|
/*lba*/ lba,
|
|
/*command*/ ATA_SETFEATURES,
|
|
/*auxiliary*/ 0,
|
|
/*data_ptr*/ NULL,
|
|
/*dxfer_len*/ 0,
|
|
/*cdb_storage*/ NULL,
|
|
/*cdb_storage_len*/ 0,
|
|
/*sense_len*/ SSD_FULL_SIZE,
|
|
/*timeout*/ timeout ? timeout : 60000,
|
|
/*is48bit*/ 1,
|
|
/*devtype*/ devtype);
|
|
|
|
if (error != 0) {
|
|
warnx("%s: build_ata_cmd() failed, likely a programmer error",
|
|
__func__);
|
|
goto bailout;
|
|
}
|
|
|
|
if (retry_count > 0)
|
|
ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER;
|
|
|
|
error = cam_send_ccb(device, ccb);
|
|
if (error != 0) {
|
|
warn("error sending ATA SET FEATURES CCB");
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) {
|
|
cam_error_print(device, ccb, CAM_ESF_ALL, CAM_EPF_ALL,stderr);
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
bailout:
|
|
return (error);
|
|
}
|
|
|
|
int
|
|
epc(struct cam_device *device, int argc, char **argv, char *combinedopt,
|
|
int retry_count, int timeout, int verbosemode __unused)
|
|
{
|
|
union ccb *ccb = NULL;
|
|
int error = 0;
|
|
int c;
|
|
int action = -1;
|
|
camcontrol_devtype devtype;
|
|
double timer_val = -1;
|
|
int timer_tenths = 0, power_cond = -1;
|
|
int delayed_entry = 0, hold = 0;
|
|
int enable = -1, save = 0;
|
|
int restore_src = -1;
|
|
int power_src = -1;
|
|
int power_only = 0;
|
|
|
|
|
|
ccb = cam_getccb(device);
|
|
if (ccb == NULL) {
|
|
warnx("%s: error allocating CCB", __func__);
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
CCB_CLEAR_ALL_EXCEPT_HDR(ccb);
|
|
|
|
while ((c = getopt(argc, argv, combinedopt)) != -1) {
|
|
switch (c) {
|
|
case 'c': {
|
|
scsi_nv_status status;
|
|
int entry_num;
|
|
|
|
status = scsi_get_nv(epc_cmd_map,
|
|
(sizeof(epc_cmd_map) / sizeof(epc_cmd_map[0])),
|
|
optarg, &entry_num, SCSI_NV_FLAG_IG_CASE);
|
|
if (status == SCSI_NV_FOUND)
|
|
action = epc_cmd_map[entry_num].value;
|
|
else {
|
|
warnx("%s: %s: %s option %s", __func__,
|
|
(status == SCSI_NV_AMBIGUOUS) ?
|
|
"ambiguous" : "invalid", "epc command",
|
|
optarg);
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
break;
|
|
}
|
|
case 'd':
|
|
enable = 0;
|
|
break;
|
|
case 'D':
|
|
delayed_entry = 1;
|
|
break;
|
|
case 'e':
|
|
enable = 1;
|
|
break;
|
|
case 'H':
|
|
hold = 1;
|
|
break;
|
|
case 'p': {
|
|
scsi_nv_status status;
|
|
int entry_num;
|
|
|
|
status = scsi_get_nv(epc_power_cond_map,
|
|
(sizeof(epc_power_cond_map) /
|
|
sizeof(epc_power_cond_map[0])), optarg,
|
|
&entry_num, SCSI_NV_FLAG_IG_CASE);
|
|
if (status == SCSI_NV_FOUND)
|
|
power_cond =epc_power_cond_map[entry_num].value;
|
|
else {
|
|
warnx("%s: %s: %s option %s", __func__,
|
|
(status == SCSI_NV_AMBIGUOUS) ?
|
|
"ambiguous" : "invalid", "power condition",
|
|
optarg);
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
break;
|
|
}
|
|
case 'P':
|
|
power_only = 1;
|
|
break;
|
|
case 'r': {
|
|
scsi_nv_status status;
|
|
int entry_num;
|
|
|
|
status = scsi_get_nv(epc_rst_val,
|
|
(sizeof(epc_rst_val) /
|
|
sizeof(epc_rst_val[0])), optarg,
|
|
&entry_num, SCSI_NV_FLAG_IG_CASE);
|
|
if (status == SCSI_NV_FOUND)
|
|
restore_src = epc_rst_val[entry_num].value;
|
|
else {
|
|
warnx("%s: %s: %s option %s", __func__,
|
|
(status == SCSI_NV_AMBIGUOUS) ?
|
|
"ambiguous" : "invalid",
|
|
"restore value source", optarg);
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
break;
|
|
}
|
|
case 's':
|
|
save = 1;
|
|
break;
|
|
case 'S': {
|
|
scsi_nv_status status;
|
|
int entry_num;
|
|
|
|
status = scsi_get_nv(epc_ps_map,
|
|
(sizeof(epc_ps_map) / sizeof(epc_ps_map[0])),
|
|
optarg, &entry_num, SCSI_NV_FLAG_IG_CASE);
|
|
if (status == SCSI_NV_FOUND)
|
|
power_src = epc_ps_map[entry_num].value;
|
|
else {
|
|
warnx("%s: %s: %s option %s", __func__,
|
|
(status == SCSI_NV_AMBIGUOUS) ?
|
|
"ambiguous" : "invalid", "power source",
|
|
optarg);
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
break;
|
|
}
|
|
case 'T': {
|
|
char *endptr;
|
|
|
|
timer_val = strtod(optarg, &endptr);
|
|
if (timer_val < 0) {
|
|
warnx("Invalid timer value %f", timer_val);
|
|
error = 1;
|
|
goto bailout;
|
|
} else if (*endptr != '\0') {
|
|
warnx("Invalid timer value %s", optarg);
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
timer_tenths = timer_val * 10;
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (action == -1) {
|
|
warnx("Must specify an action");
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
|
|
error = get_device_type(device, retry_count, timeout,
|
|
/*printerrors*/ 1, &devtype);
|
|
if (error != 0)
|
|
errx(1, "Unable to determine device type");
|
|
|
|
switch (devtype) {
|
|
case CC_DT_ATA:
|
|
case CC_DT_SATL:
|
|
break;
|
|
default:
|
|
warnx("The epc subcommand only works with ATA protocol "
|
|
"devices");
|
|
error = 1;
|
|
goto bailout;
|
|
break; /*NOTREACHED*/
|
|
}
|
|
|
|
switch (action) {
|
|
case ATA_SF_EPC_SET_TIMER:
|
|
if (timer_val == -1) {
|
|
warnx("Must specify a timer value (-T time)");
|
|
error = 1;
|
|
}
|
|
/* FALLTHROUGH */
|
|
case ATA_SF_EPC_SET_STATE:
|
|
if (enable == -1) {
|
|
warnx("Must specify enable (-e) or disable (-d)");
|
|
error = 1;
|
|
}
|
|
/* FALLTHROUGH */
|
|
case ATA_SF_EPC_GOTO:
|
|
if (power_cond == -1) {
|
|
warnx("Must specify a power condition with -p");
|
|
error = 1;
|
|
}
|
|
if (error != 0)
|
|
goto bailout;
|
|
break;
|
|
case ATA_SF_EPC_SET_SOURCE:
|
|
if (power_src == -1) {
|
|
warnx("Must specify a power source (-S battery or "
|
|
"-S notbattery) value");
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
break;
|
|
case ATA_SF_EPC_RESTORE:
|
|
if (restore_src == -1) {
|
|
warnx("Must specify a source for restored value, "
|
|
"-r default or -r saved");
|
|
error = 1;
|
|
goto bailout;
|
|
}
|
|
break;
|
|
case ATA_SF_EPC_ENABLE:
|
|
case ATA_SF_EPC_DISABLE:
|
|
case CCTL_EPC_GET_STATUS:
|
|
case CCTL_EPC_LIST:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
switch (action) {
|
|
case CCTL_EPC_GET_STATUS:
|
|
error = epc_getmode(device, devtype, ccb, retry_count, timeout,
|
|
power_only);
|
|
break;
|
|
case CCTL_EPC_LIST:
|
|
error = epc_list(device, devtype, ccb, retry_count, timeout);
|
|
break;
|
|
case ATA_SF_EPC_RESTORE:
|
|
case ATA_SF_EPC_GOTO:
|
|
case ATA_SF_EPC_SET_TIMER:
|
|
case ATA_SF_EPC_SET_STATE:
|
|
case ATA_SF_EPC_ENABLE:
|
|
case ATA_SF_EPC_DISABLE:
|
|
case ATA_SF_EPC_SET_SOURCE:
|
|
error = epc_set_features(device, devtype, ccb, retry_count,
|
|
timeout, action, power_cond, timer_tenths, enable, save,
|
|
delayed_entry, hold, power_src, restore_src);
|
|
break;
|
|
default:
|
|
warnx("Not implemented yet");
|
|
error = 1;
|
|
goto bailout;
|
|
break;
|
|
}
|
|
|
|
|
|
bailout:
|
|
if (ccb != NULL)
|
|
cam_freeccb(ccb);
|
|
|
|
return (error);
|
|
}
|