/* Copyright (C) 2012-2021 Carl Hetherington This file is part of DCP-o-matic. DCP-o-matic is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. DCP-o-matic is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with DCP-o-matic. If not, see . */ #include "cross.h" #include "compose.hpp" #include "dcpomatic_assert.h" #include "dcpomatic_log.h" #include #include LIBDCP_DISABLE_WARNINGS #include LIBDCP_ENABLE_WARNINGS #include #include #include "i18n.h" using std::map; using std::string; using std::vector; using boost::optional; auto constexpr MEDIA_PATH_REQUIRED_MATCHES = 3; Drive::Drive (string xml) { cxml::Document doc; doc.read_string (xml); _device = doc.string_child("Device"); for (auto i: doc.node_children("MountPoint")) { _mount_points.push_back (i->content()); } _size = doc.number_child("Size"); _vendor = doc.optional_string_child("Vendor"); _model = doc.optional_string_child("Model"); } string Drive::as_xml () const { xmlpp::Document doc; auto root = doc.create_root_node ("Drive"); root->add_child("Device")->add_child_text(_device); for (auto i: _mount_points) { root->add_child("MountPoint")->add_child_text(i.string()); } root->add_child("Size")->add_child_text(dcp::raw_convert(_size)); if (_vendor) { root->add_child("Vendor")->add_child_text(*_vendor); } if (_model) { root->add_child("Model")->add_child_text(*_model); } return doc.write_to_string("UTF-8"); } string Drive::description () const { char gb[64]; snprintf(gb, 64, "%.1f", _size / 1000000000.0); string name; if (_vendor) { name += *_vendor; } if (_model) { if (name.size() > 0) { name += " " + *_model; } else { name = *_model; } } if (name.size() == 0) { name = _("Unknown"); } return String::compose(_("%1 (%2 GB) [%3]"), name, gb, _device); } string Drive::log_summary () const { string mp; for (auto i: _mount_points) { mp += i.string() + ","; } if (mp.empty()) { mp = "[none]"; } else { mp = mp.substr (0, mp.length() - 1); } return String::compose( "Device %1 mounted on %2 size %3 vendor %4 model %5", _device, mp, _size, _vendor.get_value_or("[none]"), _model.get_value_or("[none]") ); } /* This is in _common so we can use it in unit tests */ optional analyse_osx_media_path (string path) { if (path.find("/IOHDIXController") != string::npos) { /* This is a disk image, so we completely ignore it */ LOG_DISK_NC("Ignoring this as it seems to be a disk image"); return {}; } OSXMediaPath mp; vector parts; split(parts, path, boost::is_any_of("/")); std::copy(parts.begin() + 1, parts.end(), back_inserter(mp.parts)); if (!parts.empty() && parts[0] == "IODeviceTree:") { mp.real = true; if (mp.parts.size() < MEDIA_PATH_REQUIRED_MATCHES) { /* Later we expect at least MEDIA_PATH_REQUIRED_MATCHES parts in a IODeviceTree */ LOG_DISK_NC("Ignoring this as it has a strange media path"); return {}; } } else if (!parts.empty() && parts[0] == "IOService:") { mp.real = false; } else { return {}; } return mp; } /* Take some OSXDisk objects, representing disks that `DARegisterDiskAppearedCallback` told us about, * and find those drives that we could write a DCP to. The drives returned are "real" (not synthesized) * and are whole disks (not partitions). They may be mounted, or contain mounted partitions, in which * their mounted() method will return true. */ vector osx_disks_to_drives (vector disks) { using namespace boost::algorithm; /* Mark disks containing mounted partitions as themselves mounted */ for (auto& i: disks) { if (!i.whole) { continue; } for (auto& j: disks) { if (&i != &j && !j.mount_points.empty() && starts_with(j.device, i.device)) { LOG_DISK("Marking %1 as mounted because %2 is", i.device, j.device); std::copy(j.mount_points.begin(), j.mount_points.end(), back_inserter(i.mount_points)); } } } /* Mark containers of mounted synths as themselves mounted */ for (auto& i: disks) { if (i.media_path.real) { for (auto& j: disks) { if (!j.media_path.real && !j.mount_points.empty()) { /* i is real, j is a mounted synth; if we see the first MEDIA_PATH_REQUIRED_MATCHES parts * of i anywhere in j we assume they are related and so i shares j's mount points. */ bool one_missing = false; string all_parts; DCPOMATIC_ASSERT (i.media_path.parts.size() >= MEDIA_PATH_REQUIRED_MATCHES); for (auto k = 0; k < MEDIA_PATH_REQUIRED_MATCHES; ++k) { if (find(j.media_path.parts.begin(), j.media_path.parts.end(), i.media_path.parts[k]) == j.media_path.parts.end()) { one_missing = true; } all_parts += i.media_path.parts[k] + " "; } if (!one_missing) { LOG_DISK("Marking %1 as mounted because %2 is (found %3)", i.device, j.device, all_parts); std::copy(j.mount_points.begin(), j.mount_points.end(), back_inserter(i.mount_points)); } } } } } vector drives; for (auto const& i: disks) { if (i.whole && i.media_path.real) { drives.push_back(Drive(i.device, i.mount_points, i.size, i.vendor, i.model)); LOG_DISK_NC(drives.back().log_summary()); } } return drives; }