1 /* Copyright (c) Mark Harmstone 2017 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 #include "shellext.h" 19 #include "send.h" 20 #include "resource.h" 21 #include <stddef.h> 22 #include <shlobj.h> 23 #ifdef __REACTOS__ 24 #undef DeleteFile 25 #endif 26 27 #define SEND_BUFFER_LEN 1048576 28 29 void BtrfsSend::ShowSendError(UINT msg, ...) { 30 WCHAR s[1024], t[1024]; 31 va_list ap; 32 33 if (!LoadStringW(module, msg, s, sizeof(s) / sizeof(WCHAR))) { 34 ShowError(hwnd, GetLastError()); 35 return; 36 } 37 38 va_start(ap, msg); 39 #ifndef __REACTOS__ 40 vswprintf(t, sizeof(t) / sizeof(WCHAR), s, ap); 41 #else 42 vsnwprintf(t, sizeof(t) / sizeof(WCHAR), s, ap); 43 #endif 44 45 SetDlgItemTextW(hwnd, IDC_SEND_STATUS, t); 46 47 va_end(ap); 48 } 49 50 DWORD BtrfsSend::Thread() { 51 NTSTATUS Status; 52 IO_STATUS_BLOCK iosb; 53 btrfs_send_subvol* bss; 54 btrfs_send_header header; 55 btrfs_send_command end; 56 BOOL success = FALSE; 57 ULONG bss_size, i; 58 59 buf = (char*)malloc(SEND_BUFFER_LEN); 60 61 dirh = CreateFileW(subvol.c_str(), FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); 62 if (dirh == INVALID_HANDLE_VALUE) { 63 ShowSendError(IDS_SEND_CANT_OPEN_DIR, subvol.c_str(), GetLastError(), format_message(GetLastError()).c_str()); 64 goto end3; 65 } 66 67 bss_size = offsetof(btrfs_send_subvol, clones[0]) + (clones.size() * sizeof(HANDLE)); 68 bss = (btrfs_send_subvol*)malloc(bss_size); 69 memset(bss, 0, bss_size); 70 71 if (incremental) { 72 WCHAR parent[MAX_PATH]; 73 HANDLE parenth; 74 75 parent[0] = 0; 76 77 GetDlgItemTextW(hwnd, IDC_PARENT_SUBVOL, parent, sizeof(parent) / sizeof(WCHAR)); 78 79 parenth = CreateFileW(parent, FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); 80 if (parenth == INVALID_HANDLE_VALUE) { 81 ShowSendError(IDS_SEND_CANT_OPEN_DIR, parent, GetLastError(), format_message(GetLastError()).c_str()); 82 goto end2; 83 } 84 85 bss->parent = parenth; 86 } else 87 bss->parent = NULL; 88 89 bss->num_clones = clones.size(); 90 91 for (i = 0; i < bss->num_clones; i++) { 92 HANDLE h; 93 94 h = CreateFileW(clones[i].c_str(), FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); 95 if (h == INVALID_HANDLE_VALUE) { 96 ULONG j; 97 98 ShowSendError(IDS_SEND_CANT_OPEN_DIR, clones[i].c_str(), GetLastError(), format_message(GetLastError()).c_str()); 99 100 for (j = 0; j < i; j++) { 101 CloseHandle(bss->clones[j]); 102 } 103 104 if (bss->parent) CloseHandle(bss->parent); 105 goto end2; 106 } 107 108 bss->clones[i] = h; 109 } 110 111 Status = NtFsControlFile(dirh, NULL, NULL, NULL, &iosb, FSCTL_BTRFS_SEND_SUBVOL, bss, bss_size, NULL, 0); 112 113 for (i = 0; i < bss->num_clones; i++) { 114 CloseHandle(bss->clones[i]); 115 } 116 117 if (!NT_SUCCESS(Status)) { 118 if (Status == (NTSTATUS)STATUS_INVALID_PARAMETER) { 119 BY_HANDLE_FILE_INFORMATION fileinfo; 120 if (!GetFileInformationByHandle(dirh, &fileinfo)) { 121 ShowSendError(IDS_SEND_GET_FILE_INFO_FAILED, GetLastError(), format_message(GetLastError()).c_str()); 122 if (bss->parent) CloseHandle(bss->parent); 123 goto end2; 124 } 125 126 if (!(fileinfo.dwFileAttributes & FILE_ATTRIBUTE_READONLY)) { 127 ShowSendError(IDS_SEND_NOT_READONLY); 128 if (bss->parent) CloseHandle(bss->parent); 129 goto end2; 130 } 131 132 if (bss->parent) { 133 if (!GetFileInformationByHandle(bss->parent, &fileinfo)) { 134 ShowSendError(IDS_SEND_GET_FILE_INFO_FAILED, GetLastError(), format_message(GetLastError()).c_str()); 135 CloseHandle(bss->parent); 136 goto end2; 137 } 138 139 if (!(fileinfo.dwFileAttributes & FILE_ATTRIBUTE_READONLY)) { 140 ShowSendError(IDS_SEND_PARENT_NOT_READONLY); 141 CloseHandle(bss->parent); 142 goto end2; 143 } 144 } 145 } 146 147 ShowSendError(IDS_SEND_FSCTL_BTRFS_SEND_SUBVOL_FAILED, Status, format_ntstatus(Status).c_str()); 148 if (bss->parent) CloseHandle(bss->parent); 149 goto end2; 150 } 151 152 if (bss->parent) CloseHandle(bss->parent); 153 154 stream = CreateFileW(file, FILE_WRITE_DATA | DELETE, 0, NULL, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); 155 if (stream == INVALID_HANDLE_VALUE) { 156 ShowSendError(IDS_SEND_CANT_OPEN_FILE, file, GetLastError(), format_message(GetLastError()).c_str()); 157 goto end2; 158 } 159 160 memcpy(header.magic, BTRFS_SEND_MAGIC, sizeof(BTRFS_SEND_MAGIC)); 161 header.version = 1; 162 163 if (!WriteFile(stream, &header, sizeof(header), NULL, NULL)) { 164 ShowSendError(IDS_SEND_WRITEFILE_FAILED, GetLastError(), format_message(GetLastError()).c_str()); 165 goto end; 166 } 167 168 do { 169 Status = NtFsControlFile(dirh, NULL, NULL, NULL, &iosb, FSCTL_BTRFS_READ_SEND_BUFFER, NULL, 0, buf, SEND_BUFFER_LEN); 170 171 if (NT_SUCCESS(Status)) { 172 if (!WriteFile(stream, buf, iosb.Information, NULL, NULL)) 173 ShowSendError(IDS_SEND_WRITEFILE_FAILED, GetLastError(), format_message(GetLastError()).c_str()); 174 } 175 } while (NT_SUCCESS(Status)); 176 177 if (Status != STATUS_END_OF_FILE) { 178 ShowSendError(IDS_SEND_FSCTL_BTRFS_READ_SEND_BUFFER_FAILED, Status, format_ntstatus(Status).c_str()); 179 goto end; 180 } 181 182 end.length = 0; 183 end.cmd = BTRFS_SEND_CMD_END; 184 end.csum = 0x9dc96c50; 185 186 if (!WriteFile(stream, &end, sizeof(end), NULL, NULL)) { 187 ShowSendError(IDS_SEND_WRITEFILE_FAILED, GetLastError(), format_message(GetLastError()).c_str()); 188 goto end; 189 } 190 191 SetEndOfFile(stream); 192 193 ShowSendError(IDS_SEND_SUCCESS); 194 success = TRUE; 195 196 end: 197 if (!success) { 198 FILE_DISPOSITION_INFO fdi; 199 200 fdi.DeleteFile = TRUE; 201 202 SetFileInformationByHandle(stream, FileDispositionInfo, &fdi, sizeof(FILE_DISPOSITION_INFO)); 203 } 204 205 CloseHandle(stream); 206 stream = INVALID_HANDLE_VALUE; 207 208 end2: 209 CloseHandle(dirh); 210 dirh = INVALID_HANDLE_VALUE; 211 212 end3: 213 free(buf); 214 buf = NULL; 215 216 started = FALSE; 217 218 SetDlgItemTextW(hwnd, IDCANCEL, closetext); 219 EnableWindow(GetDlgItem(hwnd, IDOK), TRUE); 220 EnableWindow(GetDlgItem(hwnd, IDC_STREAM_DEST), TRUE); 221 EnableWindow(GetDlgItem(hwnd, IDC_BROWSE), TRUE); 222 223 return 0; 224 } 225 226 static DWORD WINAPI send_thread(LPVOID lpParameter) { 227 BtrfsSend* bs = (BtrfsSend*)lpParameter; 228 229 return bs->Thread(); 230 } 231 232 void BtrfsSend::StartSend(HWND hwnd) { 233 WCHAR s[255]; 234 HWND cl; 235 ULONG num_clones; 236 237 if (started) 238 return; 239 240 GetDlgItemTextW(hwnd, IDC_STREAM_DEST, file, sizeof(file) / sizeof(WCHAR)); 241 242 if (file[0] == 0) 243 return; 244 245 if (incremental) { 246 WCHAR parent[MAX_PATH]; 247 248 GetDlgItemTextW(hwnd, IDC_PARENT_SUBVOL, parent, sizeof(parent) / sizeof(WCHAR)); 249 250 if (parent[0] == 0) 251 return; 252 } 253 254 started = TRUE; 255 ShowSendError(IDS_SEND_WRITING); 256 257 LoadStringW(module, IDS_SEND_CANCEL, s, sizeof(s) / sizeof(WCHAR)); 258 SetDlgItemTextW(hwnd, IDCANCEL, s); 259 260 EnableWindow(GetDlgItem(hwnd, IDOK), FALSE); 261 EnableWindow(GetDlgItem(hwnd, IDC_STREAM_DEST), FALSE); 262 EnableWindow(GetDlgItem(hwnd, IDC_BROWSE), FALSE); 263 264 clones.clear(); 265 266 cl = GetDlgItem(hwnd, IDC_CLONE_LIST); 267 num_clones = SendMessageW(cl, LB_GETCOUNT, 0, 0); 268 269 if ((LRESULT)num_clones != LB_ERR) { 270 ULONG i; 271 272 for (i = 0; i < num_clones; i++) { 273 WCHAR* s; 274 ULONG len; 275 276 len = SendMessageW(cl, LB_GETTEXTLEN, i, 0); 277 s = (WCHAR*)malloc((len + 1) * sizeof(WCHAR)); 278 279 SendMessageW(cl, LB_GETTEXT, i, (LPARAM)s); 280 281 clones.push_back(s); 282 283 free(s); 284 } 285 } 286 287 thread = CreateThread(NULL, 0, send_thread, this, 0, NULL); 288 289 if (!thread) 290 ShowError(NULL, GetLastError()); 291 } 292 293 void BtrfsSend::Browse(HWND hwnd) { 294 OPENFILENAMEW ofn; 295 296 file[0] = 0; 297 298 memset(&ofn, 0, sizeof(OPENFILENAMEW)); 299 ofn.lStructSize = sizeof(OPENFILENAMEW); 300 ofn.hwndOwner = hwnd; 301 ofn.hInstance = module; 302 ofn.lpstrFile = file; 303 ofn.nMaxFile = sizeof(file) / sizeof(WCHAR); 304 305 if (!GetSaveFileNameW(&ofn)) 306 return; 307 308 SetDlgItemTextW(hwnd, IDC_STREAM_DEST, file); 309 } 310 311 void BtrfsSend::BrowseParent(HWND hwnd) { 312 BROWSEINFOW bi; 313 PIDLIST_ABSOLUTE root, pidl; 314 HRESULT hr; 315 WCHAR parent[MAX_PATH], volpathw[MAX_PATH]; 316 HANDLE h; 317 NTSTATUS Status; 318 IO_STATUS_BLOCK iosb; 319 btrfs_get_file_ids bgfi; 320 321 if (!GetVolumePathNameW(subvol.c_str(), volpathw, (sizeof(volpathw) / sizeof(WCHAR)) - 1)) { 322 ShowStringError(hwnd, IDS_RECV_GETVOLUMEPATHNAME_FAILED, GetLastError(), format_message(GetLastError()).c_str()); 323 return; 324 } 325 326 hr = SHParseDisplayName(volpathw, 0, &root, 0, 0); 327 if (FAILED(hr)) { 328 ShowStringError(hwnd, IDS_SHPARSEDISPLAYNAME_FAILED); 329 return; 330 } 331 332 memset(&bi, 0, sizeof(BROWSEINFOW)); 333 334 bi.hwndOwner = hwnd; 335 bi.pidlRoot = root; 336 bi.ulFlags = BIF_RETURNONLYFSDIRS | BIF_USENEWUI | BIF_NONEWFOLDERBUTTON; 337 338 pidl = SHBrowseForFolderW(&bi); 339 340 if (!pidl) 341 return; 342 343 if (!SHGetPathFromIDListW(pidl, parent)) { 344 ShowStringError(hwnd, IDS_SHGETPATHFROMIDLIST_FAILED); 345 return; 346 } 347 348 h = CreateFileW(parent, FILE_TRAVERSE, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); 349 if (h == INVALID_HANDLE_VALUE) { 350 ShowStringError(hwnd, IDS_SEND_CANT_OPEN_DIR, parent, GetLastError(), format_message(GetLastError()).c_str()); 351 return; 352 } 353 354 Status = NtFsControlFile(h, NULL, NULL, NULL, &iosb, FSCTL_BTRFS_GET_FILE_IDS, NULL, 0, &bgfi, sizeof(btrfs_get_file_ids)); 355 if (!NT_SUCCESS(Status)) { 356 ShowStringError(hwnd, IDS_GET_FILE_IDS_FAILED, Status, format_ntstatus(Status).c_str()); 357 CloseHandle(h); 358 return; 359 } 360 361 CloseHandle(h); 362 363 if (bgfi.inode != 0x100 || bgfi.top) { 364 ShowStringError(hwnd, IDS_NOT_SUBVOL); 365 return; 366 } 367 368 SetDlgItemTextW(hwnd, IDC_PARENT_SUBVOL, parent); 369 } 370 371 void BtrfsSend::AddClone(HWND hwnd) { 372 BROWSEINFOW bi; 373 PIDLIST_ABSOLUTE root, pidl; 374 HRESULT hr; 375 WCHAR path[MAX_PATH], volpathw[MAX_PATH]; 376 HANDLE h; 377 NTSTATUS Status; 378 IO_STATUS_BLOCK iosb; 379 btrfs_get_file_ids bgfi; 380 381 if (!GetVolumePathNameW(subvol.c_str(), volpathw, (sizeof(volpathw) / sizeof(WCHAR)) - 1)) { 382 ShowStringError(hwnd, IDS_RECV_GETVOLUMEPATHNAME_FAILED, GetLastError(), format_message(GetLastError()).c_str()); 383 return; 384 } 385 386 hr = SHParseDisplayName(volpathw, 0, &root, 0, 0); 387 if (FAILED(hr)) { 388 ShowStringError(hwnd, IDS_SHPARSEDISPLAYNAME_FAILED); 389 return; 390 } 391 392 memset(&bi, 0, sizeof(BROWSEINFOW)); 393 394 bi.hwndOwner = hwnd; 395 bi.pidlRoot = root; 396 bi.ulFlags = BIF_RETURNONLYFSDIRS | BIF_USENEWUI | BIF_NONEWFOLDERBUTTON; 397 398 pidl = SHBrowseForFolderW(&bi); 399 400 if (!pidl) 401 return; 402 403 if (!SHGetPathFromIDListW(pidl, path)) { 404 ShowStringError(hwnd, IDS_SHGETPATHFROMIDLIST_FAILED); 405 return; 406 } 407 408 h = CreateFileW(path, FILE_TRAVERSE, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); 409 if (h == INVALID_HANDLE_VALUE) { 410 ShowStringError(hwnd, IDS_SEND_CANT_OPEN_DIR, path, GetLastError(), format_message(GetLastError()).c_str()); 411 return; 412 } 413 414 Status = NtFsControlFile(h, NULL, NULL, NULL, &iosb, FSCTL_BTRFS_GET_FILE_IDS, NULL, 0, &bgfi, sizeof(btrfs_get_file_ids)); 415 if (!NT_SUCCESS(Status)) { 416 ShowStringError(hwnd, IDS_GET_FILE_IDS_FAILED, Status, format_ntstatus(Status).c_str()); 417 CloseHandle(h); 418 return; 419 } 420 421 CloseHandle(h); 422 423 if (bgfi.inode != 0x100 || bgfi.top) { 424 ShowStringError(hwnd, IDS_NOT_SUBVOL); 425 return; 426 } 427 428 SendMessageW(GetDlgItem(hwnd, IDC_CLONE_LIST), LB_ADDSTRING, 0, (LPARAM)path); 429 } 430 431 void BtrfsSend::RemoveClone(HWND hwnd) { 432 LRESULT sel; 433 HWND cl = GetDlgItem(hwnd, IDC_CLONE_LIST); 434 435 sel = SendMessageW(cl, LB_GETCURSEL, 0, 0); 436 437 if (sel == LB_ERR) 438 return; 439 440 SendMessageW(cl, LB_DELETESTRING, sel, 0); 441 442 if (SendMessageW(cl, LB_GETCURSEL, 0, 0) == LB_ERR) 443 EnableWindow(GetDlgItem(hwnd, IDC_CLONE_REMOVE), FALSE); 444 } 445 446 INT_PTR BtrfsSend::SendDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { 447 switch (uMsg) { 448 case WM_INITDIALOG: 449 this->hwnd = hwndDlg; 450 451 GetDlgItemTextW(hwndDlg, IDCANCEL, closetext, sizeof(closetext) / sizeof(WCHAR)); 452 break; 453 454 case WM_COMMAND: 455 switch (HIWORD(wParam)) { 456 case BN_CLICKED: 457 switch (LOWORD(wParam)) { 458 case IDOK: 459 StartSend(hwndDlg); 460 return TRUE; 461 462 case IDCANCEL: 463 if (started) { 464 TerminateThread(thread, 0); 465 466 if (stream != INVALID_HANDLE_VALUE) { 467 FILE_DISPOSITION_INFO fdi; 468 469 fdi.DeleteFile = TRUE; 470 471 SetFileInformationByHandle(stream, FileDispositionInfo, &fdi, sizeof(FILE_DISPOSITION_INFO)); 472 CloseHandle(stream); 473 } 474 475 if (dirh != INVALID_HANDLE_VALUE) 476 CloseHandle(dirh); 477 478 started = FALSE; 479 480 SetDlgItemTextW(hwndDlg, IDCANCEL, closetext); 481 482 EnableWindow(GetDlgItem(hwnd, IDOK), TRUE); 483 EnableWindow(GetDlgItem(hwnd, IDC_STREAM_DEST), TRUE); 484 EnableWindow(GetDlgItem(hwnd, IDC_BROWSE), TRUE); 485 } else 486 EndDialog(hwndDlg, 1); 487 return TRUE; 488 489 case IDC_BROWSE: 490 Browse(hwndDlg); 491 return TRUE; 492 493 case IDC_INCREMENTAL: 494 incremental = IsDlgButtonChecked(hwndDlg, LOWORD(wParam)); 495 496 EnableWindow(GetDlgItem(hwnd, IDC_PARENT_SUBVOL), incremental); 497 EnableWindow(GetDlgItem(hwnd, IDC_PARENT_BROWSE), incremental); 498 return TRUE; 499 500 case IDC_PARENT_BROWSE: 501 BrowseParent(hwndDlg); 502 return TRUE; 503 504 case IDC_CLONE_ADD: 505 AddClone(hwndDlg); 506 return TRUE; 507 508 case IDC_CLONE_REMOVE: 509 RemoveClone(hwndDlg); 510 return TRUE; 511 } 512 break; 513 514 case LBN_SELCHANGE: 515 switch (LOWORD(wParam)) { 516 case IDC_CLONE_LIST: 517 EnableWindow(GetDlgItem(hwnd, IDC_CLONE_REMOVE), TRUE); 518 return TRUE; 519 } 520 break; 521 } 522 break; 523 } 524 525 return FALSE; 526 } 527 528 static INT_PTR CALLBACK stub_SendDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { 529 BtrfsSend* bs; 530 531 if (uMsg == WM_INITDIALOG) { 532 SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)lParam); 533 bs = (BtrfsSend*)lParam; 534 } else 535 bs = (BtrfsSend*)GetWindowLongPtr(hwndDlg, GWLP_USERDATA); 536 537 if (bs) 538 return bs->SendDlgProc(hwndDlg, uMsg, wParam, lParam); 539 else 540 return FALSE; 541 } 542 543 void BtrfsSend::Open(HWND hwnd, LPWSTR path) { 544 subvol = path; 545 546 if (DialogBoxParamW(module, MAKEINTRESOURCEW(IDD_SEND_SUBVOL), hwnd, stub_SendDlgProc, (LPARAM)this) <= 0) 547 ShowError(hwnd, GetLastError()); 548 } 549 550 void CALLBACK SendSubvolGUIW(HWND hwnd, HINSTANCE hinst, LPWSTR lpszCmdLine, int nCmdShow) { 551 HANDLE token; 552 TOKEN_PRIVILEGES tp; 553 LUID luid; 554 BtrfsSend* bs; 555 556 set_dpi_aware(); 557 558 if (!OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &token)) { 559 ShowError(hwnd, GetLastError()); 560 return; 561 } 562 563 if (!LookupPrivilegeValueW(NULL, L"SeManageVolumePrivilege", &luid)) { 564 ShowError(hwnd, GetLastError()); 565 return; 566 } 567 568 tp.PrivilegeCount = 1; 569 tp.Privileges[0].Luid = luid; 570 tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED; 571 572 if (!AdjustTokenPrivileges(token, FALSE, &tp, sizeof(TOKEN_PRIVILEGES), NULL, NULL)) { 573 ShowError(hwnd, GetLastError()); 574 return; 575 } 576 577 bs = new BtrfsSend; 578 579 bs->Open(hwnd, lpszCmdLine); 580 581 delete bs; 582 583 CloseHandle(token); 584 } 585 586 static void send_subvol(std::wstring subvol, std::wstring file, std::wstring parent, std::vector<std::wstring> clones) { 587 char* buf; 588 HANDLE dirh, stream; 589 ULONG bss_size, i; 590 btrfs_send_subvol* bss; 591 IO_STATUS_BLOCK iosb; 592 NTSTATUS Status; 593 btrfs_send_header header; 594 btrfs_send_command end; 595 BOOL success = FALSE; 596 597 buf = (char*)malloc(SEND_BUFFER_LEN); 598 599 dirh = CreateFileW(subvol.c_str(), FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); 600 if (dirh == INVALID_HANDLE_VALUE) 601 goto end3; 602 603 stream = CreateFileW(file.c_str(), FILE_WRITE_DATA | DELETE, 0, NULL, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); 604 if (stream == INVALID_HANDLE_VALUE) { 605 CloseHandle(dirh); 606 goto end3; 607 } 608 609 bss_size = offsetof(btrfs_send_subvol, clones[0]) + (clones.size() * sizeof(HANDLE)); 610 bss = (btrfs_send_subvol*)malloc(bss_size); 611 memset(bss, 0, bss_size); 612 613 if (parent != L"") { 614 HANDLE parenth; 615 616 parenth = CreateFileW(parent.c_str(), FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); 617 if (parenth == INVALID_HANDLE_VALUE) 618 goto end2; 619 620 bss->parent = parenth; 621 } else 622 bss->parent = NULL; 623 624 bss->num_clones = clones.size(); 625 626 for (i = 0; i < bss->num_clones; i++) { 627 HANDLE h; 628 629 h = CreateFileW(clones[i].c_str(), FILE_READ_ATTRIBUTES, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL); 630 if (h == INVALID_HANDLE_VALUE) { 631 ULONG j; 632 633 for (j = 0; j < i; j++) { 634 CloseHandle(bss->clones[j]); 635 } 636 637 if (bss->parent) CloseHandle(bss->parent); 638 goto end2; 639 } 640 641 bss->clones[i] = h; 642 } 643 644 Status = NtFsControlFile(dirh, NULL, NULL, NULL, &iosb, FSCTL_BTRFS_SEND_SUBVOL, bss, bss_size, NULL, 0); 645 646 for (i = 0; i < bss->num_clones; i++) { 647 CloseHandle(bss->clones[i]); 648 } 649 650 if (bss->parent) CloseHandle(bss->parent); 651 652 if (!NT_SUCCESS(Status)) 653 goto end2; 654 655 memcpy(header.magic, BTRFS_SEND_MAGIC, sizeof(BTRFS_SEND_MAGIC)); 656 header.version = 1; 657 658 if (!WriteFile(stream, &header, sizeof(header), NULL, NULL)) 659 goto end2; 660 661 do { 662 Status = NtFsControlFile(dirh, NULL, NULL, NULL, &iosb, FSCTL_BTRFS_READ_SEND_BUFFER, NULL, 0, buf, SEND_BUFFER_LEN); 663 664 if (NT_SUCCESS(Status)) 665 WriteFile(stream, buf, iosb.Information, NULL, NULL); 666 } while (NT_SUCCESS(Status)); 667 668 if (Status != STATUS_END_OF_FILE) 669 goto end2; 670 671 end.length = 0; 672 end.cmd = BTRFS_SEND_CMD_END; 673 end.csum = 0x9dc96c50; 674 675 if (!WriteFile(stream, &end, sizeof(end), NULL, NULL)) 676 goto end2; 677 678 SetEndOfFile(stream); 679 680 success = TRUE; 681 682 end2: 683 if (!success) { 684 FILE_DISPOSITION_INFO fdi; 685 686 fdi.DeleteFile = TRUE; 687 688 SetFileInformationByHandle(stream, FileDispositionInfo, &fdi, sizeof(FILE_DISPOSITION_INFO)); 689 } 690 691 CloseHandle(dirh); 692 CloseHandle(stream); 693 694 end3: 695 free(buf); 696 } 697 698 void CALLBACK SendSubvolW(HWND hwnd, HINSTANCE hinst, LPWSTR lpszCmdLine, int nCmdShow) { 699 LPWSTR* args; 700 int num_args; 701 std::wstring subvol = L"", parent = L"", file = L""; 702 std::vector<std::wstring> clones; 703 704 args = CommandLineToArgvW(lpszCmdLine, &num_args); 705 706 if (!args) 707 return; 708 709 if (num_args >= 2) { 710 HANDLE token; 711 TOKEN_PRIVILEGES tp; 712 LUID luid; 713 int i; 714 715 if (!OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &token)) 716 goto end; 717 718 if (!LookupPrivilegeValueW(NULL, L"SeManageVolumePrivilege", &luid)) 719 goto end; 720 721 tp.PrivilegeCount = 1; 722 tp.Privileges[0].Luid = luid; 723 tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED; 724 725 if (!AdjustTokenPrivileges(token, FALSE, &tp, sizeof(TOKEN_PRIVILEGES), NULL, NULL)) 726 goto end; 727 728 CloseHandle(token); 729 730 for (i = 0; i < num_args; i++) { 731 if (args[i][0] == '-') { 732 if (args[i][2] == 0 && i < num_args - 1) { 733 if (args[i][1] == 'p') { 734 parent = args[i+1]; 735 i++; 736 } else if (args[i][1] == 'c') { 737 clones.push_back(args[i+1]); 738 i++; 739 } 740 } 741 } else { 742 if (subvol == L"") 743 subvol = args[i]; 744 else if (file == L"") 745 file = args[i]; 746 } 747 } 748 749 if (subvol != L"" && file != L"") 750 send_subvol(subvol, file, parent, clones); 751 } 752 753 end: 754 LocalFree(args); 755 } 756