1 /* Copyright (c) Mark Harmstone 2016-17
2  *
3  * This file is part of WinBtrfs.
4  *
5  * WinBtrfs is free software: you can redistribute it and/or modify
6  * it under the terms of the GNU Lesser General Public Licence as published by
7  * the Free Software Foundation, either version 3 of the Licence, or
8  * (at your option) any later version.
9  *
10  * WinBtrfs is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU Lesser General Public Licence for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public Licence
16  * along with WinBtrfs.  If not, see <http://www.gnu.org/licenses/>. */
17 
18 #define ISOLATION_AWARE_ENABLED 1
19 #define STRSAFE_NO_DEPRECATE
20 
21 #include "shellext.h"
22 #ifndef __REACTOS__
23 #include <windows.h>
24 #include <strsafe.h>
25 #include <winternl.h>
26 #else
27 #define WIN32_NO_STATUS
28 #include <windef.h>
29 #include <winbase.h>
30 #include <strsafe.h>
31 #include <ndk/iofuncs.h>
32 #include <ndk/iotypes.h>
33 #endif
34 
35 #define NO_SHLWAPI_STRFCNS
36 #include <shlwapi.h>
37 #include <uxtheme.h>
38 
39 #include "volpropsheet.h"
40 #include "resource.h"
41 #ifndef __REACTOS__
42 #include "mountmgr.h"
43 #else
44 #include "mountmgr_local.h"
45 #endif
46 
47 #ifndef __REACTOS__
48 static const NTSTATUS STATUS_OBJECT_NAME_NOT_FOUND = 0xC0000034;
49 #endif
50 
51 HRESULT __stdcall BtrfsVolPropSheet::QueryInterface(REFIID riid, void **ppObj) {
52     if (riid == IID_IUnknown || riid == IID_IShellPropSheetExt) {
53         *ppObj = static_cast<IShellPropSheetExt*>(this);
54         AddRef();
55         return S_OK;
56     } else if (riid == IID_IShellExtInit) {
57         *ppObj = static_cast<IShellExtInit*>(this);
58         AddRef();
59         return S_OK;
60     }
61 
62     *ppObj = nullptr;
63     return E_NOINTERFACE;
64 }
65 
66 HRESULT __stdcall BtrfsVolPropSheet::Initialize(PCIDLIST_ABSOLUTE pidlFolder, IDataObject* pdtobj, HKEY hkeyProgID) {
67     ULONG num_files;
68     FORMATETC format = { CF_HDROP, nullptr, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
69     HDROP hdrop;
70     WCHAR fnbuf[MAX_PATH];
71 
72     if (pidlFolder)
73         return E_FAIL;
74 
75     if (!pdtobj)
76         return E_FAIL;
77 
78     stgm.tymed = TYMED_HGLOBAL;
79 
80     if (FAILED(pdtobj->GetData(&format, &stgm)))
81         return E_INVALIDARG;
82 
83     stgm_set = true;
84 
85     hdrop = (HDROP)GlobalLock(stgm.hGlobal);
86 
87     if (!hdrop) {
88         ReleaseStgMedium(&stgm);
89         stgm_set = false;
90         return E_INVALIDARG;
91     }
92 
93     num_files = DragQueryFileW((HDROP)stgm.hGlobal, 0xFFFFFFFF, nullptr, 0);
94 
95     if (num_files > 1) {
96         GlobalUnlock(hdrop);
97         return E_FAIL;
98     }
99 
100     if (DragQueryFileW((HDROP)stgm.hGlobal, 0, fnbuf, sizeof(fnbuf) / sizeof(MAX_PATH))) {
101         fn = fnbuf;
102 
103         win_handle h = CreateFileW(fn.c_str(), FILE_TRAVERSE | FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, nullptr,
104                                    OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OPEN_REPARSE_POINT, nullptr);
105 
106         if (h != INVALID_HANDLE_VALUE) {
107             NTSTATUS Status;
108             IO_STATUS_BLOCK iosb;
109             ULONG devsize, i;
110 
111             i = 0;
112             devsize = 1024;
113 
114             devices = (btrfs_device*)malloc(devsize);
115 
116             while (true) {
117                 Status = NtFsControlFile(h, nullptr, nullptr, nullptr, &iosb, FSCTL_BTRFS_GET_DEVICES, nullptr, 0, devices, devsize);
118                 if (Status == STATUS_BUFFER_OVERFLOW) {
119                     if (i < 8) {
120                         devsize += 1024;
121 
122                         free(devices);
123                         devices = (btrfs_device*)malloc(devsize);
124 
125                         i++;
126                     } else {
127                         GlobalUnlock(hdrop);
128                         return E_FAIL;
129                     }
130                 } else
131                     break;
132             }
133 
134             if (!NT_SUCCESS(Status)) {
135                 GlobalUnlock(hdrop);
136                 return E_FAIL;
137             }
138 
139             Status = NtFsControlFile(h, nullptr, nullptr, nullptr, &iosb, FSCTL_BTRFS_GET_UUID, nullptr, 0, &uuid, sizeof(BTRFS_UUID));
140             uuid_set = NT_SUCCESS(Status);
141 
142             ignore = false;
143             balance = new BtrfsBalance(fn);
144         } else {
145             GlobalUnlock(hdrop);
146             return E_FAIL;
147         }
148     } else {
149         GlobalUnlock(hdrop);
150         return E_FAIL;
151     }
152 
153     GlobalUnlock(hdrop);
154 
155     return S_OK;
156 }
157 
158 typedef struct {
159     uint64_t dev_id;
160     wstring name;
161     uint64_t alloc;
162     uint64_t size;
163 } dev;
164 
165 void BtrfsVolPropSheet::FormatUsage(HWND hwndDlg, wstring& s, btrfs_usage* usage) {
166     uint8_t i, j;
167     uint64_t num_devs, dev_size, dev_alloc, data_size, data_alloc, metadata_size, metadata_alloc;
168     btrfs_device* bd;
169     vector<dev> devs;
170     btrfs_usage* bue;
171     wstring t, u, v;
172 
173     static const uint64_t types[] = { BLOCK_FLAG_DATA, BLOCK_FLAG_DATA | BLOCK_FLAG_METADATA, BLOCK_FLAG_METADATA, BLOCK_FLAG_SYSTEM };
174     static const ULONG typestrings[] = { IDS_USAGE_DATA, IDS_USAGE_MIXED, IDS_USAGE_METADATA, IDS_USAGE_SYSTEM };
175     static const uint64_t duptypes[] = { 0, BLOCK_FLAG_DUPLICATE, BLOCK_FLAG_RAID0, BLOCK_FLAG_RAID1, BLOCK_FLAG_RAID10, BLOCK_FLAG_RAID5,
176                                          BLOCK_FLAG_RAID6, BLOCK_FLAG_RAID1C3, BLOCK_FLAG_RAID1C4 };
177     static const ULONG dupstrings[] = { IDS_SINGLE, IDS_DUP, IDS_RAID0, IDS_RAID1, IDS_RAID10, IDS_RAID5, IDS_RAID6, IDS_RAID1C3, IDS_RAID1C4 };
178 
179     static const uint64_t raid_types = BLOCK_FLAG_DUPLICATE | BLOCK_FLAG_RAID0 | BLOCK_FLAG_RAID1 | BLOCK_FLAG_RAID10 | BLOCK_FLAG_RAID5 |
180                                        BLOCK_FLAG_RAID6 | BLOCK_FLAG_RAID1C3 | BLOCK_FLAG_RAID1C4;
181 
182     s = L"";
183 
184     num_devs = 0;
185     bd = devices;
186 
187     while (true) {
188         num_devs++;
189 
190         if (bd->next_entry > 0)
191             bd = (btrfs_device*)((uint8_t*)bd + bd->next_entry);
192         else
193             break;
194     }
195 
196     bd = devices;
197 
198     dev_size = 0;
199 
200     while (true) {
201         dev d;
202 
203         if (bd->missing) {
204             if (!load_string(module, IDS_MISSING, d.name))
205                 throw last_error(GetLastError());
206         } else if (bd->device_number == 0xffffffff)
207             d.name = wstring(bd->name, bd->namelen / sizeof(WCHAR));
208         else if (bd->partition_number == 0) {
209             if (!load_string(module, IDS_DISK_NUM, u))
210                 throw last_error(GetLastError());
211 
212             wstring_sprintf(d.name, u, bd->device_number);
213         } else {
214             if (!load_string(module, IDS_DISK_PART_NUM, u))
215                 throw last_error(GetLastError());
216 
217             wstring_sprintf(d.name, u, bd->device_number, bd->partition_number);
218         }
219 
220         d.dev_id = bd->dev_id;
221         d.alloc = 0;
222         d.size = bd->size;
223 
224         devs.push_back(d);
225 
226         dev_size += bd->size;
227 
228         if (bd->next_entry > 0)
229             bd = (btrfs_device*)((uint8_t*)bd + bd->next_entry);
230         else
231             break;
232     }
233 
234     dev_alloc = 0;
235     data_size = data_alloc = 0;
236     metadata_size = metadata_alloc = 0;
237 
238     bue = usage;
239     while (true) {
240         for (uint64_t k = 0; k < bue->num_devices; k++) {
241             dev_alloc += bue->devices[k].alloc;
242 
243             if (bue->type & BLOCK_FLAG_DATA) {
244                 data_alloc += bue->devices[k].alloc;
245             }
246 
247             if (bue->type & BLOCK_FLAG_METADATA) {
248                 metadata_alloc += bue->devices[k].alloc;
249             }
250         }
251 
252         if (bue->type & BLOCK_FLAG_DATA)
253             data_size += bue->size;
254 
255         if (bue->type & BLOCK_FLAG_METADATA)
256             metadata_size += bue->size;
257 
258         if (bue->next_entry > 0)
259             bue = (btrfs_usage*)((uint8_t*)bue + bue->next_entry);
260         else
261             break;
262     }
263 
264     // device size
265 
266     if (!load_string(module, IDS_USAGE_DEV_SIZE, u))
267         throw last_error(GetLastError());
268 
269     format_size(dev_size, v, false);
270 
271     wstring_sprintf(t, u, v.c_str());
272 
273     s += t + L"\r\n";
274 
275     // device allocated
276 
277     if (!load_string(module, IDS_USAGE_DEV_ALLOC, u))
278         throw last_error(GetLastError());
279 
280     format_size(dev_alloc, v, false);
281 
282     wstring_sprintf(t, u, v.c_str());
283 
284 #ifndef __REACTOS__
285     s += t + L"\r\n"s;
286 #else
287     s += t + L"\r\n";
288 #endif
289 
290     // device unallocated
291 
292     if (!load_string(module, IDS_USAGE_DEV_UNALLOC, u))
293         throw last_error(GetLastError());
294 
295     format_size(dev_size - dev_alloc, v, false);
296 
297     wstring_sprintf(t, u, v.c_str());
298 
299 #ifndef __REACTOS__
300     s += t + L"\r\n"s;
301 #else
302     s += t + L"\r\n";
303 #endif
304 
305     // data ratio
306 
307     if (data_alloc > 0) {
308         if (!load_string(module, IDS_USAGE_DATA_RATIO, u))
309             throw last_error(GetLastError());
310 
311         wstring_sprintf(t, u, (float)data_alloc / (float)data_size);
312 
313 #ifndef __REACTOS__
314         s += t + L"\r\n"s;
315 #else
316         s += t + L"\r\n";
317 #endif
318     }
319 
320     // metadata ratio
321 
322     if (!load_string(module, IDS_USAGE_METADATA_RATIO, u))
323         throw last_error(GetLastError());
324 
325     wstring_sprintf(t, u, (float)metadata_alloc / (float)metadata_size);
326 
327     s += t + L"\r\n\r\n";
328 
329     for (i = 0; i < sizeof(types) / sizeof(types[0]); i++) {
330         for (j = 0; j < sizeof(duptypes) / sizeof(duptypes[0]); j++) {
331             bue = usage;
332 
333             while (true) {
334                 if ((bue->type & types[i]) == types[i] && ((duptypes[j] == 0 && (bue->type & raid_types) == 0) || bue->type & duptypes[j])) {
335                     wstring sizestring, usedstring, typestring, dupstring;
336 
337                     if (bue->type & BLOCK_FLAG_DATA && bue->type & BLOCK_FLAG_METADATA && (types[i] == BLOCK_FLAG_DATA || types[i] == BLOCK_FLAG_METADATA))
338                         break;
339 
340                     if (!load_string(module, typestrings[i], typestring))
341                         throw last_error(GetLastError());
342 
343                     if (!load_string(module, dupstrings[j], dupstring))
344                         throw last_error(GetLastError());
345 
346                     format_size(bue->size, sizestring, false);
347                     format_size(bue->used, usedstring, false);
348 
349                     wstring_sprintf(t, typestring, dupstring.c_str(), sizestring.c_str(), usedstring.c_str());
350 
351                     s += t + L"\r\n";
352 
353                     for (uint64_t k = 0; k < bue->num_devices; k++) {
354                         bool found = false;
355 
356                         format_size(bue->devices[k].alloc, sizestring, false);
357 
358                         for (size_t l = 0; l < min((uint64_t)SIZE_MAX, num_devs); l++) {
359                             if (devs[l].dev_id == bue->devices[k].dev_id) {
360                                 s += devs[l].name + L"\t" + sizestring + L"\r\n";
361 
362                                 devs[l].alloc += bue->devices[k].alloc;
363 
364                                 found = true;
365                                 break;
366                             }
367                         }
368 
369                         if (!found) {
370                             if (!load_string(module, IDS_UNKNOWN_DEVICE, typestring))
371                                 throw last_error(GetLastError());
372 
373                             wstring_sprintf(t, typestring, bue->devices[k].dev_id);
374 
375 #ifndef __REACTOS__
376                             s += t + L"\t"s + sizestring + L"\r\n"s;
377 #else
378                             s += t + L"\t" + sizestring + L"\r\n";
379 #endif
380                         }
381                     }
382 
383                     s += L"\r\n";
384 
385                     break;
386                 }
387 
388                 if (bue->next_entry > 0)
389                     bue = (btrfs_usage*)((uint8_t*)bue + bue->next_entry);
390                 else
391                     break;
392             }
393         }
394     }
395 
396     if (!load_string(module, IDS_USAGE_UNALLOC, t))
397         throw last_error(GetLastError());
398 
399 #ifndef __REACTOS__
400     s += t + L"\r\n"s;
401 #else
402     s += t + L"\r\n";
403 #endif
404 
405     for (size_t k = 0; k < min((uint64_t)SIZE_MAX, num_devs); k++) {
406         wstring sizestring;
407 
408         format_size(devs[k].size - devs[k].alloc, sizestring, false);
409 
410         s += devs[k].name + L"\t" + sizestring + L"\r\n";
411     }
412 }
413 
414 void BtrfsVolPropSheet::RefreshUsage(HWND hwndDlg) {
415     wstring s;
416     btrfs_usage* usage;
417 
418     win_handle h = CreateFileW(fn.c_str(), FILE_TRAVERSE | FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, nullptr,
419                                OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OPEN_REPARSE_POINT, nullptr);
420 
421     if (h != INVALID_HANDLE_VALUE) {
422         NTSTATUS Status;
423         IO_STATUS_BLOCK iosb;
424         ULONG devsize, usagesize, i;
425 
426         i = 0;
427         devsize = 1024;
428 
429         devices = (btrfs_device*)malloc(devsize);
430 
431         while (true) {
432             Status = NtFsControlFile(h, nullptr, nullptr, nullptr, &iosb, FSCTL_BTRFS_GET_DEVICES, nullptr, 0, devices, devsize);
433             if (Status == STATUS_BUFFER_OVERFLOW) {
434                 if (i < 8) {
435                     devsize += 1024;
436 
437                     free(devices);
438                     devices = (btrfs_device*)malloc(devsize);
439 
440                     i++;
441                 } else
442                     return;
443             } else
444                 break;
445         }
446 
447         if (!NT_SUCCESS(Status))
448             return;
449 
450         i = 0;
451         usagesize = 1024;
452 
453         usage = (btrfs_usage*)malloc(usagesize);
454 
455         while (true) {
456             Status = NtFsControlFile(h, nullptr, nullptr, nullptr, &iosb, FSCTL_BTRFS_GET_USAGE, nullptr, 0, usage, usagesize);
457             if (Status == STATUS_BUFFER_OVERFLOW) {
458                 if (i < 8) {
459                     usagesize += 1024;
460 
461                     free(usage);
462                     usage = (btrfs_usage*)malloc(usagesize);
463 
464                     i++;
465                 } else
466                     return;
467             } else
468                 break;
469         }
470 
471         if (!NT_SUCCESS(Status)) {
472             free(usage);
473             return;
474         }
475 
476         ignore = false;
477     } else
478         return;
479 
480     FormatUsage(hwndDlg, s, usage);
481 
482     SetDlgItemTextW(hwndDlg, IDC_USAGE_BOX, s.c_str());
483 
484     free(usage);
485 }
486 
487 INT_PTR CALLBACK BtrfsVolPropSheet::UsageDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
488     try {
489         switch (uMsg) {
490             case WM_INITDIALOG:
491             {
492                 wstring s;
493                 int i;
494                 ULONG usagesize;
495                 NTSTATUS Status;
496                 IO_STATUS_BLOCK iosb;
497 
498                 EnableThemeDialogTexture(hwndDlg, ETDT_ENABLETAB);
499 
500                 win_handle h = CreateFileW(fn.c_str(), FILE_TRAVERSE | FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, nullptr,
501                                         OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OPEN_REPARSE_POINT, nullptr);
502 
503                 if (h != INVALID_HANDLE_VALUE) {
504                     btrfs_usage* usage;
505 
506                     i = 0;
507                     usagesize = 1024;
508 
509                     usage = (btrfs_usage*)malloc(usagesize);
510 
511                     while (true) {
512                         Status = NtFsControlFile(h, nullptr, nullptr, nullptr, &iosb, FSCTL_BTRFS_GET_USAGE, nullptr, 0, usage, usagesize);
513                         if (Status == STATUS_BUFFER_OVERFLOW) {
514                             if (i < 8) {
515                                 usagesize += 1024;
516 
517                                 free(usage);
518                                 usage = (btrfs_usage*)malloc(usagesize);
519 
520                                 i++;
521                             } else
522                                 break;
523                         } else
524                             break;
525                     }
526 
527                     if (!NT_SUCCESS(Status)) {
528                         free(usage);
529                         break;
530                     }
531 
532                     FormatUsage(hwndDlg, s, usage);
533 
534                     SetDlgItemTextW(hwndDlg, IDC_USAGE_BOX, s.c_str());
535 
536                     free(usage);
537                 }
538 
539                 break;
540             }
541 
542             case WM_COMMAND:
543                 switch (HIWORD(wParam)) {
544                     case BN_CLICKED:
545                         switch (LOWORD(wParam)) {
546                             case IDOK:
547                             case IDCANCEL:
548                                 EndDialog(hwndDlg, 0);
549                             return true;
550 
551                             case IDC_USAGE_REFRESH:
552                                 RefreshUsage(hwndDlg);
553                             return true;
554                         }
555                     break;
556                 }
557             break;
558         }
559     } catch (const exception& e) {
560         error_message(hwndDlg, e.what());
561     }
562 
563     return false;
564 }
565 
566 static INT_PTR CALLBACK stub_UsageDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
567     BtrfsVolPropSheet* bvps;
568 
569     if (uMsg == WM_INITDIALOG) {
570         SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)lParam);
571         bvps = (BtrfsVolPropSheet*)lParam;
572     } else {
573         bvps = (BtrfsVolPropSheet*)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
574     }
575 
576     if (bvps)
577         return bvps->UsageDlgProc(hwndDlg, uMsg, wParam, lParam);
578     else
579         return false;
580 }
581 
582 void BtrfsVolPropSheet::ShowUsage(HWND hwndDlg) {
583    DialogBoxParamW(module, MAKEINTRESOURCEW(IDD_VOL_USAGE), hwndDlg, stub_UsageDlgProc, (LPARAM)this);
584 }
585 
586 static void add_lv_column(HWND list, int string, int cx) {
587     LVCOLUMNW lvc;
588     wstring s;
589 
590     if (!load_string(module, string, s))
591         throw last_error(GetLastError());
592 
593     lvc.mask = LVCF_TEXT|LVCF_WIDTH;
594     lvc.pszText = (WCHAR*)s.c_str();
595     lvc.cx = cx;
596     SendMessageW(list, LVM_INSERTCOLUMNW, 0, (LPARAM)&lvc);
597 }
598 
599 static int CALLBACK lv_sort(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort) {
600     if (lParam1 < lParam2)
601         return -1;
602     else if (lParam1 > lParam2)
603         return 1;
604     else
605         return 0;
606 }
607 
608 static uint64_t find_dev_alloc(uint64_t dev_id, btrfs_usage* usage) {
609     btrfs_usage* bue;
610     uint64_t alloc;
611 
612     alloc = 0;
613 
614     bue = usage;
615     while (true) {
616         uint64_t k;
617 
618         for (k = 0; k < bue->num_devices; k++) {
619             if (bue->devices[k].dev_id == dev_id)
620                 alloc += bue->devices[k].alloc;
621         }
622 
623         if (bue->next_entry > 0)
624             bue = (btrfs_usage*)((uint8_t*)bue + bue->next_entry);
625         else
626             break;
627     }
628 
629     return alloc;
630 }
631 
632 void BtrfsVolPropSheet::RefreshDevList(HWND devlist) {
633     NTSTATUS Status;
634     IO_STATUS_BLOCK iosb;
635     ULONG usagesize, devsize;
636     btrfs_usage* usage;
637     btrfs_device* bd;
638     int i;
639     uint64_t num_rw_devices;
640     {
641         win_handle h = CreateFileW(fn.c_str(), FILE_TRAVERSE | FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, nullptr,
642                                    OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OPEN_REPARSE_POINT, nullptr);
643 
644         if (h == INVALID_HANDLE_VALUE)
645             throw last_error(GetLastError());
646 
647         i = 0;
648         devsize = 1024;
649 
650         if (devices)
651             free(devices);
652 
653         devices = (btrfs_device*)malloc(devsize);
654 
655         while (true) {
656             Status = NtFsControlFile(h, nullptr, nullptr, nullptr, &iosb, FSCTL_BTRFS_GET_DEVICES, nullptr, 0, devices, devsize);
657             if (Status == STATUS_BUFFER_OVERFLOW) {
658                 if (i < 8) {
659                     devsize += 1024;
660 
661                     free(devices);
662                     devices = (btrfs_device*)malloc(devsize);
663 
664                     i++;
665                 } else
666                     return;
667             } else
668                 break;
669         }
670 
671         if (!NT_SUCCESS(Status))
672             return;
673 
674         bd = devices;
675 
676         i = 0;
677         usagesize = 1024;
678 
679         usage = (btrfs_usage*)malloc(usagesize);
680 
681         while (true) {
682             Status = NtFsControlFile(h, nullptr, nullptr, nullptr, &iosb, FSCTL_BTRFS_GET_USAGE, nullptr, 0, usage, usagesize);
683             if (Status == STATUS_BUFFER_OVERFLOW) {
684                 if (i < 8) {
685                     usagesize += 1024;
686 
687                     free(usage);
688                     usage = (btrfs_usage*)malloc(usagesize);
689 
690                     i++;
691                 } else {
692                     free(usage);
693                     return;
694                 }
695             } else
696                 break;
697         }
698 
699         if (!NT_SUCCESS(Status)) {
700             free(usage);
701             return;
702         }
703     }
704 
705     SendMessageW(devlist, LVM_DELETEALLITEMS, 0, 0);
706 
707     num_rw_devices = 0;
708 
709     i = 0;
710     while (true) {
711         LVITEMW lvi;
712         wstring s, u;
713         uint64_t alloc;
714 
715         // ID
716 
717         RtlZeroMemory(&lvi, sizeof(LVITEMW));
718         lvi.mask = LVIF_TEXT | LVIF_PARAM;
719         lvi.iItem = (int)SendMessageW(devlist, LVM_GETITEMCOUNT, 0, 0);
720         lvi.lParam = (LPARAM)bd->dev_id;
721 
722         s = to_wstring(bd->dev_id);
723         lvi.pszText = (LPWSTR)s.c_str();
724 
725         SendMessageW(devlist, LVM_INSERTITEMW, 0, (LPARAM)&lvi);
726 
727         // description
728 
729         lvi.mask = LVIF_TEXT;
730         lvi.iSubItem = 1;
731 
732         if (bd->missing) {
733             if (!load_string(module, IDS_MISSING, s))
734                 throw last_error(GetLastError());
735         } else if (bd->device_number == 0xffffffff)
736             s = wstring(bd->name, bd->namelen / sizeof(WCHAR));
737         else if (bd->partition_number == 0) {
738             if (!load_string(module, IDS_DISK_NUM, u))
739                 throw last_error(GetLastError());
740 
741             wstring_sprintf(s, u, bd->device_number);
742         } else {
743             if (!load_string(module, IDS_DISK_PART_NUM, u))
744                 throw last_error(GetLastError());
745 
746             wstring_sprintf(s, u, bd->device_number, bd->partition_number);
747         }
748 
749         lvi.pszText = (LPWSTR)s.c_str();
750 
751         SendMessageW(devlist, LVM_SETITEMW, 0, (LPARAM)&lvi);
752 
753         // readonly
754 
755         lvi.iSubItem = 2;
756         load_string(module, bd->readonly ? IDS_DEVLIST_READONLY_YES : IDS_DEVLIST_READONLY_NO, s);
757         lvi.pszText = (LPWSTR)s.c_str();
758         SendMessageW(devlist, LVM_SETITEMW, 0, (LPARAM)&lvi);
759 
760         if (!bd->readonly)
761             num_rw_devices++;
762 
763         // size
764 
765         lvi.iSubItem = 3;
766         format_size(bd->size, s, false);
767         lvi.pszText = (LPWSTR)s.c_str();
768         SendMessageW(devlist, LVM_SETITEMW, 0, (LPARAM)&lvi);
769 
770         // alloc
771 
772         alloc = find_dev_alloc(bd->dev_id, usage);
773 
774         lvi.iSubItem = 4;
775         format_size(alloc, s, false);
776         lvi.pszText = (LPWSTR)s.c_str();
777         SendMessageW(devlist, LVM_SETITEMW, 0, (LPARAM)&lvi);
778 
779         // alloc %
780 
781         wstring_sprintf(s, L"%1.1f%%", (float)alloc * 100.0f / (float)bd->size);
782         lvi.iSubItem = 5;
783         lvi.pszText = (LPWSTR)s.c_str();
784         SendMessageW(devlist, LVM_SETITEMW, 0, (LPARAM)&lvi);
785 
786         i++;
787 
788         if (bd->next_entry > 0)
789             bd = (btrfs_device*)((uint8_t*)bd + bd->next_entry);
790         else
791             break;
792     }
793 
794     free(usage);
795 
796     SendMessageW(devlist, LVM_SORTITEMS, 0, (LPARAM)lv_sort);
797 
798     EnableWindow(GetDlgItem(GetParent(devlist), IDC_DEVICE_ADD), num_rw_devices > 0);
799     EnableWindow(GetDlgItem(GetParent(devlist), IDC_DEVICE_REMOVE), num_rw_devices > 1);
800 }
801 
802 void BtrfsVolPropSheet::ResetStats(HWND hwndDlg) {
803     wstring t, sel;
804     WCHAR modfn[MAX_PATH];
805     SHELLEXECUTEINFOW sei;
806 
807     sel = to_wstring(stats_dev);
808 
809     GetModuleFileNameW(module, modfn, sizeof(modfn) / sizeof(WCHAR));
810 
811 #ifndef __REACTOS__
812     t = L"\""s + modfn + L"\",ResetStats " + fn + L"|" + sel;
813 #else
814     t = wstring(L"\"") + modfn + wstring(L"\",ResetStats ") + fn + wstring(L"|") + sel;
815 #endif
816 
817     RtlZeroMemory(&sei, sizeof(sei));
818 
819     sei.cbSize = sizeof(sei);
820     sei.hwnd = hwndDlg;
821     sei.lpVerb = L"runas";
822     sei.lpFile = L"rundll32.exe";
823     sei.lpParameters = t.c_str();
824     sei.nShow = SW_SHOW;
825     sei.fMask = SEE_MASK_NOCLOSEPROCESS;
826 
827     if (!ShellExecuteExW(&sei))
828         throw last_error(GetLastError());
829 
830     WaitForSingleObject(sei.hProcess, INFINITE);
831     CloseHandle(sei.hProcess);
832 
833     win_handle h = CreateFileW(fn.c_str(), FILE_TRAVERSE | FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, nullptr,
834                                OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OPEN_REPARSE_POINT, nullptr);
835 
836     if (h != INVALID_HANDLE_VALUE) {
837         NTSTATUS Status;
838         IO_STATUS_BLOCK iosb;
839         ULONG devsize, i;
840 
841         i = 0;
842         devsize = 1024;
843 
844         free(devices);
845         devices = (btrfs_device*)malloc(devsize);
846 
847         while (true) {
848             Status = NtFsControlFile(h, nullptr, nullptr, nullptr, &iosb, FSCTL_BTRFS_GET_DEVICES, nullptr, 0, devices, devsize);
849             if (Status == STATUS_BUFFER_OVERFLOW) {
850                 if (i < 8) {
851                     devsize += 1024;
852 
853                     free(devices);
854                     devices = (btrfs_device*)malloc(devsize);
855 
856                     i++;
857                 } else
858                     break;
859             } else
860                 break;
861         }
862     }
863 
864     EndDialog(hwndDlg, 0);
865 }
866 
867 INT_PTR CALLBACK BtrfsVolPropSheet::StatsDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
868     try {
869         switch (uMsg) {
870             case WM_INITDIALOG:
871             {
872                 WCHAR s[255];
873                 wstring t;
874                 btrfs_device *bd, *dev = nullptr;
875                 int i;
876 
877                 static int stat_ids[] = { IDC_WRITE_ERRS, IDC_READ_ERRS, IDC_FLUSH_ERRS, IDC_CORRUPTION_ERRS, IDC_GENERATION_ERRS };
878 
879                 bd = devices;
880 
881                 while (true) {
882                     if (bd->dev_id == stats_dev) {
883                         dev = bd;
884                         break;
885                     }
886 
887                     if (bd->next_entry > 0)
888                         bd = (btrfs_device*)((uint8_t*)bd + bd->next_entry);
889                     else
890                         break;
891                 }
892 
893                 if (!dev) {
894                     EndDialog(hwndDlg, 0);
895                     throw string_error(IDS_CANNOT_FIND_DEVICE);
896                 }
897 
898                 GetDlgItemTextW(hwndDlg, IDC_DEVICE_ID, s, sizeof(s) / sizeof(WCHAR));
899 
900                 wstring_sprintf(t, s, dev->dev_id);
901 
902                 SetDlgItemTextW(hwndDlg, IDC_DEVICE_ID, t.c_str());
903 
904                 for (i = 0; i < 5; i++) {
905                     GetDlgItemTextW(hwndDlg, stat_ids[i], s, sizeof(s) / sizeof(WCHAR));
906 
907                     wstring_sprintf(t, s, dev->stats[i]);
908 
909                     SetDlgItemTextW(hwndDlg, stat_ids[i], t.c_str());
910                 }
911 
912                 SendMessageW(GetDlgItem(hwndDlg, IDC_RESET_STATS), BCM_SETSHIELD, 0, true);
913                 EnableWindow(GetDlgItem(hwndDlg, IDC_RESET_STATS), !readonly);
914 
915                 break;
916             }
917 
918             case WM_COMMAND:
919                 switch (HIWORD(wParam)) {
920                     case BN_CLICKED:
921                         switch (LOWORD(wParam)) {
922                             case IDOK:
923                             case IDCANCEL:
924                                 EndDialog(hwndDlg, 0);
925                             return true;
926 
927                             case IDC_RESET_STATS:
928                                 ResetStats(hwndDlg);
929                             return true;
930                         }
931                     break;
932                 }
933             break;
934         }
935     } catch (const exception& e) {
936         error_message(hwndDlg, e.what());
937     }
938 
939     return false;
940 }
941 
942 static INT_PTR CALLBACK stub_StatsDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
943     BtrfsVolPropSheet* bvps;
944 
945     if (uMsg == WM_INITDIALOG) {
946         SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)lParam);
947         bvps = (BtrfsVolPropSheet*)lParam;
948     } else {
949         bvps = (BtrfsVolPropSheet*)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
950     }
951 
952     if (bvps)
953         return bvps->StatsDlgProc(hwndDlg, uMsg, wParam, lParam);
954     else
955         return false;
956 }
957 
958 void BtrfsVolPropSheet::ShowStats(HWND hwndDlg, uint64_t devid) {
959     stats_dev = devid;
960 
961     DialogBoxParamW(module, MAKEINTRESOURCEW(IDD_DEVICE_STATS), hwndDlg, stub_StatsDlgProc, (LPARAM)this);
962 }
963 
964 INT_PTR CALLBACK BtrfsVolPropSheet::DeviceDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
965     try {
966         switch (uMsg) {
967             case WM_INITDIALOG:
968             {
969                 HWND devlist;
970                 RECT rect;
971                 ULONG w;
972 
973                 EnableThemeDialogTexture(hwndDlg, ETDT_ENABLETAB);
974 
975                 devlist = GetDlgItem(hwndDlg, IDC_DEVLIST);
976 
977                 GetClientRect(devlist, &rect);
978                 w = rect.right - rect.left;
979 
980                 add_lv_column(devlist, IDS_DEVLIST_ALLOC_PC, w * 5 / 44);
981                 add_lv_column(devlist, IDS_DEVLIST_ALLOC, w * 6 / 44);
982                 add_lv_column(devlist, IDS_DEVLIST_SIZE, w * 6 / 44);
983                 add_lv_column(devlist, IDS_DEVLIST_READONLY, w * 7 / 44);
984                 add_lv_column(devlist, IDS_DEVLIST_DESC, w * 16 / 44);
985                 add_lv_column(devlist, IDS_DEVLIST_ID, w * 4 / 44);
986 
987                 SendMessageW(GetDlgItem(hwndDlg, IDC_DEVICE_ADD), BCM_SETSHIELD, 0, true);
988                 SendMessageW(GetDlgItem(hwndDlg, IDC_DEVICE_REMOVE), BCM_SETSHIELD, 0, true);
989                 SendMessageW(GetDlgItem(hwndDlg, IDC_DEVICE_RESIZE), BCM_SETSHIELD, 0, true);
990 
991                 RefreshDevList(devlist);
992 
993                 break;
994             }
995 
996             case WM_COMMAND:
997                 switch (HIWORD(wParam)) {
998                     case BN_CLICKED:
999                         switch (LOWORD(wParam)) {
1000                             case IDOK:
1001                             case IDCANCEL:
1002                                 KillTimer(hwndDlg, 1);
1003                                 EndDialog(hwndDlg, 0);
1004                             return true;
1005 
1006                             case IDC_DEVICE_ADD:
1007                             {
1008                                 wstring t;
1009                                 WCHAR modfn[MAX_PATH];
1010                                 SHELLEXECUTEINFOW sei;
1011 
1012                                 GetModuleFileNameW(module, modfn, sizeof(modfn) / sizeof(WCHAR));
1013 
1014 #ifndef __REACTOS__
1015                                 t = L"\""s + modfn + L"\",AddDevice "s + fn;
1016 #else
1017                                 t = wstring(L"\"") + modfn + wstring(L"\",AddDevice ") + fn;
1018 #endif
1019 
1020                                 RtlZeroMemory(&sei, sizeof(sei));
1021 
1022                                 sei.cbSize = sizeof(sei);
1023                                 sei.hwnd = hwndDlg;
1024                                 sei.lpVerb = L"runas";
1025                                 sei.lpFile = L"rundll32.exe";
1026                                 sei.lpParameters = t.c_str();
1027                                 sei.nShow = SW_SHOW;
1028                                 sei.fMask = SEE_MASK_NOCLOSEPROCESS;
1029 
1030                                 if (!ShellExecuteExW(&sei))
1031                                     throw last_error(GetLastError());
1032 
1033                                 WaitForSingleObject(sei.hProcess, INFINITE);
1034                                 CloseHandle(sei.hProcess);
1035 
1036                                 RefreshDevList(GetDlgItem(hwndDlg, IDC_DEVLIST));
1037 
1038                                 return true;
1039                             }
1040 
1041                             case IDC_DEVICE_REFRESH:
1042                                 RefreshDevList(GetDlgItem(hwndDlg, IDC_DEVLIST));
1043                                 return true;
1044 
1045                             case IDC_DEVICE_SHOW_STATS:
1046                             {
1047                                 WCHAR sel[MAX_PATH];
1048                                 HWND devlist;
1049                                 LVITEMW lvi;
1050 
1051                                 devlist = GetDlgItem(hwndDlg, IDC_DEVLIST);
1052 
1053                                 auto index = SendMessageW(devlist, LVM_GETNEXTITEM, -1, LVNI_SELECTED);
1054 
1055                                 if (index == -1)
1056                                     return true;
1057 
1058                                 RtlZeroMemory(&lvi, sizeof(LVITEMW));
1059                                 lvi.mask = LVIF_TEXT;
1060                                 lvi.iItem = (int)index;
1061                                 lvi.iSubItem = 0;
1062                                 lvi.pszText = sel;
1063                                 lvi.cchTextMax = sizeof(sel) / sizeof(WCHAR);
1064                                 SendMessageW(devlist, LVM_GETITEMW, 0, (LPARAM)&lvi);
1065 
1066                                 ShowStats(hwndDlg, _wtoi(sel));
1067                                 return true;
1068                             }
1069 
1070                             case IDC_DEVICE_REMOVE:
1071                             {
1072                                 wstring t, mess, mess2, title;
1073                                 WCHAR modfn[MAX_PATH], sel[MAX_PATH], sel2[MAX_PATH];
1074                                 HWND devlist;
1075                                 SHELLEXECUTEINFOW sei;
1076                                 LVITEMW lvi;
1077 
1078                                 devlist = GetDlgItem(hwndDlg, IDC_DEVLIST);
1079 
1080                                 auto index = SendMessageW(devlist, LVM_GETNEXTITEM, -1, LVNI_SELECTED);
1081 
1082                                 if (index == -1)
1083                                     return true;
1084 
1085                                 RtlZeroMemory(&lvi, sizeof(LVITEMW));
1086                                 lvi.mask = LVIF_TEXT;
1087                                 lvi.iItem = (int)index;
1088                                 lvi.iSubItem = 0;
1089                                 lvi.pszText = sel;
1090                                 lvi.cchTextMax = sizeof(sel) / sizeof(WCHAR);
1091                                 SendMessageW(devlist, LVM_GETITEMW, 0, (LPARAM)&lvi);
1092 
1093                                 lvi.iSubItem = 1;
1094                                 lvi.pszText = sel2;
1095                                 lvi.cchTextMax = sizeof(sel2) / sizeof(WCHAR);
1096                                 SendMessageW(devlist, LVM_GETITEMW, 0, (LPARAM)&lvi);
1097 
1098                                 if (!load_string(module, IDS_REMOVE_DEVICE_CONFIRMATION, mess))
1099                                     throw last_error(GetLastError());
1100 
1101                                 wstring_sprintf(mess2, mess, sel, sel2);
1102 
1103                                 if (!load_string(module, IDS_CONFIRMATION_TITLE, title))
1104                                     throw last_error(GetLastError());
1105 
1106                                 if (MessageBoxW(hwndDlg, mess2.c_str(), title.c_str(), MB_YESNO) != IDYES)
1107                                     return true;
1108 
1109                                 GetModuleFileNameW(module, modfn, sizeof(modfn) / sizeof(WCHAR));
1110 
1111 #ifndef __REACTOS__
1112                                 t = L"\""s + modfn + L"\",RemoveDevice "s + fn + L"|"s + sel;
1113 #else
1114                                 t = wstring(L"\"") + modfn + wstring(L"\",RemoveDevice ") + fn + wstring(L"|") + sel;
1115 #endif
1116 
1117                                 RtlZeroMemory(&sei, sizeof(sei));
1118 
1119                                 sei.cbSize = sizeof(sei);
1120                                 sei.hwnd = hwndDlg;
1121                                 sei.lpVerb = L"runas";
1122                                 sei.lpFile = L"rundll32.exe";
1123                                 sei.lpParameters = t.c_str();
1124                                 sei.nShow = SW_SHOW;
1125                                 sei.fMask = SEE_MASK_NOCLOSEPROCESS;
1126 
1127                                 if (!ShellExecuteExW(&sei))
1128                                     throw last_error(GetLastError());
1129 
1130                                 WaitForSingleObject(sei.hProcess, INFINITE);
1131                                 CloseHandle(sei.hProcess);
1132 
1133                                 RefreshDevList(GetDlgItem(hwndDlg, IDC_DEVLIST));
1134 
1135                                 return true;
1136                             }
1137 
1138                             case IDC_DEVICE_RESIZE:
1139                             {
1140                                 HWND devlist;
1141                                 LVITEMW lvi;
1142                                 wstring t;
1143                                 WCHAR modfn[MAX_PATH], sel[100];
1144                                 SHELLEXECUTEINFOW sei;
1145 
1146                                 devlist = GetDlgItem(hwndDlg, IDC_DEVLIST);
1147 
1148                                 auto index = SendMessageW(devlist, LVM_GETNEXTITEM, -1, LVNI_SELECTED);
1149 
1150                                 if (index == -1)
1151                                     return true;
1152 
1153                                 RtlZeroMemory(&lvi, sizeof(LVITEMW));
1154                                 lvi.mask = LVIF_TEXT;
1155                                 lvi.iItem = (int)index;
1156                                 lvi.iSubItem = 0;
1157                                 lvi.pszText = sel;
1158                                 lvi.cchTextMax = sizeof(sel) / sizeof(WCHAR);
1159                                 SendMessageW(devlist, LVM_GETITEMW, 0, (LPARAM)&lvi);
1160 
1161                                 GetModuleFileNameW(module, modfn, sizeof(modfn) / sizeof(WCHAR));
1162 
1163 #ifndef __REACTOS__
1164                                 t = L"\""s + modfn + L"\",ResizeDevice "s + fn + L"|"s + sel;
1165 #else
1166                                 t = wstring(L"\"") + modfn + wstring(L"\",ResizeDevice ") + fn + wstring(L"|") + sel;
1167 #endif
1168 
1169                                 RtlZeroMemory(&sei, sizeof(sei));
1170 
1171                                 sei.cbSize = sizeof(sei);
1172                                 sei.hwnd = hwndDlg;
1173                                 sei.lpVerb = L"runas";
1174                                 sei.lpFile = L"rundll32.exe";
1175                                 sei.lpParameters = t.c_str();
1176                                 sei.nShow = SW_SHOW;
1177                                 sei.fMask = SEE_MASK_NOCLOSEPROCESS;
1178 
1179                                 if (!ShellExecuteExW(&sei))
1180                                     throw last_error(GetLastError());
1181 
1182                                 WaitForSingleObject(sei.hProcess, INFINITE);
1183                                 CloseHandle(sei.hProcess);
1184 
1185                                 RefreshDevList(GetDlgItem(hwndDlg, IDC_DEVLIST));
1186                             }
1187                         }
1188                     break;
1189                 }
1190             break;
1191 
1192             case WM_NOTIFY:
1193                 switch (((LPNMHDR)lParam)->code) {
1194                     case LVN_ITEMCHANGED:
1195                     {
1196                         NMLISTVIEW* nmv = (NMLISTVIEW*)lParam;
1197 
1198                         EnableWindow(GetDlgItem(hwndDlg, IDC_DEVICE_SHOW_STATS), nmv->uNewState & LVIS_SELECTED);
1199 
1200                         if (nmv->uNewState & LVIS_SELECTED && !readonly) {
1201                             HWND devlist;
1202                             btrfs_device* bd;
1203                             bool device_readonly = false;
1204                             LVITEMW lvi;
1205                             WCHAR sel[MAX_PATH];
1206                             uint64_t devid;
1207 
1208                             devlist = GetDlgItem(hwndDlg, IDC_DEVLIST);
1209 
1210                             RtlZeroMemory(&lvi, sizeof(LVITEMW));
1211                             lvi.mask = LVIF_TEXT;
1212                             lvi.iItem = nmv->iItem;
1213                             lvi.iSubItem = 0;
1214                             lvi.pszText = sel;
1215                             lvi.cchTextMax = sizeof(sel) / sizeof(WCHAR);
1216                             SendMessageW(devlist, LVM_GETITEMW, 0, (LPARAM)&lvi);
1217                             devid = _wtoi(sel);
1218 
1219                             bd = devices;
1220 
1221                             while (true) {
1222                                 if (bd->dev_id == devid) {
1223                                     device_readonly = bd->readonly;
1224                                     break;
1225                                 }
1226 
1227                                 if (bd->next_entry > 0)
1228                                     bd = (btrfs_device*)((uint8_t*)bd + bd->next_entry);
1229                                 else
1230                                     break;
1231                             }
1232 
1233                             EnableWindow(GetDlgItem(hwndDlg, IDC_DEVICE_RESIZE), !device_readonly);
1234                         } else
1235                             EnableWindow(GetDlgItem(hwndDlg, IDC_DEVICE_RESIZE), false);
1236 
1237                         break;
1238                     }
1239                 }
1240             break;
1241         }
1242     } catch (const exception& e) {
1243         error_message(hwndDlg, e.what());
1244     }
1245 
1246     return false;
1247 }
1248 
1249 static INT_PTR CALLBACK stub_DeviceDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
1250     BtrfsVolPropSheet* bvps;
1251 
1252     if (uMsg == WM_INITDIALOG) {
1253         SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)lParam);
1254         bvps = (BtrfsVolPropSheet*)lParam;
1255     } else {
1256         bvps = (BtrfsVolPropSheet*)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
1257     }
1258 
1259     if (bvps)
1260         return bvps->DeviceDlgProc(hwndDlg, uMsg, wParam, lParam);
1261     else
1262         return false;
1263 }
1264 
1265 void BtrfsVolPropSheet::ShowDevices(HWND hwndDlg) {
1266    DialogBoxParamW(module, MAKEINTRESOURCEW(IDD_DEVICES), hwndDlg, stub_DeviceDlgProc, (LPARAM)this);
1267 }
1268 
1269 void BtrfsVolPropSheet::ShowScrub(HWND hwndDlg) {
1270     wstring t;
1271     WCHAR modfn[MAX_PATH];
1272     SHELLEXECUTEINFOW sei;
1273 
1274     GetModuleFileNameW(module, modfn, sizeof(modfn) / sizeof(WCHAR));
1275 
1276 #ifndef __REACTOS__
1277     t = L"\""s + modfn + L"\",ShowScrub "s + fn;
1278 #else
1279     t = wstring(L"\"") + modfn + wstring(L"\",ShowScrub ") + fn;
1280 #endif
1281 
1282     RtlZeroMemory(&sei, sizeof(sei));
1283 
1284     sei.cbSize = sizeof(sei);
1285     sei.hwnd = hwndDlg;
1286     sei.lpVerb = L"runas";
1287     sei.lpFile = L"rundll32.exe";
1288     sei.lpParameters = t.c_str();
1289     sei.nShow = SW_SHOW;
1290     sei.fMask = SEE_MASK_NOCLOSEPROCESS;
1291 
1292     if (!ShellExecuteExW(&sei))
1293         throw last_error(GetLastError());
1294 
1295     WaitForSingleObject(sei.hProcess, INFINITE);
1296     CloseHandle(sei.hProcess);
1297 }
1298 
1299 void BtrfsVolPropSheet::ShowChangeDriveLetter(HWND hwndDlg) {
1300     wstring t;
1301     WCHAR modfn[MAX_PATH];
1302     SHELLEXECUTEINFOW sei;
1303 
1304     GetModuleFileNameW(module, modfn, sizeof(modfn) / sizeof(WCHAR));
1305 
1306 #ifndef __REACTOS__
1307     t = L"\""s + modfn + L"\",ShowChangeDriveLetter "s + fn;
1308 #else
1309     t = wstring(L"\"") + modfn + wstring(L"\",ShowChangeDriveLetter ") + fn;
1310 #endif
1311 
1312     RtlZeroMemory(&sei, sizeof(sei));
1313 
1314     sei.cbSize = sizeof(sei);
1315     sei.hwnd = hwndDlg;
1316     sei.lpVerb = L"runas";
1317     sei.lpFile = L"rundll32.exe";
1318     sei.lpParameters = t.c_str();
1319     sei.nShow = SW_SHOW;
1320     sei.fMask = SEE_MASK_NOCLOSEPROCESS;
1321 
1322     if (!ShellExecuteExW(&sei))
1323         throw last_error(GetLastError());
1324 
1325     WaitForSingleObject(sei.hProcess, INFINITE);
1326     CloseHandle(sei.hProcess);
1327 }
1328 
1329 static INT_PTR CALLBACK PropSheetDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
1330     try {
1331         switch (uMsg) {
1332             case WM_INITDIALOG:
1333             {
1334                 PROPSHEETPAGE* psp = (PROPSHEETPAGE*)lParam;
1335                 BtrfsVolPropSheet* bps = (BtrfsVolPropSheet*)psp->lParam;
1336                 btrfs_device* bd;
1337 
1338                 EnableThemeDialogTexture(hwndDlg, ETDT_ENABLETAB);
1339 
1340                 SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)bps);
1341 
1342                 bps->readonly = true;
1343                 bd = bps->devices;
1344 
1345                 while (true) {
1346                     if (!bd->readonly) {
1347                         bps->readonly = false;
1348                         break;
1349                     }
1350 
1351                     if (bd->next_entry > 0)
1352                         bd = (btrfs_device*)((uint8_t*)bd + bd->next_entry);
1353                     else
1354                         break;
1355                 }
1356 
1357                 if (bps->uuid_set) {
1358                     WCHAR s[255];
1359                     wstring t;
1360 
1361                     GetDlgItemTextW(hwndDlg, IDC_UUID, s, sizeof(s) / sizeof(WCHAR));
1362 
1363                     wstring_sprintf(t, s, bps->uuid.uuid[0], bps->uuid.uuid[1], bps->uuid.uuid[2], bps->uuid.uuid[3], bps->uuid.uuid[4], bps->uuid.uuid[5],
1364                                     bps->uuid.uuid[6], bps->uuid.uuid[7], bps->uuid.uuid[8], bps->uuid.uuid[9], bps->uuid.uuid[10], bps->uuid.uuid[11],
1365                                     bps->uuid.uuid[12], bps->uuid.uuid[13], bps->uuid.uuid[14], bps->uuid.uuid[15]);
1366 
1367                     SetDlgItemTextW(hwndDlg, IDC_UUID, t.c_str());
1368                 } else
1369                     SetDlgItemTextW(hwndDlg, IDC_UUID, L"");
1370 
1371                 SendMessageW(GetDlgItem(hwndDlg, IDC_VOL_SCRUB), BCM_SETSHIELD, 0, true);
1372                 SendMessageW(GetDlgItem(hwndDlg, IDC_VOL_CHANGE_DRIVE_LETTER), BCM_SETSHIELD, 0, true);
1373 
1374                 return false;
1375             }
1376 
1377             case WM_NOTIFY:
1378             {
1379                 switch (((LPNMHDR)lParam)->code) {
1380                     case PSN_KILLACTIVE:
1381                         SetWindowLongPtrW(hwndDlg, DWLP_MSGRESULT, false);
1382                     break;
1383                 }
1384                 break;
1385             }
1386 
1387             case WM_COMMAND:
1388             {
1389                 BtrfsVolPropSheet* bps = (BtrfsVolPropSheet*)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
1390 
1391                 if (bps) {
1392                     switch (HIWORD(wParam)) {
1393                         case BN_CLICKED: {
1394                             switch (LOWORD(wParam)) {
1395                                 case IDC_VOL_SHOW_USAGE:
1396                                     bps->ShowUsage(hwndDlg);
1397                                 break;
1398 
1399                                 case IDC_VOL_BALANCE:
1400                                     bps->balance->ShowBalance(hwndDlg);
1401                                 break;
1402 
1403                                 case IDC_VOL_DEVICES:
1404                                     bps->ShowDevices(hwndDlg);
1405                                 break;
1406 
1407                                 case IDC_VOL_SCRUB:
1408                                     bps->ShowScrub(hwndDlg);
1409                                 break;
1410 
1411                                 case IDC_VOL_CHANGE_DRIVE_LETTER:
1412                                     bps->ShowChangeDriveLetter(hwndDlg);
1413                                 break;
1414                             }
1415                         }
1416                     }
1417                 }
1418 
1419                 break;
1420             }
1421         }
1422     } catch (const exception& e) {
1423         error_message(hwndDlg, e.what());
1424     }
1425 
1426     return false;
1427 }
1428 
1429 HRESULT __stdcall BtrfsVolPropSheet::AddPages(LPFNADDPROPSHEETPAGE pfnAddPage, LPARAM lParam) {
1430     try {
1431         PROPSHEETPAGE psp;
1432         HPROPSHEETPAGE hPage;
1433         INITCOMMONCONTROLSEX icex;
1434 
1435         if (ignore)
1436             return S_OK;
1437 
1438         icex.dwSize = sizeof(icex);
1439         icex.dwICC = ICC_LINK_CLASS;
1440 
1441         if (!InitCommonControlsEx(&icex))
1442             throw string_error(IDS_INITCOMMONCONTROLSEX_FAILED);
1443 
1444         psp.dwSize = sizeof(psp);
1445         psp.dwFlags = PSP_USEREFPARENT | PSP_USETITLE;
1446         psp.hInstance = module;
1447         psp.pszTemplate = MAKEINTRESOURCE(IDD_VOL_PROP_SHEET);
1448         psp.hIcon = 0;
1449         psp.pszTitle = MAKEINTRESOURCE(IDS_VOL_PROP_SHEET_TITLE);
1450         psp.pfnDlgProc = (DLGPROC)PropSheetDlgProc;
1451         psp.pcRefParent = (UINT*)&objs_loaded;
1452         psp.pfnCallback = nullptr;
1453         psp.lParam = (LPARAM)this;
1454 
1455         hPage = CreatePropertySheetPage(&psp);
1456 
1457         if (hPage) {
1458             if (pfnAddPage(hPage, lParam)) {
1459                 this->AddRef();
1460                 return S_OK;
1461             } else
1462                 DestroyPropertySheetPage(hPage);
1463         } else
1464             return E_OUTOFMEMORY;
1465     } catch (const exception& e) {
1466         error_message(nullptr, e.what());
1467     }
1468 
1469     return E_FAIL;
1470 }
1471 
1472 HRESULT __stdcall BtrfsVolPropSheet::ReplacePage(UINT uPageID, LPFNADDPROPSHEETPAGE pfnReplacePage, LPARAM lParam) {
1473     return S_OK;
1474 }
1475 
1476 void BtrfsChangeDriveLetter::do_change(HWND hwndDlg) {
1477     unsigned int sel = (unsigned int)SendDlgItemMessageW(hwndDlg, IDC_DRIVE_LETTER_COMBO, CB_GETCURSEL, 0, 0);
1478 
1479     if (sel >= 0 && sel < letters.size()) {
1480         wstring dd;
1481 
1482         if (fn.length() == 3 && fn[1] == L':' && fn[2] == L'\\') {
1483             dd = L"\\DosDevices\\?:";
1484 
1485             dd[12] = fn[0];
1486         } else
1487 #ifndef __REACTOS__
1488             throw runtime_error("Volume path was not root of drive.");
1489 #else
1490             error_message(nullptr, "Volume path was not root of drive.");
1491 #endif
1492 
1493         mountmgr mm;
1494         wstring dev_name;
1495 
1496         {
1497             auto v = mm.query_points(dd);
1498 
1499             if (v.empty())
1500 #ifndef __REACTOS__
1501                 throw runtime_error("Error finding device name.");
1502 #else
1503                 error_message(nullptr, "Error finding device name.");
1504 #endif
1505 
1506             dev_name = v[0].device_name;
1507         }
1508 
1509         wstring new_dd = L"\\DosDevices\\?:";
1510         new_dd[12] = letters[sel];
1511 
1512         mm.delete_points(dd);
1513 
1514         try {
1515             mm.create_point(new_dd, dev_name);
1516         } catch (...) {
1517             // if fails, try to recreate old symlink, so we're not left with no drive letter at all
1518             mm.create_point(dd, dev_name);
1519             throw;
1520         }
1521     }
1522 
1523     EndDialog(hwndDlg, 1);
1524 }
1525 
1526 INT_PTR BtrfsChangeDriveLetter::DlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
1527     try {
1528         switch (uMsg) {
1529             case WM_INITDIALOG:
1530             {
1531                 HWND cb = GetDlgItem(hwndDlg, IDC_DRIVE_LETTER_COMBO);
1532 
1533                 SendMessageW(cb, CB_RESETCONTENT, 0, 0);
1534 
1535                 mountmgr mm;
1536                 wstring drv;
1537 
1538                 drv = L"\\DosDevices\\?:";
1539 
1540                 for (wchar_t l = 'A'; l <= 'Z'; l++) {
1541                     bool found = true;
1542 
1543                     drv[12] = l;
1544 
1545                     try {
1546                         auto v = mm.query_points(drv);
1547 
1548                         if (v.empty())
1549                             found = false;
1550                     } catch (const ntstatus_error& ntstatus) {
1551                         if (ntstatus.Status == STATUS_OBJECT_NAME_NOT_FOUND)
1552                             found = false;
1553                         else
1554                             throw;
1555                     }
1556 
1557                     if (!found) {
1558                         wstring str = L"?:";
1559 
1560                         str[0] = l;
1561                         letters.push_back(l);
1562 
1563                         SendMessageW(cb, CB_ADDSTRING, 0, reinterpret_cast<LPARAM>(str.c_str()));
1564                     }
1565                 }
1566 
1567                 break;
1568             }
1569 
1570             case WM_COMMAND:
1571                 switch (HIWORD(wParam)) {
1572                     case BN_CLICKED:
1573                         switch (LOWORD(wParam)) {
1574                             case IDOK:
1575                                 do_change(hwndDlg);
1576                                 return true;
1577 
1578                             case IDCANCEL:
1579                                 EndDialog(hwndDlg, 0);
1580                                 return true;
1581                         }
1582                         break;
1583                 }
1584             break;
1585         }
1586     } catch (const exception& e) {
1587         error_message(hwndDlg, e.what());
1588     }
1589 
1590     return false;
1591 }
1592 
1593 #ifdef __REACTOS__
1594 INT_PTR CALLBACK VolPropSheetDlgproc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
1595 {
1596     BtrfsChangeDriveLetter* bcdl;
1597 
1598     if (uMsg == WM_INITDIALOG) {
1599         SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)lParam);
1600         bcdl = (BtrfsChangeDriveLetter*)lParam;
1601     } else
1602         bcdl = (BtrfsChangeDriveLetter*)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
1603 
1604     return bcdl->DlgProc(hwndDlg, uMsg, wParam, lParam);
1605 }
1606 #endif
1607 
1608 void BtrfsChangeDriveLetter::show() {
1609 #ifndef __REACTOS__
1610     DialogBoxParamW(module, MAKEINTRESOURCEW(IDD_DRIVE_LETTER), hwnd, [](HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) {
1611         BtrfsChangeDriveLetter* bcdl;
1612 
1613         if (uMsg == WM_INITDIALOG) {
1614             SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)lParam);
1615             bcdl = (BtrfsChangeDriveLetter*)lParam;
1616         } else
1617             bcdl = (BtrfsChangeDriveLetter*)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
1618 
1619         return bcdl->DlgProc(hwndDlg, uMsg, wParam, lParam);
1620     }, (LPARAM)this);
1621 #else
1622     DialogBoxParamW(module, MAKEINTRESOURCEW(IDD_DRIVE_LETTER), hwnd, VolPropSheetDlgproc, (LPARAM)this);
1623 #endif
1624 }
1625 
1626 #ifdef __cplusplus
1627 extern "C" {
1628 #endif
1629 
1630 void CALLBACK ResetStatsW(HWND hwnd, HINSTANCE hinst, LPWSTR lpszCmdLine, int nCmdShow) {
1631     try {
1632         win_handle token;
1633         NTSTATUS Status;
1634         TOKEN_PRIVILEGES tp;
1635         LUID luid;
1636         uint64_t devid;
1637         wstring cmdline, vol, dev;
1638         size_t pipe;
1639         IO_STATUS_BLOCK iosb;
1640 
1641         set_dpi_aware();
1642 
1643         cmdline = lpszCmdLine;
1644 
1645         if (!OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &token))
1646             throw last_error(GetLastError());
1647 
1648         if (!LookupPrivilegeValueW(nullptr, L"SeManageVolumePrivilege", &luid))
1649             throw last_error(GetLastError());
1650 
1651         tp.PrivilegeCount = 1;
1652         tp.Privileges[0].Luid = luid;
1653         tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
1654 
1655         if (!AdjustTokenPrivileges(token, false, &tp, sizeof(TOKEN_PRIVILEGES), nullptr, nullptr))
1656             throw last_error(GetLastError());
1657 
1658         pipe = cmdline.find(L"|");
1659 
1660         if (pipe == string::npos)
1661             return;
1662 
1663         vol = cmdline.substr(0, pipe);
1664         dev = cmdline.substr(pipe + 1);
1665 
1666         devid = _wtoi(dev.c_str());
1667         if (devid == 0)
1668             return;
1669 
1670         win_handle h = CreateFileW(vol.c_str(), FILE_TRAVERSE | FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, nullptr,
1671                                 OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OPEN_REPARSE_POINT, nullptr);
1672 
1673         if (h == INVALID_HANDLE_VALUE)
1674             throw last_error(GetLastError());
1675 
1676         Status = NtFsControlFile(h, nullptr, nullptr, nullptr, &iosb, FSCTL_BTRFS_RESET_STATS, &devid, sizeof(uint64_t), nullptr, 0);
1677         if (!NT_SUCCESS(Status))
1678             throw ntstatus_error(Status);
1679     } catch (const exception& e) {
1680         error_message(hwnd, e.what());
1681     }
1682 }
1683 
1684 void CALLBACK ShowChangeDriveLetterW(HWND hwnd, HINSTANCE hinst, LPWSTR lpszCmdLine, int nCmdShow) {
1685     BtrfsChangeDriveLetter bcdl(hwnd, lpszCmdLine);
1686 
1687     bcdl.show();
1688 }
1689 
1690 #ifdef __cplusplus
1691 }
1692 #endif
1693