1
0
mirror of https://github.com/veracrypt/VeraCrypt.git synced 2025-11-12 11:28:26 -06:00
Files
VeraCrypt/src/Main/Forms/DeviceSelectionDialog.cpp
El Mostafa Idrassi 621330b726 MacOSX: Fixed devices / partitions not showing in the device selection dialog (#516)
To get the size of each device / partition on the system, the method 'GetDeviceSize()' in 'src/Core/Unix/CoreUnix.cpp' first opens
the device / partition using 'open()' function to get a File Descriptor, then retrieves its size using this File Descriptor.

Starting OS X 10.11 ("El Capitan"), a feature called "System Integrity Protection (SIP)" or less formally, "rootless mode" has been added.
This feature blocks access to certain critical aspects of the OS and Hardware by 3rd-Party programs.
Specifically, low-level access to the system disks, devices and partitions is forbidden ; namely functions like 'open()' for instance fail
with the error code : "EPERM = Operation Not Permitted".

Therefore, for system devices / partitions, 'GetDeviceSize()' fails because of the failure of the 'open()' function, and throws an exception,
which is then caught inside the method 'GetHostDevices()' in '/src/Core/Unix/FreeBSD/CoreFreeBSD.cpp' : this leads to the size of the
device / partition being set to '0'.

Therefore, in the constructor of 'DeviceSelectionDialog' in 'src/Main/Forms/DeviceSelectionDialog.cpp', when the size of a device is '0',
the whole device is skipped, leading to all of its partitions not being treated or shown, even though some of these partitions may have a size which is != 0.

This commit fixes the issue by :
1 - First, checking whether the device size is '0'. If it is the case, the code loops through all the devices partitions : if there is at least one partition
with a size != 0, the device is not skipped. Otherwise, it is.
2 - Then, if the size of the device is '0', the size of the device is not shown to avoid confusing the user.
Also, since the device is not usable, the 'OK' button is not active when the device is selected.
3 - Finally, if a partition's size is '0', it is not shown since it is not usable : we cannot open it.

Signed-off-by: El Mostafa IDRASSI <el-mostafa.idrassi@prestalab.net>
2019-10-11 17:02:03 +02:00

153 lines
4.7 KiB
C++

/*
Derived from source code of TrueCrypt 7.1a, which is
Copyright (c) 2008-2012 TrueCrypt Developers Association and which is governed
by the TrueCrypt License 3.0.
Modifications and additions to the original source code (contained in this file)
and all other portions of this file are Copyright (c) 2013-2017 IDRIX
and are governed by the Apache License 2.0 the full text of which is
contained in the file License.txt included in VeraCrypt binary and source
code distribution packages.
*/
#include "System.h"
#include "Main/GraphicUserInterface.h"
#include "Main/Resources.h"
#include "DeviceSelectionDialog.h"
namespace VeraCrypt
{
DeviceSelectionDialog::DeviceSelectionDialog (wxWindow* parent)
: DeviceSelectionDialogBase (parent)
{
wxBusyCursor busy;
list <int> colPermilles;
DeviceListCtrl->InsertColumn (ColumnDevice, LangString["DEVICE"], wxLIST_FORMAT_LEFT, 1);
colPermilles.push_back (447);
#ifdef TC_WINDOWS
DeviceListCtrl->InsertColumn (ColumnDrive, LangString["DRIVE"], wxLIST_FORMAT_LEFT, 1);
colPermilles.push_back (91);
#endif
DeviceListCtrl->InsertColumn (ColumnSize, LangString["SIZE"], wxLIST_FORMAT_RIGHT, 1);
colPermilles.push_back (153);
#ifdef TC_WINDOWS
DeviceListCtrl->InsertColumn (ColumnName, LangString["LABEL"], wxLIST_FORMAT_LEFT, 1);
colPermilles.push_back (307);
#else
DeviceListCtrl->InsertColumn (ColumnMountPoint, LangString["MOUNT_POINT"], wxLIST_FORMAT_LEFT, 1);
colPermilles.push_back (396);
#endif
wxImageList *imageList = new wxImageList (16, 12, true);
imageList->Add (Resources::GetDriveIconBitmap(), Resources::GetDriveIconMaskBitmap());
DeviceListCtrl->AssignImageList (imageList, wxIMAGE_LIST_SMALL);
DeviceList = Core->GetHostDevices();
foreach_ref (HostDevice &device, DeviceList)
{
vector <wstring> fields (DeviceListCtrl->GetColumnCount());
if (DeviceListCtrl->GetItemCount() > 0)
Gui->AppendToListCtrl (DeviceListCtrl, fields);
// i.e. /dev/rdisk0 might have size = 0 in case open() fails because, for example on OSX,
// SIP is enabled on the machine ;
// This does not mean that it does not have partitions that have been successfully opened
// and have a size != 0 ;
// Therefore, we do not show the device ONLY if it does not have partitions with size != 0 ;
if (device.Size == 0)
{
bool bHasNonEmptyPartition = false;
foreach_ref (HostDevice &partition, device.Partitions)
{
if (partition.Size)
{
bHasNonEmptyPartition = true;
break;
}
}
if (!bHasNonEmptyPartition)
continue;
}
#ifdef TC_WINDOWS
fields[ColumnDevice] = StringFormatter (L"{0} {1}:", _("Harddisk"), device.SystemNumber);
fields[ColumnDrive] = device.MountPoint;
fields[ColumnName] = device.Name;
#else
fields[ColumnDevice] = wstring (device.Path) + L":";
fields[ColumnMountPoint] = device.MountPoint;
#endif
// If the size of the device is 0, we do not show the size to avoid confusing the user ;
if (device.Size)
fields[ColumnSize] = Gui->SizeToString (device.Size);
else
fields[ColumnSize] = L"";
Gui->AppendToListCtrl (DeviceListCtrl, fields, 0, &device);
foreach_ref (HostDevice &partition, device.Partitions)
{
// If a partition's size is 0, there is no need to show it in the list
// since this means it is not usable (i.e on OSX, because of SIP enabled in the machine) ;
if (!partition.Size)
continue;
fields[ColumnDevice] =
#ifndef TC_WINDOWS
wstring (L" ") +
#endif
wstring (partition.Path);
#ifdef TC_WINDOWS
fields[ColumnDrive] = partition.MountPoint;
fields[ColumnName] = partition.Name;
#else
fields[ColumnMountPoint] = partition.MountPoint;
#endif
fields[ColumnSize] = Gui->SizeToString (partition.Size);
Gui->AppendToListCtrl (DeviceListCtrl, fields, -1, &partition);
}
}
Gui->SetListCtrlWidth (DeviceListCtrl, 73);
Gui->SetListCtrlHeight (DeviceListCtrl, 16);
Gui->SetListCtrlColumnWidths (DeviceListCtrl, colPermilles);
Fit();
Layout();
Center();
StdButtonsOK->Disable();
StdButtonsOK->SetDefault();
}
void DeviceSelectionDialog::OnListItemActivated (wxListEvent& event)
{
if (StdButtonsOK->IsEnabled())
EndModal (wxID_OK);
}
void DeviceSelectionDialog::OnListItemDeselected (wxListEvent& event)
{
if (DeviceListCtrl->GetSelectedItemCount() == 0)
StdButtonsOK->Disable();
}
void DeviceSelectionDialog::OnListItemSelected (wxListEvent& event)
{
HostDevice *device = (HostDevice *) (event.GetItem().GetData());
// If a device's size is 0, we do not enable the 'OK' button since it is not usable
if (device && device->Size)
{
SelectedDevice = *device;
StdButtonsOK->Enable();
}
else
StdButtonsOK->Disable();
}
}