// Copyright 2014 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "components/metrics/machine_id_provider.h" #include <windows.h> #include <winioctl.h> #include "base/base_paths.h" #include "base/files/file_path.h" #include "base/path_service.h" #include "base/threading/thread_restrictions.h" #include "base/win/scoped_handle.h" namespace metrics { MachineIdProvider::MachineIdProvider() { } MachineIdProvider::~MachineIdProvider() { } // On windows, the machine id is based on the serial number of the drive Chrome // is running from. std::string MachineIdProvider::GetMachineId() { base::ThreadRestrictions::AssertIOAllowed(); // Use the program's path to get the drive used for the machine id. This means // that whenever the underlying drive changes, it's considered a new machine. // This is fine as we do not support migrating Chrome installs to new drives. base::FilePath executable_path; if (!PathService::Get(base::FILE_EXE, &executable_path)) { NOTREACHED(); return std::string(); } std::vector<base::FilePath::StringType> path_components; executable_path.GetComponents(&path_components); if (path_components.empty()) { NOTREACHED(); return std::string(); } base::FilePath::StringType drive_name = L"\\\\.\\" + path_components[0]; base::win::ScopedHandle drive_handle( CreateFile(drive_name.c_str(), 0, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL)); STORAGE_PROPERTY_QUERY query = {}; query.PropertyId = StorageDeviceProperty; query.QueryType = PropertyStandardQuery; // Perform an initial query to get the number of bytes being returned. DWORD bytes_returned; STORAGE_DESCRIPTOR_HEADER header = {}; BOOL status = DeviceIoControl(drive_handle, IOCTL_STORAGE_QUERY_PROPERTY, &query, sizeof(STORAGE_PROPERTY_QUERY), &header, sizeof(STORAGE_DESCRIPTOR_HEADER), &bytes_returned, NULL); if (!status) return std::string(); // Query for the actual serial number. std::vector<int8> output_buf(header.Size); status = DeviceIoControl(drive_handle, IOCTL_STORAGE_QUERY_PROPERTY, &query, sizeof(STORAGE_PROPERTY_QUERY), &output_buf[0], output_buf.size(), &bytes_returned, NULL); if (!status) return std::string(); const STORAGE_DEVICE_DESCRIPTOR* device_descriptor = reinterpret_cast<STORAGE_DEVICE_DESCRIPTOR*>(&output_buf[0]); // The serial number is stored in the |output_buf| as a null-terminated // string starting at the specified offset. const DWORD offset = device_descriptor->SerialNumberOffset; if (offset >= output_buf.size()) return std::string(); // Make sure that the null-terminator exists. const std::vector<int8>::iterator serial_number_begin = output_buf.begin() + offset; const std::vector<int8>::iterator null_location = std::find(serial_number_begin, output_buf.end(), '\0'); if (null_location == output_buf.end()) return std::string(); const char* serial_number = reinterpret_cast<const char*>(&output_buf[offset]); return std::string(serial_number); } // static MachineIdProvider* MachineIdProvider::CreateInstance() { return new MachineIdProvider(); } } // namespace metrics