blob: 3e190d5b936cd4ddc19c2590e98b37e22d5ab993 [file] [log] [blame]
dbeam005649e2015-04-02 17:24:471// Copyright 2015 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
blundellb5c6b5a2015-07-30 20:18:305#include "components/metrics/drive_metrics_provider.h"
dbeam005649e2015-04-02 17:24:476
7#include <windows.h>
8#include <ntddscsi.h>
9#include <winioctl.h>
10#include <vector>
11
12#include "base/files/file.h"
13#include "base/files/file_path.h"
14#include "base/macros.h"
15#include "base/strings/stringprintf.h"
16#include "base/win/windows_version.h"
17
blundellb5c6b5a2015-07-30 20:18:3018namespace metrics {
19
dbeam005649e2015-04-02 17:24:4720namespace {
21
22// Semi-copy of similarly named struct from ata.h in WinDDK.
23struct IDENTIFY_DEVICE_DATA {
24 USHORT UnusedWords[217];
25 USHORT NominalMediaRotationRate;
26 USHORT MoreUnusedWords[38];
27};
28COMPILE_ASSERT(sizeof(IDENTIFY_DEVICE_DATA) == 512, IdentifyDeviceDataSize);
29
30struct AtaRequest {
31 ATA_PASS_THROUGH_EX query;
32 IDENTIFY_DEVICE_DATA result;
33};
34
35} // namespace
36
37// static
38bool DriveMetricsProvider::HasSeekPenalty(const base::FilePath& path,
39 bool* has_seek_penalty) {
40 std::vector<base::FilePath::StringType> components;
41 path.GetComponents(&components);
42
43 int flags = base::File::FLAG_OPEN;
44 bool win7_or_higher = base::win::GetVersion() >= base::win::VERSION_WIN7;
45 if (!win7_or_higher)
46 flags |= base::File::FLAG_READ | base::File::FLAG_WRITE;
47
48 base::File volume(base::FilePath(L"\\\\.\\" + components[0]), flags);
49 if (!volume.IsValid())
50 return false;
51
52 if (win7_or_higher) {
53 STORAGE_PROPERTY_QUERY query = {};
54 query.QueryType = PropertyStandardQuery;
55 query.PropertyId = StorageDeviceSeekPenaltyProperty;
56
57 DEVICE_SEEK_PENALTY_DESCRIPTOR result;
58 DWORD bytes_returned;
blundellb5c6b5a2015-07-30 20:18:3059 BOOL success = DeviceIoControl(
60 volume.GetPlatformFile(), IOCTL_STORAGE_QUERY_PROPERTY, &query,
61 sizeof(query), &result, sizeof(result), &bytes_returned, NULL);
dbeam005649e2015-04-02 17:24:4762 if (success == FALSE || bytes_returned < sizeof(result))
63 return false;
64
65 *has_seek_penalty = result.IncursSeekPenalty != FALSE;
66 } else {
67 AtaRequest request = {};
68 request.query.AtaFlags = ATA_FLAGS_DATA_IN;
69 request.query.CurrentTaskFile[6] = ID_CMD;
70 request.query.DataBufferOffset = sizeof(request.query);
71 request.query.DataTransferLength = sizeof(request.result);
72 request.query.Length = sizeof(request.query);
73 request.query.TimeOutValue = 10;
74
75 DWORD bytes_returned;
blundellb5c6b5a2015-07-30 20:18:3076 BOOL success = DeviceIoControl(
77 volume.GetPlatformFile(), IOCTL_ATA_PASS_THROUGH, &request,
78 sizeof(request), &request, sizeof(request), &bytes_returned, NULL);
dbeam005649e2015-04-02 17:24:4779 if (success == FALSE || bytes_returned < sizeof(request) ||
80 request.query.CurrentTaskFile[0]) {
81 return false;
82 }
83
84 *has_seek_penalty = request.result.NominalMediaRotationRate != 1;
85 }
86
87 return true;
88}
blundellb5c6b5a2015-07-30 20:18:3089
90} // namespace metrics