#include "RemovableDriveManager.hpp" #include #include "boost/nowide/convert.hpp" #if _WIN32 #include #include #include #include #include GUID WceusbshGUID = { 0x25dbce51, 0x6c8f, 0x4a72, 0x8a,0x6d,0xb5,0x4c,0x2b,0x4f,0xc8,0x35 }; #else //linux includes #include #include #include #include #include #include #include #endif namespace Slic3r { namespace GUI { #if _WIN32 INT_PTR WINAPI WinProcCallback(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam); void RemovableDriveManager::search_for_drives() { m_current_drives.clear(); //get logical drives flags by letter in alphabetical order DWORD drives_mask = GetLogicalDrives(); for (size_t i = 0; i < 26; i++) { if(drives_mask & (1 << i)) { std::string path (1,(char)('A' + i)); path+=":"; UINT drive_type = GetDriveTypeA(path.c_str()); // DRIVE_REMOVABLE on W are sd cards and usb thumbnails (not usb harddrives) if (drive_type == DRIVE_REMOVABLE) { // get name of drive std::wstring wpath = boost::nowide::widen(path);//std::wstring(path.begin(), path.end()); std::wstring volume_name; volume_name.resize(1024); std::wstring file_system_name; file_system_name.resize(1024); LPWSTR lp_volume_name_buffer = new wchar_t; BOOL error = GetVolumeInformationW(wpath.c_str(), &volume_name[0], sizeof(volume_name), NULL, NULL, NULL, &file_system_name[0], sizeof(file_system_name)); if(error != 0) { volume_name.erase(std::find(volume_name.begin(), volume_name.end(), '\0'), volume_name.end()); /* if (volume_name == L"") { volume_name = L"REMOVABLE DRIVE"; } */ if (file_system_name != L"") { ULARGE_INTEGER free_space; GetDiskFreeSpaceExA(path.c_str(), &free_space, NULL, NULL); if (free_space.QuadPart > 0) { path += "\\"; m_current_drives.push_back(DriveData(boost::nowide::narrow(volume_name), path)); } } } } } } } void RemovableDriveManager::eject_drive(const std::string &path) { if(m_current_drives.empty()) return; for (auto it = m_current_drives.begin(); it != m_current_drives.end(); ++it) { if ((*it).path == path) { // get handle to device std::string mpath = "\\\\.\\" + path; mpath = mpath.substr(0, mpath.size() - 1); HANDLE handle = CreateFileA(mpath.c_str(), GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, nullptr, OPEN_EXISTING, 0, nullptr); if (handle == INVALID_HANDLE_VALUE) { std::cerr << "Ejecting " << mpath << " failed " << GetLastError() << " \n"; return; } DWORD deviceControlRetVal(0); //these 3 commands should eject device safely but they dont, the device does disappear from file explorer but the "device was safely remove" notification doesnt trigger. //sd cards does trigger WM_DEVICECHANGE messege, usb drives dont DeviceIoControl(handle, FSCTL_LOCK_VOLUME, nullptr, 0, nullptr, 0, &deviceControlRetVal, nullptr); DeviceIoControl(handle, FSCTL_DISMOUNT_VOLUME, nullptr, 0, nullptr, 0, &deviceControlRetVal, nullptr); // some implemenatations also calls IOCTL_STORAGE_MEDIA_REMOVAL here but it returns error to me BOOL error = DeviceIoControl(handle, IOCTL_STORAGE_EJECT_MEDIA, nullptr, 0, nullptr, 0, &deviceControlRetVal, nullptr); if (error == 0) { CloseHandle(handle); std::cerr << "Ejecting " << mpath << " failed " << deviceControlRetVal << " " << GetLastError() << " \n"; return; } CloseHandle(handle); m_did_eject = true; m_current_drives.erase(it); break; } } } bool RemovableDriveManager::is_path_on_removable_drive(const std::string &path) { if (m_current_drives.empty()) return false; std::size_t found = path.find_last_of("\\"); std::string new_path = path.substr(0, found); int letter = PathGetDriveNumberA(new_path.c_str()); for (auto it = m_current_drives.begin(); it != m_current_drives.end(); ++it) { char drive = (*it).path[0]; if (drive == ('A' + letter)) return true; } return false; } std::string RemovableDriveManager::get_drive_from_path(const std::string& path) { std::size_t found = path.find_last_of("\\"); std::string new_path = path.substr(0, found); int letter = PathGetDriveNumberA(new_path.c_str()); for (auto it = m_current_drives.begin(); it != m_current_drives.end(); ++it) { char drive = (*it).path[0]; if (drive == ('A' + letter)) return (*it).path; } return ""; } void RemovableDriveManager::register_window() { //creates new unvisible window that is recieving callbacks from system // structure to register WNDCLASSEX wndClass; wndClass.cbSize = sizeof(WNDCLASSEX); wndClass.style = CS_OWNDC | CS_HREDRAW | CS_VREDRAW; wndClass.hInstance = reinterpret_cast(GetModuleHandle(0)); wndClass.lpfnWndProc = reinterpret_cast(WinProcCallback);//this is callback wndClass.cbClsExtra = 0; wndClass.cbWndExtra = 0; wndClass.hIcon = LoadIcon(0, IDI_APPLICATION); wndClass.hbrBackground = CreateSolidBrush(RGB(192, 192, 192)); wndClass.hCursor = LoadCursor(0, IDC_ARROW); wndClass.lpszClassName = L"PrusaSlicer_aux_class"; wndClass.lpszMenuName = NULL; wndClass.hIconSm = wndClass.hIcon; if(!RegisterClassEx(&wndClass)) { DWORD err = GetLastError(); return; } HWND hWnd = CreateWindowEx( WS_EX_NOACTIVATE, L"PrusaSlicer_aux_class", L"PrusaSlicer_aux_wnd", WS_DISABLED, // style CW_USEDEFAULT, 0, 640, 480, NULL, NULL, GetModuleHandle(NULL), NULL); if(hWnd == NULL) { DWORD err = GetLastError(); } //ShowWindow(hWnd, SW_SHOWNORMAL); UpdateWindow(hWnd); } INT_PTR WINAPI WinProcCallback(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { // here we need to catch messeges about device removal // problem is that when ejecting usb (how is it implemented above) there is no messege dispached. Only after physical removal of the device. //uncomment register_window() in init() to register and comment update() in GUI_App.cpp (only for windows!) to stop recieving periodical updates LRESULT lRet = 1; static HDEVNOTIFY hDeviceNotify; switch (message) { case WM_CREATE: DEV_BROADCAST_DEVICEINTERFACE NotificationFilter; ZeroMemory(&NotificationFilter, sizeof(NotificationFilter)); NotificationFilter.dbcc_size = sizeof(DEV_BROADCAST_DEVICEINTERFACE); NotificationFilter.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE; NotificationFilter.dbcc_classguid = WceusbshGUID; hDeviceNotify = RegisterDeviceNotification(hWnd, &NotificationFilter, DEVICE_NOTIFY_WINDOW_HANDLE); break; case WM_DEVICECHANGE: { // here is the important if(wParam == DBT_DEVICEREMOVECOMPLETE) { - RemovableDriveManager::get_instance().update(0, true); } } break; default: // Send all other messages on to the default windows handler. lRet = DefWindowProc(hWnd, message, wParam, lParam); break; } return lRet; } #else void RemovableDriveManager::search_for_drives() { m_current_drives.clear(); #if __APPLE__ // if on macos obj-c class will enumerate if(m_rdmmm) { m_rdmmm->list_devices(); } #else //search /media/* folder search_path("/media/*", "/media"); //search /Volumes/* folder (OSX) //search_path("/Volumes/*", "/Volumes"); std::string path(std::getenv("USER")); std::string pp(path); //std::cout << "user: "<< path << "\n"; //if program is run with sudo, we have to search for all users // but do we want that? /* if(path == "root"){ while (true) { passwd* entry = getpwent(); if (!entry) { break; } path = entry->pw_name; pp = path; //search /media/USERNAME/* folder pp = "/media/"+pp; path = "/media/" + path + "/*"; search_path(path, pp); //search /run/media/USERNAME/* folder path = "/run" + path; pp = "/run"+pp; search_path(path, pp); } endpwent(); }else */ { //search /media/USERNAME/* folder pp = "/media/"+pp; path = "/media/" + path + "/*"; search_path(path, pp); //search /run/media/USERNAME/* folder path = "/run" + path; pp = "/run"+pp; search_path(path, pp); } #endif } void RemovableDriveManager::search_path(const std::string &path,const std::string &parent_path) { glob_t globbuf; globbuf.gl_offs = 2; int error = glob(path.c_str(), GLOB_TILDE, NULL, &globbuf); if(error == 0) { for(size_t i = 0; i < globbuf.gl_pathc; i++) { inspect_file(globbuf.gl_pathv[i], parent_path); } }else { //if error - path probably doesnt exists so function just exits //std::cout<<"glob error "<< error<< "\n"; } globfree(&globbuf); } void RemovableDriveManager::inspect_file(const std::string &path, const std::string &parent_path) { //confirms if the file is removable drive and adds it to vector //if not same file system - could be removable drive if(!compare_filesystem_id(path, parent_path)) { //free space boost::filesystem::space_info si = boost::filesystem::space(path); //std::cout << "Free space: " << fs_si.free << "Available space: " << fs_si.available << " " << path << '\n'; if(si.available != 0) { //user id struct stat buf; stat(path.c_str(), &buf); uid_t uid = buf.st_uid; std::string username(std::getenv("USER")); struct passwd *pw = getpwuid(uid); if (pw != 0 && pw->pw_name == username) m_current_drives.push_back(DriveData(boost::filesystem::basename(boost::filesystem::path(path)), path)); } } } bool RemovableDriveManager::compare_filesystem_id(const std::string &path_a, const std::string &path_b) { struct stat buf; stat(path_a.c_str() ,&buf); dev_t id_a = buf.st_dev; stat(path_b.c_str() ,&buf); dev_t id_b = buf.st_dev; return id_a == id_b; } void RemovableDriveManager::eject_drive(const std::string &path) { if (m_current_drives.empty()) return; for (auto it = m_current_drives.begin(); it != m_current_drives.end(); ++it) { if((*it).path == path) { std::string correct_path(path); for (size_t i = 0; i < correct_path.size(); ++i) { if(correct_path[i]==' ') { correct_path = correct_path.insert(i,1,'\\'); i++; } } std::cout<<"Ejecting "<<(*it).name<<" from "<< correct_path<<"\n"; // there is no usable command in c++ so terminal command is used instead // but neither triggers "succesful safe removal messege" std::string command = ""; #if __APPLE__ //m_rdmmm->eject_device(path); command = "diskutil unmount "; #else command = "umount "; #endif command += correct_path; int err = system(command.c_str()); if(err) { std::cerr<<"Ejecting failed\n"; return; } m_did_eject = true; m_current_drives.erase(it); break; } } } bool RemovableDriveManager::is_path_on_removable_drive(const std::string &path) { if (m_current_drives.empty()) return false; std::size_t found = path.find_last_of("/"); std::string new_path = path.substr(0,found); for (auto it = m_current_drives.begin(); it != m_current_drives.end(); ++it) { if(compare_filesystem_id(new_path, (*it).path)) return true; } return false; } std::string RemovableDriveManager::get_drive_from_path(const std::string& path) { std::size_t found = path.find_last_of("/"); std::string new_path = path.substr(0, found); //check if same filesystem for (auto it = m_current_drives.begin(); it != m_current_drives.end(); ++it) { if (compare_filesystem_id(new_path, (*it).path)) return (*it).path; } return ""; } #endif RemovableDriveManager::RemovableDriveManager(): m_drives_count(0), m_last_update(0), m_last_save_path(""), m_last_save_name(""), m_last_save_path_verified(false), m_is_writing(false), m_did_eject(false) #if __APPLE__ , m_rdmmm(new RDMMMWrapper()) #endif {} RemovableDriveManager::~RemovableDriveManager() { #if __APPLE__ delete m_rdmmm; #endif } void RemovableDriveManager::init() { //add_callback([](void) { RemovableDriveManager::get_instance().print(); }); #if _WIN32 //register_window(); #elif __APPLE__ m_rdmmm->register_window(); #endif update(0, true); } bool RemovableDriveManager::update(const long time,const bool check) { if(time != 0) //time = 0 is forced update { long diff = m_last_update - time; if(diff <= -2) { m_last_update = time; }else { return false; // return value shouldnt matter if update didnt run } } search_for_drives(); if (m_drives_count != m_current_drives.size()) { if (check)check_and_notify(); m_drives_count = m_current_drives.size(); } return !m_current_drives.empty(); } bool RemovableDriveManager::is_drive_mounted(const std::string &path) { for (auto it = m_current_drives.begin(); it != m_current_drives.end(); ++it) { if ((*it).path == path) { return true; } } return false; } std::string RemovableDriveManager::get_drive_path() { if (m_current_drives.size() == 0) { reset_last_save_path(); return ""; } if (m_last_save_path_verified) return m_last_save_path; return m_current_drives.back().path; } std::string RemovableDriveManager::get_last_save_path() { if (!m_last_save_path_verified) return ""; return m_last_save_path; } std::string RemovableDriveManager::get_last_save_name() { return m_last_save_name; } std::vector RemovableDriveManager::get_all_drives() { return m_current_drives; } void RemovableDriveManager::check_and_notify() { if(m_callbacks.size() != 0 && m_drives_count > m_current_drives.size() && m_last_save_path_verified && !is_drive_mounted(m_last_save_path)) { for (auto it = m_callbacks.begin(); it != m_callbacks.end(); ++it) { (*it)(); } } } void RemovableDriveManager::add_callback(std::function callback) { m_callbacks.push_back(callback); } void RemovableDriveManager::erase_callbacks() { m_callbacks.clear(); } void RemovableDriveManager::set_last_save_path(const std::string& path) { m_last_save_path_verified = false; m_last_save_path = path; } void RemovableDriveManager::verify_last_save_path() { std::string last_drive = get_drive_from_path(m_last_save_path); if (last_drive != "") { m_last_save_path_verified = true; m_last_save_path = last_drive; m_last_save_name = get_drive_name(last_drive); }else { reset_last_save_path(); } } std::string RemovableDriveManager::get_drive_name(const std::string& path) { if (m_current_drives.size() == 0) return ""; for (auto it = m_current_drives.begin(); it != m_current_drives.end(); ++it) { if ((*it).path == path) { return (*it).name; } } return ""; } bool RemovableDriveManager::is_last_drive_removed() { //std::cout<<"is last: "<