/* Copyright (C) 2016 Mike Fleetwood * * This program 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. * * This program 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 this program; if not, see . */ #include "BlockSpecial.h" #include #include #include #include #include #include namespace GParted { struct MM_Number { unsigned long m_major; unsigned long m_minor; }; typedef std::map MMNumberMapping; // Associative array caching name to major, minor number pairs // E.g. // mm_number_cache["/dev/sda"] = {8, 0} // mm_number_cache["/dev/sda1"] = {8, 1} // mm_number_cache["proc"] = {0, 0} // mm_number_cache["sysfs"] = {0, 0} static MMNumberMapping mm_number_cache; BlockSpecial::BlockSpecial() : m_name( "" ), m_major( 0UL ), m_minor( 0UL ) { } BlockSpecial::BlockSpecial( const Glib::ustring & name ) : m_name( name ), m_major( 0UL ), m_minor( 0UL ) { MMNumberMapping::const_iterator mm_num_iter = mm_number_cache.find( name ); if ( mm_num_iter != mm_number_cache.end() ) { // Use already cached major, minor pair m_major = mm_num_iter->second.m_major; m_minor = mm_num_iter->second.m_minor; return; } MM_Number pair = {0UL, 0UL}; // Call stat(name, ...) to get the major, minor pair struct stat sb; if ( stat( name.c_str(), &sb ) == 0 && S_ISBLK( sb.st_mode ) ) { m_major = major( sb.st_rdev ); m_minor = minor( sb.st_rdev ); pair.m_major = m_major; pair.m_minor = m_minor; } // Add new cache entry for name to major, minor pair mm_number_cache[name] = pair; } BlockSpecial::~BlockSpecial() { } void BlockSpecial::clear_cache() { mm_number_cache.clear(); } void BlockSpecial::register_block_special( const Glib::ustring & name, unsigned long major, unsigned long minor ) { MM_Number pair; pair.m_major = major; pair.m_minor = minor; // Add new, or update existing, cache entry for name to major, minor pair mm_number_cache[name] = pair; } bool operator==( const BlockSpecial & lhs, const BlockSpecial & rhs ) { if ( lhs.m_major > 0UL || lhs.m_minor > 0UL ) // Match block special files by major, minor device numbers. return lhs.m_major == rhs.m_major && lhs.m_minor == rhs.m_minor; else // For non-block special files fall back to name string compare. return lhs.m_name == rhs.m_name; } bool operator<( const BlockSpecial & lhs, const BlockSpecial & rhs ) { if ( lhs.m_major == 0 && rhs.m_major == 0 && lhs.m_minor == 0 && rhs.m_minor == 0 ) // Two non-block special files are ordered by name. return lhs.m_name < rhs.m_name; else // Block special files are ordered by major, minor device numbers. return lhs.m_major < rhs.m_major || ( lhs.m_major == rhs.m_major && lhs.m_minor < rhs.m_minor ); } } //GParted