summaryrefslogtreecommitdiffstats
path: root/dev_ata_cmd_set.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'dev_ata_cmd_set.cpp')
-rw-r--r--dev_ata_cmd_set.cpp119
1 files changed, 119 insertions, 0 deletions
diff --git a/dev_ata_cmd_set.cpp b/dev_ata_cmd_set.cpp
new file mode 100644
index 0000000..117e5d5
--- /dev/null
+++ b/dev_ata_cmd_set.cpp
@@ -0,0 +1,119 @@
+/*
+ * dev_ata_cmd_set.cpp
+ *
+ * Home page of code is: http://www.smartmontools.org
+ *
+ * Copyright (C) 2008-18 Christian Franke
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+
+#include "atacmds.h"
+#include "dev_ata_cmd_set.h"
+
+#include <errno.h>
+
+const char * dev_ata_cmd_set_cpp_cvsid = "$Id: dev_ata_cmd_set.cpp 4760 2018-08-19 18:45:53Z chrfranke $"
+ DEV_ATA_CMD_SET_H_CVSID;
+
+
+/////////////////////////////////////////////////////////////////////////////
+// ata_device_with_command_set
+
+// Adapter routine to implement new ATA pass through with old interface
+
+bool ata_device_with_command_set::ata_pass_through(const ata_cmd_in & in, ata_cmd_out & out)
+{
+ if (!ata_cmd_is_ok(in, true)) // data_out_support
+ return false;
+
+ smart_command_set command = (smart_command_set)-1;
+ int select = 0;
+ char * data = (char *)in.buffer;
+ char buffer[512];
+ switch (in.in_regs.command) {
+ case ATA_IDENTIFY_DEVICE:
+ command = IDENTIFY;
+ break;
+ case ATA_IDENTIFY_PACKET_DEVICE:
+ command = PIDENTIFY;
+ break;
+ case ATA_CHECK_POWER_MODE:
+ command = CHECK_POWER_MODE;
+ data = buffer; data[0] = 0;
+ break;
+ case ATA_SMART_CMD:
+ switch (in.in_regs.features) {
+ case ATA_SMART_ENABLE:
+ command = ENABLE;
+ break;
+ case ATA_SMART_READ_VALUES:
+ command = READ_VALUES;
+ break;
+ case ATA_SMART_READ_THRESHOLDS:
+ command = READ_THRESHOLDS;
+ break;
+ case ATA_SMART_READ_LOG_SECTOR:
+ command = READ_LOG;
+ select = in.in_regs.lba_low;
+ break;
+ case ATA_SMART_WRITE_LOG_SECTOR:
+ command = WRITE_LOG;
+ select = in.in_regs.lba_low;
+ break;
+ case ATA_SMART_DISABLE:
+ command = DISABLE;
+ break;
+ case ATA_SMART_STATUS:
+ command = (in.out_needed.lba_high ? STATUS_CHECK : STATUS);
+ break;
+ case ATA_SMART_AUTO_OFFLINE:
+ command = AUTO_OFFLINE;
+ select = in.in_regs.sector_count;
+ break;
+ case ATA_SMART_AUTOSAVE:
+ command = AUTOSAVE;
+ select = in.in_regs.sector_count;
+ break;
+ case ATA_SMART_IMMEDIATE_OFFLINE:
+ command = IMMEDIATE_OFFLINE;
+ select = in.in_regs.lba_low;
+ break;
+ default:
+ return set_err(ENOSYS, "Unknown SMART command");
+ }
+ break;
+ default:
+ return set_err(ENOSYS, "Non-SMART commands not implemented");
+ }
+
+ clear_err(); errno = 0;
+ int rc = ata_command_interface(command, select, data);
+ if (rc < 0) {
+ if (!get_errno())
+ set_err(errno);
+ return false;
+ }
+
+ switch (command) {
+ case CHECK_POWER_MODE:
+ out.out_regs.sector_count = data[0];
+ break;
+ case STATUS_CHECK:
+ switch (rc) {
+ case 0: // Good SMART status
+ out.out_regs.lba_high = 0xc2; out.out_regs.lba_mid = 0x4f;
+ break;
+ case 1: // Bad SMART status
+ out.out_regs.lba_high = 0x2c; out.out_regs.lba_mid = 0xf4;
+ break;
+ }
+ break;
+ default:
+ break;
+ }
+ return true;
+}
+